亲 上次那个ds1302的程序和仿真我已收到,我想请教一下

亲 上次那个ds1302的程序和仿真我已收到,我想请教一下,第1张

你好 楼上说的不错 就是用判断语句 好久没看了 不是很熟悉大致给你说一下:

比如 你想10:20 响蜂鸣器

if(TimeHour==10 & TimeMinute==20) //10点20到

beep=1;//此处执行既定功能,或调用子程序

建议:

1、插入主程序循环的末端,

2、合理处理子程序(功能)返回参数

3、如是蜂鸣器 还分有人干预 和 无人干预(响2-3分钟,自停)

总之 实现的方法很多 把这些功能编一个子程序 在主程序里 加一个调用就好

这样方便调试 希望能帮到你 很多知识 还得自己去学习摸索

嗯,我猜你问的是PC的,不是单片机

一、汇编语言的中断分以下几种:

1BIOS中断,这是固化到BIOS程序中的,每次开机BIOS会自动加载到指定内存

2186下的DOS中断,在DOS系统被加载后,系统会延用BIOS的中断向量,并向里面添加一些新的向量,这些功能便是DOS系统自带的中断服务程序

3286及以上的系统中断,PC会进入保护模式,在OS被加载后,中断由IDT控制,这一机制类似于中断向量表,只不过中断向量换成了选择子。这样的中断机制对不同型号的CPU有略微的差别,这里不细说了,我自己也没全弄明白。

二、中断实现的方式(8086下的普通中断)

听说过“优先级编码器”没?——如果同时有两个信号被接收,会指定某一个信号的优先级高,先执行它。中断就是类似的处理方法。

当CPU获取到某一高 *** 作优先级的信号时(比如时钟,每固定时间就会触发一次;比如键盘响应,用户希望通过Ctrl+C来退出任何正在执行的DOS程序),CPU会将当前正在执行的程序挂起来,转而去处理该信号(类似于Call,但略有不同,你看的书应该会讲到)。

处理中断时,系统会将其解释为一个标号,比如int 9h、int 21h等等。这个标号是一个序号,在内存某处存放着连续的一个表格,这个标号便是表格中的“行号”,只不过,每一行是两列,包括了该中断的处理程序的段基址和偏移量。中断向量表是从0000:0000开始的,每4字节为一个表项。中断标号x4就是对应的中断向量表项所存的地址,高地址是基地址,低地址是偏移。

这么说不知道你懂不懂。。。

反正总结一下你的问题吧,中断服务程序是加载到内存中的,它在加载前可能是存在BIOS芯片上,也可能是存在硬盘里的;中断向量表里只能写上中断处理程序的入口地址,要知道每个表项只有4字节;具体的中断服务程序,我不信你学汇编的书上不讲,我大概讲一下:CPU的INTR引脚获得了中断信号,得到了标号,比方说是5号,中断向量表项为0000:000A,读取这个内存,得到中断程序入口地址比方说是AAAA:BBBB,那么它会将当前的CS/IP、Flags寄存器入栈,然后转到AAAA:BBBB处去执行一直到iret指令返回原任务(或许该中断结束了这个任务,就不会返回了)。

至于保护模式的中断,相信你暂时还没遇到。到后面还有 *** 控8259A芯片来实现高级中断的,这个就不是一般需要学的了。

DS1302程序:

#include "ds1302h"

unsigned char time_buf1[8] = {20,9,3,13,18,51,00,6};//空年月日时分秒周

unsigned char time_buf[8] ;                         //空年月日时分秒周

/------------------------------------------------

向DS1302写入一字节数据

------------------------------------------------/

void Ds1302_Write_Byte(unsigned char addr, unsigned char d)

{

unsigned char i;

RST_SET;

//写入目标地址:addr

addr = addr & 0xFE;     //最低位置零

for (i = 0; i < 8; i ++)

{

if (addr & 0x01)

{

IO_SET;

}

else

{

IO_CLR;

}

SCK_SET;

SCK_CLR;

addr = addr >> 1;

}

//写入数据:d

for (i = 0; i < 8; i ++)

{

if (d & 0x01)

{

IO_SET;

}

else

{

IO_CLR;

}

SCK_SET;

SCK_CLR;

d = d >> 1;

}

RST_CLR;     //停止DS1302总线

}

/------------------------------------------------

从DS1302读出一字节数据

------------------------------------------------/

unsigned char Ds1302_Read_Byte(unsigned char addr)

