你的浏览器版本过低,可能导致网站不能正常访问!
为了你能正常使用网站功能,请使用这些浏览器。

STM32使用硬件I2C读取SHTC3温湿度传感器并显示在0.96寸OLED屏上

[复制链接]
攻城狮Melo 发布时间:2022-11-13 15:14
STM32使用硬件I2C读取SHTC3温湿度传感器的数据并显示在0.96寸OLED屏上
我用的是STM32F103C8T6,程序用的是ST标准库写的。

实现效果图
微信图片_20221113151445.jpg

微信图片_20221113151442.gif


I2C协议简介
I2C 通讯协议(Inter-Integrated Circuit)是由 Phiilps 公司开发的,由于它引脚少,硬件实现简单,可扩展性强,不需要 USART、CAN 等通讯协议的外部收发设备(那些电平转化芯片),现在被广泛地使用在系统内多个集成电路(IC)间的通讯。
I2C只有一跟数据总线 SDA(Serial Data Line),串行数据总线,只能一位一位的发送数据,属于串行通信,采用半双工通信
半双工通信:可以实现双向的通信,但不能在两个方向上同时进行,必须轮流交替进行,其实也可以理解成一种可以切换方向的单工通信,同一时刻必须只能一个方向传输,只需一根数据线.
对于I2C通讯协议把它分为物理层和协议层物理层规定通讯系统中具有机械、电子功能部分的特性(硬件部分),确保原始数据在物理媒体的传输。协议层主要规定通讯逻辑,统一收发双方的数据打包、解包标准(软件层面)。

I2C物理层
I2C 通讯设备之间的常用连接方式

微信图片_20221113151439.png

(1) 它是一个支持多设备的总线。“总线”指多个设备共用的信号线。在一个 I2C 通讯总线中,可连接多个 I2C 通讯设备,支持多个通讯主机及多个通讯从机。
(2) 一个 I2C 总线只使用两条总线线路,一条双向串行数据线SDA(Serial Data Line ),一条串行时钟线SCL(Serial Data Line )。数据线即用来表示数据,时钟线用于数据收发同步
(3) 总线通过上拉电阻接到电源。当 I2C 设备空闲时会输出高阻态,而当所有设备都空闲,都输出高阻态时,由上拉电阻把总线拉成高电平
I2C通信时单片机GPIO口必须设置为开漏输出,否则可能会造成短路。
关于更多STM32的I2C相关信息和使用方法可以看这篇文章:https://url.zeruns.tech/JC0Ah
我这里就不详细讲解了。

SHTC3温湿度传感器
微信图片_20221113151435.png

浏览数据手册可以得到一个大概信息,SHTC3是一个可以检测温度和湿度的传感器,
温度范围:-40℃~125℃
湿度范围:0%~100%
工作电压:1.6v~3.6v
通讯方式:i2c
时钟频率:三种模式分别是 0 ~ 100kHz 0 ~ 400kHz 0 ~ 1000kHz
找到如下几个关键信息

微信图片_20221113151432.png

温湿度设备地址和读写命令
在实际的使用过程中,SHTC3的设备地址需要与读写数据/命令方向位组成一个字节同时发送,字节的最低位为读写数据/命令方向位,高7位是SHTC3的设备地址。
如果要通过I2C写数据或命令给SHTC3,在I2C起始信号之后,需要发送“1110 0000”,即0xE0给SHTC3,除了通过高7位“1110 000”的设备地址寻址还通过最低位“0”通知SHTC3接下来是写数据或命令操作。
如果要通过I2C读取SHTC3中的数据,在I2C起始信号之后,需要发送“1110 0001”,即0xE1给SHTC3,除了通过高7位“1110 000”的设备地址寻址还通过最低位“1”通知SHTC3接下来是读取数据的操作。
简单来说就是,0xE0表示写数据,0xE1表示读数据。不过使用STM32硬件I2C时只需要输入0xE0就行,最低位标准库会处理的。

读取温湿度数据
微信图片_20221113151427.png

