STC15W408AS的485串口实现自发自收

STC15W408AS的485串口实现自发自收

485串口

STC的坑

STC单片机发送数据给485串口的时候,发数据一般都没有问题,但是收数据的时候,一般收不到。我改动了两版电路板,才得到正确的方式。STC新版的单片机和以前的不一样了。部分不具备直接驱动485串口的能力了。
本次例程采用的单片机为STC15W408AS
不多说,直接上程序和原理图。
这是我的一个项目程序,这里只给出部分代码,需要完成代码的点击下面:
链接:https://pan.baidu.com/s/1JsI3CQcK9qkFAscv32SHqA
提取码:7a2y
程序:
main.c
/************* 本地常量声明 **************/
#define MAIN_Fosc 11059200L //定义主时钟
#define BaudRate1 9600UL //选择波特率

#define Timer1_Reload (65536UL -(MAIN_Fosc / 4 / BaudRate1)) //Timer 1 重装值, 对应300KHZ
#define Timer2_Reload (65536UL -(MAIN_Fosc / 4 / BaudRate1)) //Timer 2 重装值, 对应300KHZ

#include “STC15Fxxxx.H”
#include “i2c.h”
#include “intrins.h”
#include “stdio.h”

#define BELL P14
#define Control P55
#define Change P33
/************* 本地变量声明 **************/
int timeCount=0;
u8 UsartRX[20]={0,0,0,0,0,0,0,0,0,0};
u8 UsartTX[20]={0,0,0,0,0,0,0,0,0,0};
u16 UsartTime=0;
u8 UsartLength=0;
u8 UsartFlag=0;

/************* 本地函数声明 **************/
void init_Time0();
void delay_10US(u32 US);
char putchar(char ch);
u16 Get_ADC10bitResult(u8 channel);
//
void main(void)
{
u8 i=0,flag=0;
u16 AD=0,data1=0,data2=0;
u32 address=0;
init_Time0();
P3M1=0X60;
P3M0=0X9F;
P5M1=0x00;
P5M0=0xFF;
S1_8bit(); //8位数据
S1_USE_P36P37(); //UART1 使用P30 P31口 默认
AUXR &= ~(1<<4); //Timer stop 波特率使用Timer2产生
AUXR |= 0x01; //S1 BRT Use Timer2;
AUXR |= (1<<2); //Timer2 set as 1T mode
TH2 = (u8)(Timer2_Reload >> 8);
TL2 = (u8)Timer2_Reload;
AUXR |= (1<<4); //Timer run enable
REN = 1; //允许接收
ES = 1; //允许中断
EA = 1; //允许全局中断
/
************************************/
//数据初始化
// 蜂鸣器
BELL=0;
//继电器
Control=0;
//启动看门狗
WDT_CONTR=0x3f;
P54=0;
while (1)
{
//复位单片机
if(timeCount%100)
{
WDT_CONTR&=0x7f;
WDT_CONTR|=0x10;
}
else if(timeCount>1000)
{
timeCount=0;
AD=Get_ADC10bitResult(2);
//printf(“电压:%d\n”,AD);
}
//判断是否接收到数据
if((UsartTime>10)&&(UsartFlag==1))
{
UsartFlag=0;

		P54=1;
		delay_10US(60);
		for(i=0;i<UsartLength;i++)
		{
			TI=0;
			if(!TI)
			{
				SBUF=UsartRX[i];			
			}
			TI=1;
			delay_10US(5);
		}
		P54=0;
		UsartLength=0;
		}
	}

}

/********************* UART1中断函数************************/
/********************* UART1中断函数************************/
void UART1_int (void) interrupt UART1_VECTOR
{
TR0=0;
if(RI)
{
RI=0;
//判断收到的数据是否为第一位
if(UsartTime>=10)
{
UsartTime=0;
UsartRX[0]=SBUF;
UsartLength=1;
UsartFlag=1;
}
else
{

			UsartTime=0;
			UsartRX[UsartLength]=SBUF;
			UsartLength++;
			UsartFlag=1;
	}
		
}

TR0=1;
}
/***************************************************
**功 能:定时器0初始化函数
**返回值:无
**参 数:无
****************************************************/
void init_Time0()
{
TMOD|=0X01;//定时器的工作模式为1,计数的位数为16位

TH0=0XFC;
TL0=0X65; //计数的个数为922,定时为1US
EA=1;//打开总中断
ET0=1;//定时器T0允许中断
TR0=1;//定时器T0开始工作

}
/***************************************************
**功 能:定时器0中断函数
**返回值:无
**参 数:无
****************************************************/
void interrupt_time0(void) interrupt 1
{
TH0=0XFC;
TL0=0X65; //计数的个数为922,定时为1US
timeCount++;
UsartTime++;
if(UsartTime>100)
{
UsartTime=11;
}
}

void delay_10US(u32 US)
{
unsigned int i;
do{
i = MAIN_Fosc / 1300000;
while(–i) ; //14T per loop
}while(–US);
}

char putchar(char ch)
{
SBUF = ch;
while(TI == 0);
TI = 0;
return ch;
}
/***************************************************
**功 能:得到电压值
**返回值:所在通道的电压值
**参 数:channel为所在的通道
****************************************************/
u16 Get_ADC10bitResult(u8 channel) //channel = 0~7
{
//AD引脚定义配置
P1ASF= 0x03; //P1.2 P1.3做ADC
ADC_CONTR = 0xE0; //90T, ADC power on
ADC_RES = 0;
ADC_RESL = 0;

ADC_CONTR = (ADC_CONTR & 0xe0) | 0x08 | channel; 	//start the ADC
NOP(4);

while((ADC_CONTR & 0x10) == 0)	;	//wait for ADC finish
ADC_CONTR &= ~0x10;		//清除ADC结束标志
P1ASF= 0x00;		//P1.2 P1.3做ADC
ADC_CONTR = 0x00;	//90T, ADC power on

// P1M1=0X02; //配置SDA为输入
// P1M0=0X00;
return (((u16)ADC_RES << 2) | (ADC_RESL & 3));
}
原理图部分,下图电阻R18 R15 R21不焊接。
在这里插入图片描述

  • 11
    点赞
  • 8
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

社会我仁哥

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值