STM8 nRF24L01程序

发布者:平静宁静最新更新时间:2016-09-06 来源: eefocus关键字:STM8  nRF24L01 手机看文章 扫描二维码
随时随地手机看文章
花了我断续的一天多时间,终于把nRF24L01从C51中移植到STM8上,主要是注意时序和I/O口的配置。
STM8 nRF24L01程序 - 小文 - 小文电子设计

 

STM8 nRF24L01程序 - 小文 - 小文电子设计

 

STM8 nRF24L01程序 - 小文 - 小文电子设计

 源程序:

/************ STM8S103F学习 ***********************/
/*芯片型号:STM8S103F3PB6                         */
/*功能描述:nrf24l01无线模块应用                  */
/*作者:    ClimberWin                            */
/*编写时间:2011.9.25                             */
/*内部晶振                                        */
/**************************************************/
//修改时间2011.9.27
//程序从C51到STM8移植成功
#include

#define   uint    unsigned int
#define   uchar    unsigned char

//****************************************IO端口定义***************************************
//****************************************NRF24L01端口定义*****************************

/////////////引脚定义//////////////

#define         CE         PD_ODR_ODR3 
#define         CSN       PD_ODR_ODR2 
#define         SCK       PC_ODR_ODR7
#define         MOSI     PC_ODR_ODR6
#define         MISO     PC_IDR_IDR5  //输入
#define         IRQ        PD_IDR_IDR4    //输入

/***************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 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 IO_config(void);

void delayms(unsigned int count);


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<450;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
    {
  if((byte & 0x80)==0)         

                {MOSI=0;}
                else
                {MOSI=1;}
                
  byte = (byte << 1);           // shift next bit into MSB..
  SCK = 1;                      // Set SCK high..
                if(MISO == 0)
  {byte |= 0;}           // capture current MISO bit
                else
                {byte |= 1;} 
  SCK = 0;                // ..then set SCK low again
    }
    return(byte);               // return read uchar
}

 /*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)
{
 uint 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 IO_config(void)
{
   PC_DDR_DDR5=0;
   PC_DDR_DDR6=1;
   PC_DDR_DDR7=1;
   
   PD_DDR_DDR2=1;
   PD_DDR_DDR3=1;
   PD_DDR_DDR4=0;
   
   
   PC_CR1_C15=1;//输入上拉
   PC_CR1_C16=1;
   PC_CR1_C17=1;
   
   PD_CR1_C12=1;
   PD_CR1_C13=1;
   PD_CR1_C14=0;//输入上拉

   PC_CR2 = 0x00; 
   PD_CR2 = 0x00; 
    
}
/***********************************/


/*****************主程序********************/
void main(void)
{
        unsigned char TxBuf[20]={0};  // 
 unsigned char RxBuf[20]={0};
 unsigned char led_num; 
       // unsigned char rx_temp; 
  
      delayms(100);
     IO_config(); 
      
       

    init_NRF24L01() ;
   TxBuf[1] =0x55 ;
 nRF24L01_TxPacket(TxBuf); // Transmit Tx buffer data
 delayms(1000);
 led_num=0x00;
 while(1)
 {
     
  TxBuf[1] =led_num ;
  nRF24L01_TxPacket(TxBuf); // Transmit Tx buffer data   
  led_num++;
  delayms(500);       
//***********************************************************************************************
  SetRX_Mode();       // 每次开始发送 然后无按键直接跳转到接收 一直循环。
  nRF24L01_RxPacket(RxBuf);
  if(RX_DR==0)
                {
                 // rx_temp=0x55;
                  delayms(10);
                }
                else
                {
                   delayms(10);
                }
  
  
 }
     
}

关键字:STM8  nRF24L01 引用地址:STM8 nRF24L01程序

上一篇:STM8S103F2 NOKIA5110+AD采集
下一篇:STM8S103F---Nokia5110液晶显示

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