可知,不同的命令,除了获取的数据顺序不一样,还有一个Clock Stretching Enable 和 Disable的区别。
Clock Stretching是时钟拉伸的意思。如果使用Clock Stretching Enable命令的话,那么发送完测量命令之后,在SHTC3测量温度湿度数据的过程中,SHTC3会拉低I2C的时钟线SCL,通过这样来禁止主机发送命令给SHTC3,只有当SHTC3完成温度湿度数据测量时,SHTC3才会释放时钟线SCL。
如果使用Clock Stretching Disable命令的话,在SHTC3测量数据的过程中,SHTC3并不会拉低I2C的时钟线SCL,只是如果主机在SHTC3测量数据的过程中发送命令或数据的时候,SHTC3是不会响应主机的,主机可以通过SHTC3是否有响应信号来判断SHTC3是否完成数据的测量。
从数据手册可知,一个测量周期包概括四个步骤:
  • 发送唤醒命令。
  • 发送测量命令
  • 读取测量完成之后的数据。
  • 发送休眠命令。  


以上唤醒命令和休眠命令在数据手册中查询。

微信图片_20221113151421.png

总结如下:
  • 唤醒SHTC3:先发送写入指令(0xE0),再发送唤醒指令高位(0x35),再发送唤醒指令低位(0x17)。
  • 等待唤醒:数据手册上写的最大唤醒时间是240us,等待的时间大于这个就行了。
  • 发送采集指令:先发送写入指令(0xE0),再发送采集指令的高位和低位。采集指令有多个,根据需要自行选择。
  • 接收数据:发送读取指令(0xE1),连续接收6个字节数据。如果采集的指令是先存温度,那么这6个字节的第1-2个字节就是温度数值,第3个字节是温度校验。第4-5个字节是湿度数值,第6个字节是湿度校验。如果采集的指令是先存湿度,则前3个字节和后3个字节相反。
  • 进入睡眠:发送写入指令,再发送睡眠指令进入睡眠。


数据的计算
由shtc3数据手册可知

微信图片_20221113151416.png

例如:采集到的湿度数值是0x6501,换算成十进制是25857。
则:湿度 = 100 * 25857 / 65536 = 39.45 (单位:%)
采集到的温度数值是0x6600,换算成十进制是26112。
则:温度 = -45 + 175 * 26112 / 65536 = 24.72 (单位:℃)

程序
这里就放出main.c、shtc3.c和oled.c这三个主要的代码,其他的请下载下面链接的压缩包。
SHTC3和OLED模块的SCL接PB6,SDA接PB7。

