找回密码
 立即注册

QQ登录

只需一步,快速开始

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

15单片机进行通过NRF24L01的双向通信有问题

[复制链接]
ID:922838 发表于 2021-10-23 15:30 | 显示全部楼层 |阅读模式
这个是我写的程序,大部分是网上找的程序,用的是15单片机,然后两个MCU通过NRF24L01通信,但不知道为什么就是通信不了,就接收不到数据。有没有大佬找找我的问题。

单片机源程序如下:
  1. #include <reg51.h>
  2. #include <intrins.h>
  3. #define TX_ADR_WIDTH 5 //5字节宽度的发送/接收地址
  4. #define TX_PLOAD_WIDTH 4//数据通道有效数据宽度
  5. #define uchar        unsigned char
  6. #define uint        unsigned int
  7. #define out P0
  8. uchar code lcd1602[]={0x30,0x31,0x32,0x33,0x34,0x35,0x36,0x37,0x38,0x39};

  9. uchar code TX_ADDRESS[TX_ADR_WIDTH] = {0x34,0x43,0x10,0x10,0x01};  // 定义一个静态发送地址
  10. uchar RX_BUF[TX_PLOAD_WIDTH];
  11. uchar TX_BUF[TX_PLOAD_WIDTH];
  12. uchar flag;
  13. uchar bdata sta;
  14. sbit RX_DR =sta^6;
  15. sbit TX_DS =sta^5;
  16. sbit MAX_RT=sta^4;

  17. /*sbit CE   = P1^5;
  18. sbit CSN  = P1^4;
  19. sbit SCK  = P1^2;
  20. sbit MOSI = P1^1;
  21. sbit MISO = P1^3;
  22. sbit IRQ  = P1^0;*/
  23. sbit CE   = P2^4;
  24. sbit CSN  = P2^0;
  25. sbit SCK  = P2^3;
  26. sbit MOSI = P2^1;
  27. sbit MISO = P3^3;
  28. sbit IRQ  = P2^2;
  29. //SPI要使用的IO
  30. sbit MOSIO=P3^4;
  31. sbit R_CLK=P3^5;
  32. sbit S_CLK=P3^6;
  33. //SPI(NFR24L01)命令
  34. #define RFAD_REG 0x00      //定义读命令以注册
  35. #define WRITE_REG 0x20     //定义要写的命令
  36. #define RD_RX_PLOAD 0x61   //定义RX有效负载寄存器地址
  37. #define WR_TX_PLOAD 0xA0   //定义TX有效负载寄存器地址
  38. #define FLUSH_TX    0xE1   //定义刷新 Tx寄存器命令
  39. #define FLUSH_RX    0xE2   //定义刷新 Rx寄存器命令
  40. #define REUSE_TX_PL 0xE3   //定义重复TX负载寄存器命令
  41. #define NOP                                        0xFF   //定义无操作

  42. //SPI(NRF24L01)寄存器
  43. #define CONFIG     0x00    //配置寄存器地址
  44. #define EN_AA      0x01    //“启用自动确认”的注册地址
  45. #define EN_RXADDR  0x02    //'Enabled RX addresses'寄存器地址
  46. #define SETUP_AW   0x03    //“设置地址宽度”寄存器地址
  47. #define SETUP_RETR 0x04    //“自动设置。Retrans的寄存器地址
  48. #define RF_CH                         0x05                 //        '射频信道'寄存器地址
  49. #define RF_SETUP         0x06          //射频设置寄存器地址
  50. #define STATUS                  0x07                 //        状态寄存器地址
  51. #define OBSERVE_TX 0x08                 //“收到Tx”寄存器地址
  52. #define CD                                 0x09    //"载波检测'寄存器地址
  53. #define RX_ADDR_P0 0x0A    //RX地址0管道的寄存器地址
  54. #define RX_ADDR_P1 0x0B    //RX地址1管道的寄存器地址
  55. #define RX_ADDR_P2 0x0C    //RX地址2管道的寄存器地址
  56. #define RX_ADDR_P3 0x0D    //RX地址3管道的寄存器地址
  57. #define RX_ADDR_P4 0x0E    //RX地址4管道的寄存器地址
  58. #define RX_ADDR_P5 0x0F    //RX地址5管道的寄存器地址
  59. #define TX_ADDR    0x10                 //“Tx address”的寄存器地址
  60. #define RX_PW_P0         0x11                 //"RX有效载荷宽度,0管'寄存器地址
  61. #define RX_PE_P1         0x12                 //"RX有效载荷宽度,1管'寄存器地址
  62. #define RX_PE_P2   0x13                 //"RX有效载荷宽度,2管'寄存器地址
  63. #define RX_PE_P3   0x14                 //"RX有效载荷宽度,3管'寄存器地址
  64. #define RX_PE_P4   0x15                 //"RX有效载荷宽度,4管'寄存器地址
  65. #define RX_PE_P5   0x16                 //"RX有效载荷宽度,5管'寄存器地址
  66. #define FIFO_STATUS 0x17   //“FIFO状态寄存器”的寄存器地址


  67. void delay500ms(void);
  68. void lcd_initial (void);                                //LCD1602初始化
  69. void  write_data(uchar dat);                //写数据函数
  70. void check_busy (void);                                        //忙检测
  71. void write_command (uchar com); //写命令函数
  72. void string (uchar ad,uchar *s);//字符写入,(字符开始地址,字符)
  73. void delay100ms(void);                                        //延时100MS
  74. void delay10ms(void);                                          //延时10MS
  75. void delay1ms(void);                                                //延时1ms
  76. uchar ajlb(uchar x);                        //按键滤波
  77. uchar JCAJ(void);                                                //检测按键按下
  78. uchar HC08JS = 0;
  79. bit                flag1 = 0;
  80. sbit        E=P2^5;
  81. sbit        RW=P2^6;
  82. sbit        RS=P2^7;
  83. sbit        FMQ=P1^2;
  84. sbit        BG=P2^4;
  85. sbit  DQ=P1^0;
  86. sbit K1=P1^3;
  87. sbit K2=P1^4;
  88. sbit K3=P1^5;
  89. sbit K4=P1^6;
  90. sfr AUXR   = 0x8E;
  91. sfr T2H = 0xd6;
  92. sfr T2L = 0xd7;


  93. void lcd_initial (void)   //LCD1602初始化
  94. {
  95.         write_command(0x38);//写入0x38(命令6):两行显示,5X7点阵,8位数据接口
  96.         _nop_();                                                //空操作
  97.         write_command(0x0C);//写入0x38(命令4):开整体显示,光标关,无闪烁
  98.         _nop_();                                                //空操作
  99.         write_command(0x04);//写入0x38(命令3):
  100.         _nop_();                                                //空操作
  101.         write_command(0x01);//写入0x38(命令1):清屏
  102.         delay1ms();
  103. }
  104. void check_busy (void)//忙检测
  105. {
  106.         uchar dt;
  107.         do
  108.         {
  109.                 dt=0xff;                                        //dt为变量单元,初值为0xff
  110.                 E=0;
  111.                 RS=0;
  112.                 RW=1;
  113.                 E=1;
  114.                 dt=out;                                                //out为LCD数据口,将out的状态送入dt
  115.         }
  116.         while(dt&0x80);                //如果忙标志位BF=1,继续循环检测,等待BF=0
  117.         E=0;                                                                //BF=0,LCD不忙,结束检测
  118. }
  119. void write_command (uchar com)//写命令函数
  120. {
  121.         check_busy();
  122.         E=0;
  123.         RS=0;                                                                                                //RS与E同时为零时。才可以写入命令
  124.         RW=0;
  125.         out=com;                                                                                //将命令com写入数据口
  126.         E=1;                                                                                                //写入命令时,E为上升沿,所以E开始为0
  127.         _nop_();                                       
  128.         E=0;                                                                                                //E由高变低,LCD开始执行命令
  129.         delay1ms();
  130. }
  131. void  write_data(uchar dat)                //写数据函数
  132. {
  133.         check_busy();
  134.         E=0;
  135.         RS=1;
  136.         RW=0;
  137.         out=dat;
  138.         E=1;
  139.         _nop_();
  140.         E=0;
  141.         delay1ms();
  142. }
  143. void string (uchar ad,uchar *s)
  144. {
  145.         write_command(ad);
  146.         while(*s>0)
  147.         {
  148.                 write_data(*s++);                                        //输出字符串,且指针加一
  149.                 delay1ms();
  150.         }
  151. }
  152. void string1(uchar ad,uchar a)
  153. {uchar b=0,c=0;
  154.         write_command(ad);
  155.         ad=ad+0x01;
  156.         b=a/10;
  157.         b=lcd1602[b];
  158.         write_data(b);
  159.         c=a%10;
  160.         c=lcd1602[c];
  161.         write_data(c);
  162.         
  163. }
  164. void lcd_xs(uchar a)
  165. {
  166.         if(a==0)
  167.         {string(0x83,"->NRF24L01");
  168.          string(0xC5,"HC-08");
  169.         }
  170.                 else if(a==1)
  171.         {string(0x83,"->HC-08");
  172.          string(0xC5,"EV1527");
  173.         }
  174.                 else if(a==2)
  175.         {string(0x83,"->EV1527");
  176.          string(0xC5,"NRF24L01");
  177.         }
  178. }
  179. void delay5us(void)   //误差 0us
  180. {
  181.     unsigned char a;
  182.     for(a=12;a>0;a--);
  183. }

  184. void delay5(uchar n)                        //延时5us程序
  185. {
  186.         do
  187.         {
  188.                 delay5us();
  189.                 n--;
  190.         }
  191.         while(n);
  192. }

  193. void delay1ms(void)   //误差 0us
  194. {
  195.     unsigned char a,b;
  196.     for(b=222;b>0;b--)
  197.         for(a=12;a>0;a--);
  198. }

  199. void delay10ms(void)   //误差 -0.000000000001us
  200. {
  201.     unsigned char a,b;
  202.     for(b=202;b>0;b--)
  203.         for(a=147;a>0;a--);
  204. }
  205. void delay50ms(void)   //误差 -0.000000000003us
  206. {
  207.     unsigned char a,b,c;
  208.     for(c=6;c>0;c--)
  209.         for(b=116;b>0;b--)
  210.             for(a=214;a>0;a--);
  211. }

  212. void delay100ms(void)   //误差 -0.000000000006us
  213. {
  214.     unsigned char a,b,c;
  215.     for(c=246;c>0;c--)
  216.         for(b=116;b>0;b--)
  217.             for(a=9;a>0;a--);
  218. }

  219. //////////////////////////////////////////////////////DS18B20程序开头
  220. void init_ds18b20(void)                        //18B20初始化
  221. {
  222.         uchar x=0;
  223.         DQ=0;
  224.         delay5(120);
  225.         DQ=1;
  226.         delay5(16);
  227.         delay5(80);
  228. }
  229. uchar readbyte(void)                //读取1字节数据
  230. {
  231.         uchar i=0;
  232.         uchar date=0;
  233.         for(i=8;i>0;i--)
  234.         {
  235.                 DQ=0;
  236.                 delay5(1);
  237.                 DQ=1;
  238.                 date>>=1;
  239.                 if(DQ)
  240.                 date|=0x80;
  241.                 delay5(11);               
  242.         }
  243.         return(date);
  244. }
  245. void writebyte(uchar dat)                //写1B函数
  246. {
  247.         uchar i=0;
  248.         for(i=8;i>0;i--)
  249.         {
  250.                 DQ=0;
  251.                 DQ=dat&0x01;
  252.                 delay5(12);
  253.                 DQ=1;
  254.                 dat>>=1;
  255.                 delay5(5);
  256.         }
  257. }
  258. uchar retemp(void)                        //读取温度
  259. {
  260.         uchar a,b,tt;
  261.         uint t;
  262.         init_ds18b20();
  263.         writebyte(0xcc);
  264.         writebyte(0x44);
  265.         init_ds18b20();
  266.         writebyte(0xcc);
  267.         writebyte(0xBE);
  268.         a=readbyte();
  269.         b=readbyte();
  270.         t=b;
  271.         t<<=8;
  272.         t=t|a;
  273.         tt=t*0.0625;
  274.         
  275.         return(tt);
  276. }
  277. ////////////////////////////////////////////DS18B20程序结尾
  278. //NFR24L01发送
  279. void init_io (void)                                                //2401初始化
  280. {
  281.         CE =0;                                                                                        //待机
  282.         CSN=1;                                                                                        //SPI禁止
  283.         SCK=0;                                                                                        //SPI时钟置低
  284.         IRQ=1;                                                                                        //中断复位
  285. }
  286. uchar SPI_RW(uchar byte)                        //根据spi协议,写一节数据到2401,同时读出一节数据
  287. {
  288.         uchar i;
  289.         for(i=0;i<8;i++)                                                        //循环8次
  290.         {
  291.                 MOSI=(byte&0x80);                                        //byte最高位输出到MOSI
  292.                 byte<<=1;                                                                        //低一位移到最高位
  293.                 SCK=1;
  294.                 byte |=MISO;                                                        //读MSIO到byte最低位
  295.                 SCK=0;
  296.         }
  297.         return(byte);                                                                //发回读出的一字节
  298. }
  299. uchar SPI_RW_Reg(uchar reg, uchar value)                //写数据value到reg寄存器
  300. {
  301.         uchar status;
  302.           CSN = 0;                   // CSN置低,开始传输数据
  303.           status = SPI_RW(reg);      // 选择寄存器,同时返回状态字
  304.           SPI_RW(value);             // 然后写数据到该寄存器
  305.           CSN = 1;                   // CSN拉高,结束数据传输
  306.           return(status);            // 返回状态寄存器
  307. }
  308. uchar SPI_Read(uchar reg)                        //从reg寄存器读一字节
  309. {
  310.         uchar reg_val;
  311.         CSN=0;                                                                                        //CSN置低,开始传输数据
  312.         SPI_RW(reg);
  313.         reg_val=SPI_RW(0);
  314.         CSN=1;
  315.         return(reg_val);
  316. }
  317. uchar SPI_Read_Buf(uchar reg, uchar *pBuf, uchar bytes)        //从reg寄存器读出bytes个字节,通常用来读取接收通道数据或接收/发送地址
  318. {
  319.         uchar status,i;
  320.         CSN=0;
  321.         status=SPI_RW(reg);
  322.         for(i=0;i<bytes;i++)
  323.         pBuf[i]=SPI_RW(0);
  324.         CSN=1;
  325.         return(status);
  326. }
  327. uchar SPI_Write_Buf(uchar reg, uchar *pBuf, uchar bytes)                //把pBuf缓存中的数据写入到nRF24L01,通常用来写入发射通道数据或接收/发送地址
  328. {
  329.         uchar status, i;
  330.           CSN = 0;                    // CSN置低,开始传输数据
  331.           status = SPI_RW(reg);       // 选择寄存器,同时返回状态字
  332.           for(i=0; i<bytes; i++)
  333.             SPI_RW(pBuf[i]);        // 逐个字节写入nRF24L01
  334.           CSN = 1;                    // CSN拉高,结束数据传输
  335.           return(status);             // 返回状态寄存器
  336. }

  337. void RX_Mode(void)                        //这个函数设置nRF24L01为接收模式,等待接收发送设备的数据包
  338. {
  339.         CE = 0;
  340.           SPI_Write_Buf(WRITE_REG + RX_ADDR_P0, TX_ADDRESS, TX_ADR_WIDTH);  // 接收设备接收通道0使用和发送设备相同的发送地址
  341.           SPI_RW_Reg(WRITE_REG + EN_AA, 0x01);               // 使能接收通道0自动应答
  342.           SPI_RW_Reg(WRITE_REG + EN_RXADDR, 0x01);           // 使能接收通道0
  343.           SPI_RW_Reg(WRITE_REG + RF_CH, 40);                 // 选择射频通道0x40
  344.           SPI_RW_Reg(WRITE_REG + RX_PW_P0, TX_PLOAD_WIDTH);  // 接收通道0选择和发送通道相同有效数据宽度
  345.           SPI_RW_Reg(WRITE_REG + RF_SETUP, 0x07);            // 数据传输率1Mbps,发射功率0dBm,低噪声放大器增益
  346.           SPI_RW_Reg(WRITE_REG + CONFIG, 0x0f);              // CRC使能,16位CRC校验,上电,接收模式
  347.           CE = 1;                                            // 拉高CE启动接收设备
  348. }
  349. void TX_Mode(uchar * BUF)
  350. {
  351.         CE = 0;
  352.           SPI_Write_Buf(WRITE_REG + TX_ADDR, TX_ADDRESS, TX_ADR_WIDTH);     // 写入发送地址
  353.           SPI_Write_Buf(WRITE_REG + RX_ADDR_P0, TX_ADDRESS, TX_ADR_WIDTH);  // 为了应答接收设备,接收通道0地址和发送地址相同
  354.           SPI_Write_Buf(WR_TX_PLOAD, BUF, TX_PLOAD_WIDTH);                  // 写数据包到TX FIFO
  355.           SPI_RW_Reg(WRITE_REG + EN_AA, 0x01);       // 使能接收通道0自动应答
  356.           SPI_RW_Reg(WRITE_REG + EN_RXADDR, 0x01);   // 使能接收通道0
  357.           SPI_RW_Reg(WRITE_REG + SETUP_RETR, 0x0a);  // 自动重发延时等待250us+86us,自动重发10次
  358.           SPI_RW_Reg(WRITE_REG + RF_CH, 40);         // 选择射频通道0x40
  359.           SPI_RW_Reg(WRITE_REG + RF_SETUP, 0x07);    // 数据传输率1Mbps,发射功率0dBm,低噪声放大器增益
  360.           SPI_RW_Reg(WRITE_REG + CONFIG, 0x0e);      // CRC使能,16位CRC校验,上电
  361.         CE = 1;
  362. }
  363. uchar Check_ACK(bit clear)                                //检查接收设备有无接收到数据包,设定没有收到应答信号是否重发
  364. {
  365.         while(IRQ);
  366.         sta = SPI_RW(NOP);                    // 返回状态寄存器
  367.         if(MAX_RT)
  368.                 if(clear)                         // 是否清除TX FIFO,没有清除在复位MAX_RT中断标志后重发
  369.                         SPI_RW(FLUSH_TX);
  370.         SPI_RW_Reg(WRITE_REG + STATUS, sta);  // 清除TX_DS或MAX_RT中断标志
  371.         IRQ = 1;
  372.         if(TX_DS)
  373.                 return(0x00);
  374.         else
  375.                 return(0xff);
  376. }
  377. void NRF24L01_CX(void)                //NRF24L01程序执行
  378. {uchar b=1,temp,a=0,temp1=0x00;
  379.         FMQ=1;
  380.         lcd_initial();                                        //调用lcd初始化
  381.         string(0x80,"24L01TXD:");
  382.         string(0xc0,"RXD:");
  383.         init_io();
  384.         while(b)
  385.         {               
  386.                 temp=retemp();
  387.                 string1(0x8A,temp);        
  388.                 TX_BUF[0] = ~temp;          // 数据送到缓存
  389.                 TX_Mode(TX_BUF);                        // 把nRF24L01设置为发送模式并发送数据
  390.                 //IRQ=0;
  391.                 a=JCAJ();
  392.                 if(a==4)
  393.                         b=0;
  394.                 Check_ACK(1);               // 等待发送完毕,清除TX FIFO
  395.                 RX_Mode();                                // 设置为接收模式
  396.                 sta = SPI_Read(STATUS);          // 读状态寄存器
  397.                  if(RX_DR)                                  // 判断是否接受到数据
  398.                 {
  399.                         SPI_Read_Buf(RD_RX_PLOAD, RX_BUF, TX_PLOAD_WIDTH);  // 从RX FIFO读出数据
  400.                         flag = 1;
  401.                 }
  402.                 SPI_RW_Reg(WRITE_REG + STATUS, sta);  // 清除RX_DS中断标志
  403.                 if(flag)                           // 接受完成
  404.                         {
  405.                         flag = 0;                       // 清标志
  406.                         temp1 = RX_BUF[0];           // 数据送到temp1
  407.                 }
  408.                         string1(0xCA,temp1);
  409.                 delay50ms();
  410.                 a=JCAJ();
  411.                 if(a==4)
  412.                         b=0;
  413.         }
  414.         
  415. }uchar JCAJ(void)                                                //检测按键按下
  416. {
  417.         uchar x=0,y=0;
  418.         if(K1==0)
  419.         {y=1;
  420.         x=ajlb(y);
  421.                 return x;
  422.         }
  423.         else if(K2==0)
  424.         {y=2;
  425.         x=ajlb(y);
  426.                 return x;
  427.         }
  428.         else if(K3==0)
  429.         {y=3;
  430.         x=ajlb(y);
  431.                 return x;
  432.         }
  433.         else if(K4==0)
  434.         {y=4;
  435.         x=ajlb(y);
  436.                 return x;
  437.         }
  438.         return x;
  439. }
  440. uchar ajlb(uchar x)                        //按键滤波
  441. {
  442.         delay10ms();
  443.         if(x==1)
  444.         {
  445.                 if(K1==0)
  446.                 {return x;}
  447.                 else
  448.                 {x=0;
  449.                  return x;}
  450.         }
  451.         else if(x==2)
  452.         {
  453.                 if(K2==0)
  454.                 {return x;}
  455.                 else
  456.                 {x=0;
  457.                  return x;}
  458.         }
  459.         else if(x==3)
  460.         {
  461.                 if(K3==0)
  462.                 {return x;}
  463.                 else
  464.                 {x=0;
  465.                  return x;}
  466.         }
  467.         else if(x==4)
  468.         {
  469.                 if(K4==0)
  470.                 {return x;}
  471.                 else
  472.                 {x=0;
  473.                  return x;}
  474.         }
  475.                 return x;
  476. }

  477. void main ()
  478. {
  479.                                                                         
  480. NRF24L01_CX();
  481. }
复制代码
回复

使用道具 举报

ID:973695 发表于 2021-10-24 22:21 | 显示全部楼层
粗略看了一下,程序有几个问题:1、两个单片机的程序收发地址应该设置不同;2、工作时得有握手协议来协调收发状态的切换。
建议你先单独写收、发两个程序,调试通过了再合并一起。

评分

参与人数 1黑币 +20 收起 理由
admin + 20

查看全部评分

回复

使用道具 举报

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

本版积分规则

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

Powered by 单片机教程网

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