Difference between revisions of "Real Time Clock(DS1307) with AVR"
Raghavendra (Talk | contribs) |
Raghavendra (Talk | contribs) |
||
Line 163: | Line 163: | ||
[[File:Schematic AVR Interfacing RTC.JPG|680px]] | [[File:Schematic AVR Interfacing RTC.JPG|680px]] | ||
+ | |||
+ | [[File:0Real time clock.gif|none]] | ||
{{DISQUS}} | {{DISQUS}} | ||
[[Category:AVR Tutorials]] | [[Category:AVR Tutorials]] |
Revision as of 11:03, 29 March 2016
Contents
Basics
The Real time clock DS1307 IC basically is stand alone time clock. Well, basically we can use a micrcontroller to keep time, but the value would go off as soon as it is powered off.
The RTC DS1307 is a handy solution to keep time all the way, when it is powered by a coin cell.
It is uses I²C (Inter-Integrated Circuit) protocol, referred to as I-squared-C, I-two-C, or IIC for communication with the micrcontroller.
Check the Basics of I2C here, if you are not familiar with it. For details of I2C in AVR, go through AVR Communication Protocols tutorial.
The first thing that the MCU sends to the slave (RTC) is the device ID. The device ID for DS1307, shown below. It also tells weather we want to write to or read from the RTC.
7 | 6 | 5 | 4 | 3 | 2 | 1 | 0 |
1 | 1 | 0 | 1 | 0 | 0 | 0 | R/W |
- bit-0 is 0 than we Write to RTC
- bit-0 is 1 we Read from RTC.
This is defined in the code as:
#define C_Ds1307ReadMode_U8 0xD1u // DS1307 ID #define C_Ds1307WriteMode_U8 0xD0u // DS1307 ID
The RTC keeps the date and time arranged in it's memory as shown below:
ADDRESS | FUNCTION | RANGE |
---|---|---|
00h | Seconds | 00–59 |
01h | Minutes | 00–59 |
02h | Hours | 01-12/00-24 |
03h | Day | 01–07 |
04h | Date | 01–31 |
05h | Month | 01–12 |
06h | Year | 00–99 |
07h | Control | |
08h to 3Fh | RAM | 00h–FFh |
Write to the addresses above we can set the time, and once we set it, we can read it any time we need.
The address 0x07 is a control registered as described below:
7 | 6 | 5 | 4 | 3 | 2 | 1 | 0 |
OUT | 0 | 0 | SQWE | 0 | 0 | RS1 | RS0 |
We write 0x00 to Control register to disable SQW-Out. We do not use any other bits from it, so you need not worry.
Initialize
Now we can initialize the RTC with the code below
void RTC_Init(void) { I2C_Init(); // Initialize the I2c module. I2C_Start(); // Start I2C communication I2C_Write(C_Ds1307WriteMode_U8); // Connect to DS1307 by sending its ID on I2c Bus I2C_Write(C_Ds1307ControlRegAddress_U8);// Select the Ds1307 ControlRegister to configure Ds1307 I2C_Write(0x00); // Write 0x00 to Control register to disable SQW-Out I2C_Stop(); // Stop I2C communication after initializing DS1307 }
Set Date and Time
void RTC_SetDateTime(rtc_t *rtc) { I2C_Start(); // Start I2C communication I2C_Write(C_Ds1307WriteMode_U8); // connect to DS1307 by sending its ID on I2c Bus I2C_Write(C_Ds1307SecondRegAddress_U8); // Request sec RAM address at 00H I2C_Write(rtc->sec); // Write sec from RAM address 00H I2C_Write(rtc->min); // Write min from RAM address 01H I2C_Write(rtc->hour); // Write hour from RAM address 02H I2C_Write(rtc->weekDay); // Write weekDay on RAM address 03H I2C_Write(rtc->date); // Write date on RAM address 04H I2C_Write(rtc->month); // Write month on RAM address 05H I2C_Write(rtc->year); // Write year on RAM address 06h I2C_Stop(); // Stop I2C communication after Setting the Date }
Note: The date and time read from Ds1307 will be of BCD format, like:
- 0x12,0x39,0x26 for 12hr,39min and 26sec.
- 0x15,0x08,0x47 for 15th day,8th month and 47th year
Get Date and Time
void RTC_GetDateTime(rtc_t *rtc) { I2C_Start(); // Start I2C communication I2C_Write(C_Ds1307WriteMode_U8); // connect to DS1307 by sending its ID on I2c Bus I2C_Write(C_Ds1307SecondRegAddress_U8); // Request Sec RAM address at 00H I2C_Stop(); // Stop I2C communication after selecting Sec Register I2C_Start(); // Start I2C communication I2C_Write(C_Ds1307ReadMode_U8); // connect to DS1307(Read mode) by sending its ID rtc->sec = I2C_Read(1); // read second and return Positive ACK rtc->min = I2C_Read(1); // read minute and return Positive ACK rtc->hour= I2C_Read(1); // read hour and return Negative/No ACK rtc->weekDay = I2C_Read(1); // read weekDay and return Positive ACK rtc->date= I2C_Read(1); // read Date and return Positive ACK rtc->month=I2C_Read(1); // read Month and return Positive ACK rtc->year =I2C_Read(0); // read Year and return Negative/No ACK I2C_Stop(); // Stop I2C communication after reading the Date }
Both the above functions use a simple structure shown below for easy access
typedef struct { uint8_t sec; uint8_t min; uint8_t hour; uint8_t weekDay; uint8_t date; uint8_t month; uint8_t year; }rtc_t;
Example
Now, let's put together the all that we have discussed in a simple example to read and show the time on character LCD.
Hookup
Code
Schematic