找回密码
 立即注册

QQ登录

只需一步,快速开始

搜索
查看: 2391|回复: 1
收起左侧

USB无线串口转NRF24L01数传通信,DS18B20采集温度,并通过串口助手显示

[复制链接]
ID:808304 发表于 2020-11-17 19:16 | 显示全部楼层 |阅读模式
最近做了一个用USB无线串口模块,串口转NRF24L01 数传通信,通过DS18B20采集外界温度,并通过传口助手显示采集的温度用到的模块用到的器材
USB无线串口模块,串口转NRF24L01 数传通信
温度采集模块,DS18B20
2块NRF24L01
51单片机
软件:串口助手
USB无线串口模块,串口转NRF24L01 数传通信.jpg DS18B20.jpg 测试结果.PNG

具体程序如下
  1. #include<reg51.h>
  2. #include<intrins.h>
  3. #define uchar unsigned char
  4. #define uint  unsigned int


  5. /**********  NRF24L01寄存器操作命令  ***********/
  6. #define READ_REG        0x00  //读配置寄存器,低5位为寄存器地址
  7. #define WRITE_REG       0x20  //写配置寄存器,低5位为寄存器地址
  8. #define RD_RX_PLOAD     0x61  //读RX有效数据,1~32字节
  9. #define WR_TX_PLOAD     0xA0  //写TX有效数据,1~32字节
  10. #define FLUSH_TX        0xE1  //清除TX FIFO寄存器.发射模式下用
  11. #define FLUSH_RX        0xE2  //清除RX FIFO寄存器.接收模式下用
  12. #define REUSE_TX_PL     0xE3  //重新使用上一包数据,CE为高,数据包被不断发送.
  13. #define NOP             0xFF  //空操作,可以用来读状态寄存器         
  14. /**********  NRF24L01寄存器地址   *************/
  15. #define CONFIG          0x00  //配置寄存器地址                             
  16. #define EN_AA           0x01  //使能自动应答功能
  17. #define EN_RXADDR       0x02  //接收地址允许
  18. #define SETUP_AW        0x03  //设置地址宽度(所有数据通道)
  19. #define SETUP_RETR      0x04  //建立自动重发
  20. #define RF_CH           0x05  //RF通道
  21. #define RF_SETUP        0x06  //RF寄存器
  22. #define STATUS          0x07  //状态寄存器
  23. #define OBSERVE_TX      0x08  // 发送检测寄存器
  24. #define CD              0x09  // 载波检测寄存器
  25. #define RX_ADDR_P0      0x0A  // 数据通道0接收地址
  26. #define RX_ADDR_P1      0x0B  // 数据通道1接收地址
  27. #define RX_ADDR_P2      0x0C  // 数据通道2接收地址
  28. #define RX_ADDR_P3      0x0D  // 数据通道3接收地址
  29. #define RX_ADDR_P4      0x0E  // 数据通道4接收地址
  30. #define RX_ADDR_P5      0x0F  // 数据通道5接收地址
  31. #define TX_ADDR         0x10  // 发送地址寄存器
  32. #define RX_PW_P0        0x11  // 接收数据通道0有效数据宽度(1~32字节)
  33. #define RX_PW_P1        0x12  // 接收数据通道1有效数据宽度(1~32字节)
  34. #define RX_PW_P2        0x13  // 接收数据通道2有效数据宽度(1~32字节)
  35. #define RX_PW_P3        0x14  // 接收数据通道3有效数据宽度(1~32字节)
  36. #define RX_PW_P4        0x15  // 接收数据通道4有效数据宽度(1~32字节)
  37. #define RX_PW_P5        0x16  // 接收数据通道5有效数据宽度(1~32字节)
  38. #define FIFO_STATUS     0x17  // FIFO状态寄存器
  39. /*————————————————————————————————————————————————————————————————————*/

  40. /******   STATUS寄存器bit位定义      *******/
  41. #define MAX_TX          0x10            //达到最大发送次数中断
  42. #define TX_OK           0x20            //TX发送完成中断
  43. #define RX_OK           0x40            //接收到数据中断
  44. /*——————————————————————————————————————————————————*/

  45. /*********     24L01发送接收数据宽度定义          ***********/
  46. #define TX_ADR_WIDTH    5     //5字节地址宽度
  47. #define RX_ADR_WIDTH    5     //5字节地址宽度
  48. #define TX_PLOAD_WIDTH  32    //32字节有效数据宽度
  49. #define RX_PLOAD_WIDTH  32    //32字节有效数据宽度

  50. const uchar TX_ADDRESS[TX_ADR_WIDTH]={0xFF,0xFF,0xFF,0xFF,0xFF}; //发送地址
  51. const uchar RX_ADDRESS[RX_ADR_WIDTH]={0xFF,0xFF,0xFF,0xFF,0xFF}; //发送地址

  52. sbit NRF_CE   = P0^7;
  53. sbit NRF_CSN  = P0^6;
  54. sbit NRF_MISO = P0^3;
  55. sbit NRF_MOSI = P0^4;
  56. sbit NRF_SCK  = P0^5;
  57. sbit NRF_IRQ  = P0^2;
  58. sbit LED=P1^0;
  59. sbit DQ=P3^7; //ds18b20 与单片机连接口
  60. //sbit S1=P3^0;
  61. //sbit S2=P3^1;
  62. //sbit S3=P3^2;
  63. uchar rece_buf[32];
  64. uchar data disdata[6];// 百、十、个、小数位1、小数位2、        小数位3、
  65. uint tvalue; // 温度值
  66. uchar tflag; // 温度正负标志

  67. void delay_us(uchar num)
  68. {
  69.         uchar i;
  70.         for(i=0;i>num;i++)
  71.         _nop_();
  72. }
  73. void delay_150us()
  74. {
  75.         uint i;

  76.         for(i=0;i>150;i++);
  77. }
  78. void delay(uint t)
  79. {
  80.         uchar k;
  81.         while(t--)
  82.         for(k=0;k<200;k++);
  83. }
  84. /***************************************************************/

  85. /*******************************************************************/
  86. uchar SPI_RW(uchar byte)
  87. {
  88.         uchar bit_ctr;
  89.         for(bit_ctr=0;bit_ctr<8;bit_ctr++)  // 输出8位
  90.         {
  91.                 NRF_MOSI=(byte&0x80);                         // MSB TO MOSI
  92.                 byte=(byte<<1);                                        // shift next bit to MSB
  93.                 NRF_SCK=1;
  94.                 byte|=NRF_MISO;                                // capture current MISO bit
  95.                 NRF_SCK=0;
  96.         }
  97.         return byte;
  98. }

  99. /*********************************************/
  100. /* 函数功能:给24L01的寄存器写值(一个字节) */
  101. /* 入口参数:reg   要写的寄存器地址          */
  102. /*           value 给寄存器写的值            */
  103. /* 出口参数:status 状态值                   */
  104. /*********************************************/
  105. uchar NRF24L01_Write_Reg(uchar reg,uchar value)
  106. {
  107.         uchar status;

  108.         NRF_CSN=0;                  //CSN=0;   
  109.           status = SPI_RW(reg);                //发送寄存器地址,并读取状态值
  110.         SPI_RW(value);
  111.         NRF_CSN=1;                  //CSN=1;

  112.         return status;
  113. }
  114. /*************************************************/
  115. /* 函数功能:读24L01的寄存器值 (一个字节)      */
  116. /* 入口参数:reg  要读的寄存器地址               */
  117. /* 出口参数:value 读出寄存器的值                */
  118. /*************************************************/
  119. uchar NRF24L01_Read_Reg(uchar reg)
  120. {
  121.         uchar value;

  122.         NRF_CSN=0;              //CSN=0;   
  123.           SPI_RW(reg);                        //发送寄存器值(位置),并读取状态值
  124.         value = SPI_RW(NOP);
  125.         NRF_CSN=1;                     //CSN=1;

  126.         return value;
  127. }
  128. /*********************************************/
  129. /* 函数功能:读24L01的寄存器值(多个字节)   */
  130. /* 入口参数:reg   寄存器地址                */
  131. /*           *pBuf 读出寄存器值的存放数组    */
  132. /*           len   数组字节长度              */
  133. /* 出口参数:status 状态值                   */
  134. /*********************************************/
  135. uchar NRF24L01_Read_Buf(uchar reg,uchar *pBuf,uchar len)
  136. {
  137.         uchar status,u8_ctr;
  138.         NRF_CSN=0;                           //CSN=0      
  139.           status=SPI_RW(reg);                                //发送寄存器地址,并读取状态值             
  140.         for(u8_ctr=0;u8_ctr<len;u8_ctr++)
  141.         pBuf[u8_ctr]=SPI_RW(0XFF);                //读出数据
  142.         NRF_CSN=1;                                 //CSN=1
  143.           return status;                                //返回读到的状态值
  144. }
  145. /**********************************************/
  146. /* 函数功能:给24L01的寄存器写值(多个字节)  */
  147. /* 入口参数:reg  要写的寄存器地址            */
  148. /*           *pBuf 值的存放数组               */
  149. /*           len   数组字节长度               */
  150. /**********************************************/
  151. uchar NRF24L01_Write_Buf(uchar reg, uchar *pBuf, uchar len)
  152. {
  153.         uchar status,u8_ctr;
  154.         NRF_CSN=0;
  155.           status = SPI_RW(reg);                        //发送寄存器值(位置),并读取状态值
  156.           for(u8_ctr=0; u8_ctr<len; u8_ctr++)
  157.         SPI_RW(*pBuf++);                                 //写入数据
  158.         NRF_CSN=1;
  159.           return status;                          //返回读到的状态值
  160. }                                                                                                    

  161. /*********************************************/
  162. /* 函数功能:24L01接收数据                   */
  163. /* 入口参数:rxbuf 接收数据数组              */
  164. /* 返回值: 0   成功收到数据                 */
  165. /*          1   没有收到数据                 */
  166. /*********************************************/
  167. uchar NRF24L01_RxPacket(uchar *rxbuf)
  168. {
  169.         uchar state;
  170.          
  171.         state=NRF24L01_Read_Reg(STATUS);                          //读取状态寄存器的值             
  172.         NRF24L01_Write_Reg(WRITE_REG+STATUS,state); //清除TX_DS或MAX_RT中断标志
  173.         if(state&RX_OK)                                                                //接收到数据
  174.         {
  175.                 NRF_CE = 0;
  176.                 NRF24L01_Read_Buf(RD_RX_PLOAD,rxbuf,RX_PLOAD_WIDTH);//读取数据
  177.                 NRF24L01_Write_Reg(FLUSH_RX,0xff);                                        //清除RX FIFO寄存器
  178.                 NRF_CE = 1;
  179.                 delay_150us();
  180.                 return 0;
  181.         }          
  182.         return 1;//没收到任何数据
  183. }
  184. /**********************************************/
  185. /* 函数功能:设置24L01为发送模式              */
  186. /* 入口参数:txbuf  发送数据数组              */
  187. /* 返回值; 0x10    达到最大重发次数,发送失败*/
  188. /*          0x20    成功发送完成              */
  189. /*          0xff    发送失败                  */
  190. /**********************************************/
  191. uchar NRF24L01_TxPacket(uchar *txbuf)
  192. {
  193.         uchar state;
  194.    
  195.         NRF_CE=0;                                                                                                //CE拉低,使能24L01配置
  196.           NRF24L01_Write_Buf(WR_TX_PLOAD,txbuf,TX_PLOAD_WIDTH);        //写数据到TX BUF  32个字节
  197.         NRF_CE=1;                                                                                                //CE置高,使能发送          
  198.         while(NRF_IRQ==1);                                                                                //等待发送完成
  199.         state=NRF24L01_Read_Reg(STATUS);                                                  //读取状态寄存器的值          
  200.         NRF24L01_Write_Reg(WRITE_REG+STATUS,state);                         //清除TX_DS或MAX_RT中断标志
  201.         if(state&MAX_TX)                                                                                //达到最大重发次数
  202.         {
  203.                 NRF24L01_Write_Reg(FLUSH_TX,0xff);                                        //清除TX FIFO寄存器
  204.                 return MAX_TX;
  205.         }
  206.         if(state&TX_OK)                                                                                        //发送完成
  207.         {
  208.                 return TX_OK;
  209.         }
  210.         return 0xff;                                                                                        //发送失败
  211. }

  212. /********************************************/
  213. /* 函数功能:检测24L01是否存在              */
  214. /* 返回值;  0  存在                        */
  215. /*           1  不存在                      */
  216. /********************************************/           
  217. uchar NRF24L01_Check(void)
  218. {
  219.         uchar check_in_buf[5]={0x11,0x22,0x33,0x44,0x55};
  220.         uchar check_out_buf[5]={0x00};

  221.         NRF_SCK=0;
  222.         NRF_CSN=1;   
  223.         NRF_CE=0;

  224.         NRF24L01_Write_Buf(WRITE_REG+TX_ADDR, check_in_buf, 5);

  225.         NRF24L01_Read_Buf(READ_REG+TX_ADDR, check_out_buf, 5);

  226.         if((check_out_buf[0] == 0x11)&&\
  227.            (check_out_buf[1] == 0x22)&&\
  228.            (check_out_buf[2] == 0x33)&&\
  229.            (check_out_buf[3] == 0x44)&&\
  230.            (check_out_buf[4] == 0x55))return 0;
  231.         else return 1;
  232. }                       


  233. void NRF24L01_RT_Init(void)
  234. {       
  235.         NRF_CE=0;                  
  236.           NRF24L01_Write_Reg(WRITE_REG+RX_PW_P0,RX_PLOAD_WIDTH);//选择通道0的有效数据宽度
  237.         NRF24L01_Write_Reg(FLUSH_RX,0xff);                                                                        //清除RX FIFO寄存器   
  238.           NRF24L01_Write_Buf(WRITE_REG+TX_ADDR,(uchar*)TX_ADDRESS,TX_ADR_WIDTH);//写TX节点地址
  239.           NRF24L01_Write_Buf(WRITE_REG+RX_ADDR_P0,(uchar*)RX_ADDRESS,RX_ADR_WIDTH); //设置TX节点地址,主要为了使能ACK          
  240.           NRF24L01_Write_Reg(WRITE_REG+EN_AA,0x01);     //使能通道0的自动应答   
  241.           NRF24L01_Write_Reg(WRITE_REG+EN_RXADDR,0x01); //使能通道0的接收地址  
  242.           NRF24L01_Write_Reg(WRITE_REG+SETUP_RETR,0x1a);//设置自动重发间隔时间:500us + 86us;最大自动重发次数:10次
  243.           NRF24L01_Write_Reg(WRITE_REG+RF_CH,0);        //设置RF通道为2.400GHz  频率=2.4+0GHz
  244.           NRF24L01_Write_Reg(WRITE_REG+RF_SETUP,0x0F);  //设置TX发射参数,0db增益,2Mbps,低噪声增益开启   
  245.           NRF24L01_Write_Reg(WRITE_REG+CONFIG,0x0f);    //配置基本工作模式的参数;PWR_UP,EN_CRC,16BIT_CRC,接收模式,开启所有中断
  246.         NRF_CE=1;                                                                          //CE置高,使能发送
  247. }

  248. void SEND_BUF(uchar *buf)
  249. {
  250.         NRF_CE=0;
  251.         NRF24L01_Write_Reg(WRITE_REG+CONFIG,0x0e);
  252.         NRF_CE=1;
  253.         delay_us(15);
  254.         NRF24L01_TxPacket(buf);
  255.         NRF_CE=0;
  256.         NRF24L01_Write_Reg(WRITE_REG+CONFIG, 0x0f);
  257.         NRF_CE=1;       
  258. }
  259. /******************************ds18b20 程序 ******************************************************************/
  260. void delay_18b20(uint i) // 延时 1 微秒
  261. {
  262. while(i--);
  263. }
  264. void ds18b20rst() //ds18b20 初始化子函数
  265. // 要求"数据线拉高 - 延时 - 数据线拉低 - 延时大于 480 微妙 -数据线拉高 - 延时等待 "
  266. {
  267. uchar x=0;
  268. DQ = 1; // 信号线 DQ复位
  269. delay_18b20(4); // 延时
  270. DQ = 0; //DQ 拉低
  271. delay_18b20(100); // 精确延时大于 480us
  272. DQ = 1; // 拉高
  273. delay_18b20(40);
  274. }
  275. void ds18b20wr(uchar wdata) /* 写数据子函数 , 无返回值,含参数 */
  276. {
  277. uchar i=0;
  278. for (i=8; i>0; i--)// 要写完一个字节,故需要重复 8 次以下操作
  279. {
  280. DQ = 0; // 数据线拉低
  281. DQ = wdata&0x01;//wdata 是一个形参,将其与 0000 0001 进行按位与
  282. // 按从低到高的顺序发送数据(一次发送一位 )
  283. delay_18b20(10);
  284. DQ = 1; // 最后将数据线拉高
  285. wdata>>=1; // 将 wdata 右移 1 位
  286. }
  287. }
  288. uchar ds18b20rd() // 读数据子函数 , 是有返回值 dat
  289. {
  290. uchar i=0;
  291. uchar dat = 0;
  292. for (i=8;i>0;i--)// 要读完一个字节,故需要重复 8 次以下操作
  293. {
  294. DQ = 0; // 给脉冲信号
  295. dat>>=1;
  296. DQ = 1; // 给脉冲信号
  297. if(DQ)
  298. dat|=0x80;
  299. delay_18b20(10);
  300. }
  301. return(dat); // 返回 dat
  302. }
  303. read_temp() // 读取温度值并转换的子函数,有返回值温度值 tvalue
  304. {
  305. uchar a,b;
  306. ds18b20rst(); // 调用 ds18b20 初始化子函数
  307. ds18b20wr(0xcc);// 调用写数据子函数,向 ds18b20 写命令 0xcc
  308. //ccH 表示跳过 ROM读序列号,适用于单机工作,直接向 18b20 发送温度变换命令
  309. ds18b20wr(0x44);// 调用写数据子函数,向 ds18b20 写命令 0x44
  310. //44H 表示启动 ds18b20 温度转换 , 结果自行存入 9 字节的 RAM中
  311. ds18b20rst(); // 调用 ds18b20 初始化子函数
  312. ds18b20wr(0xcc);// 同上
  313. ds18b20wr(0xbe);// 调用写数据子函数,向 ds18b20 写命令 0xbe
  314. //beH 表示读取 RAM中 9 字节的温度数据
  315. a=ds18b20rd(); // 调用读数据子函数,并将所得数据给 a
  316. b=ds18b20rd(); //
  317. tvalue=b; // 把 b 的值给 tvalue
  318. tvalue<<=8; //tvalue 左移 8 位
  319. tvalue=tvalue|a;//tvalue 与 a 进行按位或
  320. if(tvalue<0x0fff)//
  321. tflag=0; // 前五位为 0 时,读取的温度为正,标志位为 0,此时只要
  322. // 将测得数值乘以 0.0625 即可得到实际温度
  323. else // 前五位为 1 时,读取的温度为负,标志位为 1,此时需要
  324. { // 将测得数值取反后再加 1,再乘以 0.0625 即可得到实际温度
  325. tvalue=~tvalue+1;
  326. tflag=1; // 此时表示负温度
  327. }
  328. tvalue=tvalue*(0.625);// 温度值扩大 10 倍,精确到 1 位小数
  329. return(tvalue); // 返回温度值
  330. }
  331. /**********************************************/


  332. void main()
  333. {uchar flagdat;
  334.         while(NRF24L01_Check()); // 等待检测到NRF24L01,程序才会向下执行
  335.         NRF24L01_RT_Init();               

  336.         while(1)
  337.         {
  338.                 if(NRF_IRQ==0)                 // 如果无线模块接收到数据
  339.                 {               
  340.                         if(NRF24L01_RxPacket(rece_buf)==0)
  341.                         {                          
  342.                                 if(        rece_buf[1]=='1')                           //第1位以后是收到的命令数据,rece_buf[0]是数据位数长度
  343.                                         LED=0;
  344.                                 if(        rece_buf[1]=='2')                         //第1位以后是收到的命令数据,rece_buf[0]是数据位数长度
  345.                                         LED=1;               
  346.                         }
  347.                 }
  348.                 read_temp(); // 调用 ds18b20 读取温度       
  349.        

  350.                   rece_buf[1]='T';
  351.                   rece_buf[2]='=';
  352.       
  353.                  rece_buf[3]='+'; // 百位数;
  354.                         rece_buf[4]=tvalue%1000/100+0x30;// 十位数;
  355.                         rece_buf[5]=tvalue%100/10+0x30; // 个位数;
  356.       rece_buf[6] ='.';
  357.                   rece_buf[7] =tvalue%10+0x30; // 小数位
  358.      rece_buf[8]='C';
  359.                  rece_buf[9]=0x0d;
  360.                  rece_buf[10]=0x0a;//串口助手换行
  361.                         rece_buf[0]=10;                                                  //一共要发送10个字节,rece_buf[0]必须是10!!!!!!
  362.                         SEND_BUF(rece_buf);
  363.                

  364.                 delay(1000);
  365.         }
  366. }
复制代码

全部资料51hei下载地址:
DS18B20采集温度,并通过串口助手显示采集的温度.zip (43.69 KB, 下载次数: 48)

评分

参与人数 1黑币 +50 收起 理由
admin + 50 共享资料的黑币奖励!

查看全部评分

回复

使用道具 举报

ID:858367 发表于 2021-1-8 09:17 | 显示全部楼层
楼主辛苦,资料不错
回复

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

手机版|小黑屋|51黑电子论坛 |51黑电子论坛6群 QQ 管理员QQ:125739409;技术交流QQ群281945664

Powered by 单片机教程网

快速回复 返回顶部 返回列表