找回密碼
 立即注冊(cè)

QQ登錄

只需一步,快速開始

搜索
查看: 6331|回復(fù): 6
打印 上一主題 下一主題
收起左側(cè)

注釋祥細(xì)的NRF24L01程序 不區(qū)分收發(fā)

[復(fù)制鏈接]
跳轉(zhuǎn)到指定樓層
樓主
ID:114320 發(fā)表于 2016-5-8 02:49 | 只看該作者 回帖獎(jiǎng)勵(lì) |倒序?yàn)g覽 |閱讀模式
       網(wǎng)絡(luò)上收集的nRF24L01程序,注釋祥細(xì),不區(qū)分收發(fā),自己優(yōu)化了程序穩(wěn)定性,獻(xiàn)給需要的朋友,希望你們順利調(diào)通!
需要的自己下載: NRF24L01測試程序【不區(qū)分收發(fā) 頭文件版 C51】.rar (59.87 KB, 下載次數(shù): 74)



下面是部分NRF24L01程序預(yù)覽:

主程序:

  1. //=========================================================================
  2. //【注釋】:
  3. //        此工程內(nèi)的程序由STC12C5A60S2 11.0592MHz平臺(tái)測試成功
  4. //        使用前請(qǐng)根據(jù)實(shí)際情況更改“NRF24L01.H”和“SPI.H”內(nèi)的引腳配置,有如下6個(gè):CE,IRQ,MOSI,MISO,SCK,CSN
  5. //        按鍵與LED的引腳配置也根據(jù)實(shí)際情況更改
  6. //        發(fā)送與接收可共用該程序
  7. //        采用頭文件的方式編寫,使得程序更簡潔明了,利于分工合作,新手朋友可以學(xué)習(xí)這種編程方法
  8. //        編譯出現(xiàn)的警告,是有子函數(shù)未調(diào)用的警告,沒有關(guān)系的。
  9. //        【功能介紹】:A單片機(jī)的按鍵按下,B單片機(jī)LED燈亮,否則滅;B單片機(jī)的按鍵按下,A單片機(jī)LED燈亮,否則滅。
  10. //=========================================================================

  11. #include "mcu.h"
  12. #include "NRF24L01.H"

  13. #define                LED_ON                                P3|=(1<<6)                                //P36置一,LED亮,這種置一方法類似于STM32,推薦使用
  14. #define                LED_OFF                        P3&=~(1<<6)                        //P36置零,LED滅
  15. #define                 KEY_STAUS                (P2&(1<<0))                                //P20為按鍵 ==0為按下,!=0 為彈起

  16. //===============
  17. //延時(shí)函數(shù)
  18. //===============
  19. void delayms(uint ms)//延時(shí)?個(gè) ms
  20. {
  21.     unsigned char a,b;
  22.         while(ms--)
  23.         {
  24.             for(b=64;b>0;b--)           // 僅作為粗略延時(shí) 中斷繁忙時(shí)差距很大
  25.                 for(a=45;a>0;a--);
  26.         }
  27. }
  28. //======================
  29. //主函數(shù)
  30. //======================
  31. void main(void)
  32. {
  33.         uint while_times = 0;
  34.         init_NRF24L01();
  35.         delayms(300);
  36.        
  37.         while(1)
  38.         {       
  39.                 //===== 發(fā)送模式 =====
  40.                 nrf_TxMod();
  41.                 if(KEY_STAUS == 0)                //按鍵按下,
  42.                 {
  43.                         TxBuf[0] = 1;                //把1存入TxBuf[0]中,然后發(fā)送出去;接收程序判斷RxBuf[0]的值,等于1的話點(diǎn)亮LED
  44.                                                                                 //【注:RxBuf數(shù)組和TxBuf數(shù)組中的元素是對(duì)應(yīng)的】
  45.                 }
  46.                 else
  47.                 {
  48.                         TxBuf[0] = 0;               
  49.                 }
  50.                 nrf_trans(TxBuf);                        //將待發(fā)送的數(shù)據(jù)寫入NRF24L01
  51.                 while_times = 30;                //檢測是否發(fā)送成功 循環(huán)檢測?次  【可更改,讓接收循環(huán)次數(shù)大于發(fā)送循環(huán)次數(shù)效果較好】
  52.                 while(while_times--  ) //發(fā)送超時(shí),或者發(fā)送成功,跳出循環(huán) 進(jìn)入接收模式
  53.                 {
  54.                         get_nrf_sta();                        //獲取狀態(tài)標(biāo)志
  55.                         if(TX_DS == 1)                        //發(fā)送成功,跳出循環(huán)
  56.                                 break;
  57.                 }
  58.                 //===== 接收模式 =====
  59.                 nrf_RxMod();
  60.                 while_times = 120;                //檢測是否接收成功 循環(huán)檢測?次 【可更改,讓接收循環(huán)次數(shù)大于發(fā)送循環(huán)次數(shù)效果較好】
  61.                 while(while_times--)         //接收超時(shí)或者接收成功,跳出循環(huán) 進(jìn)入發(fā)送模式
  62.                 {
  63.                         get_nrf_sta();                        //獲取狀態(tài)標(biāo)志
  64.                         if(RX_DR == 1)                        //接收成功
  65.                         {       
  66.                                 nrf_read(RxBuf);        //接收成功后,將NRF24L01接收到的數(shù)據(jù)讀到單片機(jī)的RxBuf數(shù)組中。
  67.                                 break;                                                //跳出循環(huán)
  68.                         }
  69.                 }
  70.                 if(RX_DR == 1)                                //是因?yàn)榻邮盏綌?shù)據(jù),而不是因?yàn)槌瑫r(shí)才跳出循環(huán)
  71.                 {
  72.                         if(RxBuf[0] == 1)               
  73.                                 LED_ON;
  74.                         else if(RxBuf[0] == 0)
  75.                                 LED_OFF;
  76.                 }
  77.         }
  78. }
