lv11 嵌入式开发 UART实验 11

目录

1 UART帧格式详解

1.1 UART简介

1.2 通信基础 - 并行和串行

1.3 通信基础 - 单工和双工

1.4 通信基础 - 波特率 

1.5 UART帧格式

2 Exynos4412下的UART控制器

2.1 引脚功能设置

2.2 阅读芯片手册

3 UART寄存器详解

3.1 引脚寄存器

3.2 串口寄存器概览

3.3 ULCONn(帧格式) 

​编辑 3.4 UCONn控制寄存器(串口模式轮询、中断、DMA)

3.5 状态寄存器 

3.6 接收数据,发送数据的寄存器 

3.7 设置波特率

4 UART编程

4.1 interface.c实验代码

4.2 printf重定向

4.3 练习


1 UART帧格式详解

1.1 UART简介

Universal Asynchronous Receiver Transmitter 即     通用异步收发器,是一种通用的串行、异步通信总线     该总线有两条数据线,可以实现全双工的发送和接收     在嵌入式系统中常用于主机与辅助设备之间的通信

1.2 通信基础 - 并行和串行

并行通讯(多个数据线一起发送)

lv11 嵌入式开发 UART实验 11_第1张图片

串行通信(一个个字节发送)

lv11 嵌入式开发 UART实验 11_第2张图片

1.3 通信基础 - 单工和双工

lv11 嵌入式开发 UART实验 11_第3张图片

半双工要不能同时发送和接收,全双工可以同时发送和接收。

1.4 通信基础 - 波特率 

波特率用于描述UART通信时的通信速度,其单位为bps(bit per second)即每秒钟传送的bit的数量

1.5 UART帧格式

lv11 嵌入式开发 UART实验 11_第4张图片

串口协议中规定,先发低位再发高位

例:数据0x55 二进制01010101->发送顺序10101010

校验位很少使用,多发了一位数据,会使得通讯速度变慢

如果发送多个字节重复上面的动作,必须一个一个字节发送,附带停止位和起始位,否则发送0xFF这种数据,全是高电平,无法得知是发送了还是没发送。

串口发送的速率是根据自身的时钟速率掐时间计算。如A主机发送速率1s1个,但是接收方可能接收的数据对于A来说是0.9S1个,时间一长数据会乱。所以一般规定每次最多发8位,之后重新获取起始位,避免误差增大。

lv11 嵌入式开发 UART实验 11_第5张图片

2 Exynos4412下的UART控制器

lv11 嵌入式开发 UART实验 11_第6张图片

一般情况下处理器中都会集成UART控制器 我们使用UART进行通信时候只需对其内部的相 关寄存器进行设置即可

2.1 引脚功能设置

查看扩展板原理图

lv11 嵌入式开发 UART实验 11_第7张图片

SP3232EEA芯片的作用是把串口发出来的TTL信号转换成232信号,增加了串口的通讯距离

查看核心板原理图 

lv11 嵌入式开发 UART实验 11_第8张图片

2.2 阅读芯片手册

芯片手册4412一共包含了5个串口通道,包含一个可以与GPS通讯的Ch4。串口控制器的时钟是100Mhz,支持DMA,串口中断,每个串口包含FIFO。

 lv11 嵌入式开发 UART实验 11_第9张图片

功能框图

lv11 嵌入式开发 UART实验 11_第10张图片

总线是CPU与串口芯片相连,发送、接收及访问控制单元。

红外GPS暂不详细介绍

3 UART寄存器详解

3.1 引脚寄存器

lv11 嵌入式开发 UART实验 11_第11张图片

3.2 串口寄存器概览

5个串口寄存器内部是一样的,使用串口2基地址0x1382_0000

lv11 嵌入式开发 UART实验 11_第12张图片

3.3 ULCONn(帧格式) 

lv11 嵌入式开发 UART实验 11_第13张图片 3.4 UCONn控制寄存器(串口模式轮询、中断、DMA)

 

lv11 嵌入式开发 UART实验 11_第14张图片lv11 嵌入式开发 UART实验 11_第15张图片

接收串口数据一般有3种方式:

1.轮询,定期CPU去访问FIFO

2.中断,有数据来了通知CPU去读取

