非常感谢2位大哥指点,我好象对发送FIFO有点儿明白了.我正在做UART的实验,我现在就是想先用上接收中断,CTI中断,使能FIFO,禁止发送中断.就是想我给ARM发什么,ARM就能在上位机再给返回什么. 现在的实验现象是这样的:(设触发深度是8),发低于8个字符没问题,发挺多的也很好,就两种情况会出错,当发8的整数倍+1个字符和发8的整数倍+2个字符时就不行了.第一种情况会少2个字符,第二种情况会少1个字符.比如我想发123456789,会显示1234569;要发1234567890,会显示123456790. 真的是不明白,我感觉程序应该没错啊,别的时候一切都那么完美,咋就一到那种情况就不行呢?我本来想着是如果这样能调好,再加上发送中断,那UART的硬件及中断机制就都用上了吧.但现在还没加发送中断都拿不下,还请大家帮我看看哪里错了,谢谢大伙.程序如下: /****************************************Copyright (c)************************************************** ** Guangzou ZLG-MCU Development Co.,LTD. ** graduate school ** http://www.zlgmcu.com ** **--------------File Info------------------------------------------------------------------------------- ** File name: main.c ** Last modified Date: 2004-09-16 ** Last Version: 1.0 ** Descriptions: The main() function example template ** **------------------------------------------------------------------------------------------------------ ** Created by: Chenmingji ** Created date: 2004-09-16 ** Version: 1.0 ** Descriptions: The original version ** **------------------------------------------------------------------------------------------------------ ** Modified by: Chenxibing ** Modified date: 2005-01-17 ** Version: ** Descriptions: UART0通讯实验,中断方式,使用FIFO。 ** ********************************************************************************************************/ #include "config.h" /* 定义串口模式设置数据结构 */ typedef struct UartMode { uint8 datab; // 字长度,5/6/7/8可选 uint8 stopb; // 停止位,1/2可选 uint8 parity; // 奇偶校验位,0-无校验,1-奇校验,2-偶校验 }UARTMODE; char *rcv_buf; // UART0数据接收缓冲区 volatile uint8 rcv_new; // 接收新数据标志 uint8 sum; //计算字符串中字符的个数 /********************************************************************************************************** ** 函数名称 :IRQ_UART0() ** 函数功能 :串口0接收中断服务程序 ** 入口参数 :无 ** 出口参数 :无 ********************************************************************************************************* */ void __irq IRQ_UART0 (void) { uint8 IIR; uint8 i=0; while(((IIR = U0IIR) & 0x01) == 0) //有中断未处理完 { rcv_new = 1; // 设置接收到新的数据标志 switch (IIR & 0x0e) { case 0x04: for (;(i==0)||((i%8)!=0) ; i++) rcv_buf = U0RBR; // 读取FIFO的数据,并清除中断 break; case 0x0c: for (;(U0LSR&01)!=0;i++) rcv_buf = U0RBR; break; default: break; } } sum=i; VICVectAddr = 0x00; // 中断处理结束 } /* ********************************************************************************************************* ** 函数名称 :UART0_SendByte() ** 函数功能 :向串口0发送1字节数据 ** 入口参数 :dat 要发送的数据 ** 出口参数 :无 ********************************************************************************************************* */ void UART0_SendByte (uint8 dat) { U0THR = dat; // 要发送的数据 while ((U0LSR & 0x20) == 0); // 等待数据发送完毕 } /* ********************************************************************************************************* ** 函数名称 :UART0_SendStr() ** 函数功能 :向串口0发送1字节数据 ** 入口参数 :dat 要发送的数据 ** 出口参数 :无 ********************************************************************************************************* */ void UART0_SendStr(char const *str) {uint8 i; for(i=0;i<sum;i++) UART0_SendByte(*str++); } /* ********************************************************************************************************* ** 函数名称 :UART0_SendBuf() ** 函数功能 :向串口发送11个字符串数据 ** 入口参数 :dat 要发送的数据 ** 出口参数 :无 ********************************************************************************************************* */ void UART0_SendBuf (void) { UART0_SendStr(rcv_buf); } /* ********************************************************************************************************* ** 函数名称 :UART0_Init() ** 函数功能 :串口初始化,设置工作模式和波特率。 ** 入口参数 :baud 波特率 ** set 模式设置(UARTMODE数据结构) ** 出口参数 :1-初始化成功, 0-初始化失败 ********************************************************************************************************* */ int8 UART0_Init (uint32 baud, UARTMODE set) { uint32 bak; /* 参数过滤 */ if ((baud ==0 ) || (baud > 115200)) return (0); if ((set.datab <5) || (set.datab > 8)) return (0); if ((set.stopb == 0) || (set.stopb > 2)) return (0); if (set.parity > 4) return (0); /* 设置串口波特率 */ U0LCR = 0x80; // DLAB = 1 bak = (Fpclk >> 4) / baud; U0DLM = bak >> 8; U0DLL = bak & 0xFF; /* 设置串口模式 */ bak = set.datab - 5; // 设置字长 if (set.stopb == 2) bak |= 0x04; // 判断是否为2位停止位 if (set.parity != 0) { set.parity = set.parity - 1; bak |= 0x08; } bak |= set.parity << 4; // 设置奇偶校验 U0LCR = bak; return (1); }
/* ********************************************************************************************************* ** 函数名称 :main() ** 函数功能 :从串口UART0接收字符串"ABCDEFGH",并发送回上位机显示。 ** 调试说明 :需要PC串口显示终端软件如EasyARM.exe。 ********************************************************************************************************* */ int main (void) { UARTMODE set; set.datab = 8; set.stopb = 1; set.parity = 0; rcv_new = 0; PINSEL0 = 0x00000005; // 设置I/O连接到UART0 UART0_Init(115200,set); // 串口初始化 U0FCR = 0x81; // 使能FIFO,并设置触发点为8字节 U0IER = 0x01; // 允许RBR中断(即接收中断),禁止THRE中断(发送中断) IRQEnable(); // 使能IRQ中断 /* 使能UART0中断 */ VICIntSelect = 0x00000000; // 设置所有的通道为IRQ中断 VICVectCntl0 = 0x20 | 0x06; // UART0分配到IRQ slot0,即最高优先级 VICVectAddr0 = (uint32)IRQ_UART0; // 设置UART0向量地址 VICIntEnable = 1 << 0x06; // 使能UART0中断 while (1) { if (rcv_new==1) { rcv_new=0; UART0_SendBuf(); } } return 0; } /********************************************************************************************************* ** End Of File ********************************************************************************************************/ |