復(fù)制代碼

NRF24L01.c
  1. #include "NRF24L01.H"

  2. uchar idata nrf_sta;
  3. uchar idata RxBuf[32] =         "0";        //接收緩存 存入idata區(qū)
  4. uchar idata TxBuf[32] =         "0";        //發(fā)送緩存
  5.        
  6. uchar const TX_ADDRESS[TX_ADR_WIDTH]= {0x34,0x43,0x10,0x10,0x01};        //本地地址
  7. uchar const RX_ADDRESS[RX_ADR_WIDTH]= {0x34,0x43,0x10,0x10,0x01};        //接收地址

  8. //===== 粗略的延時(shí) =====
  9. void delayus(uint us)
  10. {
  11.         while(us--);
  12. }
  13. //================== NRF24L01初始化 ==================
  14. void init_NRF24L01(void)
  15. {
  16.     delayus(100);
  17.         CE = 0;    // 片選使能
  18.         CSN = 1;   // SPI使能
  19.         SCK = 0;   // SPI時(shí)鐘拉低
  20.         SPI_Write_Buf(WRITE_REG + TX_ADDR, TX_ADDRESS, TX_ADR_WIDTH);                    //寫發(fā)送地址       
  21.         SPI_Write_Buf(WRITE_REG + RX_ADDR_P0, RX_ADDRESS, RX_ADR_WIDTH);         //寫接收端地址
  22.         SPI_Write_Reg(WRITE_REG + EN_AA, 0x01);                                                     //通道0自動(dòng)應(yīng)答        
  23.         SPI_Write_Reg(WRITE_REG + EN_RXADDR, 0x01);                                 //允許接收地址頻道0
  24.         SPI_Write_Reg(WRITE_REG + RF_CH, 0x32);                                                     //設(shè)置信道工作頻率,收發(fā)必須一致
  25.         SPI_Write_Reg(WRITE_REG + RX_PW_P0, RX_PLOAD_WIDTH);                //設(shè)置接收數(shù)據(jù)長度
  26.         SPI_Write_Reg(WRITE_REG + RF_SETUP, 0x0f);                                           //設(shè)置發(fā)射速率為2MHZ,發(fā)射功率為最大值0dB       
  27.         SPI_Write_Reg(WRITE_REG + NRF_CONFIG, 0x7c);                                                 //IRQ引腳不顯示中斷 掉電模式  1~16CRC校驗(yàn)
  28. }
  29. //==================
  30. //讀取狀態(tài)標(biāo)志
  31. //==================
  32. void get_nrf_sta(void)
  33. {
  34.         nrf_sta = SPI_Read_Reg(STATUS);
  35.         SPI_Write_Reg(WRITE_REG+STATUS,nrf_sta);   
  36. }
  37. //==================
  38. //設(shè)置為接收模式
  39. //==================
  40. void nrf_RxMod(void)
  41. {
  42.         CE = 0;
  43.         SPI_Write_Reg(WRITE_REG+STATUS,0xff);        //清除中斷標(biāo)志
  44.         SPI_Write_Reg(FLUSH_RX,0x00);                         //清除RX_FIFO寄存器
  45.         SPI_Write_Reg(WRITE_REG + NRF_CONFIG, 0x7f);//IRQ引腳不顯示中斷 上電 接收模式   1~16CRC校驗(yàn)   
  46.         CE = 1;
  47.         delayus(100);
  48. }
  49. //==================
  50. //把接收到的數(shù)據(jù)存入數(shù)組
  51. //==================
  52. void nrf_read(uchar *rx_buf)
  53. {
  54.                 if(RX_DR == 1)                    //收到數(shù)據(jù)
  55.                 {
  56.                         CE = 0;  
  57.                         SPI_Read_Buf(RD_RX_PLOAD,rx_buf,RX_PLOAD_WIDTH);//讀取數(shù)據(jù) 存入數(shù)組
  58.                         SPI_Write_Reg(FLUSH_RX,0x00);//清除rx fifo寄存器
  59.                         CE = 1;
  60.                         delayus(100);
  61.                 }                 
  62. }
  63. //==================
  64. //設(shè)置為發(fā)送模式
  65. //==================
  66. void nrf_TxMod(void)
  67. {
  68.         CE = 0;
  69.         SPI_Write_Reg(WRITE_REG+STATUS,0xff);                         //清除中斷標(biāo)志
  70.         SPI_Write_Reg(FLUSH_TX,0x00);                                                                //清除TX_FIFO寄存器
  71.         SPI_Write_Reg(WRITE_REG + NRF_CONFIG,0x7e);                //IRQ引腳不顯示中斷 上電 發(fā)射模式  1~16CRC校驗(yàn)
  72.         CE = 1;
  73.         delayus(100);
  74. }
  75. //==================
  76. //發(fā)送  不做任何判斷只管發(fā)送
  77. //==================
  78. void nrf_trans(uchar *tx_buf)
  79. {       
  80.         CE = 0;                        //StandBy I模式
  81.         SPI_Write_Reg(WRITE_REG+STATUS,0xFF);                //清除所有中斷
  82.         SPI_Write_Reg(FLUSH_TX,0x00);                                                         //清除tx fifo寄存器        //===== 重要 =====
  83.         SPI_Write_Buf(WRITE_REG + RX_ADDR_P0, TX_ADDRESS, TX_ADR_WIDTH);                 // 裝載接收端地址
  84.         SPI_Write_Buf(WR_TX_PLOAD,tx_buf,TX_PLOAD_WIDTH);                                                                          // 裝載數(shù)據(jù)
  85.         CE = 1;                                         //置高CE激發(fā)數(shù)據(jù)發(fā)送
  86.         delayus(100);                //此延時(shí)必須有 因?yàn)閺拇龣C(jī)模式到收發(fā)模式需要時(shí)間,最大需要130us
  87. }
  88. //=========================
  89. //將float數(shù)編碼裝載 保留4位小數(shù)
  90. //占用5個(gè)字節(jié) 數(shù)據(jù)范圍+- 65535.9999
  91. //=========================
  92. void nrf_load_float(uchar a,float num)
  93. {
  94.         if(num > 0)
  95.         {
  96.                 TxBuf[a] = '+';       
  97.                 TxBuf[a+1] = (uint)num/256;
  98.                 TxBuf[a+2] = (uint)num%256;
  99.                 TxBuf[a+3] = (uint)((num - (int)num)*10000)/256;
  100.                 TxBuf[a+4] = (uint)((num - (int)num)*10000)%256;
  101.         }
  102.         else if(num < 0)
  103.         {
  104.                 num = -num;
  105.                 TxBuf[a] = '-';       
  106.                 TxBuf[a+1] = (uint)num/256;
  107.                 TxBuf[a+2] = (uint)num%256;
  108.                 TxBuf[a+3] = (uint)((num - (int)num)*10000)/256;
  109.                 TxBuf[a+4] = (uint)((num - (int)num)*10000)%256;
  110.         }
  111.         else
  112.         {
  113.                 TxBuf[a] = '0';
  114.                 TxBuf[a+1] = 0;
  115.                 TxBuf[a+2] = 0;
  116.                 TxBuf[a+3] = 0;
  117.                 TxBuf[a+4] = 0;
  118.         }
  119. }
  120. //=======================
  121. //將接收到的float數(shù)組解碼
  122. //占用5個(gè)字節(jié) 數(shù)據(jù)范圍+- 65535.9999
  123. //=======================
  124. float nrf_unload_float(uchar a)
  125. {
  126.         float num;
  127.         if(RxBuf[a] == '+'){
  128.                 num = RxBuf[a+1]*256 + RxBuf[a+2]+ (float)((int)RxBuf[a+3]*256 + RxBuf[a+4])/10000.0;
  129.         }
  130.         else if(RxBuf[a] == '-'){
  131.                 num = RxBuf[a+1]*256 + RxBuf[a+2]+ (float)((int)RxBuf[a+3]*256 + RxBuf[a+4])/10000.0;
  132.                 num = -num;
  133.         }
  134.         else if(RxBuf[a] == '0')
  135.                 num = 0;
  136.        
  137.         return (num);
  138. }
  139. //=======================
  140. //將float數(shù)編碼裝載 保留2位小數(shù)
  141. //占用3個(gè)字節(jié) 數(shù)據(jù)范圍+- 255.99
  142. //=======================
  143. void nrf_load_sfloat(uchar a,float num)
  144. {
  145.         if(num > 0){
  146.                 TxBuf[a] = '+';       
  147.                 TxBuf[a+1] = (uchar)num;                //轉(zhuǎn)換成uchar類型,自動(dòng)將保留低8位,去除高位。
  148.                 TxBuf[a+2] = (uint)((num - (int)num)*100);
  149.         }
  150.         else if(num < 0){
  151.                 num = -num;
  152.                 TxBuf[a] = '-';       
  153.                 TxBuf[a+1] = (uchar)num;
  154.                 TxBuf[a+2] = (uint)((num - (int)num)*100);
  155.         }
  156.         else{
  157.                 TxBuf[a] = '0';
  158.                 TxBuf[a+1] = 0;
  159.                 TxBuf[a+2] = 0;
  160.         }
  161. }
  162. //=======================
  163. //將float數(shù)解碼 保留2位小數(shù)
  164. //占用3個(gè)字節(jié) 數(shù)據(jù)范圍+- 255.99
  165. //======================
  166. float nrf_unload_sfloat(uchar a)        //a是數(shù)據(jù)包在數(shù)組內(nèi)的起始位置
  167. {
  168.         float num;
  169.         if(RxBuf[a] == '+'){
  170.                 num = RxBuf[a+1]+ (float)RxBuf[a+2]/100;
  171.         }
  172.         else if(RxBuf[a] == '-'){
  173.                 num = RxBuf[a+1]+ (float)RxBuf[a+2]/100;
  174.                 num = -num;
  175.         }
  176.         else if(RxBuf[a] == '0')
  177.                 num = 0;
  178.        
  179.         return (num);
  180. }