stm8 rtc时钟
stm8 rtc时钟可以使用内部低频时钟源,或者外部低速32768Hz时钟源,关于rtc部分代码如下: #include stdio.h #include stm8l15x.h void init_rtc(void) { RTC_InitTypeDef RTC_InitStr; RTC_TimeTypeDef RTC_TimeStr; RTC_DateTypeDef RTC_DateStr; #if 0 /* LSI */ CLK_RTCClockConfig(CLK_RTCCLKSource_LSI, CLK_RTCCLKDiv_1); CLK_
[单片机]
STM8 UART 接收器
STM8 UART 接受器 UART可以接收8位或9位的数据字。如果M位置1,字长为9位,其中MSB存放在寄存器UART_CR1的R8位。 字符接收 在UART接收期间,数据的最低有效位首先从RX脚移进。在此模式里,UART_DR寄存器有一个缓冲器(TDR),位于内部总线和接收移位寄存器之间。 配置步骤: 1.编程UART_CR1的M位来定义字长。 2.在UART_CR3中编程停止位的位数。 3.按下列顺序编写波特率寄存器选择要求的波特率。 a) UART_BRR2 b) UART_BRR1 4.将UART_CR2的REN置1。这将激活接收器,使它开始寻找起始位。 当一字符被接收到时 RXNE位被置位。它表明移位寄存的
[单片机]
<font color='red'>STM8</font> UART 接收器
STM8 中断屏蔽和处理流程
STM8 中断屏蔽和处理流程 中断屏蔽是通过CC寄存器的位I1和位I0以及设置每个中断向量(表13)的软件优先级的ITC_SPRx来管理的。处理流程如图17所示: (表13:软件优先级) (图17:中断处理流程图) 当一个中断请求必须被响应时: 1.在当前正在执行指令结束之后,正常的操作被悬起; 2.PC,X,Y,A和CC寄存器被自动压栈; 3.根据ITC_SPRx寄存器中的值对应的中断服务向量,CC寄存器中的位I1和I0被相应设置; 4.通过中断向量载入中断服务子程序的入口地址,接着对中断服务子程序的第一条指令取址(参考表16中断映射表来了解向量地址的更详细情况)。 中断服务子程序必须以IRET指令结束,该指令会把堆栈中
[单片机]
<font color='red'>STM8</font> 中断屏蔽和处理流程
关于一款可以在STM8上运行的实时操作系统Atomthreads
前面有朋友大概了问了这样一个问题:我想在STM8单片机上,不跑操作系统,在需要处理一些业务逻辑时,还需要检测按键短按、长按、多个按键同时按下,我要如何实现? 不跑操作系统,这个问题其实是一个“很难”的问题,在单线程(裸机)情况下,处理多件事情,这个逻辑处理必须要严谨才行。 但是,有了操作系统,这个问题就不是问题了。 1写在前面 在不跑操作系统的情况,我曾经也用STM8单线程(逻辑)处理过相对复杂的业务逻辑:检测键盘(短按、长按)、控制显示、检测信号、串口收发通信等...... 虽然那些复杂的业务逻辑功能实现了,但开发的时间和周期,以及后期更新、维护的成本可想而知。 所以,本文针对资源较小的STM8,推荐大家
[单片机]
关于一款可以在<font color='red'>STM8</font>上运行的实时操作系统Atomthreads
stm8 time1精准延时
//初始化位毫秒单位 void Timer_Init() { TIM1_PSCRH = (unsigned char) ( ( (F_CPU / 1000) - 1) 8); /* 16M系统时钟经预分频f=fck/(PSCR+1) */ TIM1_PSCRL = (unsigned char) ( (F_CPU / 1000) - 1); /* PSCR=0x3E7F,f=16M/(0x3E7F+1)=1000Hz,每个计数周期1ms */ TIM1_ARRH = 0; /* 这三行不知道为什么要先初始化,如果不定时第一次用时将不正常 */ TIM1_ARRL = 1;
[单片机]
STM8 SPI的学习
#define SPI_CS_PORT GPIOC #define SPI_CS_PIN GPIO_PIN_4 #define SPI_CLK_PORT GPIOC #define SPI_CLK_PIN GPIO_PIN_5 #define SPI_MOSI_PORT GPIOC #define SPI_MOSI_PIN GPIO_PIN_6 #define SPI_MISO_PORT GPIOC #define SPI_MISO_PIN GPIO_PIN_7 SPI是串行设备接口(Serial Peripheral Interface)的英文缩写,是一种单主多从式的全双工同步串行通信协议。 SCK:同步时钟信号线
[单片机]
SWIM接口及STM8烧录过程
1. 硬件连接 SWIM接口只需要一根传输线,即可完成双向的传输。传输过程,都是由主控制端(host)发起,设备端然后做出反应。host端需要在一个总线上实现读和写,那就是说必须要同时接一个输出IO和一个输入IO(也可以用同一个,开漏的情况下,IO可以同时读写),且输出IO必须设计成开漏,总线上挂接上拉电阻,以便随时交出控制权。硬件连接如下: 上图SWIM_IN为输入IO,SWIM为输出IO。 2. 编码格式 SWIM协议使用归零码,且支持两种速率。 高速模式 低速模式 这两种模式都是使用8M的clock同步(上图中的箭头)。高速模式下,2个低电平加8个高电平表示1;8个低电平加2个高电平表示0。低
[单片机]
SWIM接口及<font color='red'>STM8</font>烧录过程
STM8S学习笔记之三(STM8 SysClk)
STM8S系统时钟设置,对于单片机来说是非常重要的,不同的用处必须应用不同的时钟。。举个例子,做AVR时在高稳定的串口通讯时用的时钟一般是3.6864M,主要是这个算波特率精确。。STM8S同样重要。。 STM8S时钟源: ●1-24MHz高速外部晶体振荡器(HSE) ●最大24MHz高速外部时钟信号(HSE user-ext) ●16MHz高速内部RC振荡器(HSI) ●128KHz低速内部RC(LSI) 各个时钟源可单独打开或关闭,从而优化功耗。 对于我这么懒得人一般都是用的内部或者外部晶振。。这个芯片时钟方面很大的一个亮点就是时钟可以自由分频。在降低功耗方面,如果有特殊
[单片机]
小广播
添点儿料...
无论热点新闻、行业分析、技术干货……
热门活动
换一批
更多
设计资源 培训 开发板 精华推荐

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

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

更多每日新闻
电子工程世界版权所有 京B2-20211791 京ICP备10001474号-1 电信业务审批[2006]字第258号函 京公网安备 11010802033920号 Copyright © 2005-2024 EEWORLD.com.cn, Inc. All rights reserved