找回密码
 立即注册

QQ登录

只需一步,快速开始

搜索
查看: 2442|回复: 6
打印 上一主题 下一主题
收起左侧

请问为什么stc8A系列单片机和ds1302通信,会出现串口数据问号?

[复制链接]
跳转到指定楼层
楼主
连接上了ds1302模块,时间会随机在中间出??这种。


#include <stc8.h>
#define uchar unsigned char
#define uint  unsigned int
//DS1302引脚定义,可根据实际情况自行修改端口定义
sbit RST=P1^3;
sbit IO=P1^2;
sbit SCK=P1^1;
//DS1302地址定义
#define ds1302_sec_add                        0x80                //秒数据地址
#define ds1302_min_add                        0x82                //分数据地址
#define ds1302_hr_add                        0x84                //时数据地址
#define ds1302_date_add                        0x86                //日数据地址
#define ds1302_month_add                0x88                //月数据地址
#define ds1302_day_add                        0x8a                //星期数据地址
#define ds1302_year_add                        0x8c                //年数据地址
#define ds1302_control_add                0x8e                //控制数据地址
#define ds1302_charger_add                0x90                                          
#define ds1302_clkburst_add                0xbe
//初始时间定义
uchar time_buf[8] = {0x20,0x10,0x09,0x14,0x23,0x59,0x50,0x02};//初始时间
uchar readtime[14];//当前时间
uchar sec_buf=0;  //秒缓存
uchar sec_flag=0; //秒标志位
//功能:延时1毫秒
//入口参数:x
//出口参数:无
//说明:晶振为11.059M
void Delay_xms(uint x)
{
  uint i,j;
  for(i=0;i<x;i++)
    for(j=0;j<112;j++);
}
//DS1302初始化函数
void ds1302_init(void)
{
        RST=0;                        //RST脚置低
        SCK=0;                        //SCK脚置低
}
//向DS1302写入一字节数据
void ds1302_write_byte(uchar addr, uchar d)
{
        uchar i;
        RST=1;                                        //启动DS1302总线        
        //写入目标地址:addr
        addr = addr & 0xFE;   //最低位置零,寄存器0位为0时写,为1时读
        for (i = 0; i < 8; i ++) {
                if (addr & 0x01) {
                        IO=1;
                        }
                else {
                        IO=0;
                        }
                SCK=1;      //产生时钟
                SCK=0;
                addr = addr >> 1;
                }        
        //写入数据:d
        for (i = 0; i < 8; i ++) {
                if (d & 0x01) {
                        IO=1;
                        }
                else {
                        IO=0;
                        }
                SCK=1;    //产生时钟
                SCK=0;
                d = d >> 1;
                }
        RST=0;                //停止DS1302总线
}

