STM32 驱动无线NRF24L01 完成串口数据传输

发布者:luanzgc最新更新时间:2016-10-05 来源: eefocus关键字:STM32  驱动无线NRF24L01  串口数据传输 手机看文章 扫描二维码
随时随地手机看文章
2401 一个简单的SPI 接口的 2.4G 射频模块 淘宝价20¥,DIY 的17¥ ,算是廉价。
STM32 驱动NRF24L01 完成串口数据传输 - java - stm32学习日志
接口CMOS电平3.3V STM32 可直接连接。接受完成 发送完成 出错 都有IRQ 低电平中断产生。程序中 我将其连接至一IO口在外部中断中处里各类事件 但也发现这种处理方式并不是特别灵活,或许直接判断更加灵活。
NRF20L01一次可以传输 1~32个字节比较灵活。最初我是根据字符串长来不停的转换每次传输的长度,这样做十分麻烦最后用截取有效串长的方法实现效果很好。
 
程序修修改过 总算稳定了 不过在传输大于32个字节的信息时出错的概率很大,原因暂时不清楚,不过能自动恢复过来。另外在带有硬件的在线仿真调试的时候一定要运行前断开外部硬件的电源再重新连接,保证外部器件的正常初始化。
 
/***********************s****************************/
u8 tran=0; //中断标志
u8 sta;  //定义一个可位寻址的变量sta
uc8 TX_ADDRESS[TX_ADR_WIDTH] = {0x34,0x43,0x10,0x10,0x01};
char RX_BUF[256];
uchar TX_BUF[256];