復(fù)制代碼




評(píng)分

參與人數(shù) 2黑幣 +17 收起 理由
初秋夜微涼 + 5 很給力!
YJGG + 12 共享資料的黑幣獎(jiǎng)勵(lì)!

查看全部評(píng)分

分享到:  QQ好友和群QQ好友和群 QQ空間QQ空間 騰訊微博騰訊微博 騰訊朋友騰訊朋友
收藏收藏1 分享淘帖 頂 踩

相關(guān)帖子

回復(fù)

使用道具 舉報(bào)

沙發(fā)
ID:23606 發(fā)表于 2016-5-26 10:50 | 只看該作者
謝謝分享
回復(fù)

使用道具 舉報(bào)

板凳
ID:79544 發(fā)表于 2016-5-28 11:05 | 只看該作者
樓主的程序是收發(fā)一體的。我最近也在折騰NRF24L01傳輸AD的程序,發(fā)送和接收都能工作,就是收不到數(shù)據(jù),不知道樓主有這方面的程序嗎?
回復(fù)

使用道具 舉報(bào)

地板
ID:123712 發(fā)表于 2017-1-13 15:37 | 只看該作者
很好的資料,正在學(xué)習(xí),感謝樓主分享
回復(fù)

使用道具 舉報(bào)

5#
ID:48413 發(fā)表于 2017-2-21 14:52 | 只看該作者
學(xué)習(xí)一下謝謝分享
回復(fù)

使用道具 舉報(bào)

6#
ID:245836 發(fā)表于 2017-11-7 15:50 | 只看該作者
感謝分享
回復(fù)

使用道具 舉報(bào)

7#
ID:929517 發(fā)表于 2023-10-25 07:09 | 只看該作者
怎么資料沒了啊 我想找的就是這個(gè)
回復(fù)

使用道具 舉報(bào)

本版積分規(guī)則

手機(jī)版|小黑屋|51黑電子論壇 |51黑電子論壇6群 QQ 管理員QQ:125739409;技術(shù)交流QQ群281945664

Powered by 單片機(jī)教程網(wǎng)

快速回復(fù) 返回頂部 返回列表