ATmega128(DS1302)

发布者:代码律动最新更新时间:2016-09-29 来源: eefocus关键字:ATmega128  DS1302 手机看文章 扫描二维码
随时随地手机看文章
//ATmega128的DS1302实时时钟芯片驱动程序

//编译环境 AVR Studio 4.17/AVR GCC
//系统时钟7.3728MHZ,设置熔丝位为外部高频石英晶体振荡,启动时间4.1ms
//作者:张奇 zhangqi123zq@126.com
//日期:2010.02.20

//***********************************************************************
//   包含文件
//***********************************************************************

#include
#include
#define  F_CPU 7372800  /* 单片机主频为7.3728MHz,用于延时子程序 */
#include
#include
#include
//***********************************************************************
// 定义变量区
//***********************************************************************
#define uchar unsigned char
#define uint  unsigned int

#define delay_us(x)     _delay_us(x)        //AVR GCC延时函数 x(us)
#define delay_ms(x)     _delay_ms(x)        //AVR GCC延时函数 x(ms)
/*复位脚*/
#define RST_CLR PORTE &= ~(1 << PE7)              //电平置低
#define RST_SET PORTE |= (1 << PE7)               //电平置高
#define RST_IN DDRE &= ~(1 << PE7)               //方向输入
#define RST_OUT DDRE |= (1 << PE7)                //方向输出

/*双向数据*/
#define IO_CLR PORTE &= ~(1 << PE6)              //电平置低
#define IO_SET PORTE |= (1 << PE6)               //电平置高
#define IO_R PINE & (1 << PE6)                 //电平读取
#define IO_IN DDRE &= ~(1 << PE6)               //方向输入
#define IO_OUT DDRE |= (1 << PE6)                //方向输出

/*时钟信号*/
#define SCK_CLR PORTE &= ~(1 << PE5)              //时钟信号
#define SCK_SET PORTE |= (1 << PE5)               //电平置高
#define SCK_IN DDRE &= ~(1 << PE5)               //方向输入
#define SCK_OUT DDRE |= (1 << PE5)                //方向输出

/*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

#define RS_CLR PORTF &= ~(1 << PF1)       //RS置低
#define RS_SET PORTF |= (1 << PF1)        //RS置高

#define RW_CLR PORTF &= ~(1 << PF2)       //RW置低
#define RW_SET PORTF |= (1 << PF2)        //RW置高

#define EN_CLR PORTF &= ~(1 << PF3)       //E置低
#define EN_SET PORTF |= (1 << PF3)        //E置高

#define Data_IO         PORTA                //液晶数据口
#define Data_DDR        DDRA                 //数据口方向寄存器
#define D_LE0         PORTD &= ~(1 << PD4) //数码管段控制位为0,锁存端口数据
#define D_LE1           PORTD |= (1 << PD4)  //数码管段控制位为1,锁存器输出与端口一致
#define W_LE0         PORTD &= ~(1 << PD5) //数码管位控制位为0
#define W_LE1           PORTD |= (1 << PD5)  //数码管位控制位为1

unsigned char time_buf[8] = {0x20,0x10,0x01,0x14,0x09,0x55,0x00,0x02};

//*************************************************************************
//   初始化子程序
//*************************************************************************
void system_init()
{

 Data_IO = 0xFF;            //电平设置
 Data_DDR = 0xFF;            //方向输出
 PORTF = 0xFF;                       //电平设置
 DDRF = 0xFF;                        //方向输出
    PORTD=0xFF;
 DDRD=0xFF;
 D_LE1;                             //关掉数码管,以免显示乱码                   
    W_LE1;                     
    Data_IO=0xFF;                      //关数码管            
    W_LE0; 
}

//***********************************************************************
//向DS1302写入一字节数据
//***********************************************************************
void ds1302_write_byte(unsigned char addr, unsigned char d) 
{

 unsigned char i;
 RST_SET;     //启动DS1302总线
 
                                                 //写入目标地址:addr
 IO_OUT;
 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
 IO_OUT;
 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;     //启动DS1302总线

                                                 //写入目标地址:addr
 IO_OUT;
 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*/
 IO_IN;
 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;
}

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

//***********************************************************************
//                     从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;//秒 
 time_buf[7]=ds1302_read_byte(ds1302_day_add);  //周 
}

//***********************************************************************
//               DS302初始化函数
//***********************************************************************

void ds1302_init(void) 
{
 
 RST_CLR;   //RST脚置低
 SCK_CLR;   //SCK脚置低
 RST_OUT;   //RST脚设置为输出
 SCK_OUT;   //SCK脚设置为输出
}

//***********************************************************************
// 显示屏命令写入函数
//***********************************************************************


void LCD_write_com(unsigned char com) 

 RS_CLR;
 RW_CLR;
 EN_SET;
 Data_IO = com;
 delay_ms(5);
 EN_CLR;
}
//***********************************************************************
// 显示屏数据写入函数
//***********************************************************************

