当前位置:   article > 正文

STM32+NRF2401+ESP8266采集温度数据发送到oneNET平台_开发板只有nrf2401借口怎么利用杜邦线连接esp8266

开发板只有nrf2401借口怎么利用杜邦线连接esp8266

工作不是搞这个的,只是感兴趣,哈哈,做了一个小实验;记录一下实验过程吧!

一、实验部分

1、发送部分

        使用的是正点原子STM32F103ZET6的主板,主板连接的硬件有DHT11(温度湿度模块) + BH1750(光照度采集模块)+ NRF2401(WIFI模块,主要是发送采集的数据到STMF103C8T6模块上,当然C8T6也连接一个NRF2401作为接收端);

2、接收部分

        使用的是正点原子STMF103C8T6的最小系统板,主板连接的硬件有NRF2401(WIFI模块)+ esp8266(连接oneNET平台); 

        注意:esp8266做了单独的供电;用的串口调试的时候没带起来esp8266,发送指令失败,查找原因是供电不足;

3、硬件部分

下面是使用的模块:

BH1750

DHT11

OLED

esp8266

NRF24L01

实验图片

oneNET平台

NRF24L01是一款工作在2.4-2.5GHz世界通用ISM频段的单片收发芯片,无线收发器包括:频率发生器 增强型 SchockBurstTM 模式控制器 功率放大器 晶体放大器 调制器 解调器 输出功率频道选择和协议的设置可以通过SPI接口进行设置极低的电流消耗,当工作在发射模式下发射功率为6dBm时电流消耗为9.0mA 接受模式为12.3mA掉电模式和待机模式下电流消耗模式更低。
 
球开放ISM 频段,最大0dBm 发射功率,免许可证使用。  

支持六路通道的数据接收 

1.低工作电压:1.93.6V低电压工作

2.高速率:2Mbps,由于空中传输时间很短,极大的降低了无线传输中的碰撞现象(软件设置1Mbps或者2Mbps的空中传输速率)

 3.多频点:125 频点,满足多点通信和跳频通信需要

4.超小型:内置2.4GHz天线,体积小巧,15x29mm(包括天线)

5.低功耗:当工作在应答模式通信时,快速的空中传输及启动时间,极大的降低了电流消耗。

6.低应用成本:NRF24L01 集成了所有与RF协议相关的高速信号处理部分,比如:自动重发丢失数据包和自动产生应答信号等,NRF24L01的SPI接口可以利用单片机的硬件SPI口连接或用单片机I/O口进行模拟,内部有FIFO可以与各种高低速微处理器接口,便于使用低成本单片机。

二、实验功能

        主要功能实现远距离传输数据并且显示在本地OLED屏幕上和oneNET平台上;实现物联网远距离通信显示;主要还是属于智能家居方面的吧;至于为什么使用NRF2401模块发送、接收,纯属是想用这个模块玩一玩;其实ZET6板子完全可以接上ESP8266模块直接上传;这样可以实现远程监测温湿度、水表什么的了,也可以加入控制功能;刚刚说了远距离传输,也查看了一些LORA的资料,低功耗、远距离传输很强大,也买了两个玩一玩;

        对了,程序自己写了一部分,也copy一部分,改的我头大,磕磕绊绊总算搞出来了,也算是一种学习吧!模块的连接都是用的杜邦线,看着好乱的还给调试带来了麻烦,明明好使的,结果杜邦线松了,来来回回搞了好久,最好还是打个板子,省的调试麻烦;

1、ZET6主板采集温度湿度值、光照度的值,通过WIFI发送模块发送到C8T6主板上;C8T6主板也有一个WIFI接收模块,接收到温度湿度值、光照度的值显示在OLED上;同时C8T6最小系统板接了一个ESP8266模块,通过ESP8266模块发送数据到oneNET平台上,用于显示;

2、下面是代码,感兴趣鼓捣着玩的,可能有不足的地方,欢迎指正,共同进步;

三、实验代码

发送代码:

main

  1. #include "./SYSTEM/sys/sys.h"
  2. //#include "./SYSTEM/usart/usart.h"
  3. #include "./SYSTEM/delay/delay.h"
  4. #include "./USMART/usmart.h"
  5. #include "./BSP/LED/led.h"
  6. #include "./BSP/DHT11/dht11.h"
  7. #include "./BSP/BH1750/bh1750.h"
  8. #include "./BSP/OLED/oled.h"
  9. #include "./BSP/BEEP/beep.h"
  10. #include "./BSP/TIMER/btim.h"
  11. #include "./BSP/KEY/key.h"
  12. #include "./BSP/EXTI/exti.h"
  13. #include "./BSP/NRF24L01/nrf24l01.h"
  14. uint8_t alarm_Free_Time = 10; /*报警标志*/
  15. uint8_t alarm_Flag = 0; /*报警时间阈值*/
  16. uint8_t t = 0;
  17. uint8_t temperature; /*温度*/
  18. uint8_t humidity; /*湿度*/
  19. float Light = 0; /*光照度*/
  20. uint8_t key = 0; /*key值*/
  21. unsigned char rece_buf[33];
  22. int main(void)
  23. {
  24. HAL_Init(); /* 初始化HAL库 */
  25. sys_stm32_clock_init(RCC_PLL_MUL9); /* 设置时钟, 72Mhz */
  26. delay_init(72); /* 延时初始化 */
  27. usart_init(115200); /* 串口初始化为115200 */
  28. led_init(); /* 初始化LED */
  29. dht11_init(); /* DHT11初始化 */
  30. BH1750_Init(); /* BH1750初始化 */
  31. OLED_Init(); /* OLED初始化 */
  32. BEEP_Init(); /* BEEP初始化 */
  33. OLED_ColorTurn(0); /* 0正常显示,1 反色显示*/
  34. OLED_DisplayTurn(0); /* 0正常显示 1 屏幕翻转显示*/
  35. btim_timx_int_init(4999, 7199); /* 初始化定时器6_基本定时器*/
  36. btim_tim7_int_init(4999, 7199); /* 初始化定时器7_基本定时器*/
  37. key_init(); /* 按键初始化*/
  38. extix_init(); /* 初始化中断*/
  39. nrf24l01_init(); /* 初始化NRF24L01 */
  40. nrf24l01_tx_mode(); /* 进入TX模式 */
  41. rece_buf[0]=5;
  42. rece_buf[1]= 0X55;
  43. rece_buf[2]= 0XAA;
  44. rece_buf[32] = 0; /* 加入结束符 */
  45. while (1)
  46. {
  47. dht11_read_data(&temperature, &humidity); /* 读取温湿度值 */
  48. printf("温度:%d %%,湿度:%d C \r\n",temperature,humidity);
  49. if (!i2c_CheckDevice(BH1750_Addr)) /* 检测是否存在bh1750 */
  50. {
  51. Light = LIght_Intensity(); /* 获取光强度的值 */
  52. printf("light:%f \r\n",Light);
  53. }
  54. if (alarm_Free_Time == 10)
  55. {
  56. if (temperature < 30 && humidity < 80 && Light< 10000)
  57. alarm_Flag = 0; /* 温度超过30摄氏度、湿度超过80、光强度超过10000报警 */
  58. else
  59. alarm_Flag = 1;
  60. }
  61. if (alarm_Free_Time < 10)
  62. alarm_Free_Time++;
  63. delay_ms(1000);
  64. rece_buf[3]=(uint8_t)temperature;
  65. rece_buf[4]=(uint8_t)humidity;
  66. rece_buf[5]=(uint8_t)Light;
  67. nrf24l01_tx_packet(rece_buf);
  68. key = key_scan(0);
  69. }
  70. }

