找回密码
 立即注册

QQ登录

只需一步,快速开始

搜索
查看: 315|回复: 5
收起左侧

51单片机的nrf24l01双向通信,后面可以切换收发模式,主函数可以修改 病床呼叫系统

[复制链接]
ID:1113126 发表于 2024-3-27 10:42 | 显示全部楼层 |阅读模式
  1. #include<reg51.h>
  2. #define uchar unsigned char
  3. #define uint unsigned int



  4. sbit        CE            =P1^0;
  5. sbit        CSN                =P1^1;
  6. sbit        SCK            =P1^2;
  7. sbit         MOSI        =P1^3;
  8. sbit         MISO        =P1^4;
  9. sbit        IRQ                =P1^5;

  10. sbit KEY1=P3^2;     //发送按键
  11. sbit LED=P2^0;     ////接收到数据后的功能实现灯  
  12. sbit tf2=P2^2;     //二号病床呼叫
  13. sbit tf4=P2^4;     //二号病床取消呼叫
  14. sbit tf5=P2^5;     //平安无事
  15. sbit tf6=P2^6;     //已确认处理


  16. #define TX_ADR_WIDTH    5           // 5 uints TX address width
  17. #define RX_ADR_WIDTH    5           // 5 uints RX address width
  18. #define TX_PLOAD_WIDTH  32          // 32 uints TX payload
  19. #define RX_PLOAD_WIDTH  32          // 32 uints TX payload
  20. uchar  TX_ADDRESS[TX_ADR_WIDTH]= {0xE7,0xE7,0xE7,0xE7,0x22};        //本地地址
  21. uchar  RX_ADDRESS[RX_ADR_WIDTH]= {0xE7,0xE7,0xE7,0xE7,0x22};        //接收地址
  22. ///***************************************NRF24L01寄存器指令*******************************************************
  23. #define READ_REG        0x00          // 读寄存器指令
  24. #define WRITE_REG       0x20         // 写寄存器指令
  25. #define RD_RX_PLOAD     0x61          // 读取接收数据指令
  26. #define WR_TX_PLOAD     0xA0          // 写待发数据指令
  27. #define FLUSH_TX        0xE1         // 冲洗发送 FIFO指令
  28. #define FLUSH_RX        0xE2          // 冲洗接收 FIFO指令
  29. #define REUSE_TX_PL     0xE3          // 定义重复装载数据指令
  30. #define NOP             0xFF          // 保留
  31. ///*************************************SPI(nRF24L01)寄存器地址****************************************************
  32. #define CONFIG          0x00  // 配置收发状态,CRC校验模式以及收发状态响应方式
  33. #define EN_AA           0x01  // 自动应答功能设置
  34. #define EN_RXADDR       0x02  // 可用信道设置
  35. #define SETUP_AW        0x03  // 收发地址宽度设置
  36. #define SETUP_RETR      0x04  // 自动重发功能设置
  37. #define RF_CH           0x05  // 工作频率设置
  38. #define RF_SETUP        0x06  // 发射速率、功耗功能设置
  39. #define NRFRegSTATUS    0x07  // 状态寄存器
  40. #define OBSERVE_TX      0x08  // 发送监测功能
  41. #define CD              0x09  // 地址检测           
  42. #define RX_ADDR_P0      0x0A  // 频道0接收数据地址
  43. #define RX_ADDR_P1      0x0B  // 频道1接收数据地址
  44. #define RX_ADDR_P2      0x0C  // 频道2接收数据地址
  45. #define RX_ADDR_P3      0x0D  // 频道3接收数据地址
  46. #define RX_ADDR_P4      0x0E  // 频道4接收数据地址
  47. #define RX_ADDR_P5      0x0F  // 频道5接收数据地址
  48. #define TX_ADDR         0x10  // 发送地址寄存器
  49. #define RX_PW_P0        0x11  // 接收频道0接收数据长度
  50. #define RX_PW_P1        0x12  // 接收频道1接收数据长度
  51. #define RX_PW_P2        0x13  // 接收频道2接收数据长度
  52. #define RX_PW_P3        0x14  // 接收频道3接收数据长度
  53. #define RX_PW_P4        0x15  // 接收频道4接收数据长度
  54. #define RX_PW_P5        0x16  // 接收频道5接收数据长度
  55. #define FIFO_STATUS     0x17  // FIFO栈入栈出状态寄存器设置
  56. ///*****************************子函数集*********************************************************
  57. uchar NRF24SPI_Send_Byte(uchar dat);
  58. uchar SPI_WR_Reg(uchar reg, uchar value);
  59. uchar SPI_Read_Buf(uchar reg, uchar *pBuf, uchar Len);
  60. uchar SPI_Write_Buf(uchar reg, uchar *pBuf, uchar Len);
  61. uchar nRF24L01_RxPacket(unsigned char* rx_buf);
  62. void nRF24L01_TxPacket(unsigned char * tx_buf);
  63. uchar SPI_RD_Reg(uchar reg);
  64. void init_NRF24L01(void);
  65. void TX_Mode(void);
  66. void RX_Mode(void);
  67. void NRF_Send(void);
  68. uchar NRF24L01_Check(void);
  69. ///*************************************************************************************
  70. uchar NRF24SPI_Send_Byte(uchar dat)
  71. {
  72.   uchar i;
  73.    for (i = 0; i < 8; i++) // output 8-bit
  74.    {
  75.       //发送1位数据
  76.       if (dat & 0x80)         // output 'uchar', MSB to MOSI
  77.       {
  78.          MOSI = 1;
  79.       }
  80.       else
  81.       {
  82.          MOSI = 0;
  83.       }

  84.       dat <<= 1;           // shift next bit into MSB..

  85.       //读取1位数据
  86.       SCK = 1;                      // Set SCK high..

  87.       if (MISO)
  88.       {
  89.          dat |= 1;
  90.       }             // capture current MISO bit
  91.       else
  92.       {
  93.          dat &= 0xFE;
  94.       }
  95.       SCK = 0;                    // ..then set SCK low again
  96.    }

  97.    return(dat);                  // return read uchar
  98. }

  99. void Delay_n10us(uint n)           //延时n个10us
  100. {
  101.    for(;n>0;n--)
  102.    {
  103.             unsigned char a,b;
  104.      for(b=1;b>0;b--)
  105.          for(a=2;a>0;a--);
  106.    }
  107. }
  108. ///****************************************************************************************
  109. ///*NRF24L01检测是否存在
  110. ///***************************************************************************************/
  111. uchar NRF24L01_Check(void)
  112. {
  113.         uchar bu[5]={0XA5,0XA5,0XA5,0XA5,0XA5};
  114.         uchar bu1[5];
  115.         uchar i;            
  116.         SPI_Write_Buf(WRITE_REG+TX_ADDR,bu,5);//写入5个字节的地址.        
  117.         SPI_Read_Buf(TX_ADDR,bu1,5);              //读出写入的地址         
  118.         for(i=0;i<5;i++)if(bu1[i]!=0XA5)break;                                          
  119.         if(i!=5)return 1;                               //NRF24L01不在位        
  120.         return 0;                                                //NRF24L01在位
  121. }                  
  122. ///****************************************************************************************
  123. ///*NRF24L01初始化
  124. ///***************************************************************************************/
  125. void init_NRF24L01(void)
  126. {
  127.    uchar buf[5]={0};
  128.    Delay_n10us(10);
  129.    CE = 0;    // chip enable
  130.    CSN= 0;    // Spi disable

  131.   SPI_Read_Buf(TX_ADDR, buf, TX_ADR_WIDTH);//debug 测试原来的本地地址:复位值是:0xE7 0xE7 0xE7 0xE7 0xE7

  132. //        SPI_Write_Buf(WRITE_REG + TX_ADDR, TX_ADDRESS, TX_ADR_WIDTH);    // 写本地地址        
  133. //        SPI_Write_Buf(WRITE_REG + RX_ADDR_P0, RX_ADDRESS, RX_ADR_WIDTH); // 写接收端地址

  134. //        
  135. //        SPI_WR_Reg(WRITE_REG + EN_AA, 0x01);      //  频道0自动        ACK应答允许        
  136. //        SPI_WR_Reg(WRITE_REG + EN_RXADDR, 0x01);  //  允许接收地址只有频道0,如果需要多频道可以参考Page21  
  137. //        SPI_WR_Reg(WRITE_REG + SETUP_RETR, 0x1a); // 设置自动重发时间和次数:500us + 86us, 10 retrans...
  138. //        SPI_WR_Reg(WRITE_REG + RF_CH, 22);        //   设置信道工作为2.4GHZ,收发必须一致
  139. //        SPI_WR_Reg(WRITE_REG + RX_PW_P0, RX_PLOAD_WIDTH); //设置接收数据长度,本次设置为32字节
  140. //        SPI_WR_Reg(WRITE_REG + RF_SETUP, 0x07);                   //设置发射速率为1MHZ,发射功率为最大值0dB
  141. //  
  142. //  SPI_RD_Reg(WRITE_REG + EN_AA);
  143. //  SPI_RD_Reg(WRITE_REG + EN_RXADDR);
  144. //  SPI_RD_Reg(WRITE_REG + RF_CH);
  145. //  SPI_RD_Reg(WRITE_REG + RX_PW_P0);
  146. //  SPI_RD_Reg(WRITE_REG + RF_SETUP);
  147. }
  148. ///****************************************************************************************************
  149. ///*函数:uchar SPI_Read(uchar reg)
  150. ///*功能:NRF24L01的SPI时序
  151. ///****************************************************************************************************/
  152. uchar SPI_RD_Reg(uchar reg)
  153. {
  154.         uchar reg_val;
  155.         
  156.         CSN = 0;                // CSN low, initialize SPI communication...
  157.         NRF24SPI_Send_Byte(reg);            // Select register to read from..
  158.         reg_val = NRF24SPI_Send_Byte(0);    // ..then read registervalue
  159.         CSN = 1;                // CSN high, terminate SPI communication
  160.         
  161.         return(reg_val);        // return register value
  162. }
  163. //****************************************************************************************************/
  164. //*功能:NRF24L01读写寄存器函数
  165. //****************************************************************************************************/
  166. uchar SPI_WR_Reg(uchar reg, uchar value)
  167. {
  168.         uchar status;
  169.         
  170.         CSN = 0;                   // CSN low, init SPI transaction
  171.         status = NRF24SPI_Send_Byte(reg);// select register
  172.         NRF24SPI_Send_Byte(value);             // ..and write value to it..
  173.         CSN = 1;                   // CSN high again
  174.         
  175.         return(status);            // return nRF24L01 status uchar
  176. }
  177. ///****************************************************************************************************/
  178. //*函数:uint SPI_Read_Buf(uchar reg, uchar *pBuf, uchar Len)
  179. //*功能: 用于读数据,reg:为寄存器地址,pBuf:为待读出数据地址,uchars:读出数据的个数
  180. //****************************************************************************************************/
  181. uchar SPI_Read_Buf(uchar reg, uchar *pBuf, uchar Len)
  182. {
  183.         uint status,i;
  184.         
  185.         CSN = 0;                                    // Set CSN low, init SPI tranaction
  186.         status = NRF24SPI_Send_Byte(reg);                       // Select register to write to and read status uchar
  187.         
  188.   for(i=0;i<Len;i++)
  189.   {
  190.      pBuf[i] = NRF24SPI_Send_Byte(0);
  191.   }
  192.         
  193.         CSN = 1;                           
  194.         
  195.         return(status);                    // return nRF24L01 status uchar
  196. }
  197. //*********************************************************************************************************
  198. //*函数:uint SPI_Write_Buf(uchar reg, uchar *pBuf, uchar Len)
  199. //*功能: 用于写数据:为寄存器地址,pBuf:为待写入数据地址,uchars:写入数据的个数
  200. //*********************************************************************************************************/
  201. uchar SPI_Write_Buf(uchar reg, uchar *pBuf, uchar Len)
  202. {
  203.         uint status,i;
  204.         
  205.         CSN = 0;            //SPI使能      
  206.         status = NRF24SPI_Send_Byte(reg);   
  207.         for(i=0; i<Len; i++) //
  208.         {
  209.            NRF24SPI_Send_Byte(*pBuf);
  210.                  pBuf ++;
  211.         }
  212.         CSN = 1;           //关闭SPI
  213.         return(status);    //
  214. }

  215. //****************************************************************************************************/
  216. //*函数:void SetRX_Mode(void)
  217. //*功能:数据接收配置
  218. //****************************************************************************************************/
  219. void RX_Mode(void)
  220. {
  221.     uchar buf[5]={0};
  222.         CE = 0;
  223.         
  224.         SPI_Read_Buf(TX_ADDR, buf, TX_ADR_WIDTH);//debug 测试原来的本地地址:复位值是:0xE7 0xE7 0xE7 0xE7 0xE7
  225.         //SPI_WR_Reg(WRITE_REG + CONFIG, 0x03);//SPI_WR_Reg(WRITE_REG + CONFIG, 0x0f); // IRQ收发完成中断响应,16位CRC        ,主接收
  226.         
  227.         //SPI_Write_Buf(WRITE_REG + TX_ADDR, TX_ADDRESS, TX_ADR_WIDTH);    // 写本地地址        
  228.         SPI_Write_Buf(WRITE_REG + RX_ADDR_P0, RX_ADDRESS, RX_ADR_WIDTH); // 写接收端地址
  229.         
  230.         SPI_WR_Reg(WRITE_REG + EN_AA, 0);//0x01);      //  频道0自动        ACK应答允许        
  231.         SPI_WR_Reg(WRITE_REG + EN_RXADDR,0x01);  //  允许接收地址只有频道0,如果需要多频道可以参考Page21  
  232.         SPI_WR_Reg(WRITE_REG + SETUP_RETR,0x1a); // 设置自动重发时间和次数:500us + 86us, 10 retrans...
  233.         SPI_WR_Reg(WRITE_REG + RF_CH, 40);        //   设置信道工作为2.4GHZ,收发必须一致
  234.         SPI_WR_Reg(WRITE_REG + RX_PW_P0, RX_PLOAD_WIDTH); //设置接收数据长度,本次设置为32字节
  235.         SPI_WR_Reg(WRITE_REG + RF_SETUP, 0x0F);                   //设置发射速率为2MHZ,发射功率为最大值0dB
  236.         
  237.         SPI_WR_Reg(WRITE_REG + CONFIG, 0x0F);//0x0F);        
  238.         CE = 1;
  239.         Delay_n10us(20); //200us
  240. }
  241. //******************************************************************************************************/
  242. //*函数:unsigned char nRF24L01_RxPacket(unsigned char* rx_buf)
  243. //*功能:数据读取后放如rx_buf接收缓冲区中
  244. //******************************************************************************************************/
  245. uchar nRF24L01_RxPacket(unsigned char* rx_buf)
  246. {
  247.     uchar flag=0;
  248.     uchar status;
  249.          
  250.         status=SPI_RD_Reg(NRFRegSTATUS);        // 读取状态寄存其来判断数据接收状况
  251.         
  252.         if(status & 0x40)                                // 判断是否接收到数据
  253.         {
  254.            CE = 0;                         //SPI使能
  255.                  SPI_Read_Buf(RD_RX_PLOAD,rx_buf,TX_PLOAD_WIDTH);// read receive payload from RX_FIFO buffer
  256.                  flag =1;                        //读取数据完成标志
  257.         }
  258.         SPI_WR_Reg(WRITE_REG+NRFRegSTATUS, status);   //接收到数据后RX_DR,TX_DS,MAX_RT都置高为1,通过写1来清楚中断标志
  259.         return flag;
  260. }
  261. void TX_Mode(void)
  262. {
  263.     CE = 0;
  264.         //SPI_WR_Reg(WRITE_REG + CONFIG, 0x02);   //0x0E                 // IRQ收发完成中断响应,16位CRC,主发送
  265.         
  266.         SPI_Write_Buf(WRITE_REG + TX_ADDR, TX_ADDRESS, TX_ADR_WIDTH);    // 写本地地址        
  267.         SPI_Write_Buf(WRITE_REG + RX_ADDR_P0, RX_ADDRESS, RX_ADR_WIDTH); // 写接收端地址
  268.         
  269.         SPI_WR_Reg(WRITE_REG + EN_AA, 0);//0x01);      //  频道0自动        ACK应答允许        
  270.         SPI_WR_Reg(WRITE_REG + EN_RXADDR,0);// 0x01);  //  允许接收地址只有频道0,如果需要多频道可以参考Page21  
  271.         SPI_WR_Reg(WRITE_REG + SETUP_RETR, 0);//0x1a); // 设置自动重发时间和次数:500us + 86us, 10 retrans...
  272.         SPI_WR_Reg(WRITE_REG + RF_CH, 40);        //   设置信道工作为2.4GHZ,收发必须一致
  273.         SPI_WR_Reg(WRITE_REG + RX_PW_P0, RX_PLOAD_WIDTH); //设置接收数据长度,本次设置为32字节
  274.         SPI_WR_Reg(WRITE_REG + RF_SETUP, 0x0F);                   //设置发射速率为2MHZ,发射功率为最大值0dB

  275.         SPI_WR_Reg(WRITE_REG + CONFIG, 0x0E);
  276.     CE = 1;
  277. }
  278. //***********************************************************************************************************
  279. //*函数:void nRF24L01_TxPacket(unsigned char * tx_buf)
  280. //*功能:发送 tx_buf中数据
  281. //**********************************************************************************************************/
  282. void nRF24L01_TxPacket(unsigned char * tx_buf)
  283. {
  284.         CE = 0;                        //StandBy I模式        
  285.         SPI_Write_Buf(WRITE_REG + RX_ADDR_P0, TX_ADDRESS, TX_ADR_WIDTH); // 装载接收端地址
  286.         SPI_Write_Buf(WR_TX_PLOAD, tx_buf, TX_PLOAD_WIDTH);                          // 装载数据        
  287. //        SPI_WR_Reg(WRITE_REG + CONFIG, 0x0e);                    // IRQ收发完成中断响应,16位CRC,主发送
  288.         CE = 1;                 //置高CE,激发数据发送
  289. }





  290. void delay_ms(uint z)     //延时函数
  291. {
  292.     uint x,y;
  293.     for(x=z;x>0;x--)
  294.         for(y=110;y>0;y--);
  295. }
  296. void delayms(unsigned int x)
  297. {
  298.     unsigned int i;
  299.     while(x--)
  300.         for(i=125;i>0;i--);
  301. }


  302. /////////////////************************主函数************************//////////////////

  303. void main()
  304. {
  305.      uchar Tx_Buf1[8];//发送的信息
  306.      uchar Rx_Buf[32];  //接收到的数据暂存器,最多32字节数据
  307.      init_NRF24L01();
  308.      LED=1;//初始灯6熄灭
  309.      tf2=1;                        //二号病床未呼叫标志位

  310.          label1:
  311.         SPI_WR_Reg(FLUSH_RX, 0x00);
  312.             SPI_WR_Reg(FLUSH_TX, 0x00);
  313.         RX_Mode();//接收模式
  314.         while(!nRF24L01_RxPacket(Rx_Buf))  //等待接收数据 ,返回1则接收到数据 ,在等待接收数据期间,可以随时变成发送模式
  315.         {
  316.                         tf5=0;                               //oled:平安无事
  317.         }        



  318.          if(Rx_Buf[2]==1)                  //若接收到2号病床呼叫
  319.          {  tf5=1;                         //取消oled显示
  320.             Rx_Buf[2]=0;                   //清空数据
  321.             SPI_WR_Reg(FLUSH_RX, 0x00);
  322.                 SPI_WR_Reg(FLUSH_TX, 0x00);
  323.             RX_Mode();//接收模式
  324.             while(!nRF24L01_RxPacket(Rx_Buf))  //等待接收数据 ,返回1则接收到数据 ,在等待接收数据期间,可以随时变成发送模式
  325.             {
  326.                                                       
  327.                                              
  328.                                             
  329.                            
  330.                            tf2=0;             //语音呼叫:二号病床呼叫 //oled:二号病床呼叫
  331.                                                
  332.                         
  333.                            if(KEY1==0)         
  334.                {
  335.                  while(!KEY1);
  336.                  {
  337.                    TX_Mode();                     //发送模式
  338.                    Tx_Buf1[2]=3;
  339.                                    nRF24L01_TxPacket(Tx_Buf1);    // 发送命令数据
  340.                    nRF24L01_TxPacket(Tx_Buf1);    // 发送命令数据
  341.                                    nRF24L01_TxPacket(Tx_Buf1);    // 发送命令数据
  342.                                    nRF24L01_TxPacket(Tx_Buf1);    // 发送命令数据                        
  343.                    SPI_WR_Reg(FLUSH_RX, 0x00);
  344.                        SPI_WR_Reg(FLUSH_TX, 0x00);
  345.                                  }                                //oled:已确认处理

  346.                                    tf2=1;                        //取消语音呼叫:二号病床呼叫 //取消oled:二号病床呼叫
  347.                                    tf6=0;                         //语音呼叫:已确认处理
  348.                                                            
  349.                                    delay_ms(300);  
  350.                                    
  351.                                    
  352.                                    tf6=1;
  353.                                    goto label1;
  354.                             }         

  355.                      }  

  356.                           
  357.                          if(Rx_Buf[2]==2)                  //如果二号病床取消呼叫
  358.              {  tf2=1;                    //取消语音呼叫:二号病床呼叫 //取消oled:二号病床呼叫
  359.                                 Rx_Buf[2]=0;
  360.                             SPI_WR_Reg(FLUSH_RX, 0x00);
  361.                     SPI_WR_Reg(FLUSH_TX, 0x00);
  362.                             tf4=0;                    
  363.                                 delay_ms(300);  
  364.                                 tf4=1;
  365.                                 goto label1;                             
  366.                         
  367.                            
  368.                          }
  369.                                  
  370.                  }

  371. }
