1 uart硬件介绍#
UART的全称是Universal Asynchronous Receiver and Transmitter(异步收发器)。 uart主要用于:
1 | 1.打印调试 |
串口通过三根线即可,发送、接收、地线。
1 | pc的TxD -> arm的RxD (UART write) |
2 uart的参数和格式#
1 | 波特率:表示每秒传输多少bit,bits per second(bps).一般波特率都会有9600,19200,115200等选项。 |
3 UART传输原理#
3.1 如何传输一个字符A#
‘A’的ASCII值是0x41#,二进制就是01000001,怎样把这8位数据发送给PC机呢?
1.双方约定好波特率(每一位占据的时间)
2.规定传输协议
现在处于这种模式:arm的TxD -> pc的RxD (UART read)
1.arm拉低uart总线1bit的时间(起始位)
2.arm根据数据位依次驱动TxD的电平,同时PC依次读取uart总线,数据到达PC的RxD引脚,pc依次获得数据位
为了能够进行远距离的传输数据,我们的PC是使用的RS-232逻辑电平,而arm开发板使用的TTL/CMOS逻辑电平。这里先讲解下什么是TTL逻辑电平,什么是RS-232逻辑电平。
TTL/CMOS逻辑电平:
0(低电平0-0.7v)表示逻辑'0'
1(高电平2-5v) 表示逻辑'1'
RS-232逻辑电平:
(+3V ~ +12V) 表示逻辑'0'
(-12V ~ -3V) 表示逻辑'1'
TTL逻辑电平的波形:
RS232逻辑电平的波形:
那么在起始信号开始后开始计时,arm每隔一个时钟往TxD放1bit数据,同时pc也从RxD get 1bit数据.
arm pc
TxD=data[0:], data[0:]=RxD
TxD=data[1:], data[1:]=RxD
...
TxD=data[7:], data[7:]=RxD
3.1.1 RS232#
我们知道RS232的逻辑’0’和逻辑’1’相差较大,比TTL/CMOS差距大,那么逻辑电平不容易出现反转,能传输更远的距离,在工业上用得比较多。
所以我们上面PC拿到的数据是不对的,那么需要一个TTL转RS232的电平转换芯片。
4 UART控制器#
发送数据:
内存将数据放入发送FIFO(64byte),通过发送移位器将数据一位一位的依次发送到TXDn,这样PC就可以从总线上依次get到数据。
接收数据:
当pc的TXDn端将数据发送到总线后,arm获取RXDn的引脚电平依次get到数据,逐位放进接收移位器,再放入FIFO,写入内存。
当然,也可不使用fifo,直接让内存与移位器交互,不过这样会造成浪费内存资源,内存的频率是很高滴,降低了内存的吞吐量。
5 UART控制器编程#
s3c2440支持3个UART串口,以uart0为例讲解。
那么我们需要实现以下这几个函数完成串口的最基本功能:
1 | (1)uart0_init()用于初始化串口 |
5.1 初始化UART#
5.1.1 引脚初始化#
配置uart0引脚
- 根据原理图GPH2,3用于TxD0, RxD0。
- 查看dataset,配置GPH控制寄存器,让GPH2,3配成uart模式;为了将其保持为高电平,先设置其为上拉。
1
2
3GPHCON &= ~((3<<4) | (3<<6));
GPHCON |= ((2<<4) | (2<<6));
GPHUP &= ~((1<<2) | (1<<3)); /* 使能内部上拉 */
5.1.2 UART控制器初始化#
5.1.2.1 设置时钟源#
1 | UCON0 = 0x00000005; /* 时钟源选择PCLK,中断/查询模式 */ |
5.1.2.2 设置波特率#
1 | /* uart clock=50M,假设我们想要波特率=115200, |
5.1.2.3 设置数据格式#
数据格式设置为常用的8n1,表示8个数据位, 无较验位, 1个停止位
1 | ULCON0 = 0x00000003; /* 8n1: 8个数据位(数据+校验), 无较验位, 1个停止位 */ |
5.2 putchar/getchar#
putchar就是向发送寄存器(UTXH0)写入值进去。
getchar就是从接受寄存器(URXH0)取出值。
无论是getchar还是putchar都可以通过读取状态寄存器(UTRSTAT0)来作为传输结束判断标志。
查询其第2位判断发送buff是否为空,即上一次发送是否完成,如果完成即向UTXH0写入要发送的新数据;
查询其第0位判断接收buff是否有数据接受到,如果有数据接收到,返回接收buffer的值。
1 | int putchar(int c){ |
5.3 puts#
1 | int puts(const char *s){ |