dht11

  1. #include "./BSP/DHT11/dht11.h"
  2. #include "./SYSTEM/delay/delay.h"
  3. /**
  4. * @brief 复位DHT11
  5. * @param 无
  6. * @retval 无
  7. */
  8. static void dht11_reset(void)
  9. {
  10. DHT11_DQ_OUT(0); /* 拉低DQ */
  11. delay_ms(20); /* 拉低至少18ms */
  12. DHT11_DQ_OUT(1); /* DQ=1 */
  13. delay_us(30); /* 主机拉高10~35us */
  14. }
  15. /**
  16. * @brief 等待DHT11的回应
  17. * @param 无
  18. * @retval 0, DHT11正常
  19. * 1, DHT11异常/不存在
  20. */
  21. uint8_t dht11_check(void)
  22. {
  23. uint8_t retry = 0;
  24. uint8_t rval = 0;
  25. while (DHT11_DQ_IN && retry < 100) /* DHT11会拉低约83us */
  26. {
  27. retry++;
  28. delay_us(1);
  29. }
  30. if (retry >= 100)
  31. {
  32. rval = 1;
  33. }
  34. else
  35. {
  36. retry = 0;
  37. while (!DHT11_DQ_IN && retry < 100) /* DHT11拉低后会再次拉高约87us */
  38. {
  39. retry++;
  40. delay_us(1);
  41. }
  42. if (retry >= 100) rval = 1;
  43. }
  44. return rval;
  45. }
  46. /**
  47. * @brief 从DHT11读取一个位
  48. * @param 无
  49. * @retval 读取到的位值: 0 / 1
  50. */
  51. uint8_t dht11_read_bit(void)
  52. {
  53. uint8_t retry = 0;
  54. while (DHT11_DQ_IN && retry < 100) /* 等待变为低电平 */
  55. {
  56. retry++;
  57. delay_us(1);
  58. }
  59. retry = 0;
  60. while (!DHT11_DQ_IN && retry < 100) /* 等待变高电平 */
  61. {
  62. retry++;
  63. delay_us(1);
  64. }
  65. delay_us(40); /* 等待40us */
  66. if (DHT11_DQ_IN) /* 根据引脚状态返回 bit */
  67. {
  68. return 1;
  69. }
  70. else
  71. {
  72. return 0;
  73. }
  74. }
  75. /**
  76. * @brief 从DHT11读取一个字节
  77. * @param 无
  78. * @retval 读到的数据
  79. */
  80. static uint8_t dht11_read_byte(void)
  81. {
  82. uint8_t i, data = 0;
  83. for (i = 0; i < 8; i++) /* 循环读取8位数据 */
  84. {
  85. data <<= 1; /* 高位数据先输出, 先左移一位 */
  86. data |= dht11_read_bit(); /* 读取1bit数据 */
  87. }
  88. return data;
  89. }
  90. /**
  91. * @brief 从DHT11读取一次数据
  92. * @param temp: 温度值(范围:-20~60°)
  93. * @param humi: 湿度值(范围:5%~95%)
  94. * @retval 0, 正常.
  95. * 1, 失败
  96. */
  97. uint8_t dht11_read_data(uint8_t *temp, uint8_t *humi)
  98. {
  99. uint8_t buf[5];
  100. uint8_t i;
  101. dht11_reset();
  102. if (dht11_check() == 0)
  103. {
  104. for (i = 0; i < 5; i++) /* 读取40位数据 */
  105. {
  106. buf[i] = dht11_read_byte();
  107. }
  108. if ((buf[0] + buf[1] + buf[2] + buf[3]) == buf[4])
  109. {
  110. *humi = buf[0];
  111. *temp = buf[2];
  112. }
  113. }
  114. else
  115. {
  116. return 1;
  117. }
  118. return 0;
  119. }
  120. /**
  121. * @brief 初始化DHT11的IO口 DQ 同时检测DHT11的存在
  122. * @param 无
  123. * @retval 0, 正常
  124. * 1, 不存在/不正常
  125. */
  126. uint8_t dht11_init(void)
  127. {
  128. GPIO_InitTypeDef gpio_init_struct;
  129. DHT11_DQ_GPIO_CLK_ENABLE(); /* 开启DQ引脚时钟 */
  130. gpio_init_struct.Pin = DHT11_DQ_GPIO_PIN;
  131. gpio_init_struct.Mode = GPIO_MODE_OUTPUT_OD; /* 开漏输出 */
  132. gpio_init_struct.Pull = GPIO_PULLUP; /* 上拉 */
  133. gpio_init_struct.Speed = GPIO_SPEED_FREQ_HIGH; /* 高速 */
  134. HAL_GPIO_Init(DHT11_DQ_GPIO_PORT, &gpio_init_struct); /* 初始化DHT11_DQ引脚 */
  135. /* DHT11_DQ引脚模式设置,开漏输出,上拉, 这样就不用再设置IO方向了, 开漏输出的时候(=1), 也可以读取外部信号的高低电平 */
  136. dht11_reset();
  137. return dht11_check();
  138. }