3.DMA,有数据来了直接由DMA搬到内存中,不需要CPU去干涉

3.5 状态寄存器 

状态寄存器中一般为只读寄存器,可用于判断串口接收发送的状态

lv11 嵌入式开发 UART实验 11_第16张图片

3.6 接收数据,发送数据的寄存器 

lv11 嵌入式开发 UART实验 11_第17张图片

3.7 设置波特率

lv11 嵌入式开发 UART实验 11_第18张图片

波特率公式,案例用的时钟频率是40Mhz,我们开发板的串口时钟寄存器是100Mhz

lv11 嵌入式开发 UART实验 11_第19张图片

UBRDIVn = 20(整数部分)

UFRACVAL =0.7(小数部分)*16 ≈ 11(四舍五入),因为串口按位发送会校准所以可以忽略

4 UART编程

 引脚寄存器

lv11 嵌入式开发 UART实验 11_第20张图片

帧格式 

lv11 嵌入式开发 UART实验 11_第21张图片

接收和发送模式为轮询 

lv11 嵌入式开发 UART实验 11_第22张图片

100000000/(115200 *16)-1 ≈ 53.25

UBDIV2 = 取整 = 53

UFRAVVAL2 = 0.25 * 16 = 4

4.1 interface.c实验代码


#include "exynos_4412.h"

void UART_Init(void)
{
	/*1.将GPA1_0和GPA1_1设置成UART2的接收和发送引脚 GPA1CON[7:0]*/
	GPA1.CON = GPA1.CON & (~(0xFF << 0)) | (0x22 << 0);
	/*2.设置UART2的帧格式 8位数据位 1位停止位 无校验 正常模式 ULCON2[6:0]*/
	UART2.ULCON2 = UART2.ULCON2 & (~(0x7F << 0)) | (0x3 << 0);
	/*3.设置UART2的接收和发送模式为轮询模式 UCON2[3:0]*/
	UART2.UCON2 = UART2.UCON2 & (~(0xF << 0)) | (0x5 << 0);
	/*4.设置UART2的波特率为115200 UBRDIV2/UFRACVAL2*/
	UART2.UBRDIV2 = 53;
	UART2.UFRACVAL2 = 4;
}

void UART_Send_Byte(char Dat)
{
	/*等待发送寄存器为空,即上一个数据已经发送完成 UTRSTAT2[1]*/
	while(!(UART2.UTRSTAT2 & (1 << 1)));
	/*将要发送的数据写入发送寄存器 UTXH2*/
	UART2.UTXH2 = Dat;
}

char UART_Rec_Byte(void)
{
	char Dat = 0;
	/*判断接收寄存器是否接收到了数据 UTRSTAT2[0]*/
	if(UART2.UTRSTAT2 & 1)
	{
		/*从接收寄存器中读取接收到的数据 URXH2*/
		Dat = UART2.URXH2;
		return Dat;
	}
	else
	{
		return 0;
	}
}

void UART_Send_Str(char * pstr)
{
	while(*pstr != '\0')
		UART_Send_Byte(*pstr++);
}

int main()
{
	char RecDat = 0;
	UART_Init();

	while(1)
	{
		
		RecDat = UART_Rec_Byte();
		if(RecDat == 0)
		{
			
		}
		else
		{
			RecDat = RecDat + 1;
			UART_Send_Byte(RecDat);
		}
		
		
		
		UART_Send_Str("Hello World\n");
		
        //重定向的printf
		printf("Hello World\n");
	}
	return 0;
}

注:

寄存器只能显示字符,无法显示数字,如果需要发送和接收数字要转换为对应的字符的ASCII码 

如果CPU写入的速度过快,超过串口发送器发送的速度会使得接收端的数据错乱。需要对发送状态寄存器进行判断后进行发送和接收。

lv11 嵌入式开发 UART实验 11_第23张图片

4.2 printf重定向

lv11 嵌入式开发 UART实验 11_第24张图片

printf的功能实现已经再src中实现了,函数比较复杂,不做详细介绍

printf调用了puts,puts内部调用了putc,我们修改了putc的程序实现了重定向

定向功能在uart.c中实现。

lv11 嵌入式开发 UART实验 11_第25张图片

printf的区别

