[drv_usart]完善ch32的串口驱动。 (#6336)

This commit is contained in:
liYang~ 2022-08-27 12:34:27 +08:00 committed by GitHub
parent ae66e67dee
commit ef8ae7963c
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
2 changed files with 331 additions and 117 deletions

View File

@ -5,94 +5,227 @@
* *
* Change Logs: * Change Logs:
* Date Author Notes * Date Author Notes
* 2009-01-05 Bernard the first version * 2022-08-27 liYony the first version
* 2010-03-29 Bernard remove interrupt Tx and DMA Rx mode
* 2013-05-13 aozima update for kehong-lingtai.
* 2015-01-31 armink make sure the serial transmit complete in putc()
* 2016-05-13 armink add DMA Rx mode
* 2017-01-19 aubr.cool add interrupt Tx mode
* 2017-04-13 aubr.cool correct Rx parity err
* 2017-10-20 ZYH porting to HAL Libraries(with out DMA)
* 2017-11-15 ZYH update to 3.0.0
*/ */
#include "board.h" #include "board.h"
#include <rtdevice.h> #include <rtdevice.h>
#include <drv_usart.h> #include <drv_usart.h>
/* uart driver */ #ifdef RT_USING_SERIAL
struct ch32_uart
//#define DRV_DEBUG
#define LOG_TAG "drv.uart"
#include <drv_log.h>
#if !defined(BSP_USING_UART1) && !defined(BSP_USING_UART2) && !defined(BSP_USING_UART3) && !defined(BSP_USING_UART4) && \
!defined(BSP_USING_UART5) && !defined(BSP_USING_UART6) && !defined(BSP_USING_UART7) && !defined(BSP_USING_UART8)
#error "Please define at least one BSP_USING_UARTx"
/* this driver can be disabled at menuconfig -> RT-Thread Components -> Device Drivers */
#endif
enum
{ {
USART_InitTypeDef huart; #ifdef BSP_USING_UART1
USART_TypeDef *USARTx; UART1_INDEX,
IRQn_Type irq; #endif
#ifdef BSP_USING_UART2
UART2_INDEX,
#endif
#ifdef BSP_USING_UART3
UART3_INDEX,
#endif
#ifdef BSP_USING_UART4
UART4_INDEX,
#endif
#ifdef BSP_USING_UART5
UART5_INDEX,
#endif
#ifdef BSP_USING_UART6
UART6_INDEX,
#endif
#ifdef BSP_USING_UART7
UART7_INDEX,
#endif
#ifdef BSP_USING_UART8
UART8_INDEX,
#endif
}; };
/* If you want to use other serial ports, please follow UART1 to complete other
serial ports. For clock configuration, */
static struct ch32_uart_hw_config uart_hw_config[] =
{
#ifdef BSP_USING_UART1
{
/* clock configuration, please refer to ch32v30x_rcc.h */
RCC_APB2Periph_USART1|RCC_APB2Periph_GPIOA,
/* GPIO configuration : TX_Port,TX_Pin, RX_Port,RX_Pin */
GPIOA, GPIO_Pin_9, /* Tx */GPIOA, GPIO_Pin_10, /* Rx */
/* Whether to enable port remapping, you can refer to ch32v30x_gpio.h file,
for example, USART1 needs to be turned on, you can use GPIO_Remap_USART1 */
GPIO_Remap_NONE,
},
#endif
#ifdef BSP_USING_UART2
{},
#endif
#ifdef BSP_USING_UART3
{},
#endif
#ifdef BSP_USING_UART4
{},
#endif
#ifdef BSP_USING_UART5
{},
#endif
#ifdef BSP_USING_UART6
{},
#endif
#ifdef BSP_USING_UART7
{},
#endif
#ifdef BSP_USING_UART8
{},
#endif
};
static struct ch32_uart_config uart_config[] =
{
#ifdef BSP_USING_UART1
{
"uart1",
USART1,
USART1_IRQn,
},
#endif
#ifdef BSP_USING_UART2
{
"uart2",
USART2,
USART2_IRQn,
},
#endif
#ifdef BSP_USING_UART3
{
"uart3",
USART3,
USART3_IRQn,
},
#endif
#ifdef BSP_USING_UART4
{
"uart4",
UART4,
UART4_IRQn,
},
#endif
#ifdef BSP_USING_UART5
{
"uart5",
UART5,
UART5_IRQn,
},
#endif
#ifdef BSP_USING_UART6
{
"uart6",
UART6,
UART6_IRQn,
},
#endif
#ifdef BSP_USING_UART7
{
"uart7",
UART7,
UART7_IRQn,
},
#endif
#ifdef BSP_USING_UART8
{
"uart8",
UART8,
UART8_IRQn,
},
#endif
};
static struct ch32_uart uart_obj[sizeof(uart_config) / sizeof(uart_config[0])] = {0};
static rt_err_t ch32_configure(struct rt_serial_device *serial, struct serial_configure *cfg) static rt_err_t ch32_configure(struct rt_serial_device *serial, struct serial_configure *cfg)
{ {
struct ch32_uart *uart; struct ch32_uart *uart;
GPIO_InitTypeDef GPIO_InitStructure={0};
RT_ASSERT(serial != RT_NULL); RT_ASSERT(serial != RT_NULL);
RT_ASSERT(cfg != RT_NULL); RT_ASSERT(cfg != RT_NULL);
uart = (struct ch32_uart *)serial->parent.user_data;
uart->huart.USART_BaudRate = cfg->baud_rate; uart = (struct ch32_uart *) serial->parent.user_data;
uart->huart.USART_HardwareFlowControl = USART_HardwareFlowControl_None;
uart->huart.USART_Mode = USART_Mode_Rx|USART_Mode_Tx; uart->Init.USART_BaudRate = cfg->baud_rate;
uart->Init.USART_HardwareFlowControl = USART_HardwareFlowControl_None;
uart->Init.USART_Mode = USART_Mode_Rx|USART_Mode_Tx;
switch (cfg->data_bits) switch (cfg->data_bits)
{ {
case DATA_BITS_8: case DATA_BITS_8:
uart->huart.USART_WordLength = USART_WordLength_8b; uart->Init.USART_WordLength = USART_WordLength_8b;
break; break;
case DATA_BITS_9: case DATA_BITS_9:
uart->huart.USART_WordLength = USART_WordLength_9b; uart->Init.USART_WordLength = USART_WordLength_9b;
break; break;
default: default:
uart->huart.USART_WordLength = USART_WordLength_8b; uart->Init.USART_WordLength = USART_WordLength_8b;
break; break;
} }
switch (cfg->stop_bits) switch (cfg->stop_bits)
{ {
case STOP_BITS_1: case STOP_BITS_1:
uart->huart.USART_StopBits = USART_StopBits_1; uart->Init.USART_StopBits = USART_StopBits_1;
break; break;
case STOP_BITS_2: case STOP_BITS_2:
uart->huart.USART_StopBits = USART_StopBits_2; uart->Init.USART_StopBits = USART_StopBits_2;
break; break;
default: default:
uart->huart.USART_StopBits = USART_StopBits_1; uart->Init.USART_StopBits = USART_StopBits_1;
break; break;
} }
switch (cfg->parity) switch (cfg->parity)
{ {
case PARITY_NONE: case PARITY_NONE:
uart->huart.USART_Parity = USART_Parity_No; uart->Init.USART_Parity = USART_Parity_No;
break; break;
case PARITY_ODD: case PARITY_ODD:
uart->huart.USART_Parity = USART_Parity_Odd; uart->Init.USART_Parity = USART_Parity_Odd;
break; break;
case PARITY_EVEN: case PARITY_EVEN:
uart->huart.USART_Parity = USART_Parity_Even; uart->Init.USART_Parity = USART_Parity_Even;
break; break;
default: default:
uart->huart.USART_Parity = USART_Parity_No; uart->Init.USART_Parity = USART_Parity_No;
break; break;
} }
if(uart->USARTx == USART1) /* UART hardware configuration, including clock and GPIO, etc. */
RCC_APB2PeriphClockCmd(uart->hw_config->periph_clock, ENABLE);
if(uart->hw_config->remap != GPIO_Remap_NONE)
{ {
GPIO_InitTypeDef GPIO_InitStructure; RCC_APB2PeriphClockCmd(RCC_APB2Periph_AFIO, ENABLE);
RCC_APB2PeriphClockCmd(RCC_APB2Periph_USART1|RCC_APB2Periph_GPIOA, ENABLE); GPIO_PinRemapConfig(uart->hw_config->remap, ENABLE);
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_9; }
GPIO_InitStructure.GPIO_Pin = uart->hw_config->tx_gpio_pin;
GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz; GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP; GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP;
GPIO_Init(GPIOA, &GPIO_InitStructure); GPIO_Init(uart->hw_config->tx_gpio_port, &GPIO_InitStructure);
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_10; GPIO_InitStructure.GPIO_Pin = uart->hw_config->rx_gpio_pin;
GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz; GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING; GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING;
GPIO_Init(GPIOA, &GPIO_InitStructure); GPIO_Init(uart->hw_config->rx_gpio_port, &GPIO_InitStructure);
USART_Init(uart->USARTx,&uart->huart);
USART_Cmd(uart->USARTx, ENABLE); USART_Init(uart->config->Instance,&uart->Init);
} USART_Cmd(uart->config->Instance, ENABLE);
return RT_EOK; return RT_EOK;
} }
@ -107,16 +240,16 @@ static rt_err_t ch32_control(struct rt_serial_device *serial, int cmd, void *arg
/* disable interrupt */ /* disable interrupt */
case RT_DEVICE_CTRL_CLR_INT: case RT_DEVICE_CTRL_CLR_INT:
/* disable rx irq */ /* disable rx irq */
NVIC_DisableIRQ(uart->irq); NVIC_DisableIRQ(uart->config->irq_type);
/* disable interrupt */ /* disable interrupt */
USART_ITConfig(uart->USARTx,USART_IT_RXNE,DISABLE); USART_ITConfig(uart->config->Instance,USART_IT_RXNE,DISABLE);
break; break;
/* enable interrupt */ /* enable interrupt */
case RT_DEVICE_CTRL_SET_INT: case RT_DEVICE_CTRL_SET_INT:
/* enable rx irq */ /* enable rx irq */
NVIC_EnableIRQ(uart->irq); NVIC_EnableIRQ(uart->config->irq_type);
/* enable interrupt */ /* enable interrupt */
USART_ITConfig(uart->USARTx, USART_IT_RXNE,ENABLE); USART_ITConfig(uart->config->Instance, USART_IT_RXNE,ENABLE);
break; break;
} }
return RT_EOK; return RT_EOK;
@ -127,8 +260,8 @@ static int ch32_putc(struct rt_serial_device *serial, char c)
struct ch32_uart *uart; struct ch32_uart *uart;
RT_ASSERT(serial != RT_NULL); RT_ASSERT(serial != RT_NULL);
uart = (struct ch32_uart *)serial->parent.user_data; uart = (struct ch32_uart *)serial->parent.user_data;
while (USART_GetFlagStatus(uart->USARTx, USART_FLAG_TC) == RESET); while (USART_GetFlagStatus(uart->config->Instance, USART_FLAG_TC) == RESET);
uart->USARTx->DATAR = c; uart->config->Instance->DATAR = c;
return 1; return 1;
} }
@ -139,33 +272,26 @@ static int ch32_getc(struct rt_serial_device *serial)
RT_ASSERT(serial != RT_NULL); RT_ASSERT(serial != RT_NULL);
uart = (struct ch32_uart *)serial->parent.user_data; uart = (struct ch32_uart *)serial->parent.user_data;
ch = -1; ch = -1;
if (USART_GetFlagStatus(uart->USARTx, USART_FLAG_RXNE) != RESET) if (USART_GetFlagStatus(uart->config->Instance, USART_FLAG_RXNE) != RESET)
{ {
ch = uart->USARTx->DATAR & 0xff; ch = uart->config->Instance->DATAR & 0xff;
} }
return ch; return ch;
} }
rt_size_t ch32dma_transmit(struct rt_serial_device *serial, rt_uint8_t *buf, rt_size_t size, int direction) rt_size_t ch32dma_transmit(struct rt_serial_device *serial, rt_uint8_t *buf, rt_size_t size, int direction)
{ {
return RT_EOK; return RT_EOK;
} }
/**
* Uart common interrupt process. This need add to uart ISR.
*
* @param serial serial device
*/
static void uart_isr(struct rt_serial_device *serial) static void uart_isr(struct rt_serial_device *serial)
{ {
struct ch32_uart *uart = (struct ch32_uart *) serial->parent.user_data; struct ch32_uart *uart = (struct ch32_uart *) serial->parent.user_data;
RT_ASSERT(uart != RT_NULL); RT_ASSERT(uart != RT_NULL);
if (USART_GetITStatus(uart->USARTx, USART_IT_RXNE) != RESET) if (USART_GetITStatus(uart->config->Instance, USART_IT_RXNE) != RESET)
{ {
rt_hw_serial_isr(serial, RT_SERIAL_EVENT_RX_IND); rt_hw_serial_isr(serial, RT_SERIAL_EVENT_RX_IND);
USART_ClearITPendingBit(uart->USARTx, USART_IT_RXNE); USART_ClearITPendingBit(uart->config->Instance, USART_IT_RXNE);
} }
} }
@ -178,76 +304,131 @@ static const struct rt_uart_ops ch32_uart_ops =
ch32dma_transmit ch32dma_transmit
}; };
#if defined(BSP_USING_UART1) #ifdef BSP_USING_UART1
/* UART1 device driver structure */ void USART1_IRQHandler(void) __attribute__((interrupt("WCH-Interrupt-fast")));
struct ch32_uart uart1;
struct rt_serial_device serial1;
void USART1_IRQHandler(void) __attribute__((interrupt()));
void USART1_IRQHandler(void) void USART1_IRQHandler(void)
{ {
GET_INT_SP(); GET_INT_SP();
/* enter interrupt */
rt_interrupt_enter(); rt_interrupt_enter();
uart_isr(&(uart_obj[UART1_INDEX].serial));
uart_isr(&serial1);
/* leave interrupt */
rt_interrupt_leave(); rt_interrupt_leave();
FREE_INT_SP(); FREE_INT_SP();
} }
#endif /* BSP_USING_UART1 */ #endif
#ifdef BSP_USING_UART2
void USART2_IRQHandler(void) __attribute__((interrupt("WCH-Interrupt-fast")));
void USART2_IRQHandler(void)
{
GET_INT_SP();
rt_interrupt_enter();
uart_isr(&(uart_obj[UART2_INDEX].serial));
rt_interrupt_leave();
FREE_INT_SP();
}
#endif
#ifdef BSP_USING_UART3
void USART3_IRQHandler(void) __attribute__((interrupt("WCH-Interrupt-fast")));
void USART3_IRQHandler(void)
{
GET_INT_SP();
rt_interrupt_enter();
uart_isr(&(uart_obj[UART3_INDEX].serial));
rt_interrupt_leave();
FREE_INT_SP();
}
#endif
#ifdef BSP_USING_UART4
void UART4_IRQHandler(void) __attribute__((interrupt("WCH-Interrupt-fast")));
void UART4_IRQHandler(void)
{
GET_INT_SP();
rt_interrupt_enter();
uart_isr(&(uart_obj[UART4_INDEX].serial));
rt_interrupt_leave();
FREE_INT_SP();
}
#endif
#ifdef BSP_USING_UART5
void UART5_IRQHandler(void) __attribute__((interrupt("WCH-Interrupt-fast")));
void UART5_IRQHandler(void)
{
GET_INT_SP();
rt_interrupt_enter();
uart_isr(&(uart_obj[UART5_INDEX].serial));
rt_interrupt_leave();
FREE_INT_SP();
}
#endif
#ifdef BSP_USING_UART6
void UART6_IRQHandler(void) __attribute__((interrupt("WCH-Interrupt-fast")));
void UART6_IRQHandler(void)
{
GET_INT_SP();
rt_interrupt_enter();
uart_isr(&(uart_obj[UART6_INDEX].serial));
rt_interrupt_leave();
FREE_INT_SP();
}
#endif
#ifdef BSP_USING_UART7
void UART7_IRQHandler(void) __attribute__((interrupt("WCH-Interrupt-fast")));
void UART7_IRQHandler(void)
{
GET_INT_SP();
rt_interrupt_enter();
uart_isr(&(uart_obj[UART7_INDEX].serial));
rt_interrupt_leave();
FREE_INT_SP();
}
#endif
#ifdef BSP_USING_UART8
void UART8_IRQHandler(void) __attribute__((interrupt("WCH-Interrupt-fast")));
void UART8_IRQHandler(void)
{
GET_INT_SP();
rt_interrupt_enter();
uart_isr(&(uart_obj[UART8_INDEX].serial));
rt_interrupt_leave();
FREE_INT_SP();
}
#endif
int rt_hw_usart_init(void) int rt_hw_usart_init(void)
{ {
struct ch32_uart *uart; rt_size_t obj_num = sizeof(uart_obj) / sizeof(struct ch32_uart);
struct serial_configure config = RT_SERIAL_CONFIG_DEFAULT; struct serial_configure config = RT_SERIAL_CONFIG_DEFAULT;
rt_err_t result = 0;
#if defined(BSP_USING_UART1) for (int i = 0; i < obj_num; i++)
uart=&uart1;
uart->irq=USART1_IRQn;
uart->USARTx=USART1;
config.baud_rate = BAUD_RATE_115200;
serial1.ops = &ch32_uart_ops;
serial1.config = config;
uart->huart.USART_BaudRate = 115200;
uart->huart.USART_HardwareFlowControl = USART_HardwareFlowControl_None;
uart->huart.USART_Mode = USART_Mode_Rx|USART_Mode_Tx;
uart->huart.USART_WordLength = USART_WordLength_8b;
uart->huart.USART_StopBits = USART_StopBits_1;
uart->huart.USART_Parity = USART_Parity_No;
if(uart->USARTx == USART1)
{ {
GPIO_InitTypeDef GPIO_InitStructure; /* init UART object */
RCC_APB2PeriphClockCmd(RCC_APB2Periph_USART1|RCC_APB2Periph_GPIOA, ENABLE); uart_obj[i].config = &uart_config[i];
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_9; uart_obj[i].hw_config = &uart_hw_config[i];
GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz; uart_obj[i].serial.ops = &ch32_uart_ops;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP; uart_obj[i].serial.config = config;
GPIO_Init(GPIOA, &GPIO_InitStructure); /* Hardware initialization is required, otherwise it
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_10; will not be registered into the device framework */
GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz; if(uart_obj[i].hw_config->periph_clock == 0)
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING; {
GPIO_Init(GPIOA, &GPIO_InitStructure); LOG_E("You did not perform hardware initialization for %s", uart->config->name);
USART_Init(uart->USARTx,&uart->huart); continue;
USART_Cmd(uart->USARTx, ENABLE); }
/* register UART device */
result = rt_hw_serial_register(&uart_obj[i].serial, uart_obj[i].config->name,
RT_DEVICE_FLAG_RDWR
| RT_DEVICE_FLAG_INT_RX
, &uart_obj[i]);
RT_ASSERT(result == RT_EOK);
} }
/* register UART1 device */ return result;
rt_hw_serial_register(&serial1, "uart1",
RT_DEVICE_FLAG_RDWR | RT_DEVICE_FLAG_INT_RX,
uart);
#endif /* BSP_USING_UART1 */
return 0;
} }
INIT_BOARD_EXPORT(rt_hw_usart_init);
#endif /* RT_USING_SERIAL */

