无花果1986 发表于 2025-8-7 17:04:10

15W204S串口无法工作

做了个很简单的板子,就串口发个数据,居然不正常。具体现象是,如果在SendData()程序中直接给SBUF赋值,数据就正确,如果通过外部传递进来的数,就不正常,不管传递进来是什么数,电脑串口接收到的永远是0X8F。

----------------------------*/
void Delay100ms(void)      //@11.0592MHz
{
      unsigned char data i, j, k;
      _nop_();
      _nop_();
      i = 5;
      j = 52;
      k = 195;
      do
      {
                do
                {
                        while (--k);
                } while (--j);
      } while (--i);
}
void SendData(unsigned char dat)
{
    while (busy);               //等待前面的数据发送完成
    busy = 1;         
    SBUF = dat;                  //获取校验位P (PSW.0)
}                                       
void main()
{
    SCON = 0x50;                //8位可变波特率
    T2L = (65536 - (FOSC/4/BAUD));   //设置波特率重装值
    T2H = (65536 - (FOSC/4/BAUD))>>8;
    AUXR = 0x14;                //T2为1T模式, 并启动定时器2
    AUXR |= 0x01;               //选择定时器2为串口1的波特率发生器
    ES = 1;                     //使能串口1中断
    EA = 1;
      P55=0;
    while(1)
      {      
                SendData=(0x32);
                Delay100ms();      
      }
}


void Uart() interrupt 4
{
    if (RI)
    {
      RI = 0;               //清除RI位
    }
    if (TI)
    {
      TI = 0;               //清除TI位
      busy = 0;               //清忙标志
    }
}

国芯人工智能 发表于 2025-8-7 17:08:00

根据您提供的信息,STC15W204S串口在发送数据时出现异常,表现为当通过外部传递的数值进行发送时,接收端始终接收到0x8F。而直接赋值给SBUF时则正常,这表明问题可能出在数据传递过程中或与寄存器状态控制有关。

首先,检查代码逻辑是否正确。您在SendData函数中使用了busy标志位来判断串口是否空闲,但未对busy变量进行初始化。若该变量初始值为非零,可能导致程序误判串口状态,从而引发发送异常。建议在main()函数开始处对busy进行初始化:

c
unsigned char busy = 0;

其次,波特率设置部分可能存在错误。您使用了T2L和T2H来配置定时器2的重载值,但未明确定义FOSC和BAUD的值。如果这些宏定义不正确,将导致波特率计算错误,进而影响串口通信的稳定性。请确保FOSC为实际使用的晶振频率(如11.0592MHz),并合理设置BAUD值(如9600)。

此外,SCON寄存器配置为0x50,表示串口工作在方式1(8位异步串行通信),但需确认是否启用了串口中断。您在代码中设置了ES = 1;,但未启用全局中断EA = 1;,这可能导致中断无法触发,进而影响数据发送。建议在main()中添加:

c
EA = 1;

最后,检查外部传递的数据是否被正确处理。若传递的数值在函数内部被修改或未正确传递,可能导致发送数据异常。建议在SendData函数中添加调试输出,验证传入的dat值是否正确。

综上所述,建议您检查busy变量初始化、波特率配置、中断使能以及数据传递逻辑,以解决串口通信异常的问题。

[本文内容由国芯人工智能辅助生成,仅供参考]

神农鼎 发表于 2025-8-7 17:32:34





//本示例在Keil开发环境下请选择Intel的8058芯片型号进行编译
//若无特别说明,工作频率一般为11.0592MHz


#include "reg51.h"
#include "intrins.h"

typedef unsigned char BYTE;
typedef unsigned int WORD;

#define FOSC 11059200L          //系统频率
#define BAUD 115200             //串口波特率

#define NONE_PARITY   0       //无校验
#define ODD_PARITY      1       //奇校验
#define EVEN_PARITY   2       //偶校验
#define MARK_PARITY   3       //标记校验
#define SPACE_PARITY    4       //空白校验

#define PARITYBIT NONE_PARITY   //定义校验位

sfr P0M1 = 0x93;
sfr P0M0 = 0x94;
sfr P1M1 = 0x91;
sfr P1M0 = 0x92;
sfr P2M1 = 0x95;
sfr P2M0 = 0x96;
sfr P3M1 = 0xb1;
sfr P3M0 = 0xb2;
sfr P4M1 = 0xb3;
sfr P4M0 = 0xb4;
sfr P5M1 = 0xC9;
sfr P5M0 = 0xCA;
sfr P6M1 = 0xCB;
sfr P6M0 = 0xCC;
sfr P7M1 = 0xE1;
sfr P7M0 = 0xE2;

sfr AUXR= 0x8e;               //辅助寄存器
sfr T2H   = 0xd6;               //定时器2高8位
sfr T2L   = 0xd7;               //定时器2低8位

sfr P_SW1   = 0xA2;             //外设功能切换寄存器1

#define S1_S0 0x40            //P_SW1.6
#define S1_S1 0x80            //P_SW1.7

sbit P22 = P2^2;

bit busy;

void SendData(BYTE dat);
void SendString(char *s);