复制代码
回复

使用道具 举报

ID:402383 发表于 2024-3-27 15:44 | 显示全部楼层
LZ现在程序实现的功能是1对1通讯,当前的病号只能跟目标地址为0xE7,0xE7,0xE7,0xE7,0x22,频道0通道的设备进行通讯,具有局限性。

然而实际情况上病号很多,建议继续开发主控台(办公室那边)1对多的功能,这样实用性更广。
回复

使用道具 举报

ID:807591 发表于 2024-3-27 17:12 | 显示全部楼层
这个场景不合适用这个芯片,2.4g干扰太多,挂水的人玩手机的很多,蓝牙很可能会干扰到,另外这个芯片价格略贵
回复

使用道具 举报

ID:433219 发表于 2024-3-27 20:29 | 显示全部楼层
选24L01的整体方案不太合适,,此处应用场景,应该1主多从,
回复

使用道具 举报

ID:277550 发表于 2024-3-29 10:19 | 显示全部楼层
玩一下可以。。。。。。之前有朋友实现过 100发、1接收。。。。。。
回复

使用道具 举报

ID:1114987 发表于 2024-4-11 13:10 | 显示全部楼层
wkman 发表于 2024-3-27 20:29
选24L01的整体方案不太合适,,此处应用场景,应该1主多从,

有什么推荐的模块吗
回复

使用道具 举报

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

本版积分规则

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

Powered by 单片机教程网

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