久久久久久久999_99精品久久精品一区二区爱城_成人欧美一区二区三区在线播放_国产精品日本一区二区不卡视频_国产午夜视频_欧美精品在线观看免费

 找回密碼
 立即注冊

QQ登錄

只需一步,快速開始

搜索
查看: 3703|回復: 1
打印 上一主題 下一主題
收起左側

基于單片機的NRF24L01無線接收裝置設計(PCB+原理圖+代碼)

[復制鏈接]
跳轉到指定樓層
樓主
親測可用哦,需要可下,由Proteus仿真等文件
Altium Designer畫的原理圖和PCB圖如下:(51hei附件中可下載工程文件)


單片機源程序如下:
  1. #include <reg52.h>
  2. #include <intrins.h>

  3. typedef unsigned char uchar;
  4. typedef unsigned char uint;

  5. //****************************************NRF24L01端口定義***************************************
  6. sbit CE  = P1^0;
  7. sbit CSN = P1^5;
  8. sbit SCK = P1^1;
  9. sbit MOSI = P1^4;
  10. sbit MISO = P1^2;
  11. sbit IRQ = P1^3;
  12. sbit LED = P3^2;
  13. sbit re = P2^7;
  14. //*****************************************DS1820端口設置****************************************
  15. sbit      DQ=P1^6;                                                                                                                //顯示緩沖區
  16. //******************************************************************************************
  17. uint         bdata sta;                                   //NRF24L01狀態標志
  18. sbit        RX_DR        =sta^6;                               
  19. sbit        TX_DS        =sta^5;
  20. sbit        MAX_RT        =sta^4;
  21. //*************************************NRF24L01**************************************************

  22. #define TX_ADR_WIDTH    5           // 本機地址寬度設置
  23. #define RX_ADR_WIDTH    5           // 接收方地址寬度設置

  24. #define TX_PLOAD_WIDTH  20                 // 4 字節數據長度
  25. #define RX_PLOAD_WIDTH  20                // 4 字節數據長度

  26. uint const TX_ADDRESS[TX_ADR_WIDTH]= {0x34,0x43,0x10,0x10,0x01};        //本地地址
  27. uint const RX_ADDRESS[RX_ADR_WIDTH]= {0x34,0x43,0x10,0x10,0x01};        //接收地址
  28. uchar RX_BUF[RX_PLOAD_WIDTH];
  29. uchar TX_BUF[TX_PLOAD_WIDTH];
  30. //*****************************NRF24L01寄存器指令,詳細請對照,Page18******************************

  31. #define READ_REG        0x00          // 讀寄存器指令
  32. #define WRITE_REG       0x20         // 寫寄存器指令
  33. #define RD_RX_PLOAD     0x61          // 讀取接收數據指令
  34. #define WR_TX_PLOAD     0xA0          // 寫待發數據指令
  35. #define FLUSH_TX        0xE1         // 沖洗發送 FIFO指令
  36. #define FLUSH_RX        0xE2          // 沖洗接收 FIFO指令
  37. #define REUSE_TX_PL     0xE3          // 定義重復裝載數據指令
  38. #define NOP             0xFF          // 保留
  39. //****************************SPI(nRF24L01)寄存器地址,詳細請對照,Page18-24**********************

  40. #define CONFIG          0x00   // 配置收發狀態,CRC校驗模式以及收發狀態響應方式
  41. #define EN_AA           0x01   // 自動應答功能設置
  42. #define EN_RXADDR       0x02   // 可用信道設置
  43. #define SETUP_AW        0x03   // 收發地址寬度設置
  44. #define SETUP_RETR      0x04   // 自動重發功能設置
  45. #define RF_CH           0x05   // 工作頻率設置
  46. #define RF_SETUP        0x06   // 發射速率、功耗功能設置
  47. #define STATUS          0x07   // 狀態寄存器
  48. #define OBSERVE_TX      0x08   // 發送監測功能
  49. #define CD              0x09   // 地址檢測           
  50. #define RX_ADDR_P0      0x0A   // 頻道0接收數據地址
  51. #define RX_ADDR_P1      0x0B   // 頻道1接收數據地址
  52. #define RX_ADDR_P2      0x0C   // 頻道2接收數據地址
  53. #define RX_ADDR_P3      0x0D   // 頻道3接收數據地址
  54. #define RX_ADDR_P4      0x0E   // 頻道4接收數據地址
  55. #define RX_ADDR_P5      0x0F   // 頻道5接收數據地址
  56. #define TX_ADDR         0x10   // 發送地址寄存器
  57. #define RX_PW_P0        0x11   // 接收頻道0接收數據長度
  58. #define RX_PW_P1        0x12   // 接收頻道0接收數據長度
  59. #define RX_PW_P2        0x13   // 接收頻道0接收數據長度
  60. #define RX_PW_P3        0x14   // 接收頻道0接收數據長度
  61. #define RX_PW_P4        0x15   // 接收頻道0接收數據長度
  62. #define RX_PW_P5        0x16   // 接收頻道0接收數據長度
  63. #define FIFO_STATUS     0x17   // FIFO棧入棧出狀態寄存器設置
  64. //************************************NRF24L01函數申明**********************************************

  65. void Delay(unsigned int s);
  66. void inerDelay_us(unsigned char n);
  67. void init_NRF24L01(void);
  68. uint SPI_RW(uint uchar);
  69. uchar SPI_Read(uchar reg);
  70. //void SetRX_Mode(void);
  71. uint SPI_RW_Reg(uchar reg, uchar value);
  72. uint SPI_Read_Buf(uchar reg, uchar *pBuf, uchar uchars);
  73. uint SPI_Write_Buf(uchar reg, uchar *pBuf, uchar uchars);
  74. //unsigned char nRF24L01_RxPacket(unsigned char* rx_buf);
  75. void nRF24L01_TxPacket(unsigned char * tx_buf);

  76. //*****************************************長延時*****************************************
  77. void Delay(unsigned int s)
  78. {
  79.         unsigned int i;
  80.         for(i=0; i<s; i++);
  81.         for(i=0; i<s; i++);
  82. }

  83. /******************************************************************************************
  84. /*延時函數
  85. /******************************************************************************************/
  86. void inerDelay_us(unsigned char n)
  87. {
  88.         for(;n>0;n--)
  89.                 _nop_();
  90. }
  91. //****************************************************************************************
  92. /*NRF24L01初始化
  93. //***************************************************************************************/
  94. void init_NRF24L01(void)
  95. {
  96.     inerDelay_us(100);
  97.         CE=0;    // chip enable
  98.         CSN=1;   // Spi disable
  99.         SCK=0;   // Spi clock line init high
  100.         SPI_Write_Buf(WRITE_REG + TX_ADDR, TX_ADDRESS, TX_ADR_WIDTH);    // 寫本地地址       
  101.         SPI_Write_Buf(WRITE_REG + RX_ADDR_P0, RX_ADDRESS, RX_ADR_WIDTH); // 寫接收端地址
  102.         SPI_RW_Reg(WRITE_REG + EN_AA, 0x01);      //  頻道0自動        ACK應答允許       
  103.         SPI_RW_Reg(WRITE_REG + EN_RXADDR, 0x01);  //  允許接收地址只有頻道0,如果需要多頻道可以參考Page21  
  104.         SPI_RW_Reg(WRITE_REG + RF_CH, 0);        //   設置信道工作為2.4GHZ,收發必須一致
  105.         SPI_RW_Reg(WRITE_REG + RX_PW_P0, RX_PLOAD_WIDTH); //設置接收數據長度,本次設置為4字節
  106.         SPI_RW_Reg(WRITE_REG + RF_SETUP, 0x07);                   //設置發射速率為1Mkbps,發射功率為最大值0dB
  107. }
  108. /****************************************************************************************************
  109. /*函數:uint SPI_RW(uint uchar)
  110. /*功能:NRF24L01的SPI寫時序,詳細看時序圖,Page19
  111. /****************************************************************************************************/
  112. uint SPI_RW(uint uchar)
  113. {
  114.         uint bit_ctr;
  115.            for(bit_ctr=0;bit_ctr<8;bit_ctr++) // output 8-bit
  116.            {
  117.                 MOSI = (uchar & 0x80);         // output 'uchar', MSB to MOSI
  118.                 uchar = (uchar << 1);           // shift next bit into MSB..
  119.                 SCK = 1;                      // Set SCK high..
  120.                 uchar |= MISO;                         // capture current MISO bit
  121.                 SCK = 0;                              // ..then set SCK low again
  122.            }
  123.     return(uchar);                             // return read uchar
  124. }
  125. /****************************************************************************************************
  126. /*函數:uchar SPI_Read(uchar reg)
  127. /*功能:NRF24L01的SPI時序,詳細看時序圖,Page19
  128. /****************************************************************************************************/
  129. uchar SPI_Read(uchar reg)
  130. {
  131.         uchar reg_val;
  132.        
  133.         CSN = 0;                // CSN low, initialize SPI communication...
  134.         SPI_RW(reg);            // Select register to read from..
  135.         reg_val = SPI_RW(0);    // ..then read registervalue
  136.         CSN = 1;                // CSN high, terminate SPI communication
  137.        
  138.         return(reg_val);        // return register value
  139. }
  140. /****************************************************************************************************/
  141. /*功能:NRF24L01讀寫寄存器函數,
  142. /****************************************************************************************************/
  143. uint SPI_RW_Reg(uchar reg, uchar value)
  144. {
  145.         uint status;
  146.        
  147.         CSN = 0;                   // CSN low, init SPI transaction
  148.         status = SPI_RW(reg);      // select register
  149.         SPI_RW(value);             // ..and write value to it..
  150.         CSN = 1;                   // CSN high again
  151.        
  152.         return(status);            // return nRF24L01 status uchar
  153. }
  154. /****************************************************************************************************/
  155. /*函數:uint SPI_Read_Buf(uchar reg, uchar *pBuf, uchar uchars)
  156. /*功能: 用于讀數據,reg:為寄存器地址,pBuf:為待讀出數據地址,uchars:讀出數據的個數
  157. /****************************************************************************************************/
  158. /*uint SPI_Read_Buf(uchar reg, uchar *pBuf, uchar uchars)
  159. {
  160.         uint status,uchar_ctr;
  161.        
  162.         CSN = 0;                                    // Set CSN low, init SPI tranaction
  163.         status = SPI_RW(reg);                       // Select register to write to and read status uchar
  164.        
  165.         for(uchar_ctr=0;uchar_ctr<uchars;uchar_ctr++)
  166.                 pBuf[uchar_ctr] = SPI_RW(0);    //
  167.        
  168.         CSN = 1;                           
  169.        
  170.         return(status);                    // return nRF24L01 status uchar
  171. }*/
  172. /*********************************************************************************************************
  173. /*函數:uint SPI_Write_Buf(uchar reg, uchar *pBuf, uchar uchars)
  174. /*功能: 用于寫數據:為寄存器地址,pBuf:為待寫入數據地址,uchars:寫入數據的個數
  175. /*********************************************************************************************************/
  176. uint SPI_Write_Buf(uchar reg, uchar *pBuf, uchar uchars)
  177. {
  178.         uint status,uchar_ctr;
  179.        
  180.         CSN = 0;            //SPI使能      
  181.         status = SPI_RW(reg);   
  182.         for(uchar_ctr=0; uchar_ctr<uchars; uchar_ctr++) //
  183.                 SPI_RW(*pBuf++);
  184.         CSN = 1;           //關閉SPI
  185.         return(status);    //
  186. }
  187. /*
  188. /****************************************************************************************************/
  189. /*函數:void SetRX_Mode(void)
  190. /*功能:數據接收配置
  191. /****************************************************************************************************
  192. void SetRX_Mode(void)
  193. {
  194.         CE=0;
  195.         SPI_RW_Reg(WRITE_REG + CONFIG, 0x0f);                   // IRQ收發完成中斷響應,16位CRC        ,主接收
  196.         CE = 1;
  197.         inerDelay_us(130);
  198. }
  199. /******************************************************************************************************
  200. /*函數:unsigned char nRF24L01_RxPacket(unsigned char* rx_buf)
  201. /*功能:數據讀取后放如rx_buf接收緩沖區中
  202. /******************************************************************************************************
  203. unsigned char nRF24L01_RxPacket(unsigned char* rx_buf)
  204. {
  205.     unsigned char revale=0;
  206.         sta=SPI_Read(STATUS);        // 讀取狀態寄存其來判斷數據接收狀況
  207.         if(RX_DR)                                // 判斷是否接收到數據
  208.         {
  209.             CE = 0;                         //SPI使能
  210.                 SPI_Read_Buf(RD_RX_PLOAD,rx_buf,TX_PLOAD_WIDTH);// read receive payload from RX_FIFO buffer
  211.                 revale =1;                        //讀取數據完成標志
  212.         }
  213.         SPI_RW_Reg(WRITE_REG+STATUS,sta);   //接收到數據后RX_DR,TX_DS,MAX_PT都置高為1,通過寫1來清楚中斷標志
  214.         return revale;
  215. }
  216. */
  217. /***********************************************************************************************************
  218. /*函數:void nRF24L01_TxPacket(unsigned char * tx_buf)
  219. /*功能:發送 tx_buf中數據
  220. /**********************************************************************************************************/
  221. void nRF24L01_TxPacket(unsigned char * tx_buf)
  222. {
  223.         CE=0;                        //StandBy I模式       
  224.         SPI_Write_Buf(WRITE_REG + RX_ADDR_P0, TX_ADDRESS, TX_ADR_WIDTH); // 裝載接收端地址
  225.         SPI_Write_Buf(WR_TX_PLOAD, tx_buf, TX_PLOAD_WIDTH);                          // 裝載數據       
  226.         SPI_RW_Reg(WRITE_REG + CONFIG, 0x0e);                    // IRQ收發完成中斷響應,16位CRC,主發送
  227.         CE=1;                 //置高CE,激發數據發送
  228.         inerDelay_us(10);
  229. }

  230. uchar Check_ACK()
  231. {
  232. sta=SPI_Read(STATUS);                    // 讀狀態寄存器
  233. if(TX_DS||MAX_RT)
  234. {
  235.   SPI_RW_Reg(WRITE_REG + STATUS, 0xff);  //清除TX_DS或MAX_RT中斷標志
  236.   CSN = 0;
  237.   SPI_RW(FLUSH_TX);        //清除TX_FIFO
  238.      CSN  = 1;         
  239.   return(0);
  240. }                          
  241.               
  242.     else
  243.         return(1);
  244. }
  245. uchar Temp_Value[]={0x00,0x00};
  246. //bit DS18B20_IS_OK = 1;
  247. uint temp=0;
  248. uchar Display_Digit[]={0,0,0,0,0};
  249. uchar code df_Table[]={0,1,1,2,3,3,4,4,5,6,6,7,8,8,9,9};
  250. sbit LCD_RS = P2^0;
  251. sbit LCD_RW = P2^1;
  252. sbit LCD_EN = P3^7;

  253. uchar  Temp_Disp_Title[]=       {"reshidian :  "};        //1602 液晶第一行顯示內容
  254. uchar Current_Temp_Display_Buffer[]={" TEMP:       "};  //1602 液晶第二行顯示內容

  255. void delay1ms(unsigned int ms)//延時1毫秒(不夠精確的)
  256. {unsigned int i,j;
  257.    for(i=0;i<ms;i++)
  258.     for(j=0;j<100;j++);
  259. }

  260. bit LCD_Busy_Check()  //LCD 忙標志,返回值為 1602LCD 的忙標志位,為 1 表示忙
  261. {
  262.   bit result;
  263. LCD_RS = 0;
  264. LCD_RW = 1;
  265. LCD_EN = 1;
  266. delay1ms(1);
  267. result = (bit)(P0&0x80);
  268. LCD_EN=0;
  269. return result;
  270. }
  271. void Write_LCD_Command(uchar cmd)  //1602LCD 寫指令函數
  272. {
  273. while(LCD_Busy_Check());
  274. LCD_RS = 0;
  275. LCD_RW = 0;
  276. LCD_EN = 0;
  277. _nop_();
  278. _nop_();
  279. P0 = cmd;
  280. delay1ms(1);
  281. LCD_EN = 1;
  282. delay1ms(1);
  283. LCD_EN = 0;
  284. }
  285. void Write_LCD_Data(uchar dat)  //1602LCD 寫數據函數
  286. {
  287. while(LCD_Busy_Check());
  288. LCD_RS = 1;
  289. LCD_RW = 0;
  290. LCD_EN = 0;
  291. P0 = dat;
  292.   delay1ms(1);
  293. LCD_EN = 1;  
  294.   delay1ms(1);
  295. LCD_EN = 0;  
  296.   delay1ms(1);
  297. }
  298. void LCD_Initialise()   //1602LCD 初始化
  299. {
  300.   Write_LCD_Command(0x01);
  301. delay1ms(5);
  302. Write_LCD_Command(0x08);
  303. delay1ms(5);
  304. Write_LCD_Command(0x38);
  305. delay1ms(5);
  306. Write_LCD_Command(0x0c);
  307. delay1ms(5);
  308. Write_LCD_Command(0x06);
  309. delay1ms(5);  
  310. }
  311. void Set_LCD_POS(uchar pos)   //1602LCD 設置顯示位置
  312. {
  313.   Write_LCD_Command(pos|0x80);  
  314. }
  315. void Delayl(uint x)   //延時 2
  316. {
  317.   while(x--);
  318. }
  319. uchar Init_DS18B20()  //初始化(或者說復位)DS18B20
  320. {
  321.   uchar status;
  322. DQ = 1;
  323. Delayl(8);
  324. DQ = 0;
  325. Delayl(90);
  326. DQ = 1;
  327. Delayl(8);
  328. status=DQ;Delayl(100);   
  329. DQ = 1;
  330. return status;
  331. }
  332. uchar ReadOneByte()    //從 DS18B20 讀一字節數據
  333. {
  334.   uchar i,dat=0;
  335. DQ = 1;
  336. _nop_();
  337. for(i=0;i<8;i++)
  338. {
  339.    DQ = 0;
  340. dat >>= 1;
  341. DQ = 1;
  342. _nop_();
  343. _nop_();
  344. if(DQ)
  345. dat |= 0X80;
  346. Delayl(30);
  347. DQ = 1;
  348. }
  349. return dat;
  350. }
  351. void WriteOneByte(uchar dat)   //從 DS18B20 寫一字節數據
  352. {
  353. uchar i;
  354. for(i=0;i<8;i++)
  355. {
  356. DQ = 0;
  357. DQ = dat& 0x01;
  358. Delayl(5);
  359. DQ = 1;
  360. dat >>= 1;
  361. }
  362. }

  363. void Read_Temperature()  //從 DS18B20 讀取溫度值
  364. {
  365.   
  366. Init_DS18B20();
  367. WriteOneByte(0xcc);  //跳過序列號命令
  368. WriteOneByte(0x44);  //啟動溫度轉換命令
  369. Init_DS18B20();  //復位 DS18B20 (每一次讀寫之前都要對 DS18B20 進行復位操作)  
  370. WriteOneByte(0xcc);  //跳過序列號命令
  371. WriteOneByte(0xbe); //讀取溫度寄存器
  372. Temp_Value[0] = ReadOneByte(); //讀取溫度低 8 位(先讀低字節,再讀高字節, )
  373. Temp_Value[1] = ReadOneByte();//讀取溫度高 8 位  (每次只能讀一個字節)  
  374. }
  375. void Display_Temperature()  //在 1602LCD 上顯示當前溫度
  376. {
  377.   uchar i;
  378. uchar t = 150, ng = 0;  //延時值與負數標志
  379. if((Temp_Value[1]&0xf8)==0xf8)  //高字節高 5 位如果全為 1, 則為負數, 為負數時取反                     
  380. {                            //加 1,并設置負數標志為 1
  381. Temp_Value[1] = ~Temp_Value[1];
  382. Temp_Value[0] = ~Temp_Value[0]+1;
  383. if(Temp_Value[0]==0x00)   //若低字節進位,則高字節加 1  
  384. Temp_Value[1]++;
  385. ng = 1; //設置負數標志為 1
  386. }  
  387. Display_Digit[0] = df_Table[Temp_Value[0]&0x0f];   //查表得到溫度小數部分
  388.    //獲取溫度整數部分(低字節低 4 位清零,高 4 位右移 4 位)+(高字節高 5 位清零,   
  389.    //低三位左移 4 位)
  390. temp= ((Temp_Value[0]&0xf0)>>4) | ((Temp_Value[1]&0x07)<<4);
  391. //將溫度整數部分分解為 3 位待顯示數字
  392.    Display_Digit[3] = temp/100;
  393. Display_Digit[2] = temp%100/10;
  394. Display_Digit[1] = temp%10;
  395. Display_Digit[4] = ng;
  396.    //刷新 LCD 緩沖     //加字符 0 是為了將待數字轉化為字符顯示
  397. Current_Temp_Display_Buffer[11] = Display_Digit[0] + '0';  
  398. Current_Temp_Display_Buffer[10] = '.';
  399. Current_Temp_Display_Buffer[9]  = Display_Digit[1] + '0';
  400. Current_Temp_Display_Buffer[8]  = Display_Digit[2] + '0';
  401. Current_Temp_Display_Buffer[7]  = Display_Digit[3] + '0';
  402. if(Display_Digit[3] == 0)  //高位為 0 時不顯示
  403. Current_Temp_Display_Buffer[7]  = ' ';
  404. if(Display_Digit[2] == 0&&Display_Digit[3]==0) //高位為 0, 且次高位為 0, 則次高位不顯示
  405. Current_Temp_Display_Buffer[8]  = ' ';
  406. //負號顯示在恰當位置
  407. if(Display_Digit[4]==1)
  408. Current_Temp_Display_Buffer[6]  = '-';
  409. else
  410. Current_Temp_Display_Buffer[6]  = ' ';
  411. if (re ==0)
  412. ……………………

  413. …………限于本文篇幅 余下代碼請從51黑下載附件…………
復制代碼

所有資料51hei提供下載:
基于單片機無線收發裝置的紅外溫度報警器最終.7z (11.38 MB, 下載次數: 89)


評分

參與人數 1黑幣 +50 收起 理由
admin + 50 共享資料的黑幣獎勵!

查看全部評分

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

使用道具 舉報

沙發
ID:298663 發表于 2020-3-5 17:37 | 只看該作者
你pcb圖上有一處畫的直角, 這個一般是要避免的,雖然不會影響功能,但是在高速電路設計中 會有EMI問題

評分

參與人數 1黑幣 +40 收起 理由
admin + 40 回帖助人的獎勵!

查看全部評分

回復

使用道具 舉報

您需要登錄后才可以回帖 登錄 | 立即注冊

本版積分規則

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

Powered by 單片機教程網

快速回復 返回頂部 返回列表
主站蜘蛛池模板: 欧美成人精品一区 | 国产在线视频一区二区 | 亚洲区一区二 | 欧美一级在线观看 | 亚洲视频免费在线观看 | 国产福利视频在线观看 | 亚洲精品成人av久久 | 九九在线 | 国产视频久久久久 | 国产精品久久久久久久久免费桃花 | 久久久久免费精品国产 | 国产精品一区二区三区在线播放 | 一区二区在线不卡 | 精品国产乱码久久久久久影片 | 特黄特黄a级毛片免费专区 av网站免费在线观看 | 亚洲天堂精品久久 | 国产成人高清 | 高清一区二区三区 | 新91视频网 | 免费h视频 | 成人一级黄色毛片 | 日本午夜一区二区三区 | 精品福利一区二区三区 | 亚洲国产精品日韩av不卡在线 | 99re在线视频观看 | 五月综合色啪 | 北条麻妃一区二区三区在线视频 | 欧美aⅴ| 亚洲国产精品自拍 | 亚洲一区二区视频 | 天天综合久久网 | 5060网一级毛片 | 久草免费在线视频 | 亚洲欧美在线视频 | www.午夜| 中文av电影 | 成人在线中文字幕 | 国产999精品久久久 午夜天堂精品久久久久 | 久久久蜜桃 | 精品久久久网站 | 国产一区二区三区不卡av |