View File

@ -5,12 +5,45 @@
* *
* Change Logs: * Change Logs:
* Date Author Notes * Date Author Notes
* 2009-01-05 Bernard the first version * 2022-08-27 liYony the first version
*/ */
#ifndef __USART_H__
#define __USART_H__ #ifndef __DRV_USART_H__
#include "rthw.h" #define __DRV_USART_H__
#include "rtthread.h" #include <rtthread.h>
#include "rtdevice.h"
#include <rthw.h>
/* Do not use GPIO_Remap*/
#define GPIO_Remap_NONE 0
/* ch32 hardware config class */
struct ch32_uart_hw_config
{
rt_uint32_t periph_clock;
GPIO_TypeDef *tx_gpio_port;
rt_uint16_t tx_gpio_pin;
GPIO_TypeDef *rx_gpio_port;
rt_uint16_t rx_gpio_pin;
rt_uint32_t remap;
};
/* ch32 config class */
struct ch32_uart_config
{
const char *name;
USART_TypeDef *Instance;
IRQn_Type irq_type;
};
/* ch32 uart dirver class */
struct ch32_uart
{
struct ch32_uart_hw_config *hw_config;
struct ch32_uart_config *config;
USART_InitTypeDef Init;
struct rt_serial_device serial;
};
int rt_hw_usart_init(void); int rt_hw_usart_init(void);
#endif #endif