gpio模拟串口uart接收数据(HAL库)

GPIO模拟uart的原理就是根据所需要的uart的波特率,计算出每个bit所用的时间,在此时间段内去读取gpio的输入,然后将读取的8bit拼接起来即可,当然这期间要检测起始位和结束位。

uart的起始位是下降沿,结束位是上升沿。

为了检测准确,我们读取每个bit的中位位置,所以定时时间是每个bit位所占用时间一半。

9600波特率的uart模拟,每个bit位的时间是:1/9600 s =  104us的时间,那么半个bit位的定时时间就是104/2 = 52us。所以在检测到下降沿后,接着打开定时器,定时52us,每52us读取一次

GPIO的状态作为一个字节的bit位,即可以正确读取出一个字节。

代码实现方式如下:

1. 首先定义枚举值,每次定时中断时状态机就按照下面的枚举值增加一步。

enum{
    COM_START_BIT = 1,    //起始位位
    COM_START_MID_BIT,    //起始位中位
    COM_D0_BIT,    //bit0
    COM_D0_MID_BIT,    //bit0中位
    COM_D1_BIT,    //bit1
    COM_D1_MID_BIT,    //bit1中位
    COM_D2_BIT,    //bit2
    COM_D2_MID_BIT,    //bit2中位
    COM_D3_BIT,    //bit3
    COM_D3_MID_BIT,    //bit3中位
    COM_D4_BIT,    //bit4
    COM_D4_MID_BIT,    //bit4中位
    COM_D5_BIT,    //bit5
    COM_D5_MID_BIT,    //bit5中位
    COM_D6_BIT,    //bit6
    COM_D6_MID_BIT,    //bit6中位
    COM_D7_BIT,    //bit7
    COM_D7_MID_BIT,    //bit7中位
    COM_STOP_BIT,    //终止位
    COM_STOP_MID_BIT,    //终止位中位20
};

static uint8_t recvStat = COM_STOP_MID_BIT;    //定义状态机

#define ANALOGUSART_RX_BUF_SIZE 50

static uint8_t analog_usart_rx_buf[ANALOGUSART_RX_BUF_SIZE];
__IO ANALOG_R_BUF analog_usart_rbuf={0,0,analog_usart_rx_buf,0};//定义模拟串口的接收环形缓冲区

int8_t tim7_stop(void);
int8_t tim7_st

  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
在 STM32 中,可以通过 DMA 直接将串口接收到的数据存储到内存中,从而减轻 CPU 的负担。在使用 DMA 接收串口数据时,可以通过 `HAL_UARTEx_RxEventCallback` 函数来处理 DMA 接收数据完成后的回调。 `HAL_UARTEx_RxEventCallback` 的函数原型为: ```c void HAL_UARTEx_RxEventCallback(UART_HandleTypeDef *huart); ``` 其中,`huart` 是串口的句柄指针。 在使用 DMA 接收串口数据时,需要先开启 DMA,然后在回调函数中处理接收到的数据。下面是一个使用 DMA 接收串口数据的例子: ```c #include "stm32f4xx_hal.h" #define UART_RX_BUFFER_SIZE 256 UART_HandleTypeDef huart2; DMA_HandleTypeDef hdma_usart2_rx; uint8_t uart_rx_buffer[UART_RX_BUFFER_SIZE]; void HAL_UART_MspInit(UART_HandleTypeDef* uartHandle) { GPIO_InitTypeDef GPIO_InitStruct = {0}; if(uartHandle->Instance==USART2) { /* USART2 clock enable */ __HAL_RCC_USART2_CLK_ENABLE(); __HAL_RCC_GPIOA_CLK_ENABLE(); /**USART2 GPIO Configuration PA2 ------> USART2_TX PA3 ------> USART2_RX */ GPIO_InitStruct.Pin = GPIO_PIN_2|GPIO_PIN_3; GPIO_InitStruct.Mode = GPIO_MODE_AF_PP; GPIO_InitStruct.Pull = GPIO_NOPULL; GPIO_InitStruct.Speed = GPIO_SPEED_FREQ_HIGH; GPIO_InitStruct.Alternate = GPIO_AF7_USART2; HAL_GPIO_Init(GPIOA, &GPIO_InitStruct); /* USART2 DMA Init */ /* USART2_RX Init */ hdma_usart2_rx.Instance = DMA1_Stream5; hdma_usart2_rx.Init.Channel = DMA_CHANNEL_4; hdma_usart2_rx.Init.Direction = DMA_PERIPH_TO_MEMORY; hdma_usart2_rx.Init.PeriphInc = DMA_PINC_DISABLE; hdma_usart2_rx.Init.MemInc = DMA_MINC_ENABLE; hdma_usart2_rx.Init.PeriphDataAlignment = DMA_PDATAALIGN_BYTE; hdma_usart2_rx.Init.MemDataAlignment = DMA_MDATAALIGN_BYTE; hdma_usart2_rx.Init.Mode = DMA_CIRCULAR; hdma_usart2_rx.Init.Priority = DMA_PRIORITY_LOW; hdma_usart2_rx.Init.FIFOMode = DMA_FIFOMODE_DISABLE; if (HAL_DMA_Init(&hdma_usart2_rx) != HAL_OK) { Error_Handler(); } __HAL_LINKDMA(uartHandle,hdmarx,hdma_usart2_rx); /* USART2 interrupt Init */ HAL_NVIC_SetPriority(USART2_IRQn, 0, 0); HAL_NVIC_EnableIRQ(USART2_IRQn); } } void HAL_UART_RxCpltCallback(UART_HandleTypeDef *huart) { if (huart->Instance == USART2) { // 处理接收到的数据 // ... } } int main(void) { HAL_Init(); __HAL_RCC_GPIOA_CLK_ENABLE(); // 配置串口 huart2.Instance = USART2; huart2.Init.BaudRate = 115200; huart2.Init.WordLength = UART_WORDLENGTH_8B; huart2.Init.StopBits = UART_STOPBITS_1; huart2.Init.Parity = UART_PARITY_NONE; huart2.Init.Mode = UART_MODE_RX; huart2.Init.HwFlowCtl = UART_HWCONTROL_NONE; huart2.Init.OverSampling = UART_OVERSAMPLING_16; if (HAL_UART_Init(&huart2) != HAL_OK) { Error_Handler(); } // 配置 DMA 接收 HAL_UART_Receive_DMA(&huart2, uart_rx_buffer, UART_RX_BUFFER_SIZE); while (1); } ``` 在上面的代码中,我们在 `HAL_UART_MspInit` 函数中初始化了 DMA,并将 DMA 和串口绑定起来。在 `main` 函数中,我们使用 `HAL_UART_Receive_DMA` 开启了 DMA 接收并指定了接收缓冲区的地址和大小。当 DMA 接收完成后,会自动触发 `HAL_UART_RxCpltCallback` 回调函数,我们可以在该函数中处理接收到的数据。 需要注意的是,在使用 DMA 接收串口数据时,需要确保接收缓冲区的大小大于等于最大接收数据长度,否则可能会出现数据溢出的情况。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值