//从DS1302读出一字节数据
uchar ds1302_read_byte(uchar addr) {

        uchar i,temp;        
        RST=1;                                        //启动DS1302总线
        //写入目标地址:addr
        addr = addr | 0x01;    //最低位置高,寄存器0位为0时写,为1时读
        for (i = 0; i < 8; i ++) {
                if (addr & 0x01) {
                        IO=1;
                        }
                else {
                        IO=0;
                        }
                SCK=1;
                SCK=0;
                addr = addr >> 1;
                }        
        //输出数据:temp
        for (i = 0; i < 8; i ++) {
                temp = temp >> 1;
                if (IO) {
                        temp |= 0x80;
                        }
                else {
                        temp &= 0x7F;
                        }
                SCK=1;
                SCK=0;
                }        
        RST=0;                                        //停止DS1302总线
        return temp;
}
//向DS302写入时钟数据
void ds1302_write_time(void)
{
        ds1302_write_byte(ds1302_control_add,0x00);                        //关闭写保护
        ds1302_write_byte(ds1302_sec_add,0x80);                                //暂停时钟
        //ds1302_write_byte(ds1302_charger_add,0xa9);            //涓流充电
        ds1302_write_byte(ds1302_year_add,time_buf[1]);                //年
        ds1302_write_byte(ds1302_month_add,time_buf[2]);        //月
        ds1302_write_byte(ds1302_date_add,time_buf[3]);                //日
        ds1302_write_byte(ds1302_hr_add,time_buf[4]);                //时
        ds1302_write_byte(ds1302_min_add,time_buf[5]);                //分
        ds1302_write_byte(ds1302_sec_add,time_buf[6]);                //秒
        ds1302_write_byte(ds1302_day_add,time_buf[7]);                //周
        ds1302_write_byte(ds1302_control_add,0x80);                        //打开写保护     
}
//从DS302读出时钟数据
void ds1302_read_time(void)  
{
        time_buf[1]=ds1302_read_byte(ds1302_year_add);                //年
        time_buf[2]=ds1302_read_byte(ds1302_month_add);                //月
        time_buf[3]=ds1302_read_byte(ds1302_date_add);                //日
        time_buf[4]=ds1302_read_byte(ds1302_hr_add);                //时
        time_buf[5]=ds1302_read_byte(ds1302_min_add);                //分
        time_buf[6]=(ds1302_read_byte(ds1302_sec_add))&0x7f;//秒,屏蔽秒的第7位,避免超出59
        time_buf[7]=ds1302_read_byte(ds1302_day_add);                //周         
}
//串口初始化(系统晶振为12MHz)
//void Uart_init(uchar Baud_flag)
//{
//   TMOD = 0x20;       //T1 2
//   PCON = 0x00;       //SMOD = 0
//   SCON = 0x50;              //串口1 8
//   switch(Baud_flag)
//   {           
//    case 0x00:        //2400 11.0592M:0xf4
//          TH1=0xf3;
//          TL1=0xf3;
//        break;
//        case 0x01:        //4800 11.0592M:0xfa
//          TH1=0xf9;
//          TL1=0xf9;
//        break;
//    case 0x02:        //9600 11.0592M:0xfd
//          TH1=0xfd;
//          TL1=0xfd;
//        break;
//    case 0x03:        //19200 11.0592M:0xfe
//      TH1=0xfd;
//          TL1=0xfd;
//    break;
//        default:          //默认为2400
//          TH1=0xf3;
//          TL1=0xf3;
//        break;
//   }
//   TR1 = 1;                  //启动定时器1                                                               
//   ES=1;              //开串口中断
//   EA=1;              //开总中断      
//}
void Uart_init(void)                //9600bps@11.0592MHz
{
        SCON = 0x50;                //8位数据,可变波特率
        AUXR &= 0xBF;                //定时器1时钟为Fosc/12,即12T
        AUXR &= 0xFE;                //串口1选择定时器1为波特率发生器
        TMOD &= 0x0F;                //设定定时器1为16位自动重装方式
        TL1 = 0xE8;                //设定定时初值
        TH1 = 0xFF;                //设定定时初值
        ET1 = 0;                //禁止定时器1中断
        TR1 = 1;                //启动定时器1
    ES=1;              //开串口中断
        EA = 1;
//        SCON = 0x50;                //8位数据,可变波特率
//        AUXR &= 0xBF;                //定时器1时钟为Fosc/12,即12T
//        AUXR &= 0xFE;                //串口1选择定时器1为波特率发生器
//        TMOD &= 0x0F;                //设定定时器1为16位自动重装方式
//        TL1 = 0xA0;                //设定定时初值
//        TH1 = 0xFF;                //设定定时初值
//        ET1 = 0;                //禁止定时器1中断
//        TR1 = 1;                //启动定时器1
//
//            
//        
}

