本帖最后由 nvjwiciw659 于 2020-12-6 23:59 编辑
今天做下关于ESP8266WIFI模块的准备工作,串口不定长接收数据
查了一下数据手册,CHV103有IDLE(线路空闲)中断,那就好办了.
1.定义一个结构体
typedef struct
{
u8 rx_buff[256];//串口缓存区
u8 len; //接收长度
u8 flag; //用于判断是否接收完 0为接收完
} UART2_RX;
同时编写一个程序用来初始化这个结构体,或者用来清空串口缓存
void CLEAR_UART2(void)
{
uart2_rx.flag = 1;
uart2_rx.len = 0;
memset((u8 *)uart2_rx.rx_buff, 0, sizeof(uart2_rx.rx_buff));
}
2.配置串口
/*******************************************************************************
* Function Name : USARTx_CFG
* Description : Initializes the USART2 peripheral.
* Input : None
* Return : None
*******************************************************************************/
void USARTx_CFG(void)
{
GPIO_InitTypeDef GPIO_InitStructure;
USART_InitTypeDef USART_InitStructure;
NVIC_InitTypeDef NVIC_InitStructure;
RCC_APB1PeriphClockCmd(RCC_APB1Periph_USART2, ENABLE); //使能串口2时钟
RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE); //使能GPIOA时钟
USART_DeInit(USART2);
/* USART2 TX-->A.2 RX-->A.3 */
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_2;
GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP; //设置PA2为复用推挽输出
GPIO_Init(GPIOA, &GPIO_InitStructure);
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_3;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING; //设置PA3为浮空输入
GPIO_Init(GPIOA, &GPIO_InitStructure);
USART_InitStructure.USART_BaudRate = 115200; //设置串口波特率为115200
USART_InitStructure.USART_WordLength = USART_WordLength_8b; //字长为8位数据格式
USART_InitStructure.USART_StopBits = USART_StopBits_1; //1个停止位
USART_InitStructure.USART_Parity = USART_Parity_No; //无奇偶校验位
USART_InitStructure.USART_HardwareFlowControl = USART_HardwareFlowControl_None;//无硬件流控制
USART_InitStructure.USART_Mode = USART_Mode_Tx | USART_Mode_Rx; //发送和接收模式
USART_Init(USART2, &USART_InitStructure); //初始化串口
NVIC_InitStructure.NVIC_IRQChannel = USART2_IRQn;
NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority=1; //抢占优先级为1
NVIC_InitStructure.NVIC_IRQChannelSubPriority = 1; //子优先级为1
NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE; //IRQ通道使能
NVIC_Init(&NVIC_InitStructure); //中断优先级初始化
USART_Cmd(USART2, ENABLE); //使能串口
USART_ITConfig(USART2, USART_IT_RXNE, ENABLE); //开启接收中断
USART_ITConfig(USART2, USART_IT_IDLE, ENABLE); //开启空闲中断
}
3.编写中断程序
/*******************************************************************************
* Function Name : USART2_IRQHandler
* Description : This function handles USART2 global interrupt request.
* Input : None
* Return : None
*******************************************************************************/
void USART2_IRQHandler(void)
{
if(USART_GetITStatus(USART2, USART_IT_RXNE) != RESET) //接收中断产生
{
USART_ClearITPendingBit(USART2,USART_IT_RXNE); //清除接收中断标志
uart2_rx.rx_buff[uart2_rx.len] = USART_ReceiveData(USART2); //接收数据
uart2_rx.len++;
}
if(USART_GetITStatus(USART2, USART_IT_IDLE) != RESET) //空闲中断产生
{
USART_ClearITPendingBit(USART2,USART_IT_IDLE); //清除空闲中断标志
USART_ReceiveData(USART2);
uart2_rx.flag = 0;//代表一包数据收完
}
}
注意,空闲中断标志位必须读状态寄存器再读数据寄存器的操作会清除此位
4.接收测试
USARTx_CFG();
/*初始化串口缓存结构体*/
CLEAR_UART2();
while(1)
{
if(uart2_rx.flag == 0)
{
printf("Receive:%s\r\n",uart2_rx.rx_buff);
CLEAR_UART2();
}
}
烧录,然后ttl转串口连接USART2,即A2A3,往串口2发送一包数据,接收完毕会在串口1发出
完毕
5.发送
直接偷懒用例程
void USARTx_SendByte(USART_TypeDef* pUSARTx, uint8_t data)
{
USART_SendData(pUSARTx, data);
while(USART_GetFlagStatus(pUSARTx, USART_FLAG_TXE) == RESET);
}
void USARTx_SendStr(USART_TypeDef* pUSARTx, char *str)
{
uint8_t i = 0;
do
{
USARTx_SendByte(pUSARTx, *(str+i));
i++;
}while(*(str+i) != '\0');
while(USART_GetFlagStatus(pUSARTx, USART_FLAG_TC) == RESET);
}
5.发送测试
USARTx_CFG();
/*初始化串口缓存结构体*/
CLEAR_UART2();
while(1)
{
if(uart2_rx.flag == 0)
{
USARTx_SendStr(USART2,"I Received\r\n");
printf("Receive:%s\r\n",uart2_rx.rx_buff);
CLEAR_UART2();
}
}
6.小插曲
刚开始写好程序uart2_rx.flag的变化在while(1)中一直不被判断出来,经群内陶工指导,意识到可能是编译器对该变量进行了优化,添加volatile尝试,结果就是这个原因,又涨姿势了!多谢群内几个工程师的指导!!!
|