oled

  1. #include "./BSP/OLED/oled.h"
  2. #include "stdlib.h"
  3. #include "./BSP/OLED/oledfont.h"
  4. #include "./SYSTEM/delay/delay.h"
  5. uint8_t OLED_GRAM[144][8];
  6. //反显函数
  7. void OLED_ColorTurn(uint8_t i)
  8. {
  9. if(i==0)
  10. {
  11. OLED_WR_Byte(0xA6,OLED_CMD);//正常显示
  12. }
  13. if(i==1)
  14. {
  15. OLED_WR_Byte(0xA7,OLED_CMD);//反色显示
  16. }
  17. }
  18. //屏幕旋转180度
  19. void OLED_DisplayTurn(uint8_t i)
  20. {
  21. if(i==0)
  22. {
  23. OLED_WR_Byte(0xC8,OLED_CMD);//正常显示
  24. OLED_WR_Byte(0xA1,OLED_CMD);
  25. }
  26. if(i==1)
  27. {
  28. OLED_WR_Byte(0xC0,OLED_CMD);//反转显示
  29. OLED_WR_Byte(0xA0,OLED_CMD);
  30. }
  31. }
  32. //延时
  33. void IIC_delay(void)
  34. {
  35. uint8_t t=3;
  36. while(t--);
  37. }
  38. //起始信号
  39. void I2C_Start(void)
  40. {
  41. OLED_SDA_Set();
  42. OLED_SCL_Set();
  43. IIC_delay();
  44. OLED_SDA_Clr();
  45. IIC_delay();
  46. OLED_SCL_Clr();
  47. IIC_delay();
  48. }
  49. //结束信号
  50. void I2C_Stop(void)
  51. {
  52. OLED_SDA_Clr();
  53. OLED_SCL_Set();
  54. IIC_delay();
  55. OLED_SDA_Set();
  56. }
  57. //等待信号响应
  58. void I2C_WaitAck(void) //测数据信号的电平
  59. {
  60. OLED_SDA_Set();
  61. IIC_delay();
  62. OLED_SCL_Set();
  63. IIC_delay();
  64. OLED_SCL_Clr();
  65. IIC_delay();
  66. }
  67. //写入一个字节
  68. void Send_Byte(uint8_t dat)
  69. {
  70. uint8_t i;
  71. for(i=0;i<8;i++)
  72. {
  73. if(dat&0x80)//将dat的8位从最高位依次写入
  74. {
  75. OLED_SDA_Set();
  76. }
  77. else
  78. {
  79. OLED_SDA_Clr();
  80. }
  81. IIC_delay();
  82. OLED_SCL_Set();
  83. IIC_delay();
  84. OLED_SCL_Clr();//将时钟信号设置为低电平
  85. dat<<=1;
  86. }
  87. }
  88. //发送一个字节
  89. //mode:数据/命令标志 0,表示命令;1,表示数据;
  90. void OLED_WR_Byte(uint8_t dat,uint8_t mode)
  91. {
  92. I2C_Start();
  93. Send_Byte(0x78);
  94. I2C_WaitAck();
  95. if(mode){Send_Byte(0x40);}
  96. else{Send_Byte(0x00);}
  97. I2C_WaitAck();
  98. Send_Byte(dat);
  99. I2C_WaitAck();
  100. I2C_Stop();
  101. }
  102. //开启OLED显示
  103. void OLED_DisPlay_On(void)
  104. {
  105. OLED_WR_Byte(0x8D,OLED_CMD);//电荷泵使能
  106. OLED_WR_Byte(0x14,OLED_CMD);//开启电荷泵
  107. OLED_WR_Byte(0xAF,OLED_CMD);//点亮屏幕
  108. }
  109. //关闭OLED显示
  110. void OLED_DisPlay_Off(void)
  111. {
  112. OLED_WR_Byte(0x8D,OLED_CMD);//电荷泵使能
  113. OLED_WR_Byte(0x10,OLED_CMD);//关闭电荷泵
  114. OLED_WR_Byte(0xAE,OLED_CMD);//关闭屏幕
  115. }
  116. //更新显存到OLED
  117. void OLED_Refresh(void)
  118. {
  119. uint8_t i,n;
  120. for(i=0;i<8;i++)
  121. {
  122. OLED_WR_Byte(0xb0+i,OLED_CMD); //设置行起始地址
  123. OLED_WR_Byte(0x00,OLED_CMD); //设置低列起始地址
  124. OLED_WR_Byte(0x10,OLED_CMD); //设置高列起始地址
  125. I2C_Start();
  126. Send_Byte(0x78);
  127. I2C_WaitAck();
  128. Send_Byte(0x40);
  129. I2C_WaitAck();
  130. for(n=0;n<128;n++)
  131. {
  132. Send_Byte(OLED_GRAM[n][i]);
  133. I2C_WaitAck();
  134. }
  135. I2C_Stop();
  136. }
  137. }
  138. //清屏函数
  139. void OLED_Clear(void)
  140. {
  141. uint8_t i,n;
  142. for(i=0;i<8;i++)
  143. {
  144. for(n=0;n<128;n++)
  145. {
  146. OLED_GRAM[n][i]=0;//清除所有数据
  147. }
  148. }
  149. OLED_Refresh();//更新显示
  150. }
  151. //画点
  152. //x:0~127
  153. //y:0~63
  154. //t:1 填充 0,清空
  155. void OLED_DrawPoint(uint8_t x,uint8_t y,uint8_t t)
  156. {
  157. uint8_t i,m,n;
  158. i=y/8;
  159. m=y%8;
  160. n=1<<m;
  161. if(t){OLED_GRAM[x][i]|=n;}
  162. else
  163. {
  164. OLED_GRAM[x][i]=~OLED_GRAM[x][i];
  165. OLED_GRAM[x][i]|=n;
  166. OLED_GRAM[x][i]=~OLED_GRAM[x][i];
  167. }
  168. }
  169. //画线
  170. //x1,y1:起点坐标
  171. //x2,y2:结束坐标
  172. void OLED_DrawLine(uint8_t x1,uint8_t y1,uint8_t x2,uint8_t y2,uint8_t mode)
  173. {
  174. uint16_t t;
  175. int xerr=0,yerr=0,delta_x,delta_y,distance;
  176. int incx,incy,uRow,uCol;
  177. delta_x=x2-x1; //计算坐标增量
  178. delta_y=y2-y1;
  179. uRow=x1;//画线起点坐标
  180. uCol=y1;
  181. if(delta_x>0)incx=1; //设置单步方向
  182. else if (delta_x==0)incx=0;//垂直线
  183. else {incx=-1;delta_x=-delta_x;}
  184. if(delta_y>0)incy=1;
  185. else if (delta_y==0)incy=0;//水平线
  186. else {incy=-1;delta_y=-delta_x;}
  187. if(delta_x>delta_y)distance=delta_x; //选取基本增量坐标轴
  188. else distance=delta_y;
  189. for(t=0;t<distance+1;t++)
  190. {
  191. OLED_DrawPoint(uRow,uCol,mode);//画点
  192. xerr+=delta_x;
  193. yerr+=delta_y;
  194. if(xerr>distance)
  195. {
  196. xerr-=distance;
  197. uRow+=incx;
  198. }
  199. if(yerr>distance)
  200. {
  201. yerr-=distance;
  202. uCol+=incy;
  203. }
  204. }
  205. }
  206. //x,y:圆心坐标
  207. //r:圆的半径
  208. void OLED_DrawCircle(uint8_t x,uint8_t y,uint8_t r)
  209. {
  210. int a, b,num;
  211. a = 0;
  212. b = r;
  213. while(2 * b * b >= r * r)
  214. {
  215. OLED_DrawPoint(x + a, y - b,1);
  216. OLED_DrawPoint(x - a, y - b,1);
  217. OLED_DrawPoint(x - a, y + b,1);
  218. OLED_DrawPoint(x + a, y + b,1);
  219. OLED_DrawPoint(x + b, y + a,1);
  220. OLED_DrawPoint(x + b, y - a,1);
  221. OLED_DrawPoint(x - b, y - a,1);
  222. OLED_DrawPoint(x - b, y + a,1);
  223. a++;
  224. num = (a * a + b * b) - r*r;//计算画的点离圆心的距离
  225. if(num > 0)
  226. {
  227. b--;
  228. a--;
  229. }
  230. }
  231. }
  232. //在指定位置显示一个字符,包括部分字符
  233. //x:0~127
  234. //y:0~63
  235. //size1:选择字体 6x8/6x12/8x16/12x24
  236. //mode:0,反色显示;1,正常显示
  237. void OLED_ShowChar(uint8_t x,uint8_t y,uint8_t chr,uint8_t size1,uint8_t mode)
  238. {
  239. uint8_t i,m,temp,size2,chr1;
  240. uint8_t x0=x,y0=y;
  241. if(size1==8)size2=6;
  242. else size2=(size1/8+((size1%8)?1:0))*(size1/2); //得到字体一个字符对应点阵集所占的字节数
  243. chr1=chr-' '; //计算偏移后的值
  244. for(i=0;i<size2;i++)
  245. {
  246. if(size1==8)
  247. {temp=asc2_0806[chr1][i];} //调用0806字体
  248. else if(size1==12)
  249. {temp=asc2_1206[chr1][i];} //调用1206字体
  250. else if(size1==16)
  251. {temp=asc2_1608[chr1][i];} //调用1608字体
  252. else if(size1==24)
  253. {temp=asc2_2412[chr1][i];} //调用2412字体
  254. else return;
  255. for(m=0;m<8;m++)
  256. {
  257. if(temp&0x01)OLED_DrawPoint(x,y,mode);
  258. else OLED_DrawPoint(x,y,!mode);
  259. temp>>=1;
  260. y++;
  261. }
  262. x++;
  263. if((size1!=8)&&((x-x0)==size1/2))
  264. {x=x0;y0=y0+8;}
  265. y=y0;
  266. }
  267. }
  268. //显示字符串
  269. //x,y:起点坐标
  270. //size1:字体大小
  271. //*chr:字符串起始地址
  272. //mode:0,反色显示;1,正常显示
  273. void OLED_ShowString(uint8_t x,uint8_t y,uint8_t *chr,uint8_t size1,uint8_t mode)
  274. {
  275. while((*chr>=' ')&&(*chr<='~'))//判断是不是非法字符!
  276. {
  277. OLED_ShowChar(x,y,*chr,size1,mode);
  278. if(size1==8)x+=6;
  279. else x+=size1/2;
  280. chr++;
  281. }
  282. }
  283. //m^n
  284. uint32_t OLED_Pow(uint8_t m,uint8_t n)
  285. {
  286. uint32_t result=1;
  287. while(n--)
  288. {
  289. result*=m;
  290. }
  291. return result;
  292. }
  293. //显示数字
  294. //x,y :起点坐标
  295. //num :要显示的数字
  296. //len :数字的位数
  297. //size:字体大小
  298. //mode:0,反色显示;1,正常显示
  299. void OLED_ShowNum(uint8_t x,uint8_t y,uint32_t num,uint8_t len,uint8_t size1,uint8_t mode)
  300. {
  301. uint8_t t,temp,m=0;
  302. if(size1==8)m=2;
  303. for(t=0;t<len;t++)
  304. {
  305. temp=(num/OLED_Pow(10,len-t-1))%10;
  306. if(temp==0)
  307. {
  308. OLED_ShowChar(x+(size1/2+m)*t,y,'0',size1,mode);
  309. }
  310. else
  311. {
  312. OLED_ShowChar(x+(size1/2+m)*t,y,temp+'0',size1,mode);
  313. }
  314. }
  315. }
  316. //显示汉字
  317. //x,y:起点坐标
  318. //num:汉字对应的序号
  319. //mode:0,反色显示;1,正常显示
  320. void OLED_ShowChinese(uint8_t x,uint8_t y,uint8_t num,uint8_t size1,uint8_t mode)
  321. {
  322. uint8_t m,temp;
  323. uint8_t x0=x,y0=y;
  324. uint16_t i,size3=(size1/8+((size1%8)?1:0))*size1; //得到字体一个字符对应点阵集所占的字节数
  325. for(i=0;i<size3;i++)
  326. {
  327. if(size1==16)
  328. {temp=Hzk1[num][i];}//调用16*16字体
  329. else if(size1==24)
  330. {temp=Hzk2[num][i];}//调用24*24字体
  331. else if(size1==32)
  332. {temp=Hzk3[num][i];}//调用32*32字体
  333. else if(size1==64)
  334. {temp=Hzk4[num][i];}//调用64*64字体
  335. else return;
  336. for(m=0;m<8;m++)
  337. {
  338. if(temp&0x01)OLED_DrawPoint(x,y,mode);
  339. else OLED_DrawPoint(x,y,!mode);
  340. temp>>=1;
  341. y++;
  342. }
  343. x++;
  344. if((x-x0)==size1)
  345. {x=x0;y0=y0+8;}
  346. y=y0;
  347. }
  348. }
  349. //num 显示汉字的个数
  350. //space 每一遍显示的间隔
  351. //mode:0,反色显示;1,正常显示
  352. void OLED_ScrollDisplay(uint8_t num,uint8_t space,uint8_t mode)
  353. {
  354. uint8_t i,n,t=0,m=0,r;
  355. while(1)
  356. {
  357. if(m==0)
  358. {
  359. OLED_ShowChinese(128,24,t,16,mode); //写入一个汉字保存在OLED_GRAM[][]数组中
  360. t++;
  361. }
  362. if(t==num)
  363. {
  364. for(r=0;r<16*space;r++) //显示间隔
  365. {
  366. for(i=1;i<144;i++)
  367. {
  368. for(n=0;n<8;n++)
  369. {
  370. OLED_GRAM[i-1][n]=OLED_GRAM[i][n];
  371. }
  372. }
  373. OLED_Refresh();
  374. }
  375. t=0;
  376. }
  377. m++;
  378. if(m==16){m=0;}
  379. for(i=1;i<144;i++) //实现左移
  380. {
  381. for(n=0;n<8;n++)
  382. {
  383. OLED_GRAM[i-1][n]=OLED_GRAM[i][n];
  384. }
  385. }
  386. OLED_Refresh();
  387. }
  388. }
  389. //x,y:起点坐标
  390. //sizex,sizey,图片长宽
  391. //BMP[]:要写入的图片数组
  392. //mode:0,反色显示;1,正常显示
  393. void OLED_ShowPicture(uint8_t x,uint8_t y,uint8_t sizex,uint8_t sizey,uint8_t BMP[],uint8_t mode)
  394. {
  395. uint16_t j=0;
  396. uint8_t i,n,temp,m;
  397. uint8_t x0=x,y0=y;
  398. sizey=sizey/8+((sizey%8)?1:0);
  399. for(n=0;n<sizey;n++)
  400. {
  401. for(i=0;i<sizex;i++)
  402. {
  403. temp=BMP[j];
  404. j++;
  405. for(m=0;m<8;m++)
  406. {
  407. if(temp&0x01)OLED_DrawPoint(x,y,mode);
  408. else OLED_DrawPoint(x,y,!mode);
  409. temp>>=1;
  410. y++;
  411. }
  412. x++;
  413. if((x-x0)==sizex)
  414. {
  415. x=x0;
  416. y0=y0+8;
  417. }
  418. y=y0;
  419. }
  420. }
  421. }
  422. //OLED的初始化
  423. void OLED_Init(void)
  424. {
  425. GPIO_InitTypeDef GPIO_InitStructure;
  426. __HAL_RCC_GPIOG_CLK_ENABLE(); /* 使能GPIOF时钟 */
  427. __HAL_RCC_GPIOD_CLK_ENABLE();
  428. // RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOG|RCC_APB2Periph_GPIOD, ENABLE); //使能A端口时钟
  429. GPIO_InitStructure.Pin = GPIO_PIN_12; //SCL
  430. GPIO_InitStructure.Mode = GPIO_MODE_OUTPUT_OD; //推挽输出
  431. GPIO_InitStructure.Speed = GPIO_SPEED_FREQ_MEDIUM;//速度50MHz
  432. HAL_GPIO_Init(GPIOG, &GPIO_InitStructure); //初始化GPIOF
  433. HAL_GPIO_WritePin(GPIOG, GPIO_PIN_12, GPIO_PIN_RESET);
  434. GPIO_InitStructure.Pin = GPIO_PIN_5; //DIN
  435. GPIO_InitStructure.Mode = GPIO_MODE_OUTPUT_OD; //推挽输出
  436. GPIO_InitStructure.Speed = GPIO_SPEED_FREQ_MEDIUM;//速度50MHz
  437. HAL_GPIO_Init(GPIOD, &GPIO_InitStructure); //初始化GPIOF
  438. HAL_GPIO_WritePin(GPIOD, GPIO_PIN_5, GPIO_PIN_RESET);
  439. GPIO_InitStructure.Pin = GPIO_PIN_4; //RES
  440. GPIO_InitStructure.Mode = GPIO_MODE_OUTPUT_PP; //推挽输出
  441. GPIO_InitStructure.Speed = GPIO_SPEED_FREQ_MEDIUM;//速度50MHz
  442. HAL_GPIO_Init(GPIOD, &GPIO_InitStructure); //初始化GPIOF
  443. HAL_GPIO_WritePin(GPIOD, GPIO_PIN_4, GPIO_PIN_RESET);
  444. OLED_RES_Clr();
  445. delay_ms(200);
  446. OLED_RES_Set();
  447. OLED_WR_Byte(0xAE,OLED_CMD);//--turn off oled panel
  448. OLED_WR_Byte(0x00,OLED_CMD);//---set low column address
  449. OLED_WR_Byte(0x10,OLED_CMD);//---set high column address
  450. OLED_WR_Byte(0x40,OLED_CMD);//--set start line address Set Mapping RAM Display Start Line (0x00~0x3F)
  451. OLED_WR_Byte(0x81,OLED_CMD);//--set contrast control register
  452. OLED_WR_Byte(0xCF,OLED_CMD);// Set SEG Output Current Brightness
  453. OLED_WR_Byte(0xA1,OLED_CMD);//--Set SEG/Column Mapping 0xa0左右反置 0xa1正常
  454. OLED_WR_Byte(0xC8,OLED_CMD);//Set COM/Row Scan Direction 0xc0上下反置 0xc8正常
  455. OLED_WR_Byte(0xA6,OLED_CMD);//--set normal display
  456. OLED_WR_Byte(0xA8,OLED_CMD);//--set multiplex ratio(1 to 64)
  457. OLED_WR_Byte(0x3f,OLED_CMD);//--1/64 duty
  458. OLED_WR_Byte(0xD3,OLED_CMD);//-set display offset Shift Mapping RAM Counter (0x00~0x3F)
  459. OLED_WR_Byte(0x00,OLED_CMD);//-not offset
  460. OLED_WR_Byte(0xd5,OLED_CMD);//--set display clock divide ratio/oscillator frequency
  461. OLED_WR_Byte(0x80,OLED_CMD);//--set divide ratio, Set Clock as 100 Frames/Sec
  462. OLED_WR_Byte(0xD9,OLED_CMD);//--set pre-charge period
  463. OLED_WR_Byte(0xF1,OLED_CMD);//Set Pre-Charge as 15 Clocks & Discharge as 1 Clock
  464. OLED_WR_Byte(0xDA,OLED_CMD);//--set com pins hardware configuration
  465. OLED_WR_Byte(0x12,OLED_CMD);
  466. OLED_WR_Byte(0xDB,OLED_CMD);//--set vcomh
  467. OLED_WR_Byte(0x40,OLED_CMD);//Set VCOM Deselect Level
  468. OLED_WR_Byte(0x20,OLED_CMD);//-Set Page Addressing Mode (0x00/0x01/0x02)
  469. OLED_WR_Byte(0x02,OLED_CMD);//
  470. OLED_WR_Byte(0x8D,OLED_CMD);//--set Charge Pump enable/disable
  471. OLED_WR_Byte(0x14,OLED_CMD);//--set(0x10) disable
  472. OLED_WR_Byte(0xA4,OLED_CMD);// Disable Entire Display On (0xa4/0xa5)
  473. OLED_WR_Byte(0xA6,OLED_CMD);// Disable Inverse Display On (0xa6/a7)
  474. OLED_Clear();
  475. OLED_WR_Byte(0xAF,OLED_CMD);
  476. }

