GPIO 模拟UART

转载自:
http://blog.chinaaet.com/wuyage/p/5100050276

你是否遇到过某个MCU串口不够的情况? 这时我们可以考虑用GPIO去模拟,如何具体实现呢?

首选我们需要了解串口的传输协议,

UART使用异步模式工作,不需要时钟信号,其一般格式为:起始位+数据位+校验位+停止位。其中起始位1位,数据位5~8位,校验位0或1位,停止位1、1.5或2位。不过最常用的格式是1位起始位、8位数据位、没有奇偶校验、1位停止位,简记为8/N/1

8/N/1格式的时序图如下:

11111.jpg

空闲时数据线上规定为逻辑1。

开始传输数据时先发送起始位,规定为逻辑0,接收端会检测这个下降沿,以便之后开始采样接收数据。

起始位之后是数据位,规定先发送最低位,即LSB First。因为UART没有时钟信号,故使用波特率来确定每一位的长度,不过为保证检测的准确性,实际采样频率会高于波特率,一般每一位会进行若干次采样,取中间的采样值作为这一位的结果。

奇偶校验位一般不使用。

停止位一般使用1位,规定为逻辑1,除了表示传输结束外,停止位还可以起到时钟同步的作用。

需要注意的是,这里的逻辑0并不一定是0V,这与使用的电平标准有关。对于TTL电平而言,逻辑0是0V,逻辑1是高电平(一般为3.3V或5V);对于RS-232电平而言,逻辑0是3V~15V,逻辑1是-3~-15V。

除了TX、RX、GND信号外,UART中还会有诸如RTS、CTS等流控信号,因为用得不是很多,此处就不总结了。

以发送0x23(无奇偶校验)为例来说明,传输时序如下:

xxxxx.jpg

注意是LSB First,也就是最低位先传输哦。

0x23,二进制表示为00100011,传输顺序为1->1->0->0->0->1->0->0

掌握清楚这个时序那么也就好用GPIO模拟了,除了需要两个GPIO,还需要两个定时器(分别用于接收和发送时序控制),另外需要说明的是,为方便起见,采样频率这里就设置成了波特率。

1) 对于接收,当RX引脚检测到下降沿时,进入GPIO中断,然后开启一个定时器,第一次定时器周期设置为1/波特率的一半(目的是为了在中心除判断是否为低电平,以表示是否为起始位),再之后就可以设置定时器周期为1/波特率,每隔此周期在定时器中断里去采样RX引脚电平,将数据接收完毕

2)对于发送,首先发送一个起始位,之后以1/波特率为周期,在定时器中断里去发送比特位即可。

附件为在FRDM_K64F基于SDK2.0 上的实现代码,直接使用板载的虚拟串口就可以调试,

经过测试波特率可以达到38400.

 

SDK_2.0_FRDMK64F_GPIO_UART.rar

以下是对程序的简单说明:

1)gpio_uart_demo_init() 里可以配置UART的相关参数,如波特率,奇偶校验,数据位长度

2)void gpio_uart_read(uint8_t *bufptr, uint32_t size, void (*rx_callback)(void)) 这个函数为uart 接收函数,第一个参数为数据存放buffer,第二个数据为接收长度,第三个参数为callback函数。注意目前的实现是调用此函数后,当接收完指定长度数据后,会停止接收数据。 如果之后要继续接收,需要再次调用这个函数。

3)移植到其他不同平台也比较容易,只需要配置GPIO和定时器即可。

 

补充,再添加个QN902x平台上的代码:

2.24.2017  update下,我原来GPIO初始化里没有把tx pin 拉高,这里补充上了。

qpps_2_uart_NXP_NoSleep_gpiosimulateuart.rar

  • 2
    点赞
  • 21
    收藏
    觉得还不错? 一键收藏
  • 2
    评论
好的,我可以为您解答这个问题。 在STM32上使用GPIO模拟UART来打印log,需要用到两个GPIO口,一个用于作为TX(发送)口,另一个用于作为RX(接收)口。 首先,需要在STM32的代码中定义两个GPIO口作为TX和RX口,并初始化它们的状态。然后,可以使用GPIO口的输出和输入功能来模拟UART的发送和接收。 在发送数据时,需要将数据转换为二进制格式,并将其逐位地发送到TX口。在接收数据时,需要从RX口逐位地读取数据,并将其转换为ASCII字符格式。 下面是一个使用GPIO模拟UART打印log的示例代码: ```c #include "stm32f10x.h" GPIO_InitTypeDef GPIO_InitStructure; void GPIO_UART_Init(void) { RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE); /* Configure USART Tx as alternate function push-pull */ GPIO_InitStructure.GPIO_Pin = GPIO_Pin_9; GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP; GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz; GPIO_Init(GPIOA, &GPIO_InitStructure); /* Configure USART Rx as input floating */ GPIO_InitStructure.GPIO_Pin = GPIO_Pin_10; GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING; GPIO_Init(GPIOA, &GPIO_InitStructure); } void GPIO_UART_SendChar(char ch) { uint8_t i; GPIO_WriteBit(GPIOA, GPIO_Pin_9, Bit_RESET); for(i=0;i<8;i++) { GPIO_WriteBit(GPIOA, GPIO_Pin_9, (BitAction)(ch & 0x01)); ch >>= 1; } GPIO_WriteBit(GPIOA, GPIO_Pin_9, Bit_SET); } void GPIO_UART_SendString(char *str) { while(*str) { GPIO_UART_SendChar(*str++); } } char GPIO_UART_ReceiveChar(void) { uint8_t i; char ch = 0; while(GPIO_ReadInputDataBit(GPIOA, GPIO_Pin_10)); for(i=0;i<8;i++) { ch >>= 1; if(GPIO_ReadInputDataBit(GPIOA, GPIO_Pin_10)) { ch |= 0x80; } while(!GPIO_ReadInputDataBit(GPIOA, GPIO_Pin_10)); } return ch; } int main(void) { GPIO_UART_Init(); GPIO_UART_SendString("Hello, world!\r\n"); while(1) { char ch = GPIO_UART_ReceiveChar(); GPIO_UART_SendChar(ch); } } ``` 在这个示例代码中,我们使用GPIOA的Pin9作为TX口,使用GPIOA的Pin10作为RX口。我们在GPIO_UART_Init()函数中初始化这两个GPIO口,并在GPIO_UART_SendChar()和GPIO_UART_ReceiveChar()函数中实现了模拟UART的发送和接收功能。在main()函数中,我们首先使用GPIO_UART_SendString()函数发送一条“Hello, world!”的log,然后使用GPIO_UART_ReceiveChar()函数循环接收并打印log。 希望这个示例代码可以帮助您理解如何在STM32上使用GPIO模拟UART来打印log。
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值