新闻中心

EEPW首页>嵌入式系统>设计应用> AVR串口通讯的详解

AVR串口通讯的详解

作者: 时间:2016-11-24 来源:网络 收藏
一.熟悉AVR单片机UART资源,首先从波特率和帧说起

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

本文引用地址://m.amcfsurvey.com/article/201611/320644.htm

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

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

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

UCSRC

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

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

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

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

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

五.工业设计中的串口

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

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

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

我的环境是atmega128

初始化


ucharLED_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-enableinterrupts
}


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

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)

{

//这里填写串口中断处理的内容,可以添加协议,使用状态机就可以了

//把接收到得内容放在缓冲区,然后再创建个处理缓冲区数据的函数,直接放在主循环中即可
}

//函数说明:串口发送完成中断函数
#pragma interrupt_handler uart0_tx_isr:iv_USART0_TXC
void uart0_tx_isr(void)
{
//发送数据的处理函数

}


//函数说明:uart进程函数,放在大循环中
void Uart_Process(void)
{
//接收到得数据,再具体的在系统中实现,比如上位机的监控,或者传输数据等

}


//函数说明:uart测试程序

void Uart_Test(uchar data)
{
UDR0 = 0x01;//发送数据
}


//- 功能描述:串口发送字节的函数
//- 函数属性:外部,使用户使用
//- 参数说明:mydata:要发送的一个字节
//- 返回说明:无
//- 注:发送一个字节,是串口发送的基础操作
void UART_Send_Byte(unsigned char mydata)
{
// UCSR0B = (1< UCSR0B &= ~((1< while(!(UCSR0A &(1< //等待发送缓冲区为空
UDR0 = mydata;
// delay_nms(5);
UCSR0B |= (1< //改的时候不要影响其它寄存器位,开串口中断
}



关键词:AVR串口通

评论


技术专区

关闭