bh1750

  1. #include "./BSP/OLED/oled.h"
  2. #include "stdlib.h"
  3. #include "./BSP/OLED/oledfont.h"
  4. #include "./SYSTEM/delay/delay.h"
  5. uint8_t OLED_GRAM[144][8];
  6. //反显函数
  7. void OLED_ColorTurn(uint8_t i)
  8. {
  9. if(i==0)
  10. {
  11. OLED_WR_Byte(0xA6,OLED_CMD);//正常显示
  12. }
  13. if(i==1)
  14. {
  15. OLED_WR_Byte(0xA7,OLED_CMD);//反色显示
  16. }
  17. }
  18. //屏幕旋转180度
  19. void OLED_DisplayTurn(uint8_t i)
  20. {
  21. if(i==0)
  22. {
  23. OLED_WR_Byte(0xC8,OLED_CMD);//正常显示
  24. OLED_WR_Byte(0xA1,OLED_CMD);
  25. }
  26. if(i==1)
  27. {
  28. OLED_WR_Byte(0xC0,OLED_CMD);//反转显示
  29. OLED_WR_Byte(0xA0,OLED_CMD);
  30. }
  31. }
  32. //延时
  33. void IIC_delay(void)
  34. {
  35. uint8_t t=3;
  36. while(t--);
  37. }
  38. //起始信号
  39. void I2C_Start(void)
  40. {
  41. OLED_SDA_Set();
  42. OLED_SCL_Set();
  43. IIC_delay();
  44. OLED_SDA_Clr();
  45. IIC_delay();
  46. OLED_SCL_Clr();
  47. IIC_delay();
  48. }
  49. //结束信号
  50. void I2C_Stop(void)
  51. {
  52. OLED_SDA_Clr();
  53. OLED_SCL_Set();
  54. IIC_delay();
  55. OLED_SDA_Set();
  56. }
  57. //等待信号响应
  58. void I2C_WaitAck(void) //测数据信号的电平
  59. {
  60. OLED_SDA_Set();
  61. IIC_delay();
  62. OLED_SCL_Set();
  63. IIC_delay();
  64. OLED_SCL_Clr();
  65. IIC_delay();
  66. }
  67. //写入一个字节
  68. void Send_Byte(uint8_t dat)
  69. {
  70. uint8_t i;
  71. for(i=0;i<8;i++)
  72. {
  73. if(dat&0x80)//将dat的8位从最高位依次写入
  74. {
  75. OLED_SDA_Set();
  76. }
  77. else
  78. {
  79. OLED_SDA_Clr();
  80. }
  81. IIC_delay();
  82. OLED_SCL_Set();
  83. IIC_delay();
  84. OLED_SCL_Clr();//将时钟信号设置为低电平
  85. dat<<=1;
  86. }
  87. }
  88. //发送一个字节
  89. //mode:数据/命令标志 0,表示命令;1,表示数据;
  90. void OLED_WR_Byte(uint8_t dat,uint8_t mode)
  91. {
  92. I2C_Start();
  93. Send_Byte(0x78);
  94. I2C_WaitAck();
  95. if(mode){Send_Byte(0x40);}
  96. else{Send_Byte(0x00);}
  97. I2C_WaitAck();
  98. Send_Byte(dat);
  99. I2C_WaitAck();
  100. I2C_Stop();
  101. }
  102. //开启OLED显示
  103. void OLED_DisPlay_On(void)
  104. {
  105. OLED_WR_Byte(0x8D,OLED_CMD);//电荷泵使能
  106. OLED_WR_Byte(0x14,OLED_CMD);//开启电荷泵
  107. OLED_WR_Byte(0xAF,OLED_CMD);//点亮屏幕
  108. }
  109. //关闭OLED显示
  110. void OLED_DisPlay_Off(void)
  111. {
  112. OLED_WR_Byte(0x8D,OLED_CMD);//电荷泵使能
  113. OLED_WR_Byte(0x10,OLED_CMD);//关闭电荷泵
  114. OLED_WR_Byte(0xAE,OLED_CMD);//关闭屏幕
  115. }
  116. //更新显存到OLED
  117. void OLED_Refresh(void)
  118. {
  119. uint8_t i,n;
  120. for(i=0;i<8;i++)
  121. {
  122. OLED_WR_Byte(0xb0+i,OLED_CMD); //设置行起始地址
  123. OLED_WR_Byte(0x00,OLED_CMD); //设置低列起始地址
  124. OLED_WR_Byte(0x10,OLED_CMD); //设置高列起始地址
  125. I2C_Start();
  126. Send_Byte(0x78);
  127. I2C_WaitAck();
  128. Send_Byte(0x40);
  129. I2C_WaitAck();
  130. for(n=0;n<128;n++)
  131. {
  132. Send_Byte(OLED_GRAM[n][i]);
  133. I2C_WaitAck();
  134. }
  135. I2C_Stop();
  136. }
  137. }
  138. //清屏函数
  139. void OLED_Clear(void)
  140. {
  141. uint8_t i,n;
  142. for(i=0;i<8;i++)
  143. {
  144. for(n=0;n<128;n++)
  145. {
  146. OLED_GRAM[n][i]=0;//清除所有数据
  147. }
  148. }
  149. OLED_Refresh();//更新显示
  150. }
  151. //画点
  152. //x:0~127
  153. //y:0~63
  154. //t:1 填充 0,清空
  155. void OLED_DrawPoint(uint8_t x,uint8_t y,uint8_t t)
  156. {
  157. uint8_t i,m,n;
  158. i=y/8;
  159. m=y%8;
  160. n=1<<m;
  161. if(t){OLED_GRAM[x][i]|=n;}
  162. else
  163. {
  164. OLED_GRAM[x][i]=~OLED_GRAM[x][i];
  165. OLED_GRAM[x][i]|=n;
  166. OLED_GRAM[x][i]=~OLED_GRAM[x][i];
  167. }
  168. }
  169. //画线
  170. //x1,y1:起点坐标
  171. //x2,y2:结束坐标
  172. void OLED_DrawLine(uint8_t x1,uint8_t y1,uint8_t x2,uint8_t y2,uint8_t mode)
  173. {
  174. uint16_t t;
  175. int xerr=0,yerr=0,delta_x,delta_y,distance;
  176. int incx,incy,uRow,uCol;
  177. delta_x=x2-x1; //计算坐标增量
  178. delta_y=y2-y1;
  179. uRow=x1;//画线起点坐标
  180. uCol=y1;
  181. if(delta_x>0)incx=1; //设置单步方向
  182. else if (delta_x==0)incx=0;//垂直线
  183. else {incx=-1;delta_x=-delta_x;}
  184. if(delta_y>0)incy=1;
  185. else if (delta_y==0)incy=0;//水平线
  186. else {incy=-1;delta_y=-delta_x;}
  187. if(delta_x>delta_y)distance=delta_x; //选取基本增量坐标轴
  188. else distance=delta_y;
  189. for(t=0;t<distance+1;t++)
  190. {
  191. OLED_DrawPoint(uRow,uCol,mode);//画点
  192. xerr+=delta_x;
  193. yerr+=delta_y;
  194. if(xerr>distance)
  195. {
  196. xerr-=distance;
  197. uRow+=incx;
  198. }
  199. if(yerr>distance)
  200. {
  201. yerr-=distance;
  202. uCol+=incy;
  203. }
  204. }
  205. }
  206. //x,y:圆心坐标
  207. //r:圆的半径
  208. void OLED_DrawCircle(uint8_t x,uint8_t y,uint8_t r)
  209. {
  210. int a, b,num;
  211. a = 0;
  212. b = r;
  213. while(2 * b * b >= r * r)
  214. {
  215. OLED_DrawPoint(x + a, y - b,1);
  216. OLED_DrawPoint(x - a, y - b,1);
  217. OLED_DrawPoint(x - a, y + b,1);
  218. OLED_DrawPoint(x + a, y + b,1);
  219. OLED_DrawPoint(x + b, y + a,1);
  220. OLED_DrawPoint(x + b, y - a,1);
  221. OLED_DrawPoint(x - b, y - a,1);
  222. OLED_DrawPoint(x - b, y + a,1);
  223. a++;
  224. num = (a * a + b * b) - r*r;//计算画的点离圆心的距离
  225. if(num > 0)
  226. {
  227. b--;
  228. a--;
  229. }
  230. }
  231. }
  232. //在指定位置显示一个字符,包括部分字符
  233. //x:0~127
  234. //y:0~63
  235. //size1:选择字体 6x8/6x12/8x16/12x24
  236. //mode:0,反色显示;1,正常显示
  237. void OLED_ShowChar(uint8_t x,uint8_t y,uint8_t chr,uint8_t size1,uint8_t mode)
  238. {
  239. uint8_t i,m,temp,size2,chr1;
  240. uint8_t x0=x,y0=y;
  241. if(size1==8)size2=6;
  242. else size2=(size1/8+((size1%8)?1:0))*(size1/2); //得到字体一个字符对应点阵集所占的字节数
  243. chr1=chr-' '; //计算偏移后的值
  244. for(i=0;i<size2;i++)
  245. {
  246. if(size1==8)
  247. {temp=asc2_0806[chr1][i];} //调用0806字体
  248. else if(size1==12)
  249. {temp=asc2_1206[chr1][i];} //调用1206字体
  250. else if(size1==16)
  251. {temp=asc2_1608[chr1][i];} //调用1608字体
  252. else if(size1==24)
  253. {temp=asc2_2412[chr1][i];} //调用2412字体
  254. else return;
  255. for(m=0;m<8;m++)
  256. {
  257. if(temp&0x01)OLED_DrawPoint(x,y,mode);
  258. else OLED_DrawPoint(x,y,!mode);
  259. temp>>=1;
  260. y++;
  261. }
  262. x++;
  263. if((size1!=8)&&((x-x0)==size1/2))
  264. {x=x0;y0=y0+8;}
  265. y=y0;
  266. }
  267. }
  268. //显示字符串
  269. //x,y:起点坐标
  270. //size1:字体大小
  271. //*chr:字符串起始地址
  272. //mode:0,反色显示;1,正常显示
  273. void OLED_ShowString(uint8_t x,uint8_t y,uint8_t *chr,uint8_t size1,uint8_t mode)
  274. {
  275. while((*chr>=' ')&&(*chr<='~'))//判断是不是非法字符!
  276. {
  277. OLED_ShowChar(x,y,*chr,size1,mode);
  278. if(size1==8)x+=6;
  279. else x+=size1/2;
  280. chr++;
  281. }
  282. }
  283. //m^n
  284. uint32_t OLED_Pow(uint8_t m,uint8_t n)
  285. {
  286. uint32_t result=1;
  287. while(n--)
  288. {
  289. result*=m;
  290. }
  291. return result;
  292. }
  293. //显示数字
  294. //x,y :起点坐标
  295. //num :要显示的数字
  296. //len :数字的位数
  297. //size:字体大小
  298. //mode:0,反色显示;1,正常显示
  299. void OLED_ShowNum(uint8_t x,uint8_t y,uint32_t num,uint8_t len,uint8_t size1,uint8_t mode)
  300. {
  301. uint8_t t,temp,m=0;
  302. if(size1==8)m=2;
  303. for(t=0;t<len;t++)
  304. {
  305. temp=(num/OLED_Pow(10,len-t-1))%10;
  306. if(temp==0)
  307. {
  308. OLED_ShowChar(x+(size1/2+m)*t,y,'0',size1,mode);
  309. }
  310. else
  311. {
  312. OLED_ShowChar(x+(size1/2+m)*t,y,temp+'0',size1,mode);
  313. }
  314. }
  315. }
  316. //显示汉字
  317. //x,y:起点坐标
  318. //num:汉字对应的序号
  319. //mode:0,反色显示;1,正常显示
  320. void OLED_ShowChinese(uint8_t x,uint8_t y,uint8_t num,uint8_t size1,uint8_t mode)
  321. {
  322. uint8_t m,temp;
  323. uint8_t x0=x,y0=y;
  324. uint16_t i,size3=(size1/8+((size1%8)?1:0))*size1; //得到字体一个字符对应点阵集所占的字节数
  325. for(i=0;i<size3;i++)
  326. {
  327. if(size1==16)
  328. {temp=Hzk1[num][i];}//调用16*16字体
  329. else if(size1==24)
  330. {temp=Hzk2[num][i];}//调用24*24字体
  331. else if(size1==32)
  332. {temp=Hzk3[num][i];}//调用32*32字体
  333. else if(size1==64)
  334. {temp=Hzk4[num][i];}//调用64*64字体
  335. else return;
  336. for(m=0;m<8;m++)
  337. {
  338. if(temp&0x01)OLED_DrawPoint(x,y,mode);
  339. else OLED_DrawPoint(x,y,!mode);
  340. temp>>=1;
  341. y++;
  342. }
  343. x++;
  344. if((x-x0)==size1)
  345. {x=x0;y0=y0+8;}
  346. y=y0;
  347. }
  348. }
  349. //num 显示汉字的个数
  350. //space 每一遍显示的间隔
  351. //mode:0,反色显示;1,正常显示
  352. void OLED_ScrollDisplay(uint8_t num,uint8_t space,uint8_t mode)
  353. {
  354. uint8_t i,n,t=0,m=0,r;
  355. while(1)
  356. {
  357. if(m==0)
  358. {
  359. OLED_ShowChinese(128,24,t,16,mode); //写入一个汉字保存在OLED_GRAM[][]数组中
  360. t++;
  361. }
  362. if(t==num)
  363. {
  364. for(r=0;r<16*space;r++) //显示间隔
  365. {
  366. for(i=1;i<144;i++)
  367. {
  368. for(n=0;n<8;n++)
  369. {
  370. OLED_GRAM[i-1][n]=OLED_GRAM[i][n];
  371. }
  372. }
  373. OLED_Refresh();
  374. }
  375. t=0;
  376. }
  377. m++;
  378. if(m==16){m=0;}
  379. for(i=1;i<144;i++) //实现左移
  380. {
  381. for(n=0;n<8;n++)
  382. {
  383. OLED_GRAM[i-1][n]=OLED_GRAM[i][n];
  384. }
  385. }
  386. OLED_Refresh();
  387. }
  388. }
  389. //x,y:起点坐标
  390. //sizex,sizey,图片长宽
  391. //BMP[]:要写入的图片数组
  392. //mode:0,反色显示;1,正常显示
  393. void OLED_ShowPicture(uint8_t x,uint8_t y,uint8_t sizex,uint8_t sizey,uint8_t BMP[],uint8_t mode)
  394. {
  395. uint16_t j=0;
  396. uint8_t i,n,temp,m;
  397. uint8_t x0=x,y0=y;
  398. sizey=sizey/8+((sizey%8)?1:0);
  399. for(n=0;n<sizey;n++)
  400. {
  401. for(i=0;i<sizex;i++)
  402. {
  403. temp=BMP[j];
  404. j++;
  405. for(m=0;m<8;m++)
  406. {
  407. if(temp&0x01)OLED_DrawPoint(x,y,mode);
  408. else OLED_DrawPoint(x,y,!mode);
  409. temp>>=1;
  410. y++;
  411. }
  412. x++;
  413. if((x-x0)==sizex)
  414. {
  415. x=x0;
  416. y0=y0+8;
  417. }
  418. y=y0;
  419. }
  420. }
  421. }
  422. //OLED的初始化
  423. void OLED_Init(void)
  424. {
  425. GPIO_InitTypeDef GPIO_InitStructure;
  426. __HAL_RCC_GPIOG_CLK_ENABLE(); /* 使能GPIOF时钟 */
  427. __HAL_RCC_GPIOD_CLK_ENABLE();
  428. // RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOG|RCC_APB2Periph_GPIOD, ENABLE); //使能A端口时钟
  429. GPIO_InitStructure.Pin = GPIO_PIN_12; //SCL
  430. GPIO_InitStructure.Mode = GPIO_MODE_OUTPUT_OD; //推挽输出
  431. GPIO_InitStructure.Speed = GPIO_SPEED_FREQ_MEDIUM;//速度50MHz
  432. HAL_GPIO_Init(GPIOG, &GPIO_InitStructure); //初始化GPIOF
  433. HAL_GPIO_WritePin(GPIOG, GPIO_PIN_12, GPIO_PIN_RESET);
  434. GPIO_InitStructure.Pin = GPIO_PIN_5; //DIN
  435. GPIO_InitStructure.Mode = GPIO_MODE_OUTPUT_OD; //推挽输出
  436. GPIO_InitStructure.Speed = GPIO_SPEED_FREQ_MEDIUM;//速度50MHz
  437. HAL_GPIO_Init(GPIOD, &GPIO_InitStructure); //初始化GPIOF
  438. HAL_GPIO_WritePin(GPIOD, GPIO_PIN_5, GPIO_PIN_RESET);
  439. GPIO_InitStructure.Pin = GPIO_PIN_4; //RES
  440. GPIO_InitStructure.Mode = GPIO_MODE_OUTPUT_PP; //推挽输出
  441. GPIO_InitStructure.Speed = GPIO_SPEED_FREQ_MEDIUM;//速度50MHz
  442. HAL_GPIO_Init(GPIOD, &GPIO_InitStructure); //初始化GPIOF
  443. HAL_GPIO_WritePin(GPIOD, GPIO_PIN_4, GPIO_PIN_RESET);
  444. OLED_RES_Clr();
  445. delay_ms(200);
  446. OLED_RES_Set();
  447. OLED_WR_Byte(0xAE,OLED_CMD);//--turn off oled panel
  448. OLED_WR_Byte(0x00,OLED_CMD);//---set low column address
  449. OLED_WR_Byte(0x10,OLED_CMD);//---set high column address
  450. OLED_WR_Byte(0x40,OLED_CMD);//--set start line address Set Mapping RAM Display Start Line (0x00~0x3F)
  451. OLED_WR_Byte(0x81,OLED_CMD);//--set contrast control register
  452. OLED_WR_Byte(0xCF,OLED_CMD);// Set SEG Output Current Brightness
  453. OLED_WR_Byte(0xA1,OLED_CMD);//--Set SEG/Column Mapping 0xa0左右反置 0xa1正常
  454. OLED_WR_Byte(0xC8,OLED_CMD);//Set COM/Row Scan Direction 0xc0上下反置 0xc8正常
  455. OLED_WR_Byte(0xA6,OLED_CMD);//--set normal display
  456. OLED_WR_Byte(0xA8,OLED_CMD);//--set multiplex ratio(1 to 64)
  457. OLED_WR_Byte(0x3f,OLED_CMD);//--1/64 duty
  458. OLED_WR_Byte(0xD3,OLED_CMD);//-set display offset Shift Mapping RAM Counter (0x00~0x3F)
  459. OLED_WR_Byte(0x00,OLED_CMD);//-not offset
  460. OLED_WR_Byte(0xd5,OLED_CMD);//--set display clock divide ratio/oscillator frequency
  461. OLED_WR_Byte(0x80,OLED_CMD);//--set divide ratio, Set Clock as 100 Frames/Sec
  462. OLED_WR_Byte(0xD9,OLED_CMD);//--set pre-charge period
  463. OLED_WR_Byte(0xF1,OLED_CMD);//Set Pre-Charge as 15 Clocks & Discharge as 1 Clock
  464. OLED_WR_Byte(0xDA,OLED_CMD);//--set com pins hardware configuration
  465. OLED_WR_Byte(0x12,OLED_CMD);
  466. OLED_WR_Byte(0xDB,OLED_CMD);//--set vcomh
  467. OLED_WR_Byte(0x40,OLED_CMD);//Set VCOM Deselect Level
  468. OLED_WR_Byte(0x20,OLED_CMD);//-Set Page Addressing Mode (0x00/0x01/0x02)
  469. OLED_WR_Byte(0x02,OLED_CMD);//
  470. OLED_WR_Byte(0x8D,OLED_CMD);//--set Charge Pump enable/disable
  471. OLED_WR_Byte(0x14,OLED_CMD);//--set(0x10) disable
  472. OLED_WR_Byte(0xA4,OLED_CMD);// Disable Entire Display On (0xa4/0xa5)
  473. OLED_WR_Byte(0xA6,OLED_CMD);// Disable Inverse Display On (0xa6/a7)
  474. OLED_Clear();
  475. OLED_WR_Byte(0xAF,OLED_CMD);
  476. }

