《NRF905参考程序参考教程包含多个实例和解释.docx》由会员分享,可在线阅读,更多相关《NRF905参考程序参考教程包含多个实例和解释.docx(36页珍藏版)》请在三一办公上搜索。
1、NRF905参考程序参考教程包含多个实例和解释NRF905程序参考教程。 本资料主要是将程序中各部分子程序的功能与NRF905的手册相关联,使得各位同仁在每个子程序为什么这么写都在手册中找到具体的体现,特别是寄存器配置。 内涵完整参考程序,是100%可用程序。 硬件的连接方法在参考程序之后,并介绍原理。 本人最后只是为了提供给大家一个入门的资料或是引子罢了,至于如何应用的巧妙那是后期编程巧妙的结果,目前只是给刚进门的人士一个可以快速理解和掌握的浅显易懂的教程罢了。本人疏忽之处还请见谅。 按照惯例先展示作品抛砖引玉: 四路AD采集+温度采集 彩屏显示信息数据GSM手机电话 最近做的,音频功放四路
2、电机控制大功率led控制,不解释。 以前做的,舵机和摄像头,不解释。 更早先的,VGA显示,不解释。 已发送为例子。 1.通过spi接口把寄存器相应的值写到905中 2.把要发的数据写到905中 3.把状态设置成发送 4.数据被发出 那么操作905就主要是前三步的问题,那么请带个这前三个问题深入理解下面的相关解释了。 无线通信模块的三个要素: Nrf905模式的配置 Nrf905通过寄存器配置 Nrf905需要spi通信配置寄存器 先看模式配置:程序加解释 PWR_UP 0 1 1 1 TRX_CE X 0 1 1 TX_EN X X 0 1 操作模式 断电和SPI编程 待机和SPI编程 射频
3、接收模式 射频发送模式 根据这个图表,我们发觉有四种模式。捡重点的说实现收发功能有两种模式。 这两种模式在程序段中的实现是: 设置成接受模式,程序中没写PWR_UP,如果他是低电平就变成断电,所以个程序段默认PWR_UP为高电平。 void SetRxMode(void) TXEN=0; TRX_CE=1; Delay(1); / delay for mode change(=650us) 设置发送模式,这里会有疑问,在于 TRX_CE=0;。这里给出的解释是,如果我们直接写 TRX_CE=1;这样模块立即将其内部所写好的数据发送出去。 而对于编程的人员来说编出的程序五花八门,就比如说这条,改
4、程序员的意图并不想让设置发送模式时,数据就被立即发出,所以写了 TRX_CE=0;。如果看后面的完整程序,你会发现在发送时,有 TRX_CE=1;这一步。所以说,刚才那个图表没有问题。这里可以认为是准备发送模式,而不是发送模式,一旦 TRX_CE=1;那么数据立即被发送。 void SetTxMode(void) TRX_CE=0; TXEN=1; Delay(1); / delay for mode change(=650us) 关于图表中前两种模式中,实例程序所应用的是第二种,即待机spi编程模式。不管应用两种的哪一种,都是为了spi编程。 那么给出这个模式的应用程序段: 有这么做引脚赋予
5、各种电平先不用管他,我们看到 PWR=1;TRX_CE=0;TXEN=0;这三个,在待机spi模式中TXEN=x;即可以为任何值。说明现在是待机且spi编程模式。 程序段中其他引脚功能罗列下: Csn:spi的有效与否的引脚,低电平有效。如果只是单纯的设置模式,该引脚并没用处,只是后期程序的编写,所以做下配置。 Sck:spi的时钟,现在只是设置模式,还没开始spi通信,所以付个低电平。 DR:数据是否准备好,现在没有什么可准备的。 AD,CD也是一样,等到spi通信的时候才需关系。这里做个引子吧。 void nRF905Init(void) CSN=1; / Spi disable SCK=
6、0; / Spi clock line init low DR=0; / Init DR for input AM=0; / Init AM for input CD=0; / Init CD for input PWR=1; / nRF905 power on TRX_CE=0; / Set nRF905 in standby mode TXEN=0; / set radio in Rx mode Nrf905寄存器的配置 配置905寄存器的意思是,通过spi传输一个值,放入905的寄存器中,这个值可以让905传输数据时,产生各种你想要的效果,类似于你用手调节耳机音量,你的手就相当于配置耳机
7、的寄存器。 那么我先给出主要需配置的寄存器然后再解释. 如下面这个程序段: unsigned char idata RFConf11= 0x00, /配置命令/ 0x4c, /CH_NO,配置频段在430MHZ 0x0c, /输出功率为10db,不重发,节电为正常模式 0x44, /地址宽度设置,为4字节 0x04,0x04, /接收发送有效数据长度为32字节 0xCC,0xCC,0xCC,0xCC, /接收地址 0x58, /CRC充许,8位CRC校验,外部时钟信号不使能,16M晶振 ; 0x00, /配置命令/ 后面的讲解中会说,所以大家从第二个开始看。 CH_NO的意思如下,通过以下解释
8、设置不同的值,可以让905工作在不同频段,这个需要的话再做详解,不需要,可以照搬默认值,或者程序。 CH_NO 9 和 HFREQ_PLL一起进行平率设置 (默认值= 001101100b = 108d). fRF = ( 422.4 + CH_NOd /10)*(1+HFREQ_PLLd) MHz 于是乎相关的就引出以下这个寄存器 HFREQ_ PLL 1 使 PLL工作于 433 或 868/915 MHz 模式 (默认值 = 0). 0 工作于 433MHz 频段 1 工作于868 or 915 MHz频段 在这里给出个表格,如需更改该值可以参照: 工作频率 430.0 MHz 433.
9、1 MHz 433.2 MHz 434.7 MHz 862.0 MHz 868.2 MHz 868.4 MHz HFREQ_PLL 0 0 0 0 1 1 1 CH_NO 001001100 001101011 001101100 001111011 001010110 001110101 001110110 869.8 MHz 902.2 MHz 902.4 MHz 927.8 MHz 1 1 1 1 001111101 100011111 100100000 110011111 0x0c, /输出功率为10db,不重发,节电为正常模式 这里做下说明:我们拆分看看这段话。 输出功率为10db
10、 不重发 节电为正常模式 输出功率为10db,这个对于的寄存器是: 如下表,二进制10db应该是 11 PA_PWR 2 输出功率(默认值 = 00). “00”10dBm “01”2dBm “10”+6dBm “11” +10dBm 不重发,针对的寄存器是: 不管怎么说,部分都不自动重发,故 二进制是 0 AUTO_ RETRAN 1 如果TRX_CE 和 TXEN为高时,自动重发 (默认值 = 0). 0 不重发 1 数据包重发 节电为正常模式,针对的寄存器是: 如下表,要是正常模式则二进制是 0 RX_RED_ PWR 1 接收方式节能,工作电流1.6mA.灵敏度降低 (默认值 = 0)
11、. 0 正常工作 1 节能模式 那么如下结论: 输出功率为10db-11 不重发-0 节电为正常模式-0 按顺序写则是:1100-0000 11000x0C 0x44, /地址宽度设置,为4字节 如下面两个表: 收地址宽度:4字节的2进制是 100 RX_AFW 3 接收地址宽度 (默认值 = 100). 001 1 byte RX 地址 100 4 byte RX 地址 发地址宽度:4字节的2进制是 100 TX_AFW 3 发送地址宽度(default = 100) . 001 1 byte TX 地址 100 4 byte TX地址 于是乎: 100并上100, 可认为是0100并上01
12、00,可认为是4并上4,则可认为是0x44. 0x04,0x04, /接收发送有效数据长度为32字节 这条命令是我擅自更改的,更改前是2字节,如是0x04这是32字节。这样可以使905在一个数据包内传输更多信息。 那么我给出两个寄存器。 RX_PW 6 接收数据宽度(默认 = 100000). 000001 1 byte 接收数据宽度 000010 2 TX_PW 6 byte 接收数据宽度 100000 32 byte 接收数据宽度 发送数据宽度(默认 = 100000). 000001 1 byte 发送数据宽度 000010 2 byte 发送数据宽度. 100000 32 byte 发
13、送数据宽度 这里要把码补全,10 00000010 00000x40这里实际是0x40一点没错但是程序中写的是0x04,仔细想想,也没什么特别的问题。这里我水平有限,不做说明了。 0xCC,0xCC,0xCC,0xCC, /接收地址 一看就知道,地址被从新改了下,默认地址是E7这种。 RX_ ADDRESS 32 发送地址标识,使用字节取决于 RX_AFW (默认值 = E7E7E7E7h). 0x58, /CRC充许,8位CRC校验,外部时钟信号不使能,16M晶振 CRC_EN 1 CRC 校验可用 (默认值 = 1). CRC_ MODE 1 0 不可用 1 可用 CRC 模式选择端 (默
14、认值 = 1). 0 8 位 1 16 位 UP_CLK_ EN 1 输出时钟可用 (默认值 = 1) 0 外面没有可用的时钟信号 1 外面有可用的时钟信号 晶振频率端,必须与外部的晶振频率相对应(默认值 = 100). 000 4MHz 001 8MHz 010 12MHz 011 16MHz 100 20MHz XOF 3 这块看着有点乱的话,请继续往后看。 我们既然把相关寄存器的配置解释了一边,但是如果对于一个编程序的人,或者程序开发来说,这样的罗列虽然我们能弄懂每个寄存器是咋回事,但是实际编程并自己配置寄存器的话,难度是很大的。幸好,开发手册解决一切问题,下面是一个表,表的后面我有解释
15、。 字节 0 1 寄存器内容 射频器配置寄存器 位内容7:0,最高有效初始值 位7 CH_NO7:0 0110_1100 bit7:6 not used, 0000_0000 AUTO_RETRAN, RX_RED_PWR, 2 3 4 5 6 7 8 9 PA_PWR1:0, HFREQ_PLL, CH_NO8 bit7 not used, 0100_0100 TX_AFW2:0 , bit3 not used, RX_AFW2:0 bit7:6 not used, 0010_0000 RX_PW5:0 bit7:6 not used, 0010_0000 TX_PW5:0 RX_ADDRE
16、SS (device E7 identity) byte 0 RX_ADDRESS (device E7 identity) byte 1 RX_ADDRESS (device E7 identity) byte 2 RX_ADDRESS (device E7 identity) byte 3 CRC_MODE,CRC_EN, 1110_0111 XOF2:0, UP_CLK_EN, UP_CLK_FREQ1:0 解释:这是手册中的一张表,假设寄存器的配置值是如图给的这些。那么他的传输是从0字节开始到9字节截止,按顺序把16进制码传进去,你的工作就完成了。 而你需要对那个寄存器进行微小的改动,
17、只需找到手册相关寄存器的说明进行改动就可以了。我们从上表中摘出一个小表看,小表如下: bit7 not used, 0100_0100 TX_AFW2:0 , bit3 not used, RX_AFW2:0 bit7就是该值得第七位,第七位没用上。 TXAFW意思是有三位被这个寄存器用了。 等等。 通过这种字节的划分,将寄存器的配置变成了传多个2位十六进制数,使得寄存器的配置变得博大精深,新手上手困难。 不过对于驱动其他芯片也一样,配置寄存器就是这样配置的。像是某些器件如saa7113等芯片,配置寄存器时,前面还有地址,弄得更加复杂。 所以大家要通过学习nrf905了解芯片的驱动方法这才是关
18、键。 spi通信: 如何实现spi通信,在这个问题上,如果说正常学习应该是,先知晓spi的协议,spi的时序,spi写和读的时序和协议。但是如果将其看成程序的话就比较方便。咱们用程序谈这件事情。 该程序段式spi的写程序: 从MOSI=(bit)(b & 0x80);我们分析下。假设b=abcdefgh 那b & 0x80就是abcdefgh&10000000可以想象a被提取了出来。 至于bit,其实可以没有,这里可以参考c51语言关于 与 有两个做法,一个是&,另一个是&,即位与和整个值得与。 之后我们观察sck的变化,sck是spi的时钟,我们发现从0到1然后回到0,这是sck的变化。在s
19、ck变到1之前,mosi已经有了一个值,那么当sck=1;的时候,也就是所谓的上升沿,mosi被写入,这里可以认为是写入905的内部了。 b=1;这个语句等同于b=b1;意思是b左移一位的新值付给b, 比如b=abcdefgh左移一位,那么b=bcdefgh0,再左移一位,那么b=cdefgh00,以此类推左移8次之后b=00000000. 因为每移出一位,就代表着移进一位,移进的是0。 那么观察一下0x801000 0000 这个数1000 0000与上abcdefgh之后会提取出a 左移之后,再次进行与运算,就会提取出b,循环往复abcdefgh就都提取出来了。并在每一次都把这个值付给了m
20、osi。 我想通过梳理,大家应该能看懂了,至于spi的通信协议,大家可以参考下网上资料,我想看懂了程序,再看看资料应该能彻底明白了。 void SpiWrite(uchar b) uchar i=8; while(i-) Delay(10); SCK=0; MOSI=(bit)(b & 0x80); b=1; Delay(10); SCK=1; Delay(10); SCK=0; 读Spi程序段: _nop_;_nop_;是延时,延时是一个指令周期的时间。说白了,就是延时一段时间。 uchar ddata=0;这条语句意味着ddata是完全为0的。且注意一个事情ddata的定义是uchar,那
21、么他的值最大能到0xff。 ddata|=MISO;等同于ddata=ddata|miso; miso是一个引脚的电平。经过这条语句后ddata的最低位就是miso当时的电平了,这时左移再次提取新的miso电平,当八个电平都提取之后,ddata的值就提取完成了。 同样sck是spi时钟,想提取下一个miso的值必须让时钟波动一次。uchar SpiRead(void) uchar i=8; uchar ddata=0; while(i-) ddata=1; SCK=0; _nop_;_nop_; ddata|=MISO; SCK=1; _nop_;_nop_; SCK=0; return dd
22、ata; 到此为止,905的基本问题讲完了。那么我们把它串联在一起。 我们先宏观的看下。以发送流程为例 905现处于待机spi编程状态向905中传送寄存器值 将905的状态设置为发送状态成功发出-待机状态 如果想再发个数据,那么他的流程将变成 将905的状态设置为发送状态成功发出待机状态 由此看来对同一个对象进行发送,如果大家的设置都没改的话,寄存器的值只需设置一次。剩下的就是重复发送到待机这个环节了。 那么在之前的问题,大家都了解了,剩下的就是到把数据发送这一块了。这里我分为三个部分说这件事情。 向905传输一个命令 向905装入待发送的数据、 把数据发出去 向905传输命令: 这里定义了这
23、些命令,先在语法上说下#define WC 意思等价于 wc=0x00; 那么我们先解释下这几个命令,大家理解下。 #define WC #define RC 0x00 0x10 0x20 0x21 0x00的#define WTP #define RTP #define WTA #define RTA 0x22 0x23 0x24 #define RRP 如下是wc的解释: 在讲解配置寄存器是有个值我没有讲解先在我告诉大家 指令名称 W_CONFIG (WC) 指令格式 操作 写配置寄存器 AAAA 指明哪个字节 。 写操作从哪个字节开始取决于地址AAAA 0000 AAAA unsigne
24、d char idata RFConf11= 0x00, /配置命令/ 。 0x00, /配置命令/ 这就意味着wc=0x00,意思就是从0字节开始进行写操作。举个例子看下,0字节代表的是哪个寄存器, 字节 0 寄存器内容 射频器配置寄存器 位内容7:0,最高有效初始值 位7 CH_NO7:0 0110_1100 写之后的操作就是开始写1字节,这点上没什么问题最后写到第九字节。前面有这个 表的完整版,大家可以翻着看。 #define RC 0x10的解释如下:这个看表就知道不做解释了跟wc意思差不多。 R_CONFIG (RC) 0001 AAAA 读配置寄存器 AAAA 指明哪个字节 。 读
25、操作从哪个字节开始取决于地址AAAA 后几个命令都好理解我就都列出来大家自己吸收下: 那么到此,命令部分就说完了,能用的就这几个命令。 向905装入待发送数据: 向905装入数据这件事情和刚才的命令结合着说。先看以下程序段: SpiWrite(WTP); for (i=0;i4;i+) SpiWrite(TxRxBufi); / Write 32 bytes Tx data / Write payload command 先传了wtp命令,之后把txrxbuf数组中的前4字节传了进去。 那么执行外这条之后,数据就被传到905中了。观察此函数,发现调用了spiwrite这个函数,说明装入命令和装
26、入数据都是通过spi通信进行传输的。 观察以下程序段: SpiWrite(WTA); / Write address command / Write 4 bytes address for (i=0;i4;i+) SpiWrite(TxAddressi); TxAddressi和 SpiWrite(WTA); 是其中的要素,wta是写地址命令,那么TxAddressi就是地址咯,地址前面说过了,程序中除了那个config数组中有地址的说法,另外在程序段中被单独列出的地方是code TxAddress4=0xcc,0xcc,0xcc,0xcc; 那么通过这个程序段,功能就是把地址写进去。 以上的
27、两个程序段完成了数据和地址的写入,这时只要设置成发送状态,数据就可以被发出了。那么从现在开始是重点部分,就是以上两段程序的组合,并加以延伸。 仔细看下面这段程序,主要关注红字部分。程序之后有我的相关解释。 void TxPacket(uchar *TxRxBuf) uchar i; /Config905; CSN=0; SpiWrite(WTP); for (i=0;i4;i+) SpiWrite(TxRxBufi); / Write 32 bytes Tx data / Write payload command / Spi enable for write a spi command CS
28、N=1; Delay(1); CSN=0; / Spi disable / Spi enable for write a spi command SpiWrite(WTA); / Write address command / Write 4 bytes address for (i=0;i4;i+) CSN=1; SpiWrite(TxAddressi); / Spi disable / Set TRX_CE high,start Tx data TRX_CE=1; transmission 上面这段程序,如果被main调用了之后,出现的效果是将txrxbuf数组中的数无线传输出去。也就是说
29、这段程序看懂了,905就能发数了。我们看下,这个程序有两个大段的红色字体和两个小段。 大段的红色字体前面已经做过解释,小段的红色字体解释为。 在开头的几页提到过这个函数:void SetTxMode(void)其中有两个Delay(1); / while (DR!=1); / Set TRX_CE low TRX_CE=0; 引脚的电平为:TRX_CE=0;TXEN=1;,且开头我给我一个表格,那么我把它再拿出来说下。 PWR_UP 0 1 1 1 TRX_CE X 0 1 1 TX_EN X X 0 1 操作模式 断电和SPI编程 待机和SPI编程 射频接收模式 射频发送模式 如此图发射模式
30、,TRX_CE和TX_EN全为1是发送状态但是在void SetTxMode(void)中TRX_CE=0;所以他不属于发送也不属于接收状态,但只要TRX_CE=1;也就是全为1,那么就实现了发送状态,数据就被发送了。而在程序void TxPacket(uchar *TxRxBuf)就是上面那段大程序,中恰好两个小段红字阐明。所以小红字加上void SetTxMode(void)的完整功能就是将905从准备发射到发射的流程。 至于在大段程序中CSN的值,注释中已经写了spi是否有效的管脚。所以理解起来很容易,我就不解释了。 综上所述,如果把我上面所讲的所有东西全融合在一起,那么就是响当当的90
31、5发送数据的程序了。 而对于905接收程序而言也是一样,先要有相关命令,相关地址,并且用spi总线传输相关命令,并且用spi读取接收到的数据。这里我就不在多说什么了,因为跟发送的区别不大,下面我给出程序: 这个函数不像刚才的那个函数含有状态的设置,我后面会说的。那么程序的其他部分自己琢磨下。 void RxPacket(void) uchar i; Delay(1); / TRX_CE=0; Delay(100); / Set nRF905 in standby mode TRX_CE=0; CSN=0; / Spi enable for write a spi command Delay(1
32、); SpiWrite(RRP); for (i = 0 ;i 4 ;i+) TxRxBufi=SpiRead; / Read data and save to buffer CSN=1; Delay(10); 那么这样就完成了发送和接收,但是我们目前面临着一个很有意义的问题,比如我现在发了一个数据,那么无线电波在空气中传播的速度是光速,我们可以认为对面的905瞬间得到了一个值,于是我们就将它读取。 但是当我们发送数据时,对面的905正由于mcu正在干别的事,或者用更通俗的话说,不是任何时候都能用接收函数接到数据,只有当数据被发送且传到另一个905上,另一个905接收到了,TRX_CE=1;
33、这时我们读接收到的数据才能读出来。 幸好存在一个功能,当接收到数据后,905的DR会产生电平变化。所以当dr产生变化的时候我们就开始提取数据,其他时候我们可以干别的。 所以就有以下这个函数: void RX(void) SetRxMode; / Set nRF905 in Rx mode while (CheckDR=0); 我们看见三个子函数在其中,有两个我已经说过了,但是我还想再说一下,首先你想让905接收数据,那么必须先让其处于接收状态。 当接收到数据,那么dr会变化,当确实变化了我们才执行RxPacket;这个函数提取数据。 这个与发送函数的一个很大的区别在与状态设置,发送状态是一 D
34、elay(10); RxPacket; 个很果断的状态,就像你想什么时候说话都可以,别人是否能听得见都可以说话。 但是接收状态就像你的耳朵一样,你不知道何时对方会说话,那么你就必须长期处于接收状态,否则你就听不到说话,除非你知道他什么时候会说话,听不到说话的含义就是数据丢失。 长期处于接收状态,并不是说单片机不能干别的事,而是说905不能干别的事。原因是单片机可以用中断的方式得到DR引脚的变化。 而这里为了程序的简单,我们直接用while等待也没什么问题,特此注明。如有中断需要,请自己改动。 那么如何使用这些函数,也就是宏观上程序如何控制905的,我们看下主函数: void main(void
35、) nRF905Init; Config905; while(1) if(KEY0 =0)TxRxBuf0=0x11; if(KEY0 =1)TxRxBuf0=0x22; if(KEY1 =0)TxRxBuf1=0x11; if(KEY1 =1)TxRxBuf1=0x22; SetTxMode;/ Set nRF905 in Tx mode TxPacket(TxRxBuf);/ Send data by nRF905 前面的两个红字子函数是905的初始化和寄存器配置。 而在while内部有两个红字子函数,这两个函数就实现了数据的发送。而while中的if语句中key是按键,当key的按下与不
36、按都会使TXRXBUF数组的前两个字节产生一组值,这里你可以认为咱们向TXRXBUF中写入了数据。 那么这样,并在外面加了while语句就实现了循环发送按键值的功能。 那么我们再看下接收端的主函数是什么样子。接收部分只有三个子函数,且我都讲过了相关功能,这里就不赘述了。 void main(void) nRF905Init; Config905; while(1) RX; if(TxRxBuf0=0x22)in1=1;in2=1; if(TxRxBuf0=0x33)in1=1;in2=0; if(TxRxBuf1=0x22)in3=1;in4=1; if(TxRxBuf1=0x33)in3=1
37、;in4=0; 上面所讲的一切就能实现905的收发了,那我把示例给大家。 其中子函数在收发程序中都是编写的一模一样所以我就给出一份接收函数,如想得到接收函数,把刚才讲的发送主函数粘上就成了。 #include #include #include #include /- #define uint unsigned int #define uchar unsigned char /- #define BYTE_BIT0 0x01 #define BYTE_BIT1 0x02 #define BYTE_BIT2 0x04 #define BYTE_BIT3 0x08 #define BYTE_BIT
38、4 0x10 #define BYTE_BIT5 0x20 #define BYTE_BIT6 0x40 #define BYTE_BIT7 0x80 /- bdata unsigned char DATA_BUF; #define DATA7 (DATA_BUF&BYTE_BIT7) != 0) #define DATA0 (DATA_BUF&BYTE_BIT0) != 0) sbit flag =DATA_BUF7; sbit flag1 =DATA_BUF0; /-发送数据缓冲区- #define TxRxBuf_Len 4 unsigned char TxRxBufTxRxBuf_Le
39、n= 0x29,0x30,0x31,0x32, ; /-NRF905控制IO- sbit TXEN=P10; sbit TRX_CE=P32; sbit PWR=P11; /-NRF905 SPI接口- sbit MISO=P16; sbit MOSI=P15; sbit SCK=P17; sbit CSN=P13; /-nrf905状态标志- sbit AM=P14; sbit DR=P33; sbit CD=P12; /- sbit in4=P34; sbit in3=P35; sbit in2=P36; sbit in1=P37; /- /-nrf905控制指令- #define WC #define RC 0x00 0x10 0x20 0x21 0x22 0x23 0x24 #define WTP #define RTP #define WTA #def