//串口发送
void Send_char(uchar ch)
{
SBUF=ch;
while(!TI);      //等待发送完毕
TI=0;           //清中断
}
//主函数
void main(void)
{
Delay_xms(50);//等待系统稳定
ds1302_init(); //DS1302初始化
Uart_init(); //波特率初始化为9600
Delay_xms(50);
ds1302_write_time(); //写入初始值
Delay_xms(50);
while(1)
  {
  ds1302_read_time();  //读取时间
  readtime[0]=(time_buf[0]>>4);   //分离出年千位
  readtime[1]=(time_buf[0]&0x0F); //分离出年百位
  readtime[2]=(time_buf[1]>>4);   //分离出年十位
  readtime[3]=(time_buf[1]&0x0F); //分离出年个位

  readtime[4]=(time_buf[2]>>4);   //分离出月十位
  readtime[5]=(time_buf[2]&0x0F); //分离出月个位

  readtime[6]=(time_buf[3]>>4);   //分离出日十位
  readtime[7]=(time_buf[3]&0x0F); //分离出日个位

  readtime[8]=(time_buf[4]>>4);   //分离出小时十位
  readtime[9]=(time_buf[4]&0x0F); //分离出小时个位
  readtime[10]=(time_buf[5]>>4);   //分离出分钟十位
  readtime[11]=(time_buf[5]&0x0F); //分离出分钟个位
  readtime[12]=(time_buf[6]>>4);   //分离出秒钟十位
  readtime[13]=(time_buf[6]&0x0F); //分离出秒钟个位   
  if(readtime[13]!=sec_buf)      
   {
    EA=0;
    sec_flag=0;   
        sec_buf=readtime[13];
        Send_char(' ');           
    Send_char(readtime[0]+'0');//年
        Send_char(readtime[1]+'0');     
    Send_char(readtime[2]+'0');
        Send_char(readtime[3]+'0');   

        Send_char('-');

    Send_char(readtime[4]+'0');//月
        Send_char(readtime[5]+'0');

        Send_char('-');

    Send_char(readtime[6]+'0');//日
        Send_char(readtime[7]+'0');

        Send_char(' ');

        Send_char(readtime[8]+'0');//时
        Send_char(readtime[9]+'0');

        Send_char(':');

        Send_char(readtime[10]+'0');//分
        Send_char(readtime[11]+'0');

        Send_char(':');

        Send_char(readtime[12]+'0');//秒
        Send_char(readtime[13]+'0');
        Send_char('\r');
        SBUF='\n';
    while(!TI);      //等待发送完毕
    TI=0;           //清中断
        EA=1;            
   }
   Delay_xms(1000);
  }
}


乱码.png (6.16 KB, 下载次数: 57)

乱码.png
分享到:  QQ好友和群QQ好友和群 QQ空间QQ空间 腾讯微博腾讯微博 腾讯朋友腾讯朋友
收藏收藏 分享淘帖 顶 踩
回复

使用道具 举报

沙发
ID:213173 发表于 2021-1-29 17:06 | 只看该作者
stc8速度太快,端口置位后至少延时两个时钟等待稳定再读,否则易出错。
回复

使用道具 举报

板凳
ID:168165 发表于 2021-1-29 20:29 | 只看该作者
wulin 发表于 2021-1-29 17:06
stc8速度太快,端口置位后至少延时两个时钟等待稳定再读,否则易出错。

请问是延时两个时钟周期吗?我不太懂大概多长时间,我把定时器调成12T的
回复

使用道具 举报

地板
ID:213173 发表于 2021-1-30 16:11 | 只看该作者
无穷小量 发表于 2021-1-29 20:29
请问是延时两个时钟周期吗?我不太懂大概多长时间,我把定时器调成12T的

1T单片机1个_nop_();延时就可以了。如果用到串口,系统时钟要选11.0592或是其整数倍,否则波特率误差大也容易出错。Delay_xms也要改为for(j=0;j<830;j++);。
回复

使用道具 举报

5#
ID:168165 发表于 2021-1-31 12:39 | 只看该作者
wulin 发表于 2021-1-30 16:11
1T单片机1个_nop_();延时就可以了。如果用到串口,系统时钟要选11.0592或是其整数倍,否则波特率误差大也 ...

您好,果然是那个情况,加了好多延时解决了
回复

使用道具 举报

6#
ID:282251 发表于 2021-2-8 21:14 | 只看该作者
我也出现同样问题有没代码可以参考一下
回复

使用道具 举报

7#
ID:115204 发表于 2021-2-9 07:25 | 只看该作者
guo007888 发表于 2021-2-8 21:14
我也出现同样问题有没代码可以参考一下

适当加延时即可
回复

使用道具 举报

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

本版积分规则

手机版|小黑屋|51黑电子论坛 |51黑电子论坛6群 QQ 管理员QQ:125739409;技术交流QQ群281945664

Powered by 单片机教程网

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