nRF24L01应用程序

发布者:CrystalSparkle最新更新时间:2016-09-06 来源: eefocus关键字:nRF24L01  应用程序 手机看文章 扫描二维码
随时随地手机看文章
最近用了下nRF24L01模块,用以前调试nRF2401的电路板子进行调试。参考了别人的程序代码,是模拟SPI的程序。结合nRF24L01的数据手册我对程序进行了消化理解,再做修改,结合自己的电路,调试成功了。这个是收发一体的程序。
nRF24L01应用程序 - 小文 - 小文电子设计
源程序:
#include
#include

#define uint unsigned int
#define uchar unsigned char
//****************************************IO端口定义***************************************
//****************************************NRF24L01端口定义***************************************
sbit  MISO =P1^2;
sbit  MOSI =P1^6;
sbit SCK     =P1^3;
sbit CE     =P1^4;
sbit CSN  =P1^5;
sbit IRQ  =P1^7;
//************************************按键***************************************************
sbit KEY1=P3^4;

//*********************************************NRF24L01*************************************
#define TX_ADR_WIDTH    5    // 5 uints TX address width
#define RX_ADR_WIDTH    5    // 5 uints RX address width
#define TX_PLOAD_WIDTH  20   // 20 uints TX payload
#define RX_PLOAD_WIDTH  20   // 20 uints TX payload
uchar  TX_ADDRESS[TX_ADR_WIDTH]= {0x12,0x34,0x56,0x78,0x90}; //本地地址
uchar  RX_ADDRESS[RX_ADR_WIDTH]= {0x12,0x34,0x56,0x78,0x90}; //接收地址
//***************************************NRF24L01寄存器指令*******************************************************
#define READ_REG        0x00   // 读寄存器指令
#define WRITE_REG       0x20  // 写寄存器指令
#define RD_RX_PLOAD     0x61   // 读取接收数据指令
#define WR_TX_PLOAD     0xA0   // 写待发数据指令
#define FLUSH_TX        0xE1  // 冲洗发送 FIFO指令
#define FLUSH_RX        0xE2   // 冲洗接收 FIFO指令
#define REUSE_TX_PL     0xE3   // 定义重复装载数据指令
#define NOP             0xFF   // 保留
//*************************************SPI(nRF24L01)寄存器地址****************************************************
#define CONFIG          0x00  // 配置收发状态,CRC校验模式以及收发状态响应方式
#define EN_AA           0x01  // 自动应答功能设置
#define EN_RXADDR       0x02  // 可用信道设置
#define SETUP_AW        0x03  // 收发地址宽度设置
#define SETUP_RETR      0x04  // 自动重发功能设置
#define RF_CH           0x05  // 工作频率设置
#define RF_SETUP        0x06  // 发射速率、功耗功能设置
#define STATUS          0x07  // 状态寄存器
#define OBSERVE_TX      0x08  // 发送监测功能
#define CD              0x09  // 地址检测           
#define RX_ADDR_P0      0x0A  // 频道0接收数据地址
#define RX_ADDR_P1      0x0B  // 频道1接收数据地址
#define RX_ADDR_P2      0x0C  // 频道2接收数据地址
#define RX_ADDR_P3      0x0D  // 频道3接收数据地址
#define RX_ADDR_P4      0x0E  // 频道4接收数据地址
#define RX_ADDR_P5      0x0F  // 频道5接收数据地址
#define TX_ADDR         0x10  // 发送地址寄存器
#define RX_PW_P0        0x11  // 接收频道0接收数据长度
#define RX_PW_P1        0x12  // 接收频道0接收数据长度
#define RX_PW_P2        0x13  // 接收频道0接收数据长度
#define RX_PW_P3        0x14  // 接收频道0接收数据长度
#define RX_PW_P4        0x15  // 接收频道0接收数据长度
#define RX_PW_P5        0x16  // 接收频道0接收数据长度
#define FIFO_STATUS     0x17  // FIFO栈入栈出状态寄存器设置
//**************************************************************************************
void Delay(unsigned int s);
void inerDelay_us(unsigned char n);
void init_NRF24L01(void);
uchar SPI_RW(uchar byte);
uchar SPI_Read(uchar reg);
void SetRX_Mode(void);
uchar SPI_RW_Reg(uchar reg, uchar value);
uchar SPI_Read_Buf(uchar reg, uchar *pBuf, uchar num);
uchar SPI_Write_Buf(uchar reg, uchar *pBuf, uchar num);
unsigned char nRF24L01_RxPacket(unsigned char* rx_buf);
void nRF24L01_TxPacket(unsigned char * tx_buf);
void delayms(unsigned int count);
//*****************************************长延时*****************************************
void Delay(unsigned int s)
{
 unsigned int i;
 for(i=0; i  for(i=0; i }
//******************************************************************************************
uchar  sta;   //状态标志
#define RX_DR (sta & 0x40)
#define TX_DS (sta & 0x20)
#define MAX_RT (sta & 0x10)
 
//***********毫秒延时程序********************/
void delayms(unsigned int count)
{
 unsigned int i,j;
 for(i=0;i  for(j=0;j<120;j++);
}
 
//****************************************************************************************
//NRF24L01初始化
void init_NRF24L01(void)
{
    delayms(1);
  CE=0;    // chip enable
  CSN=1;   // Spi disable 
  SCK=0;   // Spi clock line init high
 SPI_Write_Buf(WRITE_REG + TX_ADDR, TX_ADDRESS, TX_ADR_WIDTH);    // 写本地地址 
 SPI_Write_Buf(WRITE_REG + RX_ADDR_P0, RX_ADDRESS, RX_ADR_WIDTH); // 写接收端地址
 SPI_RW_Reg(WRITE_REG + EN_AA, 0x01);      //  频道0自动 ACK应答允许 
 SPI_RW_Reg(WRITE_REG + EN_RXADDR, 0x01);  //  允许接收地址只有频道0,如果需要多频道可以参考Page21  
 SPI_RW_Reg(WRITE_REG + RF_CH, 0);        //   设置信道工作为2.4GHZ,收发必须一致
 SPI_RW_Reg(WRITE_REG + RX_PW_P0, RX_PLOAD_WIDTH); //设置接收数据长度,本次设置为32字节
 SPI_RW_Reg(WRITE_REG + RF_SETUP, 0x07);     //设置发射速率为1MHZ,发射功率为最大值0dB
}
//***************************************************************************************************
//函数:uint SPI_RW(uint uchar)
//功能:NRF24L01的SPI写时序
//***************************************************************************************************/
uchar SPI_RW(uchar byte)
{
 uchar i;
    for(i=0;i<8;i++) // output 8-bit
    {
  MOSI = (byte & 0x80);         // output 'uchar', MSB to MOSI
  byte = (byte << 1);           // shift next bit into MSB..
  SCK = 1;                      // Set SCK high..
  byte |= MISO;           // capture current MISO bit
  SCK = 0;                // ..then set SCK low again
    }
    return(byte);               // return read uchar
}
//***************************************************************************************************
//函数:uchar SPI_Read(uchar reg)
//功能:NRF24L01的SPI时序
/****************************************************************************************************/
uchar SPI_Read(uchar reg)
{
 uchar reg_val;
 
 CSN = 0;                // CSN low, initialize SPI communication...
 SPI_RW(reg);            // Select register to read from..
 reg_val = SPI_RW(0);    // ..then read registervalue
 CSN = 1;                // CSN high, terminate SPI communication
 
 return(reg_val);        // return register value
}
//***************************************************************************************************/
//功能:NRF24L01读写寄存器函数
//***************************************************************************************************/
uchar SPI_RW_Reg(uchar reg, uchar value)
{
 uchar status;
 
 CSN = 0;                   // CSN low, init SPI transaction
 status = SPI_RW(reg);      // select register
 SPI_RW(value);             // ..and write value to it..
 CSN = 1;                   // CSN high again
 
 return(status);            // return nRF24L01 status uchar
}
//***************************************************************************************************/
//函数:uint SPI_Read_Buf(uchar reg, uchar *pBuf, uchar uchars)
//功能: 用于读数据,reg:为寄存器地址,pBuf:为待读出数据地址,uchars:读出数据的个数
//***************************************************************************************************/
uchar SPI_Read_Buf(uchar reg, uchar *pBuf, uchar num)
{
 uchar status,i;
 
 CSN = 0;                      // Set CSN low, init SPI tranaction
 status = SPI_RW(reg);         // Select register to write to and read status uchar
 
 for(i=0;i   pBuf[i] = SPI_RW(0);    // 
 
 CSN = 1;                           
 
 return(status);                    // return nRF24L01 status uchar
}
//********************************************************************************************************
//函数:uint SPI_Write_Buf(uchar reg, uchar *pBuf, uchar uchars)
//功能: 用于写数据:为寄存器地址,pBuf:为待写入数据地址,uchars:写入数据的个数
//********************************************************************************************************/
uchar SPI_Write_Buf(uchar reg, uchar *pBuf, uchar num)
{
 uchar status,i;
 
 CSN = 0;            //SPI使能       
 status = SPI_RW(reg);   
 for(i=0; i   SPI_RW(*pBuf++);
 CSN = 1;           //关闭SPI
 return(status);    // 
}
//***************************************************************************************************/
//函数:void SetRX_Mode(void)
//功能:数据接收配置 
//***************************************************************************************************/
void SetRX_Mode(void)
{
 CE=0;
 SPI_RW_Reg(WRITE_REG + CONFIG, 0x0f);     // IRQ收发完成中断响应,16位CRC ,主接收
 CE = 1; 
 delayms(1);
}
//*****************************************************************************************************/
//函数:unsigned char nRF24L01_RxPacket(unsigned char* rx_buf)
//功能:数据读取后放如rx_buf接收缓冲区中
/******************************************************************************************************/
unsigned char nRF24L01_RxPacket(unsigned char* rx_buf)
{
    unsigned char revale=0;
 sta=SPI_Read(STATUS); // 读取状态寄存器来判断数据接收状况
 if(RX_DR)    // 判断是否接收到数据
 {
     CE = 0;    //SPI使能
  SPI_Read_Buf(RD_RX_PLOAD,rx_buf,TX_PLOAD_WIDTH);// read receive payload from RX_FIFO buffer
  revale =1;   //读取数据完成标志
 }
 SPI_RW_Reg(WRITE_REG+STATUS,sta);   //接收到数据后RX_DR,TX_DS,MAX_PT都置高为1,通过写1来清除中断标志
 return revale;
}
//**********************************************************************************************************
//函数:void nRF24L01_TxPacket(unsigned char * tx_buf)
//功能:发送 tx_buf中数据
//*********************************************************************************************************/
void nRF24L01_TxPacket(unsigned char * tx_buf)
{
 CE=0;   //StandBy I模式 
 SPI_Write_Buf(WRITE_REG + RX_ADDR_P0, TX_ADDRESS, TX_ADR_WIDTH); // 装载接收端地址
 SPI_Write_Buf(WR_TX_PLOAD, tx_buf, TX_PLOAD_WIDTH);     // 装载数据 
 SPI_RW_Reg(WRITE_REG + CONFIG, 0x0e);      // IRQ收发完成中断响应,16位CRC,主发送
 CE=1;   //置高CE,激发数据发送
 delayms(1);
}
//************************************主函数************************************************************
void main(void)
{
 unsigned char TxBuf[20]={0};  // 
 unsigned char RxBuf[20]={0};
 unsigned char led_num; 
  delayms(100);
    init_NRF24L01() ;
 TxBuf[1] = 0x55 ;
 nRF24L01_TxPacket(TxBuf); // Transmit Tx buffer data
 Delay(6000);
 led_num=0x00;
 while(1)
 {
     if(KEY1 ==0 ) 
    {
      led_num++; 
      TxBuf[1] =led_num ;
   nRF24L01_TxPacket(TxBuf); // Transmit Tx buffer data     
      P0=~led_num;
   Delay(5000);
     }
  
   
//***********************************************************************************************
  SetRX_Mode();       // 每次开始发送 然后无按键直接跳转到接收 一直循环。
  nRF24L01_RxPacket(RxBuf);
  if(RX_DR==0) P0=~RxBuf[1];  
  
 }
 
}
关键字:nRF24L01  应用程序 引用地址:nRF24L01应用程序

上一篇:一种AT89S52的USB下载线设计
下一篇:航模遥控器改装---PPM波形产生

推荐阅读最新更新时间:2024-03-16 15:08

nRF24L01 AVR 发送接收驱动程序
#include #include #include 12864.h 智芯锐电子技术社区:http://www.zhixinrui.com //------------------------------------------------------------------------------ //spi标志 #define DDR_SPI DDRB #define DD_MOSI 5 #define DD_MISO 6 #define DD_SCK 7 #define DD_SS 4 #define CE PB3 #define IRQ PB2 //----------
[单片机]
如何在非实时linux上实现实时应用程序与内核模块之间共享存储器
linux并不是严格意义上的实时操作系统,为了实际需要,工程师们必须想尽办法来祢补这一不足,于是出现了rtlinux和rtai等并不强调商业性的软件。免费的rtlinux显然庞大而并不兼容大部分的嵌入式平台,最新版本的rtlinux也只能支持I386和PPC而已。Rtai是不错的选择,但要把它移植到你的平台上去,为了适应你的linux版本,你的CPU,你必须的花费许多的工作,比如说最近比较流行的AT91RM9200DK,光修改linux版本补丁就要花费许多的功夫。Rtlinux和rtai为了增强linux操作系统的实时性,主要是通过开辟内核模块与应用程序之间可以共享的内存快来实现的。它们在内核空间控制硬实时任务的运行,并通过一个名
[嵌入式]
开车勿打手机!安全禁令为汽车OEM带来新商机
  随着美国和加拿大部分地区开始禁止驾车者使用移动设备,分心驾驶问题变得更加重要。在4月拉斯维加斯举行的2010年国际CTIA无线展会上,多家公司展示了针对分心驾驶问题的解决方案。这些方案从安装在汽车上的硬件,到基于运营商网络、可以用于多数智能手机平台的应用程序,多种多样。    立法行动   加拿大12个省中有五个已经颁布禁令,禁止在驾车过程中使用手持设备。不列颠哥伦比亚省是最新颁布禁令的一个省,从今年初开始生效。   2001年纽约成为美国第一个禁止在车中使用手机的州。自那以后,又有六个州和地区颁布了禁令。另外,明尼苏达州2009年禁止在开车时发短信,爱荷华州从今年7月起将实施一项禁令。虽然爱荷华州将实施禁令,但在201
[手机便携]
STC单片机用户应用程序区和ISP监控程序区的区别
用户应用程序区(AP区): 是指用户自己编写的程序区 ISP监控程序区: ISP区是指芯片出厂时就已经固化在单片机内部的一段程序,STC单片机可以进行ISP串行下载程序,这就是因为芯片在出厂时已经在单片机内部固化了ISP引导码,程序首次上电时先会从ISP区开始执行代码,体现在实际实验中时,就是我们在下载程序时,先要点击下载软件界面上的下载,然后再开启单片机电源,当单片机检测到上位机有下载程序的需要时,便启用ISP下载功能给单片机下载程序。若经过短暂的时间没有检测到上位机有下载程序的需求,单片机便会从用户应用程序区(AP区)开始执行代码。
[单片机]
NRF24l01无线温度传感(2)
///////////////////////////////////////////////接收部分//////////////////////////////////////////////////////////////// #include reg52.h #include intrins.h typedef unsigned char uchar; typedef unsigned char uint; //****************************************IO端口定义*************************************** sbit MISO =P1^2; sbit
[单片机]
Procreate 宣布推出划时代的全新 iPad 应用程序「Procreate Dreams」
Procreate 宣布推出划时代的全新 iPad 应用程序「Procreate Dreams」,为每一个人提供开创性的动画工具。 过去 6 年多来最畅销 iPad 付费应用 Procreate 的创作者宣布推出全新 iPad 动画应用程序 Procreate Dreams Procreate Dreams 将具备一系列直观的动画制作工具,专业人士和初学者都将受益匪浅。 Procreate Dreams 将于 2023 年 11 月 22 日发布。 Procreate Dreams 在 App Store 上单次购买售价为 19.99 美元。无需订阅。 澳大利亚霍巴特 - Media OutReach - 20
[家用电子]
金融服务机构如何优化和保障移动应用程序体验
世界经济论坛(World Economic Forum)数据显示,20世纪90年代中期到21世纪初出生的Z世代在2019年将占到全球人口(77亿人)的32%,并且首次超过了千禧一代。Z时代人群的数量正在不断增加,他们最大的特点是喜欢使用移动设备,甚至只使用移动设备(他们出生在一个可以通过网络获得各种信息并且社交媒体、通信、信息来源、约会应用以及网络银行和公用事业等一切事物都高度依赖移动设备的互联世界中)。在中国,Z世代占全国人口的约20%,他们在手机、媒体订阅等技术产品方面的支出高于西方的同龄人。因此在今年,为了满足只使用移动设备的Z世代的需求,运营的升级和网络性能的提升将成为各个行业业务的重中之重。 尤其是银行和金融服务机构
[物联网]
金融服务机构如何优化和保障移动<font color='red'>应用程序</font>体验
如何在ADAS应用程序中使用MIPI®CSI-2端口复制记录传感器数据
随着高级驾驶员辅助系统(ADAS)促成自动驾驶,对机器视觉、查看、并行处理和数据记录的聚合视频传感器数据多副本的需求越来越多。 前置机器视觉摄像头更是需要多个副本,但它将很快适用于自主车辆的附加摄像头、雷达和光线检测/测距(LIDAR)传感器。数据记录是一个当今非常常见的复制应用(图1)。在机器视觉应用中,通常记录某些行车状况的原始传感器数据,以用于未来分析。这种情况下,拥有聚合原始传感器数据的第二副本,并将其用于数据记录,是大有裨益的;而另一个副本则用于机器视觉处理。   图1:常见的ADAS数据记录拓扑结构 复制聚合传感器数据 数据复制可在视频路径中的不同位置进行。可通过单独的电缆将每个传感器连接到机器视觉和数据记录电子控制
[嵌入式]
小广播
添点儿料...
无论热点新闻、行业分析、技术干货……
设计资源 培训 开发板 精华推荐

最新单片机文章
何立民专栏 单片机及嵌入式宝典

北京航空航天大学教授,20余年来致力于单片机与嵌入式系统推广工作。

换一换 更多 相关热搜器件
随便看看
电子工程世界版权所有 京B2-20211791 京ICP备10001474号-1 电信业务审批[2006]字第258号函 京公网安备 11010802033920号 Copyright © 2005-2024 EEWORLD.com.cn, Inc. All rights reserved