找回密码
 立即注册
查看: 226|回复: 5

串口程序 修改

[复制链接]
  • 打卡等级:常住居民II
  • 打卡总天数:87
  • 最近打卡:2025-04-30 09:11:54
已绑定手机

12

主题

56

回帖

367

积分

中级会员

积分
367
发表于 2025-2-25 10:40:10 | 显示全部楼层 |阅读模式
#include "STC15W4K.H"

#define  TRUE    1

#define  FALSE   0


volatile unsigned char Flag=FALSE;

unsigned char uart1temp;


void UART_init(void)

{                  

        // 下面代码设置定时器1

        TMOD = 0x20;        // 0010 0000 定时器1工作于方式2(8位自动重装方式)

        TH1  = 0xFA;        // 波特率:9600 /22.1184MHZ

        TL1  = 0xFA;        // 波特率:9600 /22.1184MHZ

        TR1  = 1;

        // 下面代码设置定串口

        AUXR = 0x00;             // 很关键,使用定时器1作为波特率发生器,S1ST2=0

        SCON = 0x50;         // 01010 0000 SM0.SM1=01(最普遍的8位通信),REN=1(允许接受)

                                // 下面代码设置中断

        ES   = 1;                // 关键:开启了中断就必须编写相应的中断函数,哪怕是中断空函数,

                               // 但必须有,否则程序进入中断入口地址后(这里是0023H)不能跳出,必然出错

        EA   = 1;

}


/***************************************************************************

* 描  述 : 串口1发送数据函数

* 入  参 : uint8 数据

* 返回值 : 无

**************************************************************************/

void SendDataByUart1(uint8 dat)

{

    SBUF = dat;                 //写数据到UART数据寄存器

    while(TI == 0);             //在停止位没有发送时,TI为0即一直等待

    TI = 0;                     //清除TI位(该位必须软件清零)

}


/***************************************************************************

* 描  述 : 串口1中断服务函数

* 入  参 : 无

* 返回值 : 无

**************************************************************************/

void Uart1() interrupt 4 using 1

{

        ES = 0;                            // 串口1中断关闭

        Flag=TRUE;                  //接收到数据,接收标识符有效

        if (RI)                     //串行接收到停止位的中间时刻时,该位置1

  {

      RI = 0;                 //清除RI位 (该位必须软件清零)

     uart1temp = SBUF;      

   }


     ES =  1;                   // 串口1中断打开

}


/**************************************************************************************

* 描  述 : 串口1接收到数据后发送出去

* 入  参 : 无

* 返回值 : 无

**************************************************************************************/

void UART1_Tx_Puts(void)

{

  if(Flag)                //有新数据通过串口被接收到

        {      

          ES = 0;                                             //串口1中断关闭                  

         SendDataByUart1(uart1temp);       //发送字符

        SendDataByUart1(0x0D);            //发送换行符

        SendDataByUart1(0x0A);            //发送换行符                                               

        ES = 1;                                  //串口1中断打开                       

        Flag=FALSE;                       //清除接收标识符

  }

}


int main()

{

        P3M1 &= 0xFE;        P3M0 &= 0xFE;                          //设置P3.0为准双向口

        P3M1 &= 0xFD;        P3M0 |= 0x02;                          //设置P3.1为推挽输出


        UART_init();                              // 串口初始化

        EA = 1;               //总中断打开


        while(1)

        {

                UART1_Tx_Puts();   //串口接收到一个字符后返回该字符

        }

}


经大神指点,去掉了中断里 TI 部分的语句。

这个程序是之前为了完成功能,从两个程序里拼凑出的一个程序,现在看起来的确有些繁冗。

但对于当时的我来说已经是一个很大的进步了。

串口部分的内容后续还会有很多要学习,我会继续深挖。

感谢大神对我的指导,看到大神对我提出的意见我还是很忐忑的,但是也正是因为忐忑才驱使我进步,是大神让我有机会看到了之前看不到的程序写法。

再次感谢大神的批评和指导,请大神们不吝赐教,继续鞭策、斧正。


回复

使用道具 举报 送花

  • 打卡等级:常住居民II
  • 打卡总天数:87
  • 最近打卡:2025-04-30 09:11:54
已绑定手机

12

主题

56

回帖

367

积分

中级会员

积分
367
发表于 2025-3-10 13:45:55 | 显示全部楼层
去掉中断函数中的TI部分 是不对的。虽然程序能够运行,但是应该有隐患。
回复 支持 反对