{

unsigned char i;

unsigned char temp;

RST_SET;

//写入目标地址:addr

addr = addr | 0x01;//最低位置高

for (i = 0; i < 8; i ++)

{

if (addr & 0x01)

{

IO_SET;

}

else

{

IO_CLR;

}

SCK_SET;

SCK_CLR;

addr = addr >> 1;

}

//输出数据:temp

for (i = 0; i < 8; i ++)

{

temp = temp >> 1;

if (IO_R)

{

temp |= 0x80;

}

else

{

temp &= 0x7F;

}

SCK_SET;

SCK_CLR;

}

RST_CLR; //停止DS1302总线

return temp;

}

/------------------------------------------------

向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]+tmp16;

}

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_day_add,time_buf[7]);  //周

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);   //打开写保护

}

/------------------------------------------------

从DS1302读出时钟数据

------------------------------------------------/

void Ds1302_Read_Time(void)

{

unsigned char i,tmp;

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;//秒

time_buf[7]=Ds1302_Read_Byte(ds1302_day_add);  //周

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]+tmp10;

}

}

/------------------------------------------------

DS1302初始化

------------------------------------------------/

void Ds1302_Init(void)

{

RST_CLR;   //RST脚置低

SCK_CLR;   //SCK脚置低

Ds1302_Write_Byte(ds1302_sec_add,0x00);

}

主程序MAIN:

/-----------------------------------------------

名称:DS1302时钟数码管显示

论坛:>

编写:shifang

日期:20095

修改:无

内容:DS1302实时时钟数码管显示,只显示时间。并通过4个按键加减小时、分钟,其他参数调节自行添加

------------------------------------------------/

#include<reg52h> //包含头文件,一般情况不需要改动,头文件包含特殊功能寄存器的定义

#include "ds1302h"

#define KeyPort P3 //定义按键端口

#define DataPort P0 //定义数据端口 程序中遇到DataPort 则用P0 替换

sbit LATCH1=P2^2;//定义锁存使能端口 段锁存

sbit LATCH2=P2^3;//                 位锁存

bit ReadTimeFlag;//定义读时间标志

unsigned char code dofly_DuanMa[10]={0x3f,0x06,0x5b,0x4f,0x66,0x6d,0x7d,0x07,0x7f,0x6f};// 显示段码值0~9

unsigned char code dofly_WeiMa[]={0xfe,0xfd,0xfb,0xf7,0xef,0xdf,0xbf,0x7f};//分别对应相应的数码管点亮,即位码

unsigned char TempData[8]; //存储显示值的全局变量

void DelayUs2x(unsigned char t);//us级延时函数声明

void DelayMs(unsigned char t); //ms级延时

void Display(unsigned char FirstBit,unsigned char Num);//数码管显示函数

unsigned char KeyScan(void);//键盘扫描

void Init_Timer0(void);//定时器初始化

/------------------------------------------------

主函数

------------------------------------------------/

void main (void)

{

unsigned char num;

Init_Timer0();

Ds1302_Init();

while (1)         //主循环

{

num=KeyScan();

switch(num)

{

case 1:time_buf1[4]++;if(time_buf1[4]==24)time_buf1[4]=0;

Ds1302_Write_Time();break; //正常时间 小时 加1

case 2:time_buf1[4]--;if(time_buf1[4]==255)time_buf1[4]=23;

Ds1302_Write_Time();break; //正常时间 小时减1

case 3:time_buf1[5]++;if(time_buf1[5]==60)time_buf1[5]=0;

Ds1302_Write_Time();break;//分加1

case 4:time_buf1[5]--;if(time_buf1[5]==255)time_buf1[5]=59;

Ds1302_Write_Time();break; //分减1

default:break;

}

if(ReadTimeFlag==1)

{

ReadTimeFlag=0;

Ds1302_Read_Time();

//数据的转换,因我们采用数码管0~9的显示,将数据分开

TempData[0]=dofly_DuanMa[time_buf1[4]/10]; //时

TempData[1]=dofly_DuanMa[time_buf1[4]%10];

TempData[2]=0x40;                  //加入"-"

TempData[3]=dofly_DuanMa[time_buf1[5]/10]; //分

TempData[4]=dofly_DuanMa[time_buf1[5]%10];

TempData[5]=0x40;

TempData[6]=dofly_DuanMa[time_buf1[6]/10]; //秒

TempData[7]=dofly_DuanMa[time_buf1[6]%10];

}

}

}

/------------------------------------------------

uS延时函数,含有输入参数 unsigned char t,无返回值

unsigned char 是定义无符号字符变量,其值的范围是

0~255 这里使用晶振12M,精确延时请使用汇编,大致延时

长度如下 T=tx2+5 uS

------------------------------------------------/

void DelayUs2x(unsigned char t)

{

while(--t);

}

/------------------------------------------------

