微波EDA网,见证研发工程师的成长!
首页 > 硬件设计 > 嵌入式设计 > AVR串口通讯的详解

AVR串口通讯的详解

时间:11-24 来源:互联网 点击:
一.熟悉AVR单片机UART资源,首先从波特率和帧说起

波特率:与51不同有单独的波特率发生器,不需要定时器来产生,节省了资源
波特率计算公式,这里我采用IccAVR的配置功能,直接计算生成
单片机支持的模式:异步正常模式,异步倍数模式,和同步模式,一般选第一种模式
帧格式:起始位+数据位(5-9位可选)+校验位(可选)+停止位(1、2位) 空闲
o ********* P 1 1
通讯电路空闲时为高电平

二、大概了解了下硬件资源后,就要了解软硬件的桥梁—寄存器了
1.数据寄存器:数据来了要有个接受的地方吧,数据发送要有个数据发送的信封吧。
这就是数据寄存器 UDR (RXB和TXB) 物理上为分开的,地址上是一样的。就像写信和接信时,你家的地址只

有同一个地址一样,但是写信和别人发给你的信封却有2个一样。使用时自动控制的。
数据寄存器为空时才能发送数据,否则会无效。数据进入后,进入移位寄存器,由引脚TXD一位位发出。
2.控制和状态寄存器 UCSRA
RXC TXC UDRE FE DOR PE U2X MPCM
接受完成置1 发送完成置1 数据为空标志 帧错误1 接受数据 校验位错误 倍速模式 多机通信
读取数据清0 中断时自动清0 数据完全到移 溢出1 1 1 地址位
位寄存器中1
UCSRB 设置相关中断的允许
RXCIE TXCIE UDRIE RXEN TXEN UXSZ2 RXB8 TXB8
接受中断允许 发送中断允许 空中断允许 数据接受允许 数据发送允许 位数设置 接受第9位

RXEN,TXEN设置时会改变时普通IO口,或者是当做复用口用,在发送数据时设置下,数据全部发送后才生效
RXB8,TXB8需要先读写出

UCSRC

URSEL UMSEL UPM1 UPM0 USBS UCSZ1 UCSZ0 UCPOL
寄存器选择 工作模式 校验方式 停止位 和上面的UXSZ2设置
写时需要设为1 1异步 数据个数
存在共用寄存器 00禁止11奇10偶 0为1个1为2个

波特率寄存器:UBRRL UBRRH
UBRRH和UCSRC共用 了底4位 加上UBRRL共12位 设置后正在传的数据会被打断
需要注意URSEL的设置 :0 写入的比特率高4位
1 写入的是寄存器的内容
读UBRRH,第一次是比特率的内容 在连续的2个时钟周期内再读一次就是UCRSC的内容

使用时可以查速查表,直接用ICCAVR生成工具即可。

三,相关操作 等练习个程序后补上
1 初始化 关全局中断
TXC RXC看数据是否完成 发送数据前TXC必须请零
把数据放入到发送缓冲器即可 UDR中 5-8位
2 注意下空中断 允许后需要不断写数据 否则一直不断的中断产生 一般禁止就可以了
TXEN 设置0后 所有数据发送后生效 然后就当普通IO口用了
禁止接受 会立即 丢失数据

四一些总线标准
RS232 9针D型接口
1 -3 ~-25 0 3-25V
需要使用电平转换电路 MAX232

五.工业设计中的串口

这里如果大家看到了还是注意下比较好 ,网上写的不多的,我也是从工程实践和查找大量的参考书中总结的

设计思路是基于状态机,并自定义了协议,同时协议中使用了CRC校验,和简易的加密技术

思路是:主从方式,上位机发送数据包,下位机在中断中接受,接受数据时要一位位的确认,并不断进行切换,发送的位置状态,把初步确认的数据放到接受缓冲器里,等所有的数据接收好了,程序进入大循环了,就执行主程序中加入uart操作函数,这个函数首先判断主机发送的命令和设置是否接受完成,在完成的状态下进行校验正确性,校验后,根据主机命令,组装数据包和存贮主机的设置数据,并把需要发送的数据包或设置完成数据包放在需要发送的数据缓冲区,接着改变下此时的状态:为我组好了,准备发送数据了,接着触发中断,可以直接往串口发个数据即可,正常发送后,单片机会执行其他程序,等上位机接受到这个触发数据后,下位机会中断,中断程序会根据状态,一位位的发生缓冲区的数据,直到所有的数据发送完成,发送完成后还要置位到接受数据状态0。

我的环境是atmega128

初始化

uchar LED_Temp=0xFF;
uchar OUT_temp=0x04;
static uchar Uart_Status;
static uchar R_Data_Lenth;

uchar Tx_Buf[TxBufSize];
uchar Rx_Buf[RxBufSize];
uchar *P_Uart_Rx;
uchar *P_Uart_Tx;
uchar Rx_Count;
uchar Tx_Count;

void Uart_Init(void)
{
//UCSR0B = 0x00; //先关闭
UCSR0A=0x00;
UCSR0C=0x06; //8 DATA ,1 STOP, NO PARITY
UCSR0B = (1 // RXCIE=1;TXCIE=1;UDREIE=0;RXEN=1;TXEN=1
Com_baudrate (9600);
P_Uart_Tx=Tx_Buf; //缓冲区指针定义
P_Uart_Rx=Rx_Buf;
Uart_Status=0; //开始时状态为接受起始位状态,其实这里是因为我在程序中用了通信协议

//本篇为基础,就把协议的内容删去了,仅仅提供了能运行的最简单的框架
SEI(); //re-enable interrupts
}

//函数说明:波特率设置

void Com_baudrate (unsigned int baudrate)
{
unsigned int tmp;
tmp= 8000000/baudrate/16-1;
UBRR0H=(unsigned char)(tmp>>8);
UBRR0L=(unsigned char)tmp;
}

//函数说明:串口接收中断函数

#pragma interrupt_handler uart0_rx_isr:iv_USART0_RXC
void uart0_rx_isr(void)

Copyright © 2017-2020 微波EDA网 版权所有

网站地图

Top