main.c
  1. #include "stm32f10x.h"                  // Device header
  2. #include "Delay.h"
  3. #include "OLED.h"
  4. #include "IWDG.h"
  5. #include "SHTC3.h"

  6. uint16_t numlen(uint16_t num);

  7. int main(void)
  8. {
  9.   IWDG_Configuration();  //初始化看门狗
  10.   OLED_Init();      //初始化OLED屏
  11.   SHTC3_I2C_Init();    //初始化SHTC3
  12.   
  13.   OLED_ShowString(1, 1, "T:");
  14.   OLED_ShowString(2, 1, "H:");
  15.   OLED_ShowString(4, 1, "err_count:");

  16.   uint32_t a=0;
  17.   uint16_t err_count=0;
  18.   
  19.   while (1)
  20.   {
  21.     a++;
  22.     OLED_ShowNum(3, 1, a, 9);
  23.     if(a==999999999)a=0;

  24.     float Temp,Hum;      //声明变量存放温湿度数据

  25.     if(ReadSHTC3(&Hum,&Temp))  //读取温湿度数据
  26.     {
  27.       if(Temp>=0)
  28.       {
  29.         char String[10];
  30.         sprintf(String, "%.2fC", Temp);  //格式化字符串输出到字符串变量
  31.         OLED_ShowString(1, 3, String);  //显示温度
  32.         /*
  33.         OLED_ShowNum(1,3, (uint8_t)Temp, numlen((uint8_t)Temp));//显示温度整数部分
  34.         OLED_ShowChar(1, 3+numlen((uint8_t)Temp), '.');      //显示小数点
  35.         OLED_ShowNum(1,3+numlen((uint8_t)Temp)+1, (uint8_t)(Temp*100)%100, 2);  //显示温度小数部分
  36.         OLED_ShowChar(1, 3+numlen((uint8_t)Temp)+1+2, 'C');    //显示符号
  37.         */

  38.         sprintf(String, "%.2f%%", Hum);  //格式化字符串输出到字符串变量
  39.         OLED_ShowString(2, 3, String);  //显示湿度
  40.         /*
  41.         OLED_ShowNum(2,3, (uint8_t)Hum, numlen((uint8_t)Hum));  //显示湿度整数部分
  42.         OLED_ShowChar(2, 3+numlen((uint8_t)Hum), '.');      //显示小数点
  43.         OLED_ShowNum(2,3+numlen((uint8_t)Hum)+1, (uint8_t)(Hum*100)%100, 2);  //显示湿度小数部分
  44.         OLED_ShowChar(2, 3+numlen((uint8_t)Hum)+1+2, '%');      //显示符号
  45.         */
  46.       }else
  47.       {
  48.         char String[10];
  49.         sprintf(String, "-%.2fC", Temp);//格式化字符串输出到字符串变量
  50.         OLED_ShowString(1, 3, String);  //显示温度
  51.         /*
  52.         OLED_ShowChar(1, 3, '-');      //显示负号
  53.         OLED_ShowNum(1,3+1, (uint8_t)Temp, numlen((uint8_t)Temp));  //显示温度整数部分
  54.         OLED_ShowChar(1, 3+1+numlen((uint8_t)Temp), '.');      //显示小数点
  55.         OLED_ShowNum(1,3+1+numlen((uint8_t)Temp)+1, (uint8_t)(Temp*100)%100, 2);  //显示温度小数部分
  56.         OLED_ShowChar(1, 3+1+numlen((uint8_t)Temp)+1+2, 'C');    //显示符号
  57.         */

  58.         sprintf(String, "%.2f%%", Hum);  //格式化字符串输出到字符串变量
  59.         OLED_ShowString(2, 3, String);  //显示湿度
  60.         /*
  61.         OLED_ShowNum(2,3, (uint8_t)Hum, numlen((uint8_t)Hum));  //显示湿度整数部分
  62.         OLED_ShowChar(2, 3+numlen((uint8_t)Hum), '.');      //显示小数点
  63.         OLED_ShowNum(2,3+numlen((uint8_t)Hum)+1, (uint8_t)(Hum*100)%100, 2);  //显示湿度小数部分
  64.         OLED_ShowChar(2, 3+numlen((uint8_t)Hum)+1+2, '%');      //显示符号
  65.         */
  66.       }
  67.     }
  68.     else
  69.     {
  70.       err_count++;
  71.       OLED_ShowNum(4,11, err_count, numlen(err_count));  //显示错误次数计数
  72.     }
  73.   /*
  74.   https://blog.zeruns.tech
  75.   */

  76.     Delay_ms(100);  //延时100毫秒

  77.     IWDG_FeedDog();  //喂狗(看门狗,超过1秒没有执行喂狗则自动复位)
  78.   }
  79. }

  80. /**
  81.   * @brief  计算整数长度
  82.   * @param  num 要计算长度的整数
  83.   * @retval 长度值
  84.   */
  85. uint16_t numlen(uint16_t num)
  86. {
  87.     uint16_t len = 0;        // 初始长度为0
  88.     for(; num > 0; ++len)    // 判断num是否大于0,否则长度+1
  89.         num /= 10;           // 使用除法进行运算,直到num小于1
  90.     return len;              // 返回长度的值
  91. }
复制代码

