proteus仿真之DS1302+8位数码管显示试验
来源:互联网 发布:淘宝dsr评分计算器 编辑:程序博客网 时间:2024/06/05 16:27
proteus仿真之DS1302+8位数码管显示试验。
仿真效果如下图:
源程序如下:
/*
51单片机:DS1302+数码管 Proteus 仿真程序。
功能:数码管时钟显示。
仿真结果:
(1)8位数码管显示设定的时间与日期。
(2)通过按键切换时间与日期的显示。
*/
#include <reg52.h>
sbit LE_DUAN = P2^0; //定义573锁存使能端口 数码管段锁存
sbit LE_WEI = P2^1; //定义573锁存使能端口 数码管位锁存
#define SEGPORT P0 //定义数码管连接端口
sbit SCK = P3^6; // DS1302时钟线
sbit SDA = P3^4; // DS1302数据线
sbit RST = P3^5; // DS1302复位线
//DS1302 复位重定义
#define RST_CLR RST=0 //电平置低
#define RST_SET RST=1 //电平置高
//DS1302 数据
#define SDA_CLR SDA=0 //电平置低
#define SDA_SET SDA=1 //电平置高
#define SDA_RD SDA //电平读取
//DS1302 时钟
#define SCK_CLR SCK=0 //时钟信号
#define SCK_SET SCK=1 //电平置高
#define DS1302_SEC 0x80 //秒数据地址
#define DS1302_MIN 0x82 //分数据地址
#define DS1302_HOUR 0x84 //时数据地址
#define DS1302_DATE 0x86 //日数据地址
#define DS1302_MON 0x88 //月数据地址
#define DS1302_DAY 0x8a //星期数据地址
#define DS1302_YEAR 0x8c //年数据地址
#define DS1302_CTRL 0x8e //控制数据地址
#define DS1302_CHARGE 0x90 //涓流充电
bit ReadRTC_Flag; //读DS1302标志。1为读 0为不读。
unsigned char TimeMode; //日期与时间切换标志。
unsigned char time_buf1[8] = {40,14,2,14,10,59,50,7}; // -年月日时分秒周 2014-02-14 10:59:50 7周
unsigned char time_buf[8] ; // -年月日时分秒周
unsigned char TempData[8] ;
unsigned char code Seg_Wei[] = {0xfe,0xfd,0xfb,0xf7,0xef,0xdf,0xbf,0x7f}; //数码管的位码,低电平有效。
unsigned char code Seg_Duan[] = {0x3f,0x06,0x5b,0x4f,0x66,0x6d,0x7d,0x07,0x7f,0x6f,0x40}; //共阳数码管显示段码值 0~9,-
void DS1302_Init(void);
void DS1302_Write_Byte(unsigned char addr, unsigned char d);
unsigned char DS1302_Read_Byte(unsigned char addr) ;
void DS1302_Read_Time(void);
void DS1302_Write_Time(void);
void Seg_Disp(unsigned char FirstBit,unsigned char Num);
void InitTIMER0(void);//inital timer0
void Delay_1ms(unsigned int i);
unsigned char GetKey(void);
void main(void)
{
unsigned char Key;
InitTIMER0();
DS1302_Init();
DS1302_Write_Time();
P2=0xff; //51默认为输入
while(1)
{
Key = GetKey();
if(Key&0x01)
{
TimeMode = ~TimeMode; //有按键后,模式取反,改变当前显示模式
}
if(ReadRTC_Flag==1)
{
ReadRTC_Flag=0;
DS1302_Read_Time();
if(TimeMode)
{
TempData[0]=Seg_Duan[time_buf1[1]/10]; //年 数据的转换,
TempData[1]=Seg_Duan[time_buf1[1]%10]; //因我们采用数码管0~9的显示,将数据分开
TempData[2]=0x40; //加入"-"
TempData[3]=Seg_Duan[time_buf1[2]/10]; //月
TempData[4]=Seg_Duan[time_buf1[2]%10];
TempData[5]=0x40;
TempData[6]=Seg_Duan[time_buf1[3]/10]; //日
TempData[7]=Seg_Duan[time_buf1[3]%10];
}
else
{
TempData[0]=Seg_Duan[time_buf1[4]/10]; //时 数据的转换,
TempData[1]=Seg_Duan[time_buf1[4]%10]; //因我们采用数码管0~9的显示,将数据分开
TempData[2]=0x40; //加入"-"
TempData[3]=Seg_Duan[time_buf1[5]/10]; //分
TempData[4]=Seg_Duan[time_buf1[5]%10];
TempData[5]=0x40;
TempData[6]=Seg_Duan[time_buf1[6]/10]; //秒
TempData[7]=Seg_Duan[time_buf1[6]%10];
}
}
}
}
/*------------------------------------------------
DS1302初始化
------------------------------------------------*/
void DS1302_Init(void)
{
RST_CLR; //RST脚置低
SCK_CLR; //SCK脚置低
DS1302_Write_Byte(DS1302_SEC,0x00);
}
/*------------------------------------------------
向DS1302写入一字节数据
------------------------------------------------*/
void DS1302_Write_Byte(unsigned char addr, unsigned char dat)
{
unsigned char i;
RST_SET;
addr = addr & 0xFE; //写地址 最低位为W写,低电平
for (i = 0; i < 8; i++)
{
if (addr & 0x01)
{
SDA_SET;
}
else
{
SDA_CLR;
}
SCK_SET;
SCK_CLR;
addr = addr >> 1;
}
//写入数据:dat
for (i = 0; i < 8; i ++)
{
if (dat & 0x01)
{
SDA_SET;
}
else
{
SDA_CLR;
}
SCK_SET;
SCK_CLR;
dat = dat >> 1;
}
RST_CLR; //停止DS1302总线
}
/*------------------------------------------------
从DS1302读出一字节数据
------------------------------------------------*/
unsigned char DS1302_Read_Byte(unsigned char addr)
{
unsigned char i;
unsigned char temp;
RST_SET;
addr = addr | 0x01; //最低RD,有效为高电平
for (i = 0; i < 8; i ++)
{
if (addr & 0x01)
{
SDA_SET;
}
else
{
SDA_CLR;
}
SCK_SET;
SCK_CLR;
addr = addr >> 1;
}
//输出数据:temp
for (i = 0; i < 8; i ++)
{
temp = temp >> 1;
if (SDA_RD)
{
temp |= 0x80;
}
else
{
temp &= 0x7F;
}
SCK_SET;
SCK_CLR;
}
RST_CLR; //停止DS1302总线
return temp;
}
/*------------------------------------------------
从DS1302读出时钟数据
------------------------------------------------*/
void DS1302_Read_Time(void)
{
unsigned char i,tmp;
time_buf[1]=DS1302_Read_Byte(DS1302_YEAR); //年
time_buf[2]=DS1302_Read_Byte(DS1302_MON); //月
time_buf[3]=DS1302_Read_Byte(DS1302_DATE); //日
time_buf[4]=DS1302_Read_Byte(DS1302_HOUR); //时
time_buf[5]=DS1302_Read_Byte(DS1302_MIN); //分
time_buf[6]=(DS1302_Read_Byte(DS1302_SEC))&0x7F;//秒
time_buf[7]=DS1302_Read_Byte(DS1302_DAY); //周
for(i=0;i<8;i++)
{ //BCD处理
tmp=time_buf[i]/16;
time_buf1[i]=time_buf[i]%16;
time_buf1[i]=time_buf1[i]+tmp*10;
}
}
/*------------------------------------------------
向DS1302写入时钟数据
------------------------------------------------*/
void DS1302_Write_Time(void)
{
unsigned char i,tmp;
for(i=0;i<8;i++)
{ //BCD处理
tmp=time_buf1[i]/10;
time_buf[i]=time_buf1[i]%10;
time_buf[i]=time_buf[i]+tmp*16;
}
DS1302_Write_Byte(DS1302_CTRL,0x00); //关闭写保护
DS1302_Write_Byte(DS1302_SEC,0x80); //暂停
//DS1302_Write_Byte(DS1302_CHARGE,0xa9); //涓流充电
DS1302_Write_Byte(DS1302_YEAR,time_buf[1]); //年
DS1302_Write_Byte(DS1302_MON,time_buf[2]); //月
DS1302_Write_Byte(DS1302_DATE,time_buf[3]); //日
DS1302_Write_Byte(DS1302_HOUR,time_buf[4]); //时
DS1302_Write_Byte(DS1302_MIN,time_buf[5]); //分
DS1302_Write_Byte(DS1302_SEC,time_buf[6]); //秒
DS1302_Write_Byte(DS1302_DAY,time_buf[7]); //周
DS1302_Write_Byte(DS1302_CTRL,0x80); //打开写保护
}
/*------------------------------------------------
数码管显示, pos为起始位
------------------------------------------------*/
void Seg_Disp(unsigned char pos,unsigned char Num)
{
static unsigned char i=0;
SEGPORT=0; //清空数据,防止有交替重影
LE_DUAN=1; //段锁存
LE_DUAN=0;
SEGPORT=Seg_Wei[i+pos]; //取位码
LE_WEI=1; //位锁存
LE_WEI=0;
SEGPORT=TempData[i]; //取显示数据,段码
LE_DUAN=1; //段锁存
LE_DUAN=0;
i++;
if(i==Num)
i=0;
}
/*------------------------------------------------
Timer0 初始化,开中断,2ms定时
------------------------------------------------*/
void InitTIMER0(void)
{
TMOD|=0x01; //定时器设置 16位
TH0=(65536-2000)/256; //定时时间 2ms
TL0=(65536-2000)%256;
EA=1;
ET0=1;
TR0=1;
}
/*------------------------------------------------
简单延时程序:约1ms
------------------------------------------------*/
void delay_1ms(unsigned int i)
{
unsigned char j;
while(i--)
for(j=0;j<125; j++);
}
/*------------------------------------------------
按键扫描 这里按键接在 P2.2
------------------------------------------------*/
unsigned char GetKey(void)
{
unsigned char KeyTemp,CheckValue,Key=0x00;
CheckValue = P2&0x04;
if(CheckValue==0x04)
return 0x00;
delay_1ms(10);
KeyTemp = P2&0x04;
if(KeyTemp == CheckValue)
return 0x00;
if(!(CheckValue&0x04))
Key = 0x01;
return Key;
}
void tim0_isr(void) interrupt 1//中断,用于数码管扫描
{
static unsigned char num;
TH0=(65536-2000)/256; //重新赋值 2ms
TL0=(65536-2000)%256;
Seg_Disp(0,8);
num++;
if(num==50) //大致100ms
{
num=0;
ReadRTC_Flag=1; //读标志位置1
}
}
以上程序在Keil +proteus7.8中通过。
- proteus仿真之DS1302+8位数码管显示试验
- proteus仿真之8位数码管动态扫描显示试验
- proteus仿真之DS1302+LCD1602显示试验
- 基于proteus的51单片机仿真实例六十、8位数码管显示实例
- 基于proteus的51单片机仿真实例七十八、使用BCD译码器实现8位数码管显示应用实例
- 单片机按键与数码管试验程序 Proteus仿真
- 基于proteus的51单片机仿真实例五十九、1位数码管显示实例
- 基于proteus的51单片机仿真实例七十九、8位数码管驱动芯片max7221应用实例
- 1~99秒倒计时数码管显示C程序+Proteus仿真
- ds1302 数码管显示 C51
- DS1302时钟数码管显示
- proteus 数码管动态显示问题解决
- 8位数码管显示其中之一
- 8位数码管静态显示其中之二
- 仿真LED数码管显示控件
- DS18B20之Proteus仿真使用
- 8位数码管动态扫描显示
- 两位数码管显示
- 编程建议
- eclipse 10个最常用快捷键
- Aircrack-ng官方文档翻译[中英对照]---Airdecap-ng
- Linux Kernel模块内存泄露的一种查找思路
- linux 解压命令
- proteus仿真之DS1302+8位数码管显示试验
- 多个摄像头同步工作
- VC++ CopyFile函数用法
- 玩转Web之JavaScript(二)-----javaScript语法总结(二) 涉及Date与数组的语法
- 黑马程序员----IO流
- 利用npoi导出excel
- Android 4.0 截屏(Screenshot)代码流程小结
- android Socket使用笔记
- Coding4Fun的4个 ValueConverter 示例