void LCD_write_data(unsigned char data) 
{
 RS_SET;
 RW_CLR;
 EN_SET;
 Data_IO = data;
 delay_ms(5);
 EN_CLR;
}
//***********************************************************************
// 显示屏清空显示
//***********************************************************************

void LCD_clear(void) 
{
 LCD_write_com(0x01);
 delay_ms(5);
}
//***********************************************************************
// 显示屏字符串写入函数
//***********************************************************************

void LCD_write_str(unsigned char x,unsigned char y,unsigned char *s) 
{
 
    if (y == 0) 
    {
     LCD_write_com(0x80 + x);
    }
    else 
    {
     LCD_write_com(0xC0 + x);
    }
    
    while (*s) 
    {
     LCD_write_data( *s);
     s ++;
    }
}
//***********************************************************************
// 显示屏单字符写入函数
//***********************************************************************

void LCD_write_char(unsigned char x,unsigned char y,unsigned char data) 
{
 
    if (y == 0) 
    {
     LCD_write_com(0x80 + x);
    }
    else 
    {
     LCD_write_com(0xC0 + x);
    }
    
    LCD_write_data( data);  
}
//***********************************************************************
// 显示屏初始化函数
//***********************************************************************

void LCD_init(void) 
{
 system_init();
 LCD_write_com(0x38);    //显示模式设置
 delay_ms(5);
 LCD_write_com(0x08);    //显示关闭
 delay_ms(5);
    LCD_write_com(0x01);    //显示清屏
 delay_ms(5);
    LCD_write_com(0x06);    //显示光标移动设置
 delay_ms(5);
 LCD_write_com(0x0C);    //显示开及光标设置
    delay_ms(5);
}

//***********************************************************************
// 主函数
//***********************************************************************

void main(void)

{

 unsigned char temp;
  delay_ms(200);
 LCD_init();                           //1602液晶初始化
 ds1302_init();                        //DS1302初始化
 delay_ms(10);
 ds1302_write_time();                 //向DS1302写初始数据,如年月和时间等

 while (1) 
 {
  delay_ms(200);
  ds1302_read_time();                //读DS1302数据
   
  temp = (time_buf[0] >> 4) + '0';   //在LCD上写要显示的数据
  LCD_write_char(0, 0, temp);/*年*/
  temp = (time_buf[0] & 0x0F) + '0';
  LCD_write_char(1, 0, temp);
  temp = (time_buf[1] >> 4) + '0';
  LCD_write_char(2, 0, temp);
  temp = (time_buf[1] & 0x0F) + '0';
  LCD_write_char(3, 0, temp);
  LCD_write_char(4, 0, '-');
  
  temp = (time_buf[2] >> 4) + '0';
  LCD_write_char(5, 0, temp);/*月*/
  temp = (time_buf[2] & 0x0F) + '0';
  LCD_write_char(6, 0, temp);
  LCD_write_char(7, 0, '-');
  
  temp = (time_buf[3] >> 4) + '0';
  LCD_write_char(8, 0, temp);/*日*/
  temp = (time_buf[3] & 0x0F) + '0';
  LCD_write_char(9, 0, temp);
  
  temp = (time_buf[4] >> 4) + '0';
  LCD_write_char(8, 1, temp);/*时*/
  temp = (time_buf[4] & 0x0F) + '0';
  LCD_write_char(9, 1, temp);
  LCD_write_char(10, 1, ':');
  
  temp = (time_buf[5] >> 4) + '0';
  LCD_write_char(11, 1, temp);/*分*/
  temp = (time_buf[5] & 0x0F) + '0';
  LCD_write_char(12, 1, temp);
  LCD_write_char(13, 1, ':');
  
  temp = (time_buf[6] >> 4) + '0';
  LCD_write_char(14, 1, temp);/*秒*/
  temp = (time_buf[6] & 0x0F) + '0';
  LCD_write_char(15, 1, temp);
 }
}

关键字:ATmega128  DS1302 引用地址:ATmega128(DS1302)

上一篇:ATmega128(EEPROM)
下一篇:ATmega128(ICC7.22 "delay.h")

推荐阅读最新更新时间:2024-03-16 15:12