SHTC3.c
  1. #include "stm32f10x.h"
  2. #include "Delay.h"

  3. /*SHTC3地址*/
  4. #define SHTC3_ADDRESS 0xE0

  5. /*设置使用哪一个I2C*/
  6. #define I2Cx I2C1

  7. /*
  8. https://blog.zeruns.tech
  9. */

  10. /**
  11.   * @brief  CRC校验,CRC多项式为:x^8+x^5+x^4+1,即0x31
  12.   * @param  DAT 要校验的数据
  13.   * @retval 校验码
  14.   */
  15. uint8_t SHTC3_CRC_CAL(uint16_t DAT)
  16. {
  17.   uint8_t i,t,temp;
  18.   uint8_t CRC_BYTE;

  19.   CRC_BYTE = 0xFF;
  20.   temp = (DAT>>8) & 0xFF;

  21.   for(t = 0; t < 2; t++)
  22.   {
  23.     CRC_BYTE ^= temp;
  24.     for(i = 0;i < 8;i ++)
  25.     {
  26.       if(CRC_BYTE & 0x80)
  27.       {
  28.         CRC_BYTE <<= 1;
  29.         CRC_BYTE ^= 0x31;
  30.       }
  31.       else
  32.       {
  33.         CRC_BYTE <<= 1;
  34.       }
  35.     }

  36.     if(t == 0)
  37.     {
  38.       temp = DAT & 0xFF;
  39.     }
  40.   }

  41.   return CRC_BYTE;
  42. }

  43. /*发送起始信号*/
  44. void SHTC3_I2C_START(){
  45.     while( I2C_GetFlagStatus(I2Cx, I2C_FLAG_BUSY));//等待总线空闲
  46.   I2C_GenerateSTART(I2Cx, ENABLE);//发送起始信号
  47.   while( I2C_CheckEvent(I2Cx,I2C_EVENT_MASTER_MODE_SELECT)==ERROR);//检测EV5事件
  48. }

  49. /*发送停止信号*/
  50. void SHTC3_I2C_STOP(){
  51.     I2C_GenerateSTOP(I2Cx, ENABLE);//发送停止信号
  52. }

  53. /**
  54.   * @brief  发送两个字节数据
  55.   * @param  MSB 高8位
  56.   * @param  LSB 低8位
  57.   * @retval 无
  58.   */
  59. void SHTC3_WriteByte(uint8_t MSB,uint8_t LSB)
  60. {
  61.   SHTC3_I2C_START();  //发送起始信号
  62.   
  63.   I2C_Send7bitAddress(I2Cx, SHTC3_ADDRESS, I2C_Direction_Transmitter);    //发送设备写地址
  64.   while(I2C_CheckEvent(I2Cx,I2C_EVENT_MASTER_TRANSMITTER_MODE_SELECTED)==ERROR);  //检测EV6事件

  65.     I2C_SendData(I2Cx, MSB);//发送高8位数据
  66.   while (!I2C_CheckEvent(I2Cx, I2C_EVENT_MASTER_BYTE_TRANSMITTED));//检测EV8事件
  67.    
  68.   I2C_SendData(I2Cx, LSB);//发送低8位数据
  69.   while (!I2C_CheckEvent(I2Cx, I2C_EVENT_MASTER_BYTE_TRANSMITTED));//检测EV8事件
  70.   
  71.   I2C_GenerateSTOP(I2Cx, ENABLE);//发送停止信号
  72.   
  73. }

  74. /**
  75.   * @brief  读取数据
  76.   * @retval 读取到的字节数据
  77.   */
  78. uint8_t SHTC3_ReadData()
  79. {
  80.     while (!I2C_CheckEvent(I2Cx,I2C_EVENT_MASTER_BYTE_RECEIVED));//检测EV7事件
  81.   return I2C_ReceiveData(I2Cx);//读取数据并返回
  82. }

  83. /*软件复位SHTC3*/
  84. void SHTC3_SoftReset(void)                    
  85. {
  86.     SHTC3_WriteByte(0x80,0x5D);    //重置SHTC3
  87. }

  88. /*引脚初始化*/
  89. void SHTC3_I2C_Init(void)
  90. {
  91.   RCC_APB1PeriphClockCmd(RCC_APB1Periph_I2C1,ENABLE);  //使能I2C1时钟
  92.   RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB,ENABLE);//使能GPIOB时钟

  93.   /*STM32F103芯片的硬件I2C1: PB6 -- SCL; PB7 -- SDA */
  94.   GPIO_InitTypeDef  GPIO_InitStructure;               //定义结构体配置GPIO
  95.   GPIO_InitStructure.GPIO_Pin =  GPIO_Pin_6 | GPIO_Pin_7;
  96.   GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;   
  97.   GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_OD;    //设置输出模式为开漏输出,需接上拉电阻
  98.   GPIO_Init(GPIOB, &GPIO_InitStructure);              //初始化GPIO
  99.   
  100.   I2C_DeInit(I2Cx);  //将外设I2C寄存器重设为缺省值
  101.   I2C_InitTypeDef  I2C_InitStructure;                 //定义结构体配置I2C
  102.   I2C_InitStructure.I2C_Mode = I2C_Mode_I2C;      //工作模式
  103.   I2C_InitStructure.I2C_DutyCycle = I2C_DutyCycle_2;  //时钟占空比,Tlow/Thigh = 2
  104.   I2C_InitStructure.I2C_OwnAddress1 = 0x30;  //主机的I2C地址,用不到则随便写,无影响
  105.   I2C_InitStructure.I2C_Ack = I2C_Ack_Enable;  //使能应答位
  106.   I2C_InitStructure.I2C_AcknowledgedAddress = I2C_AcknowledgedAddress_7bit;//设置地址长度7位
  107.   I2C_InitStructure.I2C_ClockSpeed = 400000;  //I2C传输速度,400K,根据自己所用芯片手册查看支持的速度。  
  108.   I2C_Init(I2Cx, &I2C_InitStructure);         //初始化I2C

  109.   I2C_Cmd(I2Cx, ENABLE);  //启用I2C

  110.   SHTC3_WriteByte(0X35,0X17);//唤醒SHTC3
  111.   
  112.   Delay_us(200);
  113. }

  114. /**
  115.   * @brief  读取SHTC3数据
  116.   * @param  *Hum 湿度
  117.   * @param  *Temp 温度
  118.   * @retval 1 - 读取成功;0 - 读取失败
  119.   */
  120. uint8_t ReadSHTC3(float *Hum,float *Temp)
  121. {
  122.     uint16_t HumData,TempData,HumCRC,TempCRC;//声明变量存放读取的数据
  123.    
  124.     /*SHTC3_WriteByte(0X35,0X17);//唤醒SHTC3
  125.   
  126.   Delay_us(300);*/
  127.    
  128.     SHTC3_WriteByte(0X5C,0X24);//发送指令,先读取湿度数据,时钟拉伸(测量数据期间拉低SCL时钟线,占用总线)
  129.   
  130.     SHTC3_I2C_START();//发送起始信号
  131.    
  132.   I2C_Send7bitAddress(I2Cx,SHTC3_ADDRESS,I2C_Direction_Receiver);//发送设备读地址
  133.   
  134.   while( I2C_CheckEvent(I2Cx,I2C_EVENT_MASTER_RECEIVER_MODE_SELECTED )==ERROR);//检测EV6事件
  135.   
  136.     HumData = SHTC3_ReadData(); //读取湿度高8位数据
  137.     HumData=HumData<<8;         //左移8位
  138.   HumData |= SHTC3_ReadData();//读取湿度低8位数据
  139.     HumCRC = SHTC3_ReadData();  //读取湿度CRC校验数据

  140.     TempData = SHTC3_ReadData();//读取温度高8位数据
  141.     TempData=TempData<<8;       //左移8位
  142.   TempData |= SHTC3_ReadData();//读取温度低8位数据
  143.     TempCRC = SHTC3_ReadData(); //读取温度CRC校验数据

  144.     SHTC3_I2C_STOP();   //发送停止信号
  145.    
  146.   //SHTC3_WriteByte(0XB0,0X98);//发送休眠指令

  147.     if( SHTC3_CRC_CAL(HumData)==HumCRC && SHTC3_CRC_CAL(TempData)==TempCRC ){   //对接收到数据进行CRC校验
  148.        *Hum = (float)HumData*100/65536;        //将接收的16位二进制数据转换为10进制湿度数据
  149.        *Temp = (float)TempData*175/65536-45;   //将接收的16位二进制数据转换为10进制温度数据
  150.        return 1;
  151.     }
  152.     else{
  153.         return 0;
  154.     }
  155. }
