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; //清忙标志
}
}
根据您提供的信息,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变量初始化、波特率配置、中断使能以及数据传递逻辑,以解决串口通信异常的问题。
[本文内容由国芯人工智能辅助生成,仅供参考]
//本示例在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++); //发送当前字符
}
}
这里多了个等号:
串口程序供参考:
https://www.stcaimcu.com/thread-16321-1-1.html
https://www.stcaimcu.com/thread-4598-1-1.html DebugLab 发表于 2025-8-7 18:53
这里多了个等号:
串口程序供参考:
这个等号是我前面改成全局变量,忘记删了,去掉等号也不行,很神奇的现象。 国芯人工智能 发表于 2025-8-7 17:08
根据您提供的信息,STC15W204S串口在发送数据时出现异常,表现为当通过外部传递的数值进行发送时,接收端始 ...
EA已经开了,busy已经初始化=0,都还是不行 可以直接烧录官方例子到芯片里测试,串口收到外面传进来是数据,原样输出。
对比输入输出内容就知道收发是否正常。
乘风飞扬 发表于 2025-8-8 14:50
可以直接烧录官方例子到芯片里测试,串口收到外面传进来是数据,原样输出。
对比输入输出内容就知道收发是 ...
好的,谢谢,我试试看 乘风飞扬 发表于 2025-8-8 14:50
可以直接烧录官方例子到芯片里测试,串口收到外面传进来是数据,原样输出。
对比输入输出内容就知道收发是 ...
烧录范例这个程序不行,电脑串口发送,无法收到回复,程序能正常烧录,证明硬件应该没有问题,很奇怪的现象。
页:
[1]
2