ATmega128 时钟系统及其分布
时钟系统及其分布 Figure 18为AVR的主要时钟系统及其分布。这些时钟并不需要同时工作。为了降低功耗, 可以通过使用不同的睡眠模式来禁止无需工作的模块的时钟,如 P 41“ 电源管理及睡眠模 式” . 所示。 CPU 时钟- clkCPU CPU时钟与操作AVR内核的子系统相连,如通用工作寄存器文件、状态寄存器以及保存堆 栈指针的数据存储器。终止CPU 时钟将使内核停止工作和计算。 I/O 时钟- clkI/O I/O时钟用于主要的I/O 模块,如定时器/ 计数器、SPI 和USART。I/O 时钟还用于外部中断 模块。但是有些外部中断由异步逻辑检测,因此即使I/O 时钟停止了这些中断仍然可以得 到监控。此外,TWI
[单片机]
<font color='red'>ATmega128</font> 时钟系统及其分布
ATMEGA128 UART的使用
#include myuart.h #include avr/io.h #include #define UDRE 5 #define RXEN 4 #define TXEN 3 #define UCSZ0 1 #define RXC 7 #define RXCIE 7 ISR(USART0_RX_vect)//接收中断 { unsigned char dummy; while ( UCSR0A & (1 RXC) ) { dummy = UDR0; } USART_Transmit(dummy); } void USART_Transmit(unsigned char data)//发送字节 {
[单片机]
<font color='red'>ATMEGA128</font> UART的使用
ds1302实时时钟
现在流行的串行时钟电路很多,如DS1302、DS1307、PCF8485等。这些电路的接口简单、价格低廉、使用方便,被广泛地采用。本文介绍的实时时钟电路DS1302是DALLAS公司的一种具有涓细电流充电能力的电路,主要特点是采用串行数据传输,可为掉电保护电源提供可编程的充电功能,并且可以关闭充电功能。采用普通32.768kHz晶振。 2 DS1302的结构及工作原理   DS1302是美国DALLAS公司推出的一种高性能、低功耗、带RAM的实时时钟电路,它可以对年、月、日、周日、时、分、秒进行计时,具有闰年补偿功能,工作电压为2.5V~5.5V。采用三线接口与CPU进行同步通信,并可采用突发方式一次传送多个字节的时钟信号或R
[电源管理]
<font color='red'>ds1302</font>实时时钟
DS1302控制代码
两天的成果, /*************************************/ #include reg52.h #define uint unsigned int #define uchar unsigned char uchar year,month,day,week,hours,minutes,seconds; uchar flag1302,cnt; /************************************* 控制按键 *************************************/ sbit K1=P1^0; sbit K2=P1^2; sbit K3=P1
[单片机]
ATmega128并行控制带字库的12864程序
ATmega128的特点是io口很多,所以我们可以用并行的方式来驱动此液晶屏,增加总线的速度,此程序是是网上整理收集而来,但已经通过本人验证可以使用,故在此发表.大家在应用的时候只需更改相应的io就行了. 软件:GCC V4.20 --------------------------------------------------------------- 实验内容:写Lcd12864_ST7920。 --------------------------------------------------------------- 硬件连接: LCD12864_ST7920 ATmega128 1.GND
[单片机]
DS1302的AVR单片机C程序
下面是DS1302的AVR 单片机 的C程序,这个程序结构很不错。但是没有调试这个AVR单片机的程序。 #define ds1302_rst PC0 #define ds1302_io PC1 #define ds1302_sclk PC2 #define set_ds1302_rst_ddr() DDRC|=1 ds1302_rst #define set_ds1302_rst() PORTC|=1 ds1302_rst #define clr_ds1302_rst() PORTC&=~(1 ds1302_rst) #define set_ds1302_io_ddr() DDRC|=1 ds1302_io #def
[单片机]
MSP430之变态版3线SPI总线实现(DS1302时钟芯片用)
如题, 这个是网上所谓的变态版的3线SPI总线:一根时钟线,一根使能线,一根双向IO线. 一个模块,两个文件: //spi3.c #include typedef.h #include spi3.h /*********************************************************************** 名称:init_spi3 描述:SPI3初始化函数 参数:(无) 返回:(无) 说明: ***********************************************************************/ void init_spi3(void) {
[单片机]
ATmega128 状态寄存器
状态寄存器包含了最近执行的算术指令的结果信息。这些信息可以用来改变程序流程以 实现条件操作。状态寄存器的内容只有在ALU 运算结束后才会更新。这样,在多数情况 下就不需要专门的比较指令了,从而使系统运行更快速,代码效率更高。 在进入中断例程时状态寄存器不会自动保存;中断返回时也不会自动恢复。这些工作需 要软件来处理。 AVR 中断寄存器 – SREG – 定义如下: • Bit 7 – I: 全局中断使能 置位时使能全局中断。单独的中断使能由其他独立的控制寄存器控制。如果I 清零,则不论单独中断标志置位与否,都不会产生中断。任意一个中断发生后I 清零,而执行RETI指令后置位以使能中断。I 也可以通过SEI 和CLI
[单片机]
<font color='red'>ATmega128</font> 状态寄存器
小广播
添点儿料...
无论热点新闻、行业分析、技术干货……
设计资源 培训 开发板 精华推荐

最新单片机文章
何立民专栏 单片机及嵌入式宝典

北京航空航天大学教授,20余年来致力于单片机与嵌入式系统推广工作。

换一换 更多 相关热搜器件
电子工程世界版权所有 京B2-20211791 京ICP备10001474号-1 电信业务审批[2006]字第258号函 京公网安备 11010802033920号 Copyright © 2005-2024 EEWORLD.com.cn, Inc. All rights reserved