• 【应用】nRF24L01无线模块在单片机与FPGA上的应用


      先简单的介绍下nRF24L01无线模块

      (1) 2.4Ghz 全球开放ISM 频段免许可证使用

      (2) 最高工作速率2Mbps,高效GFSK调制,抗干扰能力强,特别适合工业控制场合

      (3) 126 频道,满足多点通信和跳频通信需要

      (4) 内置硬件CRC 检错和点对多点通信地址控制

      (5) 低功耗1.9 - 3.6V 工作,待机模式下状态为22uA;掉电模式下为900nA

      (6) 内置2.4Ghz 天线,体积小巧15mm X29mm

      (7) 模块可软件设地址,只有收到本机地址时才会输出数据(提供中断指示),可直接接各种单片机使用,软件编程非常方便

       

      通过SPI方式完成数据的交换,包括数据的发送,数据的接收。说明一下,单片机中如果没有SPI的硬件电路,我们可以使用单片机的普通IO口进行SPI的时序模拟,只要符合无线模块的时序逻辑,一样能控制无线模块的通信。FPGA是可编程逻辑,最大的特点就是灵活,用户可根据需求加入所需要的逻辑器件,当然它所包含的逻辑单元也是相当的丰富,有SPI硬件模块。这样用户就省去了SPI方式的时序逻辑,可以更好的专注于功能的开发。

      下面将详细的介绍下nRF24L01无线模块在单片机与FPGA上的应用

    单片机:这里我们使用的单片机型号为PIC16F877。

                         图1.3  NRF24L01接入PIC的原理图

        

      说明:从图1.3中可以看出,主要是图1.1中的6个信号(还有2个是地与电源)接入单片机中。而那些引脚是普通的IO口,需要用户模仿SPI时序进行控制。

      无线模块进行数据的交换就是数据的发送与数据的接收,下面将从这2个方面进行介绍。不管是数据的发送还是数据的接收,要想控制好NRF24L01无线模块,先要通过SPI方式对无线模块进行配置,只需要往它对应的寄存器里写入数值便可。

      先定义一下PIC上的宏,下面我们就可以很方便的对PIC的引脚进行操作。

    View Code
     1 #define      MISO    RC2
    2 #define MOSI RC3
    3 #define SCK RD0
    4 #define CE RD2
    5 #define CSN RD1
    6 #define IRQ RC1
    7 #define LED RD3
    8 #define KEY0 RB0
    9 #define KEY1 RB1
    10 #define KEY2 RB2
    11 #define KEY3 RB3
    12 #define KEY4 RB4
    13 #define KEY5 RB5
    14 #define KEY6 RB6
    15 #define KEY7 RB7

       

      NRF24L01无线模块的寄存器

    View Code
     1 //*******************NRF24L01寄存器指令
    2 #define READ_REG 0x00 // 读寄存器指令
    3 #define WRITE_REG 0x20 // 写寄存器指令
    4 #define RD_RX_PLOAD 0x61 // 读取接收数据指令
    5 #define WR_TX_PLOAD 0xA0 // 写待发数据指令
    6 //*******************SPI(nRF24L01)寄存器地址
    7 #define CONFIG 0x00   // 配置收发状态,
    8 #define EN_AA 0x01   // 自动应答功能设置
    9 #define EN_RXADDR 0x02   // 可用信道设置
    10 #define SETUP_AW 0x03   // 收发地址宽度设置
    11 #define SETUP_RETR 0x04   // 自动重发功能设置
    12 #define RF_CH 0x05   // 工作频率设置
    13 #define RF_SETUP 0x06   // 发射速率、功耗功能设置
    14 #define STATUS 0x07   // 状态寄存器
    15 #define RX_ADDR_P0 0x0A   // 频道0接收数据地址
    16 #define TX_ADDR 0x10   // 发送地址寄存器
    17 #define RX_PW_P0 0x11   // 接收频道0接收数据长度
    18 #define FIFO_STATUS 0x17   // FIFO栈入栈出状态寄存器设置

        

      有2类寄存器是用户可以根据自己的需求所确定的,那就是地址的长度以及内容、发送与接收数据的长度,但无线模块一次最多可以发送32个字节,这两类寄存器一般设置为3~4个字节。

    View Code
    1 #define TX_PLOAD_WIDTH 4       
    2 #define RX_PLOAD_WIDTH 4
    3 unsigned char TX_ADDRESS[TX_ADR_WIDTH]= {0x34,0x43,0x10}; //本地地址
    4 unsigned char RX_ADDRESS[RX_ADR_WIDTH]= {0x34,0x43,0x10}; //接收地址

       

      A  模拟SPI方式

    View Code
     1   /****************************************************************************************************
    2 /*函数:uint SPI_RW(uint uchar)
    3 /*功能:NRF24L01的SPI时序
    4 /****************************************************************************************************/
    5 unsigned char SPI_RW(unsigned char a)
    6 {
    7 unsigned char i;
    8 for(i=0;i<8;i++)
    9 {
    10 if((a&0x80)==0x80)
    11 MOSI=1;
    12 else MOSI=0; // output 'uchar', MSB to MOSI
    13 a=(a<<1); // shift next bit into MSB..
    14 SCK=1; // Set SCK high..
    15 if(MISO==1)
    16 a|=0x01;
    17 else a&=0xfe; // capture current MISO bit
    18 SCK=0; // ..then set SCK low again
    19 }
    20 return(a); // return read uchar
    21 }

       

      B  以SPI方式对寄存器的操作

    View Code
     1 /****************************************************************************************************
    2 /*函数:uchar SPI_Read(uchar reg)
    3 /*功能:NRF24L01的SPI读操作
    4 /****************************************************************************************************/
    5 unsigned char SPI_Read(unsigned char reg)
    6 {
    7 unsigned char reg_val;
    8 CSN=0; // CSN low, initialize SPI communication...
    9 SPI_RW(reg); // Select register to read from..
    10 reg_val=SPI_RW(0); // ..then read registervalue
    11 CSN=1; // CSN high, terminate SPI communication
    12 return(reg_val); // return register value
    13 }
    14 /****************************************************************************************************/
    15 /*功能:NRF24L01读写寄存器函数
    16 /****************************************************************************************************/
    17 unsigned char SPI_RW_Reg(unsigned char reg, unsigned char value)
    18 {
    19 unsigned char status;
    20 CSN = 0; // CSN low, init SPI transaction
    21 status=SPI_RW(reg); // select register
    22 SPI_RW(value); // ..and write value to it..
    23 CSN = 1; // CSN high again
    24 return(status); // return nRF24L01 status uchar
    25 }
    26 /****************************************************************************************************/
    27 /*函数:uint SPI_Read_Buf(uchar reg, uchar *pBuf, uchar uchars)
    28 /*功能: 用于读数据,reg:为寄存器地址,pBuf:为待读出数据地址,uchars:读出数据的个数
    29 /****************************************************************************************************/
    30 unsigned char SPI_Read_Buf(unsigned char reg, unsigned char *pBuf, unsigned char uchars)
    31 {
    32 unsigned char status,uchar_ctr;
    33 CSN = 0; // Set CSN low, init SPI tranaction
    34 status=SPI_RW(reg); // Select register to write to and read status uchar
    35
    36 for(uchar_ctr=0;uchar_ctr<uchars;uchar_ctr++)
    37 {
    38 pBuf[uchar_ctr]=SPI_RW(0);
    39 }
    40 CSN = 1;
    41
    42 return(status);
    43 }
    44 /*********************************************************************************************************
    45 /*函数:uint SPI_Write_Buf(uchar reg, uchar *pBuf, uchar uchars)
    46 /*功能: 用于写数据:为寄存器地址,pBuf:为待写入数据地址,uchars:写入数据的个数
    47 /*********************************************************************************************************/
    48 unsigned char SPI_Write_Buf(unsigned char reg, unsigned char *pBuf, unsigned char uchars)
    49 {
    50 unsigned char status,uchar_ctr;
    51
    52 CSN = 0; //SPI使能
    53 status=SPI_RW(reg);
    54 for(uchar_ctr=0; uchar_ctr<uchars; uchar_ctr++)
    55 {
    56 SPI_RW(*pBuf++);
    57 }
    58 CSN = 1; //关闭SPI
    59 return(status);
    60 }

        

      这样就可以对NRF24L01无线模块进行初始化工作,以及数据发送、数据接收。让无线模块是处于接收状态还是处于发送状态,初始化的工作有所不同,但区别不大,主要是CONFIG寄存器,可详细参考它的datesheet。

       NRF24L01发送的初始化以及发送时序

    View Code
     1 void init_NRF24L01_send(void)
    2 {
    3 delay(30);
    4 CE=0; // chip enable
    5 CSN=1; // Spi disable
    6 SCK=0; // Spi clock line init high
    7 delay(30);
    8 SPI_Write_Buf(WRITE_REG + TX_ADDR, TX_ADDRESS, TX_ADR_WIDTH); // 写本地地址
    9 SPI_Write_Buf(WRITE_REG + RX_ADDR_P0, RX_ADDRESS, RX_ADR_WIDTH); // 写接收端地址
    10 SPI_RW_Reg(WRITE_REG + EN_AA, 0x01); // 频道0自动 ACK应答允许
    11 SPI_RW_Reg(WRITE_REG + EN_RXADDR, 0x01); // 允许接收地址只有频道0,如果需要多频道可以参考Page21
    12 SPI_RW_Reg(WRITE_REG + SETUP_RETR, 0x1a); // 500us + 86us, 10 retrans...
    13 SPI_RW_Reg(WRITE_REG + RF_CH, 40); // 设置信道工作为2.4GHZ,收发必须一致
    14 SPI_RW_Reg(WRITE_REG + RX_PW_P0, RX_PLOAD_WIDTH); //设置接收数据长度,本次设置为4字节
    15 SPI_RW_Reg(WRITE_REG + RF_SETUP, 0x07); //设置发射速率为1MHZ,发射功率为最大值0dB
    16 SPI_RW_Reg(WRITE_REG + CONFIG, 0x0e); // IRQ收发完成中断响应,16位CRC,主发送
    17 // CE=1; // chip enable
    18 delay(30);
    19 }
    20
    21 /***********************************************************************************************************
    22 /*函数:void nRF24L01_TxPacket(unsigned char *tx_buf)
    23 /*功能:发送 tx_buf中数据
    24 /**********************************************************************************************************/
    25 void nRF24L01_TxPacket(unsigned char *tx_buf)
    26 {
    27 CE=0; //StandBy I模式
    28 SPI_Write_Buf(WRITE_REG + RX_ADDR_P0, TX_ADDRESS, TX_ADR_WIDTH); // 装载接收端地址
    29 SPI_Write_Buf(WR_TX_PLOAD, tx_buf, TX_PLOAD_WIDTH); // 装载数据
    30 SPI_RW_Reg(WRITE_REG + CONFIG, 0x0e); // IRQ收发完成中断响应,16位CRC,主发送
    31 CE=1; //置高CE,激发数据发送
    32 delay(100);
    33 CE=0;
    34 }

        

       NRF24L01接收的初始化以及接收时序

    View Code
     1 void init_NRF24L01_receive(void)
    2 {
    3 delay(30);
    4 CE=0; // chip enable
    5 CSN=1; // Spi disable
    6 SCK=0; // Spi clock line init high
    7 delay(30);
    8 SPI_Write_Buf(WRITE_REG + TX_ADDR, TX_ADDRESS, TX_ADR_WIDTH); // 写本地地址
    9 SPI_Write_Buf(WRITE_REG + RX_ADDR_P0, RX_ADDRESS, RX_ADR_WIDTH); // 写接收端地址
    10 SPI_RW_Reg(WRITE_REG + EN_AA, 0x01); // 频道0自动 ACK应答允许
    11 SPI_RW_Reg(WRITE_REG + EN_RXADDR, 0x01); // 允许接收地址只有频道0,如果需要多频道可以参考Page21
    12 SPI_RW_Reg(WRITE_REG + RF_CH, 40); // 设置信道工作为2.4GHZ,收发必须一致
    13 SPI_RW_Reg(WRITE_REG + RX_PW_P0, RX_PLOAD_WIDTH); //设置接收数据长度,本次设置为32字节
    14 SPI_RW_Reg(WRITE_REG + RF_SETUP, 0x07); //设置发射速率为1MHZ,发射功率为最大值0dB
    15 SPI_RW_Reg(WRITE_REG + CONFIG, 0x0f); // IRQ收发完成中断响应,16位CRC,主接受
    16 CE=1;
    17 delay(40);
    18 }
    19
    20
    21 /******************************************************************************************************/
    22 /*函数:unsigned char nRF24L01_RxPacket(unsigned char* rx_buf)
    23 /*功能:数据读取后放如rx_buf接收缓冲区中
    24 /******************************************************************************************************/
    25 unsigned char nRF24L01_RxPacket(unsigned char* rx_buf)
    26 {
    27 unsigned char revale=0;
    28 sta=SPI_Read(STATUS); // 读取状态寄存其来判断数据接收状况
    29 if(RX_DR) // 判断是否接收到数据
    30 {
    31 CE = 0; //SPI使能
    32 SPI_Read_Buf(RD_RX_PLOAD,rx_buf,TX_PLOAD_WIDTH);// read receive payload from RX_FIFO buffer
    33 revale =1; //读取数据完成标志
    34 }
    35 SPI_RW_Reg(WRITE_REG+STATUS,0xff);
    36 return revale;
    37 }

      

    下面总结一下NRF24L01在FPGA的应用

      由于FPGA自带SPI硬件,只需要在SOPC Bulider中添加SPI模块即可,在顶层图中我们就可以看到图1.4,另外,我们再添加两个IO口,这样我们就不必再模拟SPI方式,在FPGA中,有一个很好的API函数alt_avalon_spi_command();其函数原型为:

    1 int alt_avalon_spi_command(alt_u32base,alt_u32slave,
    2 alt_u32write_length,
    3 constalt_u8*wdata,
    4 alt_u32read_length,
    5 alt_u8*read_data,
    6 alt_u32flags)

      

    该函数执行以下功能:
        1、 SPI 从机片选信号有效(拉低) ;
        2、 从 wdata 指针读取数据,通过 SPI 接口传输总共 write_length 字节的数据,丢弃 MISO接口输入的数据;
        3、 读 read_length 个字节的数据,存储到 read_data 指针指向的地址。读传输过程中 MOSI 被置为 0;
        4、 撤销 SPI 从机片选信号(拉高)。
        头文件<altera_avalon_spi.h>,该头文件定义了 SPI 核的寄存器映射和访问硬件可用的一些特征常量。

    这个函数的最大缺点就是不可以在中断中使用,但这并不影响对它的使用。NRF24L01在单片机和FPGA上的应用的本质是一样的,主要区别就是对上面的A、B  SPI方式进行改写。

    A   就是用alt_avalon_spi_command();代替,是不是很方便呢。:-D

    B  以SPI方式对寄存器的操作

    View Code
     1 /*********************************************************************
    2 ** 函数名称: void SPI_RW_Reg(unsigned char reg, unsigned char value)()
    3 ** 函数功能: 访问无线模块寄存器,并也对其写数值控制
    4 ** 参数:2个,第一个为寄存器地址,第二个为向寄存器写的数值
    5 *********************************************************************/
    6 void SPI_RW_Reg ( unsigned char reg, unsigned char value )
    7 {
    8 alt_avalon_spi_command ( SPI_BASE,0,1,&reg,0,NULL,1 ); // select register
    9 alt_avalon_spi_command ( SPI_BASE,0,1,&value,0,NULL,0 );
    10 }
    11
    12 /*********************************************************************
    13 ** 函数名称: void SPI_Write_Buf(unsigned char reg, unsigned char *pBuf, unsigned char bytes)
    14 ** 函数功能: 访问寄存器,并向其写入bytes字节的数值
    15 ** 参数:3个,寄存器地址,数据、长度
    16 *********************************************************************/
    17 void SPI_Write_Buf ( unsigned char reg, unsigned char *pBuf, unsigned char bytes )
    18 {
    19 alt_avalon_spi_command ( SPI_BASE,0,1,&reg,0,NULL,1 );
    20 alt_avalon_spi_command ( SPI_BASE,0,bytes,pBuf,0,NULL,0 );
    21 }
    22 /********************************************************************
    23 ** 函数名称: unsigned char SPI_Read(unsigned char reg)
    24 ** 函数功能: 访问寄存器地址,并返回该寄存器的数值
    25 ** 参数:寄存器地址
    26 *********************************************************************/
    27 unsigned char SPI_Read ( unsigned char reg )
    28 {
    29 unsigned char reg_val;
    30 alt_avalon_spi_command ( SPI_BASE,0,1,&reg,0,NULL,1 );
    31 alt_avalon_spi_command ( SPI_BASE,0,0,NULL,1,&reg_val,0 );
    32 return ( reg_val );
    33 }
    34 /*
    35 *********************************************************************
    36 ** 函数名称: void SPI_Read_Buf(unsigned char reg, unsigned char *pBuf, unsigned char uchars)
    37 ** 函数功能: 访问寄存器,并从其读出bytes字节的数值
    38 ** 参数:3个,寄存器地址,数据、长度
    39 *********************************************************************
    40 */
    41 void SPI_Read_Buf ( unsigned char reg, unsigned char *pBuf, unsigned char uchars )
    42 {
    43 alt_avalon_spi_command ( SPI_BASE,0,1,&reg,0,NULL,1 );
    44 alt_avalon_spi_command ( SPI_BASE,0,0,NULL,uchars,pBuf,0 );
    45 }

       

      跟单片机相比,是不是觉得看得清晰点呢。这就是这个函数的方便之处了。有一点要注意一下,这个函数的最后一个参数的作用,以前没注意,走过一段弯路,它的作用就是相当于上面单片机中的CSN信号,如果需要对SPI从器件进行连续访问,则不释放该信号可以提高访问速度。

      好了,差不多就总结到这里了。无线模块的应用很广泛,可以用作无线遥控器、数据的无线烧写等用途。有想法的可以一起讨论。


     

  • 相关阅读:
    HTTP处理程序介绍
    c# Enum获取name,value和description
    如何成为优秀的软件人才
    关于系统设计分层
    从DLL中加载启动窗体
    摩斯密码
    休息下
    关于博文转载
    整合TextBox与Label 创建新控件EFLabelText
    ProC连接Oracle
  • 原文地址:https://www.cnblogs.com/kongtiao/p/2137286.html
Copyright © 2020-2023  润新知