void main()
{
    P0M0 = 0x00;
    P0M1 = 0x00;
    P1M0 = 0x00;
    P1M1 = 0x00;
    P2M0 = 0x00;
    P2M1 = 0x00;
    P3M0 = 0x00;
    P3M1 = 0x00;
    P4M0 = 0x00;
    P4M1 = 0x00;
    P5M0 = 0x00;
    P5M1 = 0x00;
    P6M0 = 0x00;
    P6M1 = 0x00;
    P7M0 = 0x00;
    P7M1 = 0x00;

    ACC = P_SW1;
    ACC &= ~(S1_S0 | S1_S1);    //S1_S0=0 S1_S1=0
    P_SW1 = ACC;                //(P3.0/RxD, P3.1/TxD)
   
//ACC = P_SW1;
//ACC &= ~(S1_S0 | S1_S1);    //S1_S0=1 S1_S1=0
//ACC |= S1_S0;               //(P3.6/RxD_2, P3.7/TxD_2)
//P_SW1 = ACC;
//
//ACC = P_SW1;
//ACC &= ~(S1_S0 | S1_S1);    //S1_S0=0 S1_S1=1
//ACC |= S1_S1;               //(P1.6/RxD_3, P1.7/TxD_3)
//P_SW1 = ACC;

#if (PARITYBIT == NONE_PARITY)
    SCON = 0x50;                //8位可变波特率
#elif (PARITYBIT == ODD_PARITY) || (PARITYBIT == EVEN_PARITY) || (PARITYBIT == MARK_PARITY)
    SCON = 0xda;                //9位可变波特率,校验位初始为1
#elif (PARITYBIT == SPACE_PARITY)
    SCON = 0xd2;                //9位可变波特率,校验位初始为0
#endif

    T2L = (65536 - (FOSC/4/BAUD));   //设置波特率重装值
    T2H = (65536 - (FOSC/4/BAUD))>>8;
    AUXR = 0x14;                //T2为1T模式, 并启动定时器2
    AUXR |= 0x01;               //选择定时器2为串口1的波特率发生器
    ES = 1;                     //使能串口1中断
    EA = 1;

    SendString("STC15F2K60S2\r\nUart Test !\r\n");
    while(1);
}

/*----------------------------
UART 中断服务程序
-----------------------------*/
void Uart() interrupt 4
{
    if (RI)
    {
      RI = 0;               //清除RI位
      P0 = SBUF;            //P0显示串口数据
      P22 = RB8;            //P2.2显示校验位
    }
    if (TI)
    {
      TI = 0;               //清除TI位
      busy = 0;               //清忙标志
    }
}

/*----------------------------
发送串口数据
----------------------------*/
void SendData(BYTE dat)
{
    while (busy);               //等待前面的数据发送完成
    ACC = dat;                  //获取校验位P (PSW.0)
    if (P)                      //根据P来设置校验位
    {
#if (PARITYBIT == ODD_PARITY)
      TB8 = 0;                //设置校验位为0
#elif (PARITYBIT == EVEN_PARITY)
      TB8 = 1;                //设置校验位为1
#endif
    }
    else
    {
#if (PARITYBIT == ODD_PARITY)
      TB8 = 1;                //设置校验位为1
#elif (PARITYBIT == EVEN_PARITY)
      TB8 = 0;                //设置校验位为0
#endif
    }
    busy = 1;
    SBUF = ACC;               //写数据到UART数据寄存器
}

/*----------------------------
发送字符串
----------------------------*/
void SendString(char *s)
{
    while (*s)                  //检测字符串结束标志
    {
      SendData(*s++);         //发送当前字符
    }
}


神农鼎 发表于 2025-8-7 17:34:46


DebugLab 发表于 2025-8-7 18:53:06

这里多了个等号:

串口程序供参考:
https://www.stcaimcu.com/thread-16321-1-1.html
https://www.stcaimcu.com/thread-4598-1-1.html

无花果1986 发表于 2025-8-7 19:06:53

DebugLab 发表于 2025-8-7 18:53
这里多了个等号:

串口程序供参考:


这个等号是我前面改成全局变量,忘记删了,去掉等号也不行,很神奇的现象。

无花果1986 发表于 2025-8-7 19:09:57

国芯人工智能 发表于 2025-8-7 17:08
根据您提供的信息,STC15W204S串口在发送数据时出现异常,表现为当通过外部传递的数值进行发送时,接收端始 ...

EA已经开了,busy已经初始化=0,都还是不行

乘风飞扬 发表于 2025-8-8 14:50:35

可以直接烧录官方例子到芯片里测试,串口收到外面传进来是数据,原样输出。
对比输入输出内容就知道收发是否正常。

无花果1986 发表于 2025-8-8 17:05:15

乘风飞扬 发表于 2025-8-8 14:50
可以直接烧录官方例子到芯片里测试,串口收到外面传进来是数据,原样输出。
对比输入输出内容就知道收发是 ...

好的,谢谢,我试试看

无花果1986 发表于 2025-8-8 20:00:15

乘风飞扬 发表于 2025-8-8 14:50
可以直接烧录官方例子到芯片里测试,串口收到外面传进来是数据,原样输出。
对比输入输出内容就知道收发是 ...

烧录范例这个程序不行,电脑串口发送,无法收到回复,程序能正常烧录,证明硬件应该没有问题,很奇怪的现象。
页: [1] 2
查看完整版本: 15W204S串口无法工作