NRF24L01

  1. /**
  2. ****************************************************************************************************
  3. * @file nrf24l01.c
  4. * @author 正点原子团队(ALIENTEK)
  5. * @version V1.0
  6. * @date 2020-04-26
  7. * @brief NRF24L01 驱动代码
  8. * @license Copyright (c) 2020-2032, 广州市星翼电子科技有限公司
  9. ****************************************************************************************************
  10. * @attention
  11. *
  12. * 实验平台:正点原子 STM32F103开发板
  13. * 在线视频:www.yuanzige.com
  14. * 技术论坛:www.openedv.com
  15. * 公司网址:www.alientek.com
  16. * 购买地址:openedv.taobao.com
  17. *
  18. * 修改说明
  19. * V1.0 20200426
  20. * 第一次发布
  21. *
  22. ****************************************************************************************************
  23. */
  24. #include "./BSP/SPI/spi.h"
  25. #include "./BSP/NRF24L01/nrf24l01.h"
  26. #include "./SYSTEM/delay/delay.h"
  27. extern SPI_HandleTypeDef g_spi2_handler; /* SPI2句柄 */
  28. const uint8_t TX_ADDRESS[TX_ADR_WIDTH] = {0xFF,0xFF,0xFF,0xFF,0xFF}; /* 发送地址 */
  29. const uint8_t RX_ADDRESS[RX_ADR_WIDTH] = {0xFF,0xFF,0xFF,0xFF,0xFF}; /* 发送地址 */
  30. /**
  31. * @brief 针对NRF24L01修改SPI2驱动
  32. * @param 无
  33. * @retval 无
  34. */
  35. void nrf24l01_spi_init(void)
  36. {
  37. __HAL_SPI_DISABLE(&g_spi2_handler); /* 先关闭SPI2 */
  38. g_spi2_handler.Init.CLKPolarity = SPI_POLARITY_LOW; /* 串行同步时钟的空闲状态为低电平 */
  39. g_spi2_handler.Init.CLKPhase = SPI_PHASE_1EDGE; /* 串行同步时钟的第1个跳变沿(上升或下降)数据被采样 */
  40. HAL_SPI_Init(&g_spi2_handler);
  41. __HAL_SPI_ENABLE(&g_spi2_handler); /* 使能SPI2 */
  42. }
  43. /**
  44. * @brief 初始化24L01的IO口
  45. * @note 将SPI2模式改成SCK空闲低电平,及SPI 模式0
  46. * @param 无
  47. * @retval 无
  48. */
  49. void nrf24l01_init(void)
  50. {
  51. GPIO_InitTypeDef gpio_init_struct;
  52. NRF24L01_CE_GPIO_CLK_ENABLE(); /* CE脚时钟使能 */
  53. NRF24L01_CSN_GPIO_CLK_ENABLE(); /* CSN脚时钟使能 */
  54. NRF24L01_IRQ_GPIO_CLK_ENABLE(); /* IRQ脚时钟使能 */
  55. gpio_init_struct.Pin = NRF24L01_CE_GPIO_PIN;
  56. gpio_init_struct.Mode = GPIO_MODE_OUTPUT_PP; /* 推挽输出 */
  57. gpio_init_struct.Pull = GPIO_PULLUP; /* 上拉 */
  58. gpio_init_struct.Speed = GPIO_SPEED_FREQ_HIGH; /* 高速 */
  59. HAL_GPIO_Init(NRF24L01_CE_GPIO_PORT, &gpio_init_struct); /* 初始化CE引脚 */
  60. gpio_init_struct.Pin = NRF24L01_CSN_GPIO_PIN;
  61. HAL_GPIO_Init(NRF24L01_CSN_GPIO_PORT, &gpio_init_struct);/* 初始化CSN引脚 */
  62. gpio_init_struct.Pin = NRF24L01_IRQ_GPIO_PIN;
  63. gpio_init_struct.Mode = GPIO_MODE_INPUT; /* 输入 */
  64. gpio_init_struct.Pull = GPIO_PULLUP; /* 上拉 */
  65. gpio_init_struct.Speed = GPIO_SPEED_FREQ_HIGH; /* 高速 */
  66. HAL_GPIO_Init(NRF24L01_IRQ_GPIO_PORT, &gpio_init_struct);/* 初始化CE引脚 */
  67. spi2_init(); /* 初始化SPI2 */
  68. nrf24l01_spi_init(); /* 针对NRF的特点修改SPI的设置 */
  69. NRF24L01_CE(0); /* 使能24L01 */
  70. NRF24L01_CSN(1); /* SPI片选取消 */
  71. }
  72. /**
  73. * @brief 检测24L01是否存在
  74. * @param 无
  75. * @retval 0, 成功; 1, 失败;
  76. */
  77. uint8_t nrf24l01_check(void)
  78. {
  79. uint8_t buf[5] = {0XA5, 0XA5, 0XA5, 0XA5, 0XA5};
  80. uint8_t i;
  81. spi2_set_speed(SPI_SPEED_32); /* spi速度为7.5Mhz(24L01的最大SPI时钟为10Mhz) */
  82. nrf24l01_write_buf(NRF_WRITE_REG + TX_ADDR, buf, 5); /* 写入5个字节的地址. */
  83. nrf24l01_read_buf(TX_ADDR, buf, 5); /* 读出写入的地址 */
  84. for (i = 0; i < 5; i++)
  85. {
  86. if (buf[i] != 0XA5) break;
  87. }
  88. if (i != 5) return 1; /* 检测24L01错误 */
  89. return 0; /* 检测到24L01 */
  90. }
  91. /**
  92. * @brief NRF24L01写寄存器
  93. * @param reg : 寄存器地址
  94. * @param value : 写入寄存器的值
  95. * @retval 状态寄存器值
  96. */
  97. static uint8_t nrf24l01_write_reg(uint8_t reg, uint8_t value)
  98. {
  99. uint8_t status;
  100. NRF24L01_CSN(0); /* 使能SPI传输 */
  101. status = spi2_read_write_byte(reg); /* 发送寄存器号 */
  102. spi2_read_write_byte(value); /* 写入寄存器的值 */
  103. NRF24L01_CSN(1); /* 禁止SPI传输 */
  104. return status; /* 返回状态值 */
  105. }
  106. /**
  107. * @brief NRF24L01读寄存器
  108. * @param reg : 寄存器地址
  109. * @retval 读取到的寄存器值;
  110. */
  111. static uint8_t nrf24l01_read_reg(uint8_t reg)
  112. {
  113. uint8_t reg_val;
  114. NRF24L01_CSN(0); /* 使能SPI传输 */
  115. spi2_read_write_byte(reg); /* 发送寄存器号 */
  116. reg_val = spi2_read_write_byte(0XFF); /* 读取寄存器内容 */
  117. NRF24L01_CSN(1); /* 禁止SPI传输 */
  118. return reg_val; /* 返回状态值 */
  119. }
  120. /**
  121. * @brief 在指定位置读出指定长度的数据
  122. * @param reg : 寄存器地址
  123. * @param pbuf : 数据指针
  124. * @param len : 数据长度
  125. * @retval 状态寄存器值
  126. */
  127. static uint8_t nrf24l01_read_buf(uint8_t reg, uint8_t *pbuf, uint8_t len)
  128. {
  129. uint8_t status, i;
  130. NRF24L01_CSN(0); /* 使能SPI传输 */
  131. status = spi2_read_write_byte(reg); /* 发送寄存器值(位置),并读取状态值 */
  132. for (i = 0; i < len; i++)
  133. {
  134. pbuf[i] = spi2_read_write_byte(0XFF); /* 读出数据 */
  135. }
  136. NRF24L01_CSN(1); /* 关闭SPI传输 */
  137. return status; /* 返回读到的状态值 */
  138. }
  139. /**
  140. * @brief 在指定位置写指定长度的数据
  141. * @param reg : 寄存器地址
  142. * @param pbuf : 数据指针
  143. * @param len : 数据长度
  144. * @retval 状态寄存器值
  145. */
  146. static uint8_t nrf24l01_write_buf(uint8_t reg, uint8_t *pbuf, uint8_t len)
  147. {
  148. uint8_t status, i;
  149. NRF24L01_CSN(0); /* 使能SPI传输 */
  150. status = spi2_read_write_byte(reg);/* 发送寄存器值(位置),并读取状态值 */
  151. for (i = 0; i < len; i++)
  152. {
  153. spi2_read_write_byte(*pbuf++); /* 写入数据 */
  154. }
  155. NRF24L01_CSN(1); /* 关闭SPI传输 */
  156. return status; /* 返回读到的状态值 */
  157. }
  158. /**
  159. * @brief 启动NRF24L01发送一次数据(数据长度 = TX_PLOAD_WIDTH)
  160. * @param ptxbuf : 待发送数据首地址
  161. * @retval 发送完成状态
  162. * @arg 0 : 发送成功
  163. * @arg 1 : 达到最大发送次数,失败
  164. * @arg 0XFF : 其他错误
  165. */
  166. uint8_t nrf24l01_tx_packet(uint8_t *ptxbuf)
  167. {
  168. uint8_t sta;
  169. uint8_t rval = 0XFF;
  170. NRF24L01_CE(0);
  171. nrf24l01_write_buf(WR_TX_PLOAD, ptxbuf, TX_PLOAD_WIDTH); /* 写数据到TX BUF TX_PLOAD_WIDTH个字节 */
  172. NRF24L01_CE(1); /* 启动发送 */
  173. while (NRF24L01_IRQ != 0); /* 等待发送完成 */
  174. sta = nrf24l01_read_reg(STATUS); /* 读取状态寄存器的值 */
  175. nrf24l01_write_reg(NRF_WRITE_REG + STATUS, sta); /* 清除TX_DS或MAX_RT中断标志 */
  176. if (sta & MAX_TX) /* 达到最大重发次数 */
  177. {
  178. nrf24l01_write_reg(FLUSH_TX, 0xff); /* 清除TX FIFO寄存器 */
  179. rval = 1;
  180. }
  181. if (sta & TX_OK)/* 发送完成 */
  182. {
  183. rval = 0; /* 标记发送成功 */
  184. }
  185. return rval; /* 返回结果 */
  186. }
  187. /**
  188. * @brief 启动NRF24L01接收一次数据(数据长度 = RX_PLOAD_WIDTH)
  189. * @param prxbuf : 接收数据缓冲区首地址
  190. * @retval 接收完成状态
  191. * @arg 0 : 接收成功
  192. * @arg 1 : 失败
  193. */
  194. uint8_t nrf24l01_rx_packet(uint8_t *prxbuf)
  195. {
  196. uint8_t sta;
  197. uint8_t rval = 1;
  198. sta = nrf24l01_read_reg(STATUS); /* 读取状态寄存器的值 */
  199. nrf24l01_write_reg(NRF_WRITE_REG + STATUS, sta); /* 清除RX_DS中断标志 */
  200. if (sta & RX_OK) /* 接收到数据 */
  201. {
  202. nrf24l01_read_buf(RD_RX_PLOAD, prxbuf, RX_PLOAD_WIDTH); /* 读取数据 */
  203. nrf24l01_write_reg(FLUSH_RX, 0xff); /* 清除RX FIFO寄存器 */
  204. rval = 0; /* 标记接收完成 */
  205. }
  206. return rval; /* 返回结果 */
  207. }
  208. /**
  209. * @brief NRF24L01进入接收模式
  210. * @note 设置RX地址,写RX数据宽度,选择RF频道,波特率和LNA HCURR
  211. * 当CE变高后,即进入RX模式,并可以接收数据了
  212. * @param 无
  213. * @retval 无
  214. */
  215. void nrf24l01_rx_mode(void)
  216. {
  217. NRF24L01_CE(0);
  218. nrf24l01_write_buf(NRF_WRITE_REG + RX_ADDR_P0, (uint8_t *)RX_ADDRESS, RX_ADR_WIDTH); /* 写RX节点地址 */
  219. nrf24l01_write_reg(NRF_WRITE_REG + EN_AA, 0x01); /* 使能通道0的自动应答 */
  220. nrf24l01_write_reg(NRF_WRITE_REG + EN_RXADDR, 0x01); /* 使能通道0的接收地址 */
  221. nrf24l01_write_reg(NRF_WRITE_REG + RF_CH, 0); /* 设置RF通信频率 */
  222. nrf24l01_write_reg(NRF_WRITE_REG + RX_PW_P0, RX_PLOAD_WIDTH); /* 选择通道0的有效数据宽度 */
  223. nrf24l01_write_reg(NRF_WRITE_REG + RF_SETUP, 0x0f); /* 设置TX发射参数,0db增益,2Mbps,低噪声增益开启 */
  224. nrf24l01_write_reg(NRF_WRITE_REG + CONFIG, 0x0f); /* 配置基本工作模式的参数;PWR_UP,EN_CRC,16BIT_CRC,接收模式 */
  225. NRF24L01_CE(1); /* CE为高,进入接收模式 */
  226. }
  227. /**
  228. * @brief NRF24L01进入发送模式
  229. * @note 设置TX地址,写TX数据宽度,设置RX自动应答的地址,填充TX发送数据,选择RF频道,波特率和
  230. * LNA HCURR,PWR_UP,CRC使能
  231. * 当CE变高后,即进入TX模式,并可以发送数据了, CE为高大于10us,则启动发送.
  232. * @param 无
  233. * @retval 无
  234. */
  235. void nrf24l01_tx_mode(void)
  236. {
  237. NRF24L01_CE(0);
  238. nrf24l01_write_buf(NRF_WRITE_REG + TX_ADDR, (uint8_t *)TX_ADDRESS, TX_ADR_WIDTH); /* 写TX节点地址 */
  239. nrf24l01_write_buf(NRF_WRITE_REG + RX_ADDR_P0, (uint8_t *)RX_ADDRESS, RX_ADR_WIDTH); /* 设置RX节点地址,主要为了使能ACK */
  240. nrf24l01_write_reg(NRF_WRITE_REG + EN_AA, 0x01); /* 使能通道0的自动应答 */
  241. nrf24l01_write_reg(NRF_WRITE_REG + EN_RXADDR, 0x01); /* 使能通道0的接收地址 */
  242. nrf24l01_write_reg(NRF_WRITE_REG + SETUP_RETR, 0x1a); /* 设置自动重发间隔时间:500us + 86us;最大自动重发次数:10次 */
  243. nrf24l01_write_reg(NRF_WRITE_REG + RF_CH, 0); /* 设置RF通道为40 */
  244. nrf24l01_write_reg(NRF_WRITE_REG + RF_SETUP, 0x0f); /* 设置TX发射参数,0db增益,2Mbps,低噪声增益开启 */
  245. nrf24l01_write_reg(NRF_WRITE_REG + CONFIG, 0x0e); /* 配置基本工作模式的参数;PWR_UP,EN_CRC,16BIT_CRC,接收模式,开启所有中断 */
  246. NRF24L01_CE(1); /* CE为高,10us后启动发送 */
  247. }