/**************************************************/
void RF_SPI_Config(void)
{
 SPI_InitTypeDef  SPI_InitStructure;
    GPIO_InitTypeDef GPIO_InitStructure;
 EXTI_InitTypeDef EXTI_InitStructure;
 RCC_APB1PeriphClockCmd(RCC_APB1Periph_SPI2,ENABLE);
 RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB,ENABLE);
 RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOC,ENABLE);
 RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOD,ENABLE); 
 
 /* PB15-MOSI2,PB13-SCK2*/
    GPIO_InitStructure.GPIO_Pin = GPIO_Pin_13 |GPIO_Pin_14 | GPIO_Pin_15;
    GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP;
    GPIO_Init(GPIOB, &GPIO_InitStructure);
 //IRQ
 GPIO_SetBits(GPIOB, GPIO_Pin_0);
 GPIO_InitStructure.GPIO_Pin = GPIO_Pin_0;
    GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IPU;
    GPIO_Init(GPIOB, &GPIO_InitStructure);
 GPIO_EXTILineConfig(GPIO_PortSourceGPIOB, GPIO_PinSource0);
 /* 配置中断线0为下降触发*/ 
 EXTI_InitStructure.EXTI_Line = EXTI_Line0;
 EXTI_InitStructure.EXTI_Mode = EXTI_Mode_Interrupt;
 EXTI_InitStructure.EXTI_Trigger = EXTI_Trigger_Falling;
 EXTI_InitStructure.EXTI_LineCmd = ENABLE;
 EXTI_Init(&EXTI_InitStructure);
    /*PB2-CS*/
 GPIO_SetBits(GPIOB, GPIO_Pin_2);//预置为高
    GPIO_InitStructure.GPIO_Pin = GPIO_Pin_2;
    GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;
    GPIO_Init(GPIOB, &GPIO_InitStructure);
 /*PC4-A0*/
 GPIO_SetBits(GPIOC, GPIO_Pin_4);//预置为高
    GPIO_InitStructure.GPIO_Pin = GPIO_Pin_4;
    GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;
    GPIO_Init(GPIOC, &GPIO_InitStructure);
 /*LED*/
 GPIO_SetBits(GPIOB, GPIO_Pin_12);//预置为高
    GPIO_InitStructure.GPIO_Pin = GPIO_Pin_12;
    GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;
    GPIO_Init(GPIOB, &GPIO_InitStructure);
    /* SPI2 configuration */
 SPI_Cmd(SPI2, DISABLE);             //必须先禁能,才能改变MODE
    SPI_InitStructure.SPI_Direction = SPI_Direction_2Lines_FullDuplex;  //两线全双工
    SPI_InitStructure.SPI_Mode = SPI_Mode_Master;       //主
    SPI_InitStructure.SPI_DataSize = SPI_DataSize_8b;      //8位
    SPI_InitStructure.SPI_CPOL = SPI_CPOL_Low;        //CPOL=0 时钟悬空低
    SPI_InitStructure.SPI_CPHA = SPI_CPHA_1Edge;       //CPHA=0 数据捕获第1个
    SPI_InitStructure.SPI_NSS = SPI_NSS_Soft;        //软件NSS
    SPI_InitStructure.SPI_BaudRatePrescaler = SPI_BaudRatePrescaler_64 ; //64分频
    SPI_InitStructure.SPI_FirstBit = SPI_FirstBit_MSB;      //高位在前
    SPI_InitStructure.SPI_CRCPolynomial = 7;        //CRC7
    
 SPI_Init(SPI2, &SPI_InitStructure);
    SPI_Cmd(SPI2, ENABLE);
}
/**************************************************************
但切记不可忽略SPI的硬件接收,因为读SPI_DR才能清除RXEN
***************************************************************/
 u8 SPI_RW(u8 byte)
{
 /*等待发送寄存器空*/
 while((SPI2->SR & SPI_I2S_FLAG_TXE)==RESET);
    /*发送一个字节*/
 SPI2->DR = byte;
 /* 等待接收寄存器有效*/
 while((SPI2->SR & SPI_I2S_FLAG_RXNE)==RESET);
 return(SPI2->DR);
}
 /**************************************************
函数:SPI_RW_Reg()
描述:写数据value到reg寄存器
*************************************************/
u8 SPI_RW_Reg(u8 reg, u8 value)
{
 u8 status;
   CSN_L;                   // CSN置低,开始传输数据
   status = SPI_RW(reg);      // 选择寄存器,同时返回状态字
   SPI_RW(value);             // 然后写数据到该寄存器
   CSN_H;                   // CSN拉高,结束数据传输
   return(status);            // 返回状态寄存器
}
 /**************************************************
函数: init_io()
描述:初始化IO
*************************************************/
void RX_Mode(void);
void init_io(void)
{
 CE_L;        // 待机
 CSN_H;        // SPI禁止
 LED1;  // 关闭指示灯
 RX_Mode(); //接收
}
/**************************************************
函数:SPI_Read()
描述:从reg寄存器读一字节
*************************************************/
u8 SPI_Read(u8 reg)
{
 u8 reg_val;
   CSN_L;                    // CSN置低,开始传输数据
   SPI_RW(reg);                // 选择寄存器
   reg_val = SPI_RW(0);        // 然后从该寄存器读数据
   CSN_H;                    // CSN拉高,结束数据传输
   return(reg_val);            // 返回寄存器数据
}
/**************************************************
函数:SPI_Read_Buf()
描述:从reg寄存器读出bytes个字节,通常用来读取接收通道
   数据或接收/发送地址
*************************************************/
uchar SPI_Read_Buf(uchar reg, char * pBuf, uchar bytes)
{
 uchar status, i;
   CSN_L;                    // CSN置低,开始传输数据
   status = SPI_RW(reg);       // 选择寄存器,同时返回状态字
   for(i=0; i      pBuf[i] = SPI_RW(0);    // 逐个字节从nRF24L01读出
   CSN_H;                    // CSN拉高,结束数据传输
   return(status);             // 返回状态寄存器
}
 /**************************************************
函数:SPI_Write_Buf()
描述:把pBuf缓存中的数据写入到nRF24L01,通常用来写入发
   射通道数据或接收/发送地址
*************************************************/
uchar SPI_Write_Buf(uchar reg, uchar * pBuf, uchar bytes)
{
 uchar status, i;
   CSN_L;                    // CSN置低,开始传输数据
   status = SPI_RW(reg);       // 选择寄存器,同时返回状态字
   for(i=0; i      SPI_RW(pBuf[i]);        // 逐个字节写入nRF24L01
   CSN_H;                    // CSN拉高,结束数据传输
   return(status);             // 返回状态寄存器
}
/**************************************************
函数:RX_Mode()
描述:这个函数设置nRF24L01为接收模式,等待接收发送设备的数据包
*************************************************/
void RX_Mode(void)
{
 CE_L;
   SPI_Write_Buf(RF_WRITE_REG + RX_ADDR_P0, (u8*)TX_ADDRESS, TX_ADR_WIDTH);  // 接收设备接收通道0使用和发送设备相同的发送地址
   SPI_RW_Reg(RF_WRITE_REG + EN_AA, 0x01);               // 使能接收通道0自动应答
   SPI_RW_Reg(RF_WRITE_REG + EN_RXADDR, 0x01);           // 使能接收通道0
   SPI_RW_Reg(RF_WRITE_REG + RF_CH, 40);                 // 选择射频通道0x40
   SPI_RW_Reg(RF_WRITE_REG + RX_PW_P0, TX_PLOAD_WIDTH);  // 接收通道0选择和发送通道相同有效数据宽度
   SPI_RW_Reg(RF_WRITE_REG + RF_SETUP, 0x07);            // 数据传输率1Mbps,发射功率0dBm,低噪声放大器增益
   SPI_RW_Reg(RF_WRITE_REG + CONFIG, 0x0f);              // CRC使能,16位CRC校验,上电,接收模式
   CE_H;                                          // 拉高CE启动接收设备
}
 /**************************************************
函数:TX_Mode()
描述:
    这个函数设置nRF24L01为发送模式,(CE=1持续至少10us),
 130us后启动发射,数据发送结束后,发送模块自动转入接收
 模式等待应答信号。
*************************************************/
void TX_Mode(uchar * BUF)
{
 CE_L;
   SPI_Write_Buf(RF_WRITE_REG + TX_ADDR, (u8*)TX_ADDRESS, TX_ADR_WIDTH);     // 写入发送地址
   SPI_Write_Buf(RF_WRITE_REG + RX_ADDR_P0, (u8*)TX_ADDRESS, TX_ADR_WIDTH);  // 为了应答接收设备,接收通道0地址和发送地址相同
   SPI_Write_Buf(WR_TX_PLOAD, BUF, TX_PLOAD_WIDTH);                  // 写数据包到TX FIFO
   SPI_RW_Reg(RF_WRITE_REG + EN_AA, 0x01);       // 使能接收通道0自动应答
   SPI_RW_Reg(RF_WRITE_REG + EN_RXADDR, 0x01);   // 使能接收通道0
   SPI_RW_Reg(RF_WRITE_REG + SETUP_RETR, 0x0a);  // 自动重发延时等待250us+86us,自动重发10次
   SPI_RW_Reg(RF_WRITE_REG + RF_CH, 40);         // 选择射频通道0x40
   SPI_RW_Reg(RF_WRITE_REG + RF_SETUP, 0x07);    // 数据传输率1Mbps,发射功率0dBm,低噪声放大器增益
   SPI_RW_Reg(RF_WRITE_REG + CONFIG, 0x0e);      // CRC使能,16位CRC校验,上电
 CE_H;CE_H;delay_ms(1);
}
/**************************************************
函数:Check_ACK()
描述:
    检查接收设备有无接收到数据包,设定没有收到应答信
 号是否重发
***************************************************/
uchar Check_ACK(u8 clear)
{
 while(IRQ);
 sta = SPI_RW(NOP);                    // 返回状态寄存器
 if(MAX_RT)
  if(clear)                         // 是否清除TX FIFO,若没有清除在清除MAX_RT中断标志后重发
   SPI_RW(FLUSH_TX);
 SPI_RW_Reg(RF_WRITE_REG + STATUS, sta);  // 清除TX_DS或MAX_RT中断标志
 IRQ_H;
 if(TX_DS)
  return(0x00);
 else
  return(0xff);
}

 void sent_data(u8* fp,u16 flong)
{
 u16 i=65535;
 TX_Mode((u8*)&flong); //传送长度
 while(!tran&&i>1)i--; //等待完成
 tran=0;
 flong=flong/33+1;
 for(i=0;i<20000;i++);//130uS*2延时
 while(flong)
 {
  if(MAX_RT) return;//无应答返回
  TX_Mode(fp);   //传送数据
  while(!tran&&i>1)i--; //等待完成
  tran=0;
  for(i=0;i<20000;i++);//130uS*2延时
  fp+=32;flong--;
 }
}
extern u8 RX_NU;
void test (void)
{
 if (Uart2_Get_Flag!=0&&Timer2==0)
 {
   sent_data(TX_BUF,(u16)Uart2_Get_Flag);
   Uart2_Get_Flag=0; 
 }
 
 if(Timer2==0&&RX_NU==2)
 { 
  RX_NU=1;
  USART2_Puts("传输错误 ");
  USART2_Puts("\r\n");
 }  
}
 
两个中断 串口 和 外部中断
/*******************************************************************************
* Function Name  : EXTI0_IRQHandler
* Description    : This function handles External interrupt Line 0 request.
* Input          : None
* Output         : None
* Return         : None
*******************************************************************************/
 extern u8 sta;
 extern char RX_BUF[256];
 extern uchar TX_BUF[256];
 extern u8 SPI_RW_Reg(u8 reg, u8 value);
 extern void RX_Mode(void);
 extern uchar SPI_Read_Buf(uchar reg, char * pBuf, uchar bytes);
 u8 RX_NU=1;//1接收长度 2接收数据
 u16 rectnu,onerc; //接收串长,接收次数
 char* PRX_BUF=RX_BUF;
void EXTI0_IRQHandler(void)
{
    EXTI_ClearITPendingBit(EXTI_Line0);
 tran=1;
 CSN_L;
 sta=SPI_RW(NOP);     // 返回状态寄存器
 CSN_H;
                 
 if(MAX_RT) 
 {
 USART2_Puts("对方无应答  ");
 CSN_L;
 SPI_RW(FLUSH_TX);       // 清除TX FIFO,若没有清除在清除MAX_RT中断标志后重发
 CSN_H;
 SPI_RW_Reg(RF_WRITE_REG + STATUS, sta);
 }
                          
 if(TX_DS)
 {
   SPI_RW_Reg(RF_WRITE_REG + STATUS, sta);  // 清除TX_DS或MAX_RT中断标志
 }
    if(RX_DR)              // 判断是否接受到数据
 {  
 
   if(RX_NU==1)
   {
    CE_L;
    SPI_Read_Buf(RD_RX_PLOAD, RX_BUF, TX_PLOAD_WIDTH);  // 从RX FIFO读出数据     
       SPI_RW_Reg(RF_WRITE_REG + STATUS, sta);  // 清除RX_DS中断标志
    rectnu=RX_BUF[0];rectnu|=RX_BUF[1]<<8;   //接收串长
    onerc=rectnu/33+1;                   //计算接收次数         
             RX_NU=2;RX_Mode();Timer2=500;   /*超时时间*/
    return;
   }
   if(RX_NU==2)
   {  
    CE_L;
    SPI_Read_Buf(RD_RX_PLOAD, PRX_BUF, TX_PLOAD_WIDTH);  // 从RX FIFO读出数据  
       SPI_RW_Reg(RF_WRITE_REG + STATUS, sta);  // 清除RX_DS中断标志
    onerc--;PRX_BUF+=32;       //接收计数 接收指针移动
    if(!onerc)
    {
     RX_BUF[rectnu]='\0';    //截取有效串长
     USART2_Puts(RX_BUF);    //串口发送接收到的字符
     USART2_Puts("\r\n");   
     PRX_BUF=RX_BUF;   //恢复指针
     RX_NU=1;
    }
    RX_Mode();Timer2=200;   /*超时时间*/
    return;
   }
  }
  LED;
  RX_Mode();

}

 /*******************************************************************************
* Function Name  : USART2_IRQHandler
* Description    : This function handles USART2 global interrupt request.
* Input          : None
* Output         : None
* Return         : None
*******************************************************************************/
extern u16 Uart2_Get_Flag;
extern u8 Uart2_Get_Data;
extern u8 TX_BUF[256];
void USART2_IRQHandler(void)
{
 //接收中断
if(USART_GetITStatus(USART2,USART_IT_RXNE)==SET)
 {
  Timer2=500; //500MS后nfr2401发送
  USART_ClearITPendingBit(USART2,USART_IT_RXNE);
  TX_BUF[Uart2_Get_Flag]=USART_ReceiveData(USART2);
  Uart2_Get_Flag++;
  //USART2_Puts(TX_BUF); 
 }
 
 //溢出-如果发生溢出需要先读SR,再读DR寄存器 则可清除不断入中断的问题
 if(USART_GetFlagStatus(USART2,USART_FLAG_ORE)==SET)
 {
  USART_ClearFlag(USART2,USART_FLAG_ORE); //读SR
  USART_ReceiveData(USART2);    //读DR
 }
}

 

在 外部中断中 可以一次传输最少要两次 第一次RX_NU==1 用来接收此次传输的数据长度 用来截取有效串长,第二次RX_NU==2 就收数据放到数组里 若是数据长于32个字节就分多次传输。其中引入 超时机制 若是200/500 MS 后传输还没完成(RX_NU再次 等于1)就放弃接收数据 直接接收再次新的串长度。这个机制很重要但还有待完善。

串口中也一样500MS 还没新的数据进来就说明串口接收完成 之后就发送数据。Uart2_Get_Flag 当然就是串长了。

NRF24L01 可以参考其数据手册 都是中文好说好说~~这里就不提了。

#ifndef _API_DEF_
#define _API_DEF_

// Define interface to nRF24L01

//* Define SPI pins
/*sbit CE   = P1^0;  // Chip Enable pin signal (output)
sbit CSN  = P1^1;  // Slave Select pin, (output to CSN, nRF24L01)
sbit IRQ  = P1^3;  // Interrupt signal, from nRF24L01 (input)
sbit MISO = P1^4;  // Master In, Slave Out pin (input)
sbit MOSI = P1^5;  // Serial Clock pin, (output)
sbit SCK  = P1^7;  // Master Out, Slave In pin (output)
*/

// SPI(nRF24L01) commands
#define RF_READ_REG    0x00  // Define read command to register
#define RF_WRITE_REG   0x20  // Define write command to register
#define RD_RX_PLOAD 0x61  // Define RX payload register address
#define WR_TX_PLOAD 0xA0  // Define TX payload register address
#define FLUSH_TX    0xE1  // Define flush TX register command
#define FLUSH_RX    0xE2  // Define flush RX register command
#define REUSE_TX_PL 0xE3  // Define reuse TX payload register command
#define NOP         0xFF  // Define No Operation, might be used to read status register

// SPI(nRF24L01) registers(addresses)
#define CONFIG      0x00  // 'Config' register address
#define EN_AA       0x01  // 'Enable Auto Acknowledgment' register address
#define EN_RXADDR   0x02  // 'Enabled RX addresses' register address
#define SETUP_AW    0x03  // 'Setup address width' register address
#define SETUP_RETR  0x04  // 'Setup Auto. Retrans' register address
#define RF_CH       0x05  // 'RF channel' register address
#define RF_SETUP    0x06  // 'RF setup' register address
#define STATUS      0x07  // 'Status' register address
#define OBSERVE_TX  0x08  // 'Observe TX' register address
#define CD          0x09  // 'Carrier Detect' register address
#define RX_ADDR_P0  0x0A  // 'RX address pipe0' register address
#define RX_ADDR_P1  0x0B  // 'RX address pipe1' register address
#define RX_ADDR_P2  0x0C  // 'RX address pipe2' register address
#define RX_ADDR_P3  0x0D  // 'RX address pipe3' register address
#define RX_ADDR_P4  0x0E  // 'RX address pipe4' register address
#define RX_ADDR_P5  0x0F  // 'RX address pipe5' register address
#define TX_ADDR     0x10  // 'TX address' register address
#define RX_PW_P0    0x11  // 'RX payload width, pipe0' register address
#define RX_PW_P1    0x12  // 'RX payload width, pipe1' register address
#define RX_PW_P2    0x13  // 'RX payload width, pipe2' register address
#define RX_PW_P3    0x14  // 'RX payload width, pipe3' register address
#define RX_PW_P4    0x15  // 'RX payload width, pipe4' register address
#define RX_PW_P5    0x16  // 'RX payload width, pipe5' register address
#define FIFO_STATUS 0x17  // 'FIFO Status Register' register address


#endif   /* _API_DEF_ */

#ifndef HAL_H
#define HAL_H
//硬件初始化
extern void ChipHalInit(void);
extern void ChipOutHalInit(void);
extern void delay_ms(u16 dly_ms);
extern void USART_Configuration(void);
extern void RF_SPI_Config(void);
extern volatile u16 Timer1,Timer2;
//RF_2401
extern void init_io(void);
extern void test (void);
extern u8 tran;//中断进入标志
extern u8 SPI_Read(u8 reg);  
extern u8 SPI_RW(u8 byte);
#define TX_ADR_WIDTH   5  // 5字节宽度的发送/接收地址
#define TX_PLOAD_WIDTH 32 // 接收字节数32个字
#define uchar u8
/*控制引脚*/
#define CE_H  GPIOC->BSRR=GPIO_Pin_4
#define CE_L  GPIOC->BRR=GPIO_Pin_4 

#define CSN_H  GPIOB->BSRR=GPIO_Pin_2
#define CSN_L  GPIOB->BRR=GPIO_Pin_2 

#define IRQ   (GPIO_ReadInputDataBit(GPIOB,GPIO_Pin_0))  
#define IRQ_H  GPIOB->BSRR=GPIO_Pin_0

#define LED1  GPIOB->BSRR=GPIO_Pin_12
#define LED0  GPIOB->BRR=GPIO_Pin_12
#define LED   GPIOB->ODR=((GPIOD->ODR)^GPIO_Pin_12)

#define  RX_DR  ((sta>>6)&0X01)
#define  TX_DS  ((sta>>5)&0X01)
#define  MAX_RT  ((sta>>4)&0X01)

//USART
extern void USART2_Putc(unsigned char c);
extern void USART2_Puts(char * str);
extern u16 Uart2_Get_Flag; //串口2接收到数据长度
extern u8 Uart2_Get_Data; //串口2接收的数据
#endif

 

中断用前注意优先级的设置

#include "STM32Lib\\stm32f10x.h" 
void NVIC_Configuration(void)
{
 NVIC_InitTypeDef NVIC_InitStructure;
 
 /* 配置中断使用组合1*/
 NVIC_PriorityGroupConfig(NVIC_PriorityGroup_1);
 
 /* EXTI0*/
 NVIC_InitStructure.NVIC_IRQChannel =EXTI0_IRQn;
 NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 0;
 NVIC_InitStructure.NVIC_IRQChannelSubPriority = 0;
 NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;
 NVIC_Init(&NVIC_InitStructure);

  /* Configure one bit for preemption priority */
 NVIC_PriorityGroupConfig(NVIC_PriorityGroup_1);
 
 /*UART2*/
 NVIC_InitStructure.NVIC_IRQChannel = USART2_IRQn;
 NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 0;
 NVIC_InitStructure.NVIC_IRQChannelSubPriority = 0;
 NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;
 NVIC_Init(&NVIC_InitStructure);
  
}

关键字:STM32  驱动无线NRF24L01  串口数据传输 引用地址:STM32 驱动无线NRF24L01 完成串口数据传输

上一篇:STM32 代码中类型修饰符 volatile 的作用
下一篇:STM32驱动MAX6675读取温度

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

详细解析STM32中的堆栈机制
刚拿到STM32时,你只编写一个死循环 编译后,就会发现这么个程序已用了1600多的RAM,这要是在51单片机上,会心疼死了,这1600多的RAM跑哪儿去了,分析.map文件,你会发现是堆和栈占用的 在startup_stm32f10x_md.s文件中,它的前面几行就有以下定义: 这下明白了吧,STM32在启动的时候,RAM首先分配给使用到的全局变量,还有调用库占用的一些数据(不太清楚是什么数据),然后再将剩余的空间分配给Heap和Stack。由于内存空间是启动时实现分配好的,所以当动态分配内存的需求过多的时候,就会产生堆栈空间不足的问题。 查阅网上的资料,理解堆和栈的区别: - (1)栈区(stack):由编译器自动
[单片机]
详细解析<font color='red'>STM32</font>中的堆栈机制
STM32之EXTI
STM32因为具有NVIC使其中断体系大大提升,NVIC使用来管理异常与中断的,并且NVIC模块中还包含SysTick。 这次我们来讲一下EXTI外部中断。 1.在用到时我们必须要设置中断向量表,我们要将中断向量表存储在一个固定的位置,究竟是RAM还是FLASH就得看你的需求了。 2.STM32中中断优先级分组总共有四种分组方式,我们只能用其中的一组。所以在设置完毕中断向量表的存储位置后我们就得选择合适我们的中断优先级分组。(这一部分可以我的STM32之NVIC一文) 3.然后你要使用哪个外部中断就对此中断设置抢占优先级,亚优先级,以及打开。 4.然后就是相应中断的模式,方式,等配置了 5.设置相应的引脚为输入
[单片机]
<font color='red'>STM32</font>之EXTI
Windows下Keil MDK5配置STM32开发环境
Windows下使用Keil MDK5进行开发和编译, 配合ST-LINK工具进行烧录 stm32f103c8t6 参数 ARM 32-bit Cortex-M3 72 MHz maximum frequency 64k flash 20k ram LQFP封装48pin 安装说明 文件准备 mdk525.exe 不建议使用5.12等早期版本, 在更新时窗口容易卡, 且失败总会弹出需要手工消除. keygen2032 大部分找到的keygen, 有效期都是2020年的, 没法用, 必须要能生成2032有效期的版本 st-link驱动 https://www.st.com/zh/development-t
[单片机]
STM32 CustomHID 的实现
如何建立一个自定义的HID工程呢?下面就来讲讲。 首先先介绍下工程的架构,工程的总体架构下图所示,按照下图架构建工程: 分析下工程布局,首先是APP,这个组里存放着主文件mian.c,管理所有中断服务程序stm3210x_it.c,及其管理外设库头文件的stm32f10x_conf.h。BSP这个组里存放着BSP.c,外设的洗衣初始化都在这个函数中定义,比如说串口的配置,LED灯的配置,系统时钟的配置,各类NVIC的中断配置。在这个文件中,会定义一个BSP_Init()函数,所有配置的都在这个函数中调用,例如: void BSP_Init(void) { RCC_Configuration(); Set_USBCloc
[单片机]
<font color='red'>STM32</font> CustomHID 的实现
STM32中断源位置
在...\CMSIS\Device\ST\STM32F10x中的stm32f10x.h, 然后在结构体IRQn_Type找到对应单片机型号的代码片段。 比如STM32F103C8T6属于STM32F10X_MD, 那对应 的中断源为: 再比如STM32F103ZET6对应的是STM32F10X_HD
[单片机]
<font color='red'>STM32</font>中断源位置
STM32单片机可以用来做什么?能实现哪些功能?
已从事单片机开发十几年,刚开始接触单片机时,感觉挺有意思。 可以用自己的思维写程序,控制硬件去实现一些智能化的操作。 刚点亮第一个LED,就开始幻想以后能做任何自己想要的产品,那感觉多爽! 但是你会发现学完51单片机、或者STM32单片机以后,还是啥也做不出来。 不是你比较倒霉,每个工程师都是这样过来的。 就像你想写出一篇好作文,光练字和学成语是没用的,你得有思维,有经历。 做产品也是一样,单片机就是工具,思维才是灵魂。 思维怎么来? 答案肯定是多做项目。 所以,想成为一个具备独立开发的工程师,学习单片机只是你成长过程的 冰山一角 。 大部分的时间,应该用来做项目,从简单到复杂。 项目数量决定成长速度,项目质量决定成长高度。
[单片机]
STM32的SysTick时钟源来自哪里
有位朋友在后台大概问了这样一个问题:STM32的SysTick时钟源是来自Cortex系统定时器吗? 引伸: 为什么STM32CubeMX中Cortex系统定时器可选择1分频(和8分频)? 1写在前面 看到这个问题,我在想,这位朋友可能没有认真看手册,同时也存在一个误区。 我顺便也搜索了一下,网上很多文章都说到:SysTick时钟源是来自Cortex系统定时器,就是那个有8分频的时钟 但是,我们实际应用中,SysTick时钟源真的是来自这个Cortex系统定时器吗? 2 SysTick时钟初始化代码 不管是使用标准外设库,还是HAL库,你初始化SysTick,都会调用内核中的SysTick_Co
[单片机]
<font color='red'>STM32</font>的SysTick时钟源来自哪里
STM32单片机中断详解
中断,在单片机中占有非常重要的地位。代码默认地从上向下执行,遇到条件或者其他语句,会按照指定的地方跳转。而在单片机执行代码的过程中,难免会有一些突发的情况需要处理,这样就会打断当前的代码,待处理完突发情况之后,程序会回到被打断的地方继续执行。 1 EXTI控制器 外部中断/事件控制器(EXTI)管理了控制器的 23 个中断/事件线。每个中断/事件线都对应有一个边沿检测器,可以实现输入信号的上升沿检测和下降沿的检测。EXTI 可以实现对每个中断/事件线进行单独配置,可以单独配置为中断或者事件,以及触发事件的属性。 外部信号进入经过1的边沿检测电路,检测是否符合(有2和3的上升沿和下降沿选择寄存器决定),产生信号,然后和4软件
[单片机]
<font color='red'>STM32</font>单片机中断详解
小广播
添点儿料...
无论热点新闻、行业分析、技术干货……
设计资源 培训 开发板 精华推荐

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

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

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