mS延时函数,含有输入参数 unsigned char t,无返回值

unsigned char 是定义无符号字符变量,其值的范围是

0~255 这里使用晶振12M,精确延时请使用汇编

------------------------------------------------/

void DelayMs(unsigned char t)

{

while(t--)

{

//大致延时1mS

DelayUs2x(245);

DelayUs2x(245);

}

}

/------------------------------------------------

显示函数,用于动态扫描数码管

输入参数 FirstBit 表示需要显示的第一位,如赋值2表示从第三个数码管开始显示

如输入0表示从第一个显示。

Num表示需要显示的位数,如需要显示99两位数值则该值输入2

------------------------------------------------/

void Display(unsigned char FirstBit,unsigned char Num)

{

static unsigned char i=0;

DataPort=0;   //清空数据,防止有交替重影

LATCH1=1;     //段锁存

LATCH1=0;

DataPort=dofly_WeiMa[i+FirstBit]; //取位码

LATCH2=1;     //位锁存

LATCH2=0;

DataPort=TempData[i]; //取显示数据,段码

LATCH1=1;     //段锁存

LATCH1=0;

i++;

if(i==Num)

i=0;

}

/------------------------------------------------

定时器初始化子程序

------------------------------------------------/

void Init_Timer0(void)

{

TMOD |= 0x01;   //使用模式1,16位定时器,使用"|"符号可以在使用多个定时器时不受影响

//TH0=0x00;       //给定初值

//TL0=0x00;

EA=1;            //总中断打开

ET0=1;           //定时器中断打开

TR0=1;           //定时器开关打开

}

/------------------------------------------------

定时器中断子程序

------------------------------------------------/

void Timer0_isr(void) interrupt 1

{

static unsigned int num;

TH0=(65536-2000)/256;    //重新赋值 2ms

TL0=(65536-2000)%256;

Display(0,8);       // 调用数码管扫描

num++;

if(num==50)        //大致100ms

{

num=0;

ReadTimeFlag=1; //读标志位置1

}

}

/------------------------------------------------

按键扫描函数,返回扫描键值

------------------------------------------------/

unsigned char KeyScan(void)

{

unsigned char keyvalue;

if(KeyPort!=0xff)

{

DelayMs(10);

if(KeyPort!=0xff)

{

keyvalue=KeyPort;

while(KeyPort!=0xff);

switch(keyvalue)

{

case 0xfe:return 1;break;

case 0xfd:return 2;break;

case 0xfb:return 3;break;

case 0xf7:return 4;break;

case 0xef:return 5;break;

case 0xdf:return 6;break;

case 0xbf:return 7;break;

case 0x7f:return 8;break;

default:return 0;break;

}

}

}

return 0;

}

有两种方法,这里介绍一种:

思路是将1302的某个寄存器定义为是否首次开机检测标志,比如存入0xaa数值。

上电时读取1302的这个寄存器,如果是0xaa,说明不是首次,便不再初始化,否则初始化,并向开机定义的寄存器中写入0xaa。

#define wrstar 0xce //写1302初始化标志

#define rdstar 0xcf //读1302初始化标志

void init1302()

{

rd(rdstar); //检测是否首次启用,否则初始化 这是查询RAM标志位

if(num!=0xaa) //RAM标志位不是预存的0xaa 则初始化

{ wr(wrwp,0x00); //允许写

wr(wrsec,0x30); //秒

wr(wrmin,0x00);//分

wr(wrhour,0x12);//

wr(wrday,0x15);//

wr(wrmoon,0x03);//

wr(wryear,0x11);//

wr(wrweek,0x01);//

wr(wra1hour,0x24);//闹铃1

wr(wra1min,0x00);//闹铃1

wr(wra2hour,0x24);//闹铃2

wr(wra2min,0x00);//闹铃2

wr(wra3hour,0x24);//闹铃3

wr(wra3min,0x00);//闹铃3

wr(wrstar,0xaa);// 初始化标志位写入0xaa

wr(wrwp,0x80); //禁止写

}

}

以上就是关于亲 上次那个ds1302的程序和仿真我已收到,我想请教一下全部的内容,包括:亲 上次那个ds1302的程序和仿真我已收到,我想请教一下、汇编语言的中断服务的中断服务程序是什么、求一个DS1302的闹钟程序,显示当前时间,用LED显示的程序等相关内容解答,如果想了解更多相关内容,可以关注我们,你们的支持是我们更新的动力!

欢迎分享,转载请注明来源:内存溢出

原文地址: https://outofmemory.cn/zz/9839266.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2023-05-02
下一篇 2023-05-02

发表评论

登录后才能评论

评论列表(0条)

保存