1来源:没有操作系统在printf.c中,而有操作系统是调用的C库的区别

2输出定向不一样:重定向输出到串口,输出到linux终端

4.3 练习

1.若使用UART协议发送一个字节的数据0x63,画出信号线上的时序图

注:8位数据位、无校验位、一位停止位



2.编程实现电脑远程控制LED状态

注:在终端上输入‘2’,LED2点亮,再次输入‘2’,LED2熄灭... ...


#include "exynos_4412.h"

void LED_Init(void)
{
	GPX2.CON = GPX2.CON & (~(0xF << 28)) | (0x1 << 28); //LED2 GPX2_7 output

	//GPX1.CON = GPX1.CON & (~0xF) | 0x1;                 //LED3 GPX1_0 output

	//GPF3.CON = GPF3.CON & (~(0xF << 16)) | (0x1 << 16); //LED4 GPF3_4 output

	//GPF3.CON = GPF3.CON & (~(0xF << 20)) | (0x1 << 20); //LED5 GPF3_5 output
}

void Led_on(int num)

{

	switch(num)

	{

		case 2:

			GPX2.DAT = GPX2.DAT | (1 << 7);

		case 3:

			GPX1.DAT = GPX1.DAT | (1 << 0);

		case 4:

			GPF3.DAT = GPF3.DAT | (1 << 4);

		case 5:

			GPF3.DAT = GPF3.DAT | (1 << 5);

		default:

			break;

	}

}



void Led_off(int num)

{

	switch(num)

	{

		case 2:

			GPX2.DAT = GPX2.DAT & ~(1 << 7);

		case 3:

			GPX1.DAT = GPX1.DAT & ~(1 << 0);

		case 4:

			GPF3.DAT = GPF3.DAT & ~(1 << 4);

		case 5:

			GPF3.DAT = GPF3.DAT & ~(1 << 5);

		default:

			break;

	}

}


void UART_Init(void)
{
	/*1.将GPA1_0和GPA1_1设置成UART2的接收和发送引脚 GPA1CON[7:0]*/
	GPA1.CON = GPA1.CON & (~(0xFF << 0)) | (0x22 << 0);
	/*2.设置UART2的帧格式 8位数据位 1位停止位 无校验 正常模式 ULCON2[6:0]*/
	UART2.ULCON2 = UART2.ULCON2 & (~(0x7F << 0)) | (0x3 << 0);
	/*3.设置UART2的接收和发送模式为轮询模式 UCON2[3:0]*/
	UART2.UCON2 = UART2.UCON2 & (~(0xF << 0)) | (0x5 << 0);
	/*4.设置UART2的波特率为115200 UBRDIV2/UFRACVAL2*/
	UART2.UBRDIV2 = 53;
	UART2.UFRACVAL2 = 4;
}

void UART_Send_Byte(char Dat)
{
	/*等待发送寄存器为空,即上一个数据已经发送完成 UTRSTAT2[1]*/
	while(!(UART2.UTRSTAT2 & (1 << 1)));
	/*将要发送的数据写入发送寄存器 UTXH2*/
	UART2.UTXH2 = Dat;
}

char UART_Rec_Byte(void)
{
	char Dat = 0;
	/*判断接收寄存器是否接收到了数据 UTRSTAT2[0]*/
	if(UART2.UTRSTAT2 & 1)
	{
		/*从接收寄存器中读取接收到的数据 URXH2*/
		Dat = UART2.URXH2;
		return Dat;
	}
	else
	{
		return 0;
	}
}

void UART_Send_Str(char * pstr)
{
	while(*pstr != '\0')
		UART_Send_Byte(*pstr++);
}

int main()
{
	char RecDat = 0;
	int toggle = 0;
	UART_Init();
	LED_Init();

	while(1)
	{
		
		RecDat = UART_Rec_Byte();
		if(RecDat == '2')
		{
			if(toggle == 0)
			{
				Led_on(2);
				toggle = 1;
			}
			else
			{
				Led_off(2);
				toggle = 0;
			}
		}

		
		/*
		UART_Send_Str("Hello World\n");
		*/

		
	}
	return 0;
}

你可能感兴趣的:(嵌入式开发,嵌入式硬件,arm开发)