使用道具 举报 送花

  • 打卡等级:常住居民II
  • 打卡总天数:93
  • 最近打卡:2025-04-28 14:14:49

4

主题

35

回帖

147

积分

注册会员

积分
147
发表于 2025-3-10 14:12:55 | 显示全部楼层
TI标志位要清除,我已经吃过亏了

回复 支持 反对

使用道具 举报 送花

  • 打卡等级:常住居民II
  • 打卡总天数:87
  • 最近打卡:2025-04-30 09:11:54
已绑定手机

12

主题

56

回帖

367

积分

中级会员

积分
367
发表于 2025-3-11 09:18:52 | 显示全部楼层
nkgd*** 发表于 2025-3-10 14:12
TI标志位要清除,我已经吃过亏了

回复 支持 反对

使用道具 举报 送花

  • 打卡等级:以坛为家II
  • 打卡总天数:523
  • 最近打卡:2025-05-03 00:05:21

10

主题

1159

回帖

5125

积分

论坛元老

积分
5125
发表于 2025-3-11 10:08:04 | 显示全部楼层
用官方例程试试。

/*************        功能说明        **************

双串口全双工中断方式收发通讯程序。

通过PC向MCU发送数据, MCU收到后通过串口把收到的数据原样返回.

******************************************/


#define         MAIN_Fosc                        22118400L        //定义主时钟
#include        "STC15Fxxxx.H"

#define                Baudrate1                        115200L
#define                UART1_BUF_LENGTH        32


u8        TX1_Cnt;        //发送计数
u8        RX1_Cnt;        //接收计数
bit        B_TX1_Busy;        //发送忙标志

u8         idata RX1_Buffer[UART1_BUF_LENGTH];        //接收缓冲


void        UART1_config(u8 brt);        // 选择波特率, 2: 使用Timer2做波特率, 其它值: 使用Timer1做波特率.
void         PrintString1(u8 *puts);


//========================================================================
// 函数: void main(void)
// 描述: 主函数。
// 参数: none.
// 返回: none.
// 版本: VER1.0
// 日期: 2014-11-28
// 备注:
//========================================================================
void main(void)
{
        P0M1 = 0;        P0M0 = 0;        //设置为准双向口
        P1M1 = 0;        P1M0 = 0;        //设置为准双向口
        P2M1 = 0;        P2M0 = 0;        //设置为准双向口
        P3M1 = 0;        P3M0 = 0;        //设置为准双向口
        P4M1 = 0;        P4M0 = 0;        //设置为准双向口
        P5M1 = 0;        P5M0 = 0;        //设置为准双向口
        P6M1 = 0;        P6M0 = 0;        //设置为准双向口
        P7M1 = 0;        P7M0 = 0;        //设置为准双向口

        UART1_config(1);        // 选择波特率, 2: 使用Timer2做波特率, 其它值: 使用Timer1做波特率.
        EA = 1;        //允许总中断

        PrintString1("STC15F2K60S2 UART1 Test Prgramme!\r\n");        //SUART1发送一个字符串

        while (1)
        {
                if((TX1_Cnt != RX1_Cnt) && (!B_TX1_Busy))        //收到数据, 发送空闲
                {
                        SBUF = RX1_Buffer[TX1_Cnt];                //把收到的数据远样返回
                        B_TX1_Busy = 1;
                        if(++TX1_Cnt >= UART1_BUF_LENGTH)        TX1_Cnt = 0;
                }
        }
}


//========================================================================
// 函数: void PrintString1(u8 *puts)
// 描述: 串口1发送字符串函数。
// 参数: puts:  字符串指针.
// 返回: none.
// 版本: VER1.0
// 日期: 2014-11-28
// 备注:
//========================================================================
void PrintString1(u8 *puts)        //发送一个字符串
{
    for (; *puts != 0;        puts++)           //遇到停止符0结束
        {
                SBUF = *puts;
                B_TX1_Busy = 1;
                while(B_TX1_Busy);
        }
}

