串口1中断服务函数的解析

//串口1中断服务函数的解析
void USART1_IRQHandler(void)                
u8 Res;


if(USART_GetITStatus(USART1, USART_IT_RXNE) != RESET)   //接收到的数据必须是0x0d 0x0a结尾,RESET=0
{
Res =USART_ReceiveData(USART1);                                  //读取接收到的数据

if((USART_RX_STA&0x8000)==0)                                           //接收未完成
{
if(USART_RX_STA&0x4000)                                            //接收到了0x0d
{
if(Res!=0x0a)USART_RX_STA=0;                          //接收错误重新开始
else USART_RX_STA|=0x8000;                       //接收完成了
}
else                                                                                     //还没接收到0x0d
{
if(Res==0x0d)USART_RX_STA|=0x4000;
else
{
USART_RX_BUF[USART_RX_STA&0X3FFF]=Res ;
USART_RX_STA++;
if(USART_RX_STA>(USART_REC_LEN-1))USART_RX_STA=0;   //在接收数据超过 USART_REC_LEN 的时候,则会
 丢弃前面的数据,重新接收。 

}  
}
}    
     } 
 
 
// typedef enum {RESET = 0, SET = !RESET} FlagStatus, ITStatus;              //RESET=0;
// u16 USART_RX_STA=0;                                                                                    //接收状态标记


///
 当接收到从电脑发过来的数据,把接收到的数据保存在 USART_RX_BUF 中,
 同时在接收状态寄存器(USART_RX_STA)中计数接收到的有效数据个数,当
 收到回车(回车的表示由 2 个字节组成:0X0D 和 0X0A)的第一个字节 0X0D 
 时,计数器将不再增加,等待0X0A 的到来,而如果 0X0A 没有来到,则认为
 这次接收失败,重新开始下一次接收。如果顺利接收到 0X0A,则标记 USART_RX_STA 
 的第 15 位,这样完成一次接收,并等待该位被其他程序清除,从而开始下一次的接收,
 而如果迟迟没有收到 0X0D,那么在接收数据超过 USART_REC_LEN 的时候,则会
 丢弃前面的数据,重新接收。
 
      计算机向串口发送一串字符,一般不止一个,例如发送”abcdefg回车“。那么串口中断
      函数会执行9次,回车要执行两次串口中断。
      当串口中断函数第一次执行时,USART1->DR里面装的是字符a,下面以串口第一次执行来分析这
      个串口中断函数。

if(USART_GetITStatus(USART1, USART_IT_RXNE) != RESET);      //这是判断读数据寄存器是否空,因为接受到了a,所以不是空的,这个判断成立。


Res =USART_ReceiveData(USART1);                                                //(USART1->DR); 既然接受到了字符a,那么就要把他读取出来


if((USART_RX_STA&0x8000)==0);                //因为现在接受的是第一个字符,所以接收肯定没有完成,USART_RX_STA还是它的初始化值,于是第15位还是0,
这个判断语句成立。于是要执行下面这句话  

  
if(USART_RX_STA&0x4000) ;                       // USART_RX_STA的第14位仍然是0,所以这个判断不成立,所以会执行下面这句话   


if(Res==0x0d);                                               //当然这个判断也不成立,所以要执行下面这句话
 
USART_RX_BUF[USART_RX_STA&0X3FFF]=Res ;     //计算出接收的是第几个字符,然后装到缓存里面
                                                                                                 //0x3fff=0011 1111 1111 1111



USART_RX_STA++;                                                            //加1表明已经接收好了几个数据
  • 44
    点赞
  • 170
    收藏
    觉得还不错? 一键收藏
  • 2
    评论
//出入均设有缓冲区,大小可任意设置。 //可供使用的函数名: //char getbyte(void);从接收缓冲区取一个byte,如不想等待则在调用前检测inbufsign是否为1。 //getline(char idata *line, unsigned char n); 获取一行数据回车结束,必须定义最大输入字符数 //putbyte(char c);放入一个字节到发送缓冲区 //putbytes(unsigned char *outplace,j);放一串数据到发送缓冲区,自定义长度 //putstring(unsigned char code *puts);发送一个定义在程序存储区的字符串到串口 //puthex(unsigned char c);发送一个字节的hex码,分成两个字节发。 //putchar(uchar c,uchar j);输出一个无符号字符数的十进制表示,必须标示小数点的位置,自动删除前面无用的零 //putint(uint ui,uchar j);输出一个无符号整型数的十进制表示,必须标示小数点的位置,自动删除前面无用的零 //delay(unsigned char d); 延时n x 100ns //putinbuf(uchar c);人工输入一个字符到输入缓冲区 //CR;发送一个回车换行 //************************************************************************* #include <w77e58.h> #define uchar unsigned char #define uint unsigned int #define OLEN 32 /* size of serial transmission buffer */ idata unsigned char outbuf[OLEN]; /* storage for transmission buffer */ unsigned char idata *outlast=outbuf; //最后由中断传输出去的字节位置 unsigned char idata *putlast=outbuf; //最后放入发送缓冲区的字节位置 #define ILEN 12 /* size of serial receiving buffer */ idata unsigned char inbuf[ILEN]; unsigned char idata *inlast=inbuf; //最后由中断进入接收缓冲区的字节位置 unsigned char idata *getlast=inbuf; //最后取走的字节位置 bit outbufsign0; //最后一个数据覵BUF发完标志 发完=0 bit outbufsign; //输出缓冲区非空标志 有=1 bit inbufsign; //接收缓冲区非空标志 有=1 bit inbufful; //输入缓冲区满标志 满=1 #define CR putstring("\r\n") //CR=回车换行 //***************************** //延时n x 100ns void delay(unsigned char d) //在源程序开头定义是否用w77e58或22。1184M晶振 {unsigned char j; do{ d--; //110592 & 89c52 #ifndef cpuw77e58 #ifndef xtal221184 j=21; //k=38 cpu80320 100us k="21" cpu 8052 #else j=42; #endif #else #ifndef xtal221184 j=38; #else j=76;

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值