接收部分

main

  1. /* USER CODE BEGIN Header */
  2. /**
  3. ******************************************************************************
  4. * @file : main.c
  5. * @brief : Main program body
  6. ******************************************************************************
  7. * @attention
  8. *
  9. * Copyright (c) 2023 STMicroelectronics.
  10. * All rights reserved.
  11. *
  12. * This software is licensed under terms that can be found in the LICENSE file
  13. * in the root directory of this software component.
  14. * If no LICENSE file comes with this software, it is provided AS-IS.
  15. *
  16. ******************************************************************************
  17. */
  18. /* USER CODE END Header */
  19. /* Includes ------------------------------------------------------------------*/
  20. #include "main.h"
  21. #include "usart.h"
  22. #include "gpio.h"
  23. /* Private includes ----------------------------------------------------------*/
  24. /* USER CODE BEGIN Includes */
  25. #include "onenet.h"
  26. #include "esp8266.h"
  27. #include "nrf24l01.h"
  28. //C库
  29. #include <string.h>
  30. #include <stdio.h>
  31. void SystemClock_Config(void);
  32. extern uint8_t Uart2_RxData; //接收中断缓冲
  33. unsigned char rece_buf[33];
  34. /**
  35. * @brief The application entry point.
  36. * @retval int
  37. */
  38. int main(void)
  39. {
  40. HAL_Init();
  41. SystemClock_Config();
  42. MX_GPIO_Init();
  43. MX_USART1_UART_Init();
  44. MX_USART2_UART_Init();
  45. nrf24l01_init(); /* 初始化NRF24L01 */
  46. HAL_UART_Receive_IT(&huart2,(uint8_t *)&Uart2_RxData, 1);//开启串口中断
  47. /*设置NRF2401接收模式*/
  48. nrf24l01_rx_mode();
  49. /* esp8266连接wifi+连接Onenet */
  50. HAL_Delay(2000);
  51. ESP8266_Init(); //初始化ESP8266,连接wifi
  52. HAL_Delay(2000);
  53. while(OneNet_DevLink()) //连接OneNET
  54. HAL_Delay(2000);
  55. printf("是否存在nrf2401");
  56. while(nrf24l01_check())
  57. {
  58. printf("不存在nrf2401");
  59. }
  60. while (1)
  61. {
  62. if(nrf24l01_rx_packet(rece_buf)==0)
  63. {
  64. rece_buf[32]=0;
  65. printf("%s\r\n",rece_buf);
  66. }
  67. OneNet_SendData(); //发送数据
  68. ESP8266_Clear(); //清空数据缓存区
  69. HAL_Delay(1000); //3s发送一次
  70. }
  71. }
  72. /**
  73. * @brief System Clock Configuration
  74. * @retval None
  75. */
  76. void SystemClock_Config(void)
  77. {
  78. RCC_OscInitTypeDef RCC_OscInitStruct = {0};
  79. RCC_ClkInitTypeDef RCC_ClkInitStruct = {0};
  80. /** Initializes the RCC Oscillators according to the specified parameters
  81. * in the RCC_OscInitTypeDef structure.
  82. */
  83. RCC_OscInitStruct.OscillatorType = RCC_OSCILLATORTYPE_HSE;
  84. RCC_OscInitStruct.HSEState = RCC_HSE_ON;
  85. RCC_OscInitStruct.HSEPredivValue = RCC_HSE_PREDIV_DIV1;
  86. RCC_OscInitStruct.HSIState = RCC_HSI_ON;
  87. RCC_OscInitStruct.PLL.PLLState = RCC_PLL_ON;
  88. RCC_OscInitStruct.PLL.PLLSource = RCC_PLLSOURCE_HSE;
  89. RCC_OscInitStruct.PLL.PLLMUL = RCC_PLL_MUL9;
  90. if (HAL_RCC_OscConfig(&RCC_OscInitStruct) != HAL_OK)
  91. {
  92. Error_Handler();
  93. }
  94. /** Initializes the CPU, AHB and APB buses clocks
  95. */
  96. RCC_ClkInitStruct.ClockType = RCC_CLOCKTYPE_HCLK|RCC_CLOCKTYPE_SYSCLK
  97. |RCC_CLOCKTYPE_PCLK1|RCC_CLOCKTYPE_PCLK2;
  98. RCC_ClkInitStruct.SYSCLKSource = RCC_SYSCLKSOURCE_PLLCLK;
  99. RCC_ClkInitStruct.AHBCLKDivider = RCC_SYSCLK_DIV1;
  100. RCC_ClkInitStruct.APB1CLKDivider = RCC_HCLK_DIV2;
  101. RCC_ClkInitStruct.APB2CLKDivider = RCC_HCLK_DIV1;
  102. if (HAL_RCC_ClockConfig(&RCC_ClkInitStruct, FLASH_LATENCY_2) != HAL_OK)
  103. {
  104. Error_Handler();
  105. }
  106. }
  107. /* USER CODE BEGIN 4 */
  108. /* USER CODE END 4 */
  109. /**
  110. * @brief This function is executed in case of error occurrence.
  111. * @retval None
  112. */
  113. void Error_Handler(void)
  114. {
  115. /* USER CODE BEGIN Error_Handler_Debug */
  116. /* User can add his own implementation to report the HAL error return state */
  117. __disable_irq();
  118. while (1)
  119. {
  120. }
  121. /* USER CODE END Error_Handler_Debug */
  122. }
  123. #ifdef USE_FULL_ASSERT
  124. /**
  125. * @brief Reports the name of the source file and the source line number
  126. * where the assert_param error has occurred.
  127. * @param file: pointer to the source file name
  128. * @param line: assert_param error line source number
  129. * @retval None
  130. */
  131. void assert_failed(uint8_t *file, uint32_t line)
  132. {
  133. /* USER CODE BEGIN 6 */
  134. /* User can add his own implementation to report the file name and line number,
  135. ex: printf("Wrong parameters value: file %s on line %d\r\n", file, line) */
  136. /* USER CODE END 6 */
  137. }
  138. #endif /* USE_FULL_ASSERT */

声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/繁依Fanyi0/article/detail/494253
推荐阅读
相关标签
  

闽ICP备14008679号