//========================================================================
// 函数: SetTimer2Baudraye(u16 dat)
// 描述: 设置Timer2做波特率发生器。
// 参数: dat: Timer2的重装值.
// 返回: none.
// 版本: VER1.0
// 日期: 2014-11-28
// 备注:
//========================================================================
void        SetTimer2Baudraye(u16 dat)        // 选择波特率, 2: 使用Timer2做波特率, 其它值: 使用Timer1做波特率.
{
        AUXR &= ~(1<<4);        //Timer stop
        AUXR &= ~(1<<3);        //Timer2 set As Timer
        AUXR |=  (1<<2);        //Timer2 set as 1T mode
        TH2 = dat / 256;
        TL2 = dat % 256;
        IE2  &= ~(1<<2);        //禁止中断
        AUXR |=  (1<<4);        //Timer run enable
}

//========================================================================
// 函数: void        UART1_config(u8 brt)
// 描述: UART1初始化函数。
// 参数: brt: 选择波特率, 2: 使用Timer2做波特率, 其它值: 使用Timer1做波特率.
// 返回: none.
// 版本: VER1.0
// 日期: 2014-11-28
// 备注:
//========================================================================
void        UART1_config(u8 brt)        // 选择波特率, 2: 使用Timer2做波特率, 其它值: 使用Timer1做波特率.
{
        /*********** 波特率使用定时器2 *****************/
        if(brt == 2)
        {
                AUXR |= 0x01;                //S1 BRT Use Timer2;
                SetTimer2Baudraye(65536UL - (MAIN_Fosc / 4) / Baudrate1);
        }

        /*********** 波特率使用定时器1 *****************/
        else
        {
                TR1 = 0;
                AUXR &= ~0x01;                //S1 BRT Use Timer1;
                AUXR |=  (1<<6);        //Timer1 set as 1T mode
                TMOD &= ~(1<<6);        //Timer1 set As Timer
                TMOD &= ~0x30;                //Timer1_16bitAutoReload;
                TH1 = (u8)((65536UL - (MAIN_Fosc / 4) / Baudrate1) / 256);
                TL1 = (u8)((65536UL - (MAIN_Fosc / 4) / Baudrate1) % 256);
                ET1 = 0;        //禁止中断
                INT_CLKO &= ~0x02;        //不输出时钟
                TR1  = 1;
        }
        /*************************************************/

        SCON = (SCON & 0x3f) | 0x40;        //UART1模式, 0x00: 同步移位输出, 0x40: 8位数据,可变波特率, 0x80: 9位数据,固定波特率, 0xc0: 9位数据,可变波特率
//        PS  = 1;        //高优先级中断
        ES  = 1;        //允许中断
        REN = 1;        //允许接收
        P_SW1 &= 0x3f;
        P_SW1 |= 0x80;                //UART1 switch to, 0x00: P3.0 P3.1, 0x40: P3.6 P3.7, 0x80: P1.6 P1.7 (必须使用内部时钟)
//        PCON2 |=  (1<<4);        //内部短路RXD与TXD, 做中继, ENABLE,DISABLE

        B_TX1_Busy = 0;
        TX1_Cnt = 0;
        RX1_Cnt = 0;
}


//========================================================================
// 函数: void UART1_int (void) interrupt UART1_VECTOR
// 描述: UART1中断函数。
// 参数: nine.
// 返回: none.
// 版本: VER1.0
// 日期: 2014-11-28
// 备注:
//========================================================================
void UART1_int (void) interrupt 4
{
        if(RI)
        {
                RI = 0;
                RX1_Buffer[RX1_Cnt] = SBUF;
                if(++RX1_Cnt >= UART1_BUF_LENGTH)        RX1_Cnt = 0;        //防溢出
        }

        if(TI)
        {
                TI = 0;
                B_TX1_Busy = 0;
        }
}



回复 支持 反对

使用道具 举报 送花

  • 打卡等级:常住居民II
  • 打卡总天数:87
  • 最近打卡:2025-04-30 09:11:54
已绑定手机

12

主题

56

回帖

367

积分

中级会员

积分
367
发表于 2025-3-11 13:09:41 | 显示全部楼层
angm*** 发表于 2025-3-11 10:08
用官方例程试试。

/*************        功能说明        **************

这不是冲哥的程序吗?
回复 支持 反对

使用道具 举报 送花

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

QQ|手机版|小黑屋|深圳国芯人工智能有限公司 ( 粤ICP备2022108929号-2 )

GMT+8, 2025-5-3 01:01 , Processed in 0.164602 second(s), 84 queries .

Powered by Discuz! X3.5

© 2001-2025 Discuz! Team.

快速回复 返回顶部 返回列表