====== PIC18: PCF8563 I2C Real Time Clock Calendar ====== {{:pic18_clock:clock-calendar_0.png?300 |}}The **[[http://www.laurtec.it/progetti-elettronici/sistemi-automatici/76-scheda-di-sviluppo-per-pic-freedom-ii|Freedom II demoboard]]** includes also a **PCF8563 I2C Real Time Clock Calendar (RTCC)**; let's learn how to use it to display time and date and set an alarm. As soon as the RTCC is powered on it begins keeping time and date, starting from 00:00 (hh:mimi:ss) and from 00/00/00 (dd/momo/yy); the chip makes available those data thanks to a bunch of _registers_, which can be seen on the right of the below image: {{:pic18_clock:rtcc-block-diagram.png|}} So to read date and time we could just create a loop that does //I2C sequential reads// (see **[[pic18_i2c_eeprom_dump_via_rs_232_serial|previous article on I2C]]**) towards the PCF8563 and then display the values on an LCD; while it certainly can be done, it is quite an inefficient way to use the PIC, because it has to continually poll the RTCC. What we are going to do instead is using the RTCC //interrupts// to tell the PIC when the time/date have changed (or an alarm threshold has been reached) so that it can efficiently do the I2C reading. ===== RTCC Interrupts ===== As seen in the above diagram //pin 3// (INT) is //active low// and is activated whenever a timer or alarm interrupt is raised; PORTB2 in the Freedom is connected to it so that **we can use its INT2 External to capture an RTCC interrupt event**; let's see how we can do this in practice. ==== Set RTCC for Timer and Alarm interrupts ==== The PCF8563 has to be _prepared_ to raise interrupts; that's done by the //I2C_RTCCtrlStatus2Set// and //I2C_RTCTimerInit// routines. {{:pic18_clock:rtcc-registers01.png|}} **I2C_RTCCtrlStatus2Set** modifies **CtrlStatus2 register (01h)**, by clearing AF (alarm flag) and TF (timer flag) bits and setting AIE (Alarm Interrupt Enable) and TIE (Timer Interrupt Enable) bits: I2C_RTCCtrlStatus2Set ; Setup I2C registers for Timer and Alarm Interrupt: call I2C_Start movlw litRTCWR call I2C_SendSSPBUF movlw 0x01 call I2C_SendSSPBUF movlw 0x03 ; Set TI, timer and alarm flags cleared, timer interrupt enabled call I2C_SendSSPBUF call I2C_Stop return **I2C_RTCTimerInit** operates on **Timer Control register (0Eh)** by setting //TE// bit (Timer Enable) and the timer frequency to 64Hz: I2C_RTCTimerInit call I2C_RTCCtrlStatus2Set ; Load other I2C:RTC registers call I2C_Start ; I2C Start movlw litRTCWR call I2C_SendSSPBUF movlw 0x0E ; Register Timer Countdown (0x0E) call I2C_SendSSPBUF movlw 0x81 ; Timer Enabled, 64Hz clock frequency call I2C_SendSSPBUF movlw 0x20 ; Load register 0x0F (Timer Countdown) with value 0x20 (dec. 32) call I2C_SendSSPBUF call I2C_Stop ; set alarm time call I2C_Start ; I2C Start movlw litRTCWR call I2C_SendSSPBUF movlw 0x09 ; Register call I2C_SendSSPBUF movlw 0x08 ; AE = 0: alarm enabled; 8 minutes call I2C_SendSSPBUF call I2C_Stop return {{:pic18_clock:rtcc-registers06.png|}} What a 64HZ clock frequency means is that **the Timer Countdown register (0Fh)**, loaded with an arbitrary **value of 0x20** (dec. 32, see above), **decreases of one unit every 1/64s** and when it **reaches 0 a timer interrupt is generated.** This occurs every 1/2s (32/64), so every half a second the RTCC warns the PIC that the time _could_ be changed. === Alarms === Let's take a look at the **registers from 09h to 0Ch**: {{:pic18_clock:rtcc-registers02.png|}} An **alarm interrupt** is raised whenever **//all// alarm //enabled// conditions are met**; that means that if we set the register bits AE (alarm enable) for hours and minutes, for instance, and we put the BCD values 4 and 32 an alarm will raise only when it's 4:32 (both enabled conditions are evaluated). ==== Interpreting the interrupts ==== Now the PIC starts to see interrupts for timer and alarm coming from the RTCC; in **isr.asm** this is taken care of by this piece of code: IsrINT2 bcf INTCON3,INT2IF bsf REG_FLAG,INT2ExtInt ; Set flag INT2ExtInt in the custom register REG_FLAG, which is checked in main.asm goto EndIsr which sets the **flag INT2ExtInt**, checked in **main.asm**: ; check RTC interrupt btfss REG_FLAG,INT2ExtInt ; check if there's an interrupt from INT2 goto EndMain ; NO: goto EndMain call I2C_RTCCtrlStatus2Read ; YES: read CtrlStatus2 The **sub I2C_RTCCtrlStatus2Read**, inside **i2c-rtcc.asm** reads RTCC //Control/Status 2 (01h)//, puts its value into **RTCReg0x01** and goes **back to main**, where only bits 2 and 3 (TF and AF: timer and alarm flags) are extracted using a mask. Then the program jumps to two different routines, on the evaluation of the AF or TF flag: movlw b'00001100' ; use mask to get only AF and TF flags of CtrlStatus2 andwf RTCReg0x01,F ; and mask with RTCReg0x01 and leave the result in the register btfsc RTCReg0x01,3 ; check if bit 3 (AF) is clear call RTCAlarmInterrupt ; NO: it is set, so call sub I2C_RTCAlarmInt btfsc RTCReg0x01,2 ; check if bit 1 (TF) is clear call RTCTimerInterrupt ; NO: it is set, so call sub I2C_RTCTimerInt Then the normal, not interrupted, state is restored by: * calling **I2C_RTCCtrlStatus2Set** sub, which clears AF and TF bits in the RTCC (see above) * **clearing REG_FLAG,INT2ExtInt**, so that the next interrupt can be handled. ==== BCD to byte conversion to view values onto LCD display ==== Once a Timer interrupt is raised sub //RTCTimerInterrupt// (in main.asm) is called: RTCTimerInterrupt call RTCView ; Timer Interrupt occurred: view RTC values on LCD //RTCView// sub does the following: RTCView call I2C_RTCRead ; call sub that reads RTC values via I2C call I2C_BCD2BYTE ; call sub that converts BCD values to a byte ; in order to display them onto the LCD The values read by //I2C_RTCRead// are **BCD** ones and saved into file registers named SecondsBCD, MinutesBCD and so on; BCDs are representations of numbers in which each digit is a 4bit (one nibble) number. So for instance the BCD decimal number //27// is 0010 (2) 0111 (7) , which is .39 in normal (not BCD) decimal format. We have to //convert BCDs// before viewing them on the LCD; that's what //I2C_BCD2BYTE// does. {{:pic18_clock:bcd2byte_02_0.png|}} The sub splits the BCD values into two parts, by using proper //masks//, one for the units, the other for the tenths. Then tenths are swapped, multiplied by 10 and added to units to obtain the byte value to be displayed onto the LCD: I2C_BCD2BYTE ; Obtain values from BCDs movlw b'00001111' ; use mask andwf DayBCD,W ; AND BCD value and mask movwf DayUnits ; copy value to register movlw b'00110000' ;use mask andwf DayBCD,W ;AND BCD value and mask movwf DayTenths ;copy value to register swapf DayTenths,F ;swap the result into the lower nibble movlw .10 ; multiplier value mulwf DayTenths ; DayTenths * 10 movff PRODL,DayByte ; copy the low byte result of the previous multiplication to DayByte movf DayUnits,W ; copy DayUnits to W addwf DayByte,F ; copy W (DayUnits) to DayByte ===== Set date and time ===== Now that the display is showing date and time by using interrupts, wouldn't it be cool to also be able to **set date, time and alarm values**? To do that we could use the Freedom II board buttons in this way: * **BT1**: toggles date/time modification status * **BT2**: increases current date/time value (hours,minutes,seconds,...) * **BT3**: decreases current date/time value (hours,minutes,seconds,...) * **BT4**: skips to next date/time value So to tell the PIC we want to modify the values we have first to press //BT1// in order to activate the modify mode; then we can move to a specific position with //BT4//, according to the following scheme: {{:pic18_clock:lcd-positions.png|}} so pressing //BT4// once means moving to position //1// (minutes), once again to move to position //2// (seconds), then //3// (day), //4// (month), //5// (year), //6// (alarm hours), //7// (alarm minutes), //8// (alarm day) and lastly back to //0// (hours); the values really shown on **AE** positions are **_0_ when alarm is _enabled_ or _1_ when it is //disabled//**. Then the values can be increased and decreased with //BT2// and //BT3// respectively; the routines for these buttons consider different minimum and maximum values, depending on the value to be changed: a maximum value of //23// for hours, //59// for minutes and seconds... a minimum value of //01// for month, //01// for day and so on. Actually the maximum values for //alarms// are different; for instance the maximum value for minutes is //60//, which means //alarm disabled// for that particular alarm. Both routines end in the same way: call I2C_BYTE2BCD ; Convert BYTES to BCDs call I2C_RTCWrite ; Write BCDs to RTC the sub **I2C_BYTE2BCD**, which is inside i2c-rtcc.asm converts the changed //byte// value to a _BCD_. Let's see how to achieve that. ==== Byte to BCD conversion before writing values to RTCC registers ==== Say we have just changed the //byte// value for seconds to //58//; we have to **convert it in BCD, because RTCC registers expect BCD numbers only**. The BCD representation of .58 is //.88//; this last value has to be saved into the seconds register of the RTCC; with BCDs the first digit (5) goes into the //upper nibble// and the second (8) into the //lower nibble//: {{:pic18_clock:byte2bcd.png|}} So 58 has to be split into //5// and //8//; that's a **division by ten**, in which //5// is the result (//quotient//) and //8// is the //remainder//. And that's taken care of by the aptly named **DivisionBy10** sub (inside math.asm); then quotient and remainder are properly handled into //I2C_BYTE2BCD// to form the SecondsBCD (.88), which is **written**, along with the other BCDs, **to the RTCC by the //I2C_RTCWrite// sub**. > //DISCLAIMER: This is an unfinished project. I got stuck at a certain point and didn't have time or energy to debug and complete it; the code was starting to become really convoluted and needs refinement, to say the least... All the interrupt parts works; the challenging part was the one which sets date and time. BT4 doesn't skip smoothly to the next value, plus there is no way to determine the actual position of the value to be changed. AE bits are overwritten.//