复制代码

OLED.c
  1. #include "stm32f10x.h"
  2. #include "OLED_Font.h"

  3. /*OLED屏地址*/
  4. #define OLED_ADDRESS 0x78

  5. /*设置哪一个使用I2C*/
  6. #define I2Cx I2C1

  7. /*
  8. https://blog.zeruns.tech
  9. */

  10. /*引脚初始化*/
  11. void OLED_I2C_Init(void)
  12. {
  13.   RCC_APB1PeriphClockCmd(RCC_APB1Periph_I2C1,ENABLE);  //使能I2C1时钟
  14.   RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB,ENABLE);//使能GPIOB时钟

  15.   /*STM32F103芯片的硬件I2C: PB6 -- SCL; PB7 -- SDA */
  16.   GPIO_InitTypeDef  GPIO_InitStructure;
  17.   GPIO_InitStructure.GPIO_Pin =  GPIO_Pin_6 | GPIO_Pin_7;
  18.   GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
  19.   GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_OD;    //设置输出模式为开漏输出,需接上拉电阻
  20.   GPIO_Init(GPIOB, &GPIO_InitStructure);
  21.   
  22.   I2C_DeInit(I2Cx);  //将外设I2C寄存器重设为缺省值
  23.   I2C_InitTypeDef  I2C_InitStructure;
  24.   I2C_InitStructure.I2C_Mode = I2C_Mode_I2C;      //工作模式
  25.   I2C_InitStructure.I2C_DutyCycle = I2C_DutyCycle_2;  //时钟占空比,Tlow/Thigh = 2
  26.   I2C_InitStructure.I2C_OwnAddress1 = 0x30;  //主机的I2C地址,用不到则随便写,无影响
  27.   I2C_InitStructure.I2C_Ack = I2C_Ack_Enable;  //使能应答位
  28.   I2C_InitStructure.I2C_AcknowledgedAddress = I2C_AcknowledgedAddress_7bit;//设置地址长度7位
  29.   I2C_InitStructure.I2C_ClockSpeed = 400000;  //I2C传输速度,400K,根据自己所用芯片手册查看支持的速度。  
  30.   I2C_Init(I2Cx, &I2C_InitStructure);

  31.   I2C_Cmd(I2Cx, ENABLE);
  32. }

  33. void I2C_WriteByte(uint8_t addr,uint8_t data)
  34. {
  35.   
  36.   while( I2C_GetFlagStatus(I2Cx, I2C_FLAG_BUSY));
  37.   
  38.   //发送起始信号
  39.   I2C_GenerateSTART(I2Cx, ENABLE);
  40.   //检测EV5事件
  41.   while( I2C_CheckEvent(I2Cx,I2C_EVENT_MASTER_MODE_SELECT)==ERROR);
  42.   //发送设备写地址
  43.   I2C_Send7bitAddress(I2Cx, OLED_ADDRESS, I2C_Direction_Transmitter);
  44.   //检测EV6事件
  45.   while( I2C_CheckEvent(I2Cx,I2C_EVENT_MASTER_TRANSMITTER_MODE_SELECTED)==ERROR);
  46.   
  47.   //发送要操作设备内部的地址
  48.   I2C_SendData(I2Cx, addr);
  49.   //检测EV8_2事件
  50.   while (!I2C_CheckEvent(I2Cx, I2C_EVENT_MASTER_BYTE_TRANSMITTED));
  51.   
  52.   I2C_SendData(I2Cx, data);//发送数据
  53.   while (!I2C_CheckEvent(I2Cx, I2C_EVENT_MASTER_BYTE_TRANSMITTED));

  54.   //发送停止信号
  55.   I2C_GenerateSTOP(I2Cx, ENABLE);
  56. }

  57. /**
  58.   * @brief  OLED写命令
  59.   * @param  Command 要写入的命令
  60.   * @retval 无
  61.   */
  62. void OLED_WriteCommand(unsigned char Command)//写命令
  63. {
  64.   I2C_WriteByte(0x00, Command);
  65. }

  66. /**
  67.   * @brief  OLED写数据
  68.   * @param  Data 要写入的数据
  69.   * @retval 无
  70. */
  71. void OLED_WriteData(unsigned char Data)//写数据
  72. {
  73.   I2C_WriteByte(0x40, Data);
  74. }

  75. /**
  76.   * @brief  OLED设置光标位置
  77.   * @param  Y 以左上角为原点,向下方向的坐标,范围:0~7
  78.   * @param  X 以左上角为原点,向右方向的坐标,范围:0~127
  79.   * @retval 无
  80.   */
  81. void OLED_SetCursor(uint8_t Y, uint8_t X)
  82. {
  83.   OLED_WriteCommand(0xB0 | Y);          //设置Y位置
  84.   OLED_WriteCommand(0x10 | ((X & 0xF0) >> 4));  //设置X位置低4位
  85.   OLED_WriteCommand(0x00 | (X & 0x0F));      //设置X位置高4位
  86. }

  87. /**
  88.   * @brief  OLED清屏
  89.   * @param  无
  90.   * @retval 无
  91.   */
  92. void OLED_Clear(void)
  93. {  
  94.   uint8_t i, j;
  95.   for (j = 0; j < 8; j++)
  96.   {
  97.     OLED_SetCursor(j, 0);
  98.     for(i = 0; i < 128; i++)
  99.     {
  100.       OLED_WriteData(0x00);
  101.     }
  102.   }
  103. }

  104. /**
  105.   * @brief  OLED部分清屏
  106.   * @param  Line 行位置,范围:1~4
  107.   * @param  start 列开始位置,范围:1~16
  108.   * @param  end 列开始位置,范围:1~16
  109.   * @retval 无
  110.   */
  111. void OLED_Clear_Part(uint8_t Line, uint8_t start, uint8_t end)
  112. {  
  113.   uint8_t i,Column;
  114.   for(Column = start; Column <= end; Column++)
  115.   {
  116.     OLED_SetCursor((Line - 1) * 2, (Column - 1) * 8);    //设置光标位置在上半部分
  117.     for (i = 0; i < 8; i++)
  118.     {
  119.       OLED_WriteData(0x00);      //显示上半部分内容
  120.     }
  121.     OLED_SetCursor((Line - 1) * 2 + 1, (Column - 1) * 8);  //设置光标位置在下半部分
  122.     for (i = 0; i < 8; i++)
  123.     {
  124.       OLED_WriteData(0x00);    //显示下半部分内容
  125.     }
  126.   }
  127. }

  128. /**
  129.   * @brief  OLED显示一个字符
  130.   * @param  Line 行位置,范围:1~4
  131.   * @param  Column 列位置,范围:1~16
  132.   * @param  Char 要显示的一个字符,范围:ASCII可见字符
  133.   * @retval 无
  134.   */
  135. void OLED_ShowChar(uint8_t Line, uint8_t Column, char Char)
  136. {        
  137.   uint8_t i;
  138.   OLED_SetCursor((Line - 1) * 2, (Column - 1) * 8);    //设置光标位置在上半部分
  139.   for (i = 0; i < 8; i++)
  140.   {
  141.     OLED_WriteData(OLED_F8x16[Char - ' '][i]);      //显示上半部分内容
  142.   }
  143.   OLED_SetCursor((Line - 1) * 2 + 1, (Column - 1) * 8);  //设置光标位置在下半部分
  144.   for (i = 0; i < 8; i++)
  145.   {
  146.     OLED_WriteData(OLED_F8x16[Char - ' '][i + 8]);    //显示下半部分内容
  147.   }
  148. }

  149. /**
  150.   * @brief  OLED显示字符串
  151.   * @param  Line 起始行位置,范围:1~4
  152.   * @param  Column 起始列位置,范围:1~16
  153.   * @param  String 要显示的字符串,范围:ASCII可见字符
  154.   * @retval 无
  155.   */
  156. void OLED_ShowString(uint8_t Line, uint8_t Column, char *String)
  157. {
  158.   uint8_t i;
  159.   for (i = 0; String[i] != '\0'; i++)
  160.   {
  161.     OLED_ShowChar(Line, Column + i, String[i]);
  162.   }
  163. }

  164. /**
  165.   * @brief  OLED次方函数
  166.   * @retval 返回值等于X的Y次方
  167.   */
  168. uint32_t OLED_Pow(uint32_t X, uint32_t Y)
  169. {
  170.   uint32_t Result = 1;
  171.   while (Y--)
  172.   {
  173.     Result *= X;
  174.   }
  175.   return Result;
  176. }

  177. /**
  178.   * @brief  OLED显示数字(十进制,正数)
  179.   * @param  Line 起始行位置,范围:1~4
  180.   * @param  Column 起始列位置,范围:1~16
  181.   * @param  Number 要显示的数字,范围:0~4294967295
  182.   * @param  Length 要显示数字的长度,范围:1~10
  183.   * @retval 无
  184.   */
  185. void OLED_ShowNum(uint8_t Line, uint8_t Column, uint32_t Number, uint8_t Length)
  186. {
  187.   uint8_t i;
  188.   for (i = 0; i < Length; i++)              
  189.   {
  190.     OLED_ShowChar(Line, Column + i, Number / OLED_Pow(10, Length - i - 1) % 10 + '0');
  191.   }
  192. }

  193. /**
  194.   * @brief  OLED显示数字(十进制,带符号数)
  195.   * @param  Line 起始行位置,范围:1~4
  196.   * @param  Column 起始列位置,范围:1~16
  197.   * @param  Number 要显示的数字,范围:-2147483648~2147483647
  198.   * @param  Length 要显示数字的长度,范围:1~10
  199.   * @retval 无
  200.   */
  201. void OLED_ShowSignedNum(uint8_t Line, uint8_t Column, int32_t Number, uint8_t Length)
  202. {
  203.   uint8_t i;
  204.   uint32_t Number1;
  205.   if (Number >= 0)
  206.   {
  207.     OLED_ShowChar(Line, Column, '+');
  208.     Number1 = Number;
  209.   }
  210.   else
  211.   {
  212.     OLED_ShowChar(Line, Column, '-');
  213.     Number1 = -Number;
  214.   }
  215.   for (i = 0; i < Length; i++)              
  216.   {
  217.     OLED_ShowChar(Line, Column + i + 1, Number1 / OLED_Pow(10, Length - i - 1) % 10 + '0');
  218.   }
  219. }

  220. /**
  221.   * @brief  OLED显示数字(十六进制,正数)
  222.   * @param  Line 起始行位置,范围:1~4
  223.   * @param  Column 起始列位置,范围:1~16
  224.   * @param  Number 要显示的数字,范围:0~0xFFFFFFFF
  225.   * @param  Length 要显示数字的长度,范围:1~8
  226.   * @retval 无
  227.   */
  228. void OLED_ShowHexNum(uint8_t Line, uint8_t Column, uint32_t Number, uint8_t Length)
  229. {
  230.   uint8_t i, SingleNumber;
  231.   for (i = 0; i < Length; i++)              
  232.   {
  233.     SingleNumber = Number / OLED_Pow(16, Length - i - 1) % 16;
  234.     if (SingleNumber < 10)
  235.     {
  236.       OLED_ShowChar(Line, Column + i, SingleNumber + '0');
  237.     }
  238.     else
  239.     {
  240.       OLED_ShowChar(Line, Column + i, SingleNumber - 10 + 'A');
  241.     }
  242.   }
  243. }

  244. /**
  245.   * @brief  OLED显示数字(二进制,正数)
  246.   * @param  Line 起始行位置,范围:1~4
  247.   * @param  Column 起始列位置,范围:1~16
  248.   * @param  Number 要显示的数字,范围:0~1111 1111 1111 1111
  249.   * @param  Length 要显示数字的长度,范围:1~16
  250.   * @retval 无
  251.   */
  252. void OLED_ShowBinNum(uint8_t Line, uint8_t Column, uint32_t Number, uint8_t Length)
  253. {
  254.   uint8_t i;
  255.   for (i = 0; i < Length; i++)              
  256.   {
  257.     OLED_ShowChar(Line, Column + i, Number / OLED_Pow(2, Length - i - 1) % 2 + '0');
  258.   }
  259. }

  260. /**
  261.   * @brief  OLED初始化
  262.   * @param  无
  263.   * @retval 无
  264.   */
  265. void OLED_Init(void)
  266. {
  267.   uint32_t i, j;
  268.   
  269.   for (i = 0; i < 1000; i++)      //上电延时
  270.   {
  271.     for (j = 0; j < 1000; j++);
  272.   }
  273.   
  274.   OLED_I2C_Init();      //端口初始化
  275.   
  276.   OLED_WriteCommand(0xAE);  //关闭显示
  277.   
  278.   OLED_WriteCommand(0xD5);  //设置显示时钟分频比/振荡器频率
  279.   OLED_WriteCommand(0x80);
  280.   
  281.   OLED_WriteCommand(0xA8);  //设置多路复用率
  282.   OLED_WriteCommand(0x3F);
  283.   
  284.   OLED_WriteCommand(0xD3);  //设置显示偏移
  285.   OLED_WriteCommand(0x00);
  286.   
  287.   OLED_WriteCommand(0x40);  //设置显示开始行
  288.   
  289.   OLED_WriteCommand(0xA1);  //设置左右方向,0xA1正常 0xA0左右反置
  290.   
  291.   OLED_WriteCommand(0xC8);  //设置上下方向,0xC8正常 0xC0上下反置

  292.   OLED_WriteCommand(0xDA);  //设置COM引脚硬件配置
  293.   OLED_WriteCommand(0x12);
  294.   
  295.   OLED_WriteCommand(0x81);  //设置对比度控制
  296.   OLED_WriteCommand(0xCF);

  297.   OLED_WriteCommand(0xD9);  //设置预充电周期
  298.   OLED_WriteCommand(0xF1);

  299.   OLED_WriteCommand(0xDB);  //设置VCOMH取消选择级别
  300.   OLED_WriteCommand(0x30);

  301.   OLED_WriteCommand(0xA4);  //设置整个显示打开/关闭

  302.   OLED_WriteCommand(0xA6);  //设置正常/倒转显示

  303.   OLED_WriteCommand(0x8D);  //设置充电泵
  304.   OLED_WriteCommand(0x14);

  305.   OLED_WriteCommand(0xAF);  //开启显示
  306.    
  307.   OLED_Clear();        //OLED清屏
  308. }
复制代码
   

转载自: zeruns
收藏 评论0 发布时间:2022-11-13 15:14

举报

0个回答

所属标签

相似分享

官网相关资源

关于
我们是谁
投资者关系
意法半导体可持续发展举措
创新与技术
意法半导体官网
联系我们
联系ST分支机构
寻找销售人员和分销渠道
社区
媒体中心
活动与培训
隐私策略
隐私策略
Cookies管理
行使您的权利
官方最新发布
STM32N6 AI生态系统
STM32MCU,MPU高性能GUI
ST ACEPACK电源模块
意法半导体生物传感器
STM32Cube扩展软件包
关注我们
st-img 微信公众号
st-img 手机版