15c4fa2eeSHeiko Schocher /* 25c4fa2eeSHeiko Schocher * (C) Copyright 2010 35c4fa2eeSHeiko Schocher * Heiko Schocher, DENX Software Engineering, hs@denx.de 45c4fa2eeSHeiko Schocher * 55c4fa2eeSHeiko Schocher * See file CREDITS for list of people who contributed to this 65c4fa2eeSHeiko Schocher * project. 75c4fa2eeSHeiko Schocher * 85c4fa2eeSHeiko Schocher * This program is free software; you can redistribute it and/or 95c4fa2eeSHeiko Schocher * modify it under the terms of the GNU General Public License as 105c4fa2eeSHeiko Schocher * published by the Free Software Foundation; either version 2 of 115c4fa2eeSHeiko Schocher * the License, or (at your option) any later version. 125c4fa2eeSHeiko Schocher * 135c4fa2eeSHeiko Schocher * This program is distributed in the hope that it will be useful, 145c4fa2eeSHeiko Schocher * but WITHOUT ANY WARRANTY; without even the implied warranty of 155c4fa2eeSHeiko Schocher * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the 165c4fa2eeSHeiko Schocher * GNU General Public License for more details. 175c4fa2eeSHeiko Schocher * 185c4fa2eeSHeiko Schocher * You should have received a copy of the GNU General Public License 195c4fa2eeSHeiko Schocher * along with this program; if not, write to the Free Software 205c4fa2eeSHeiko Schocher * Foundation, Inc., 59 Temple Place, Suite 330, Boston, 215c4fa2eeSHeiko Schocher * MA 02111-1307 USA 225c4fa2eeSHeiko Schocher */ 235c4fa2eeSHeiko Schocher #include <common.h> 245c4fa2eeSHeiko Schocher #include <command.h> 255c4fa2eeSHeiko Schocher #include <i2c.h> 265c4fa2eeSHeiko Schocher #include <rtc.h> 275c4fa2eeSHeiko Schocher 2871d19f30SHeiko Schocher #define RTC_RV3029_CTRL1 0x00 2971d19f30SHeiko Schocher #define RTC_RV3029_CTRL1_EERE (1 << 3) 3071d19f30SHeiko Schocher 3171d19f30SHeiko Schocher #define RTC_RV3029_CTRL_STATUS 0x03 3271d19f30SHeiko Schocher #define RTC_RV3029_CTRLS_EEBUSY (1 << 7) 3371d19f30SHeiko Schocher 345c4fa2eeSHeiko Schocher #define RTC_RV3029_CTRL_RESET 0x04 355c4fa2eeSHeiko Schocher #define RTC_RV3029_CTRL_SYS_R (1 << 4) 365c4fa2eeSHeiko Schocher 375c4fa2eeSHeiko Schocher #define RTC_RV3029_CLOCK_PAGE 0x08 385c4fa2eeSHeiko Schocher #define RTC_RV3029_PAGE_LEN 7 395c4fa2eeSHeiko Schocher 405c4fa2eeSHeiko Schocher #define RV3029C2_W_SECONDS 0x00 415c4fa2eeSHeiko Schocher #define RV3029C2_W_MINUTES 0x01 425c4fa2eeSHeiko Schocher #define RV3029C2_W_HOURS 0x02 435c4fa2eeSHeiko Schocher #define RV3029C2_W_DATE 0x03 445c4fa2eeSHeiko Schocher #define RV3029C2_W_DAYS 0x04 455c4fa2eeSHeiko Schocher #define RV3029C2_W_MONTHS 0x05 465c4fa2eeSHeiko Schocher #define RV3029C2_W_YEARS 0x06 475c4fa2eeSHeiko Schocher 485c4fa2eeSHeiko Schocher #define RV3029C2_REG_HR_12_24 (1 << 6) /* 24h/12h mode */ 495c4fa2eeSHeiko Schocher #define RV3029C2_REG_HR_PM (1 << 5) /* PM/AM bit in 12h mode */ 505c4fa2eeSHeiko Schocher 5171d19f30SHeiko Schocher #define RTC_RV3029_EEPROM_CTRL 0x30 5271d19f30SHeiko Schocher #define RTC_RV3029_TRICKLE_1K (1 << 4) 5371d19f30SHeiko Schocher #define RTC_RV3029_TRICKLE_5K (1 << 5) 5471d19f30SHeiko Schocher #define RTC_RV3029_TRICKLE_20K (1 << 6) 5571d19f30SHeiko Schocher #define RTC_RV3029_TRICKLE_80K (1 << 7) 5671d19f30SHeiko Schocher 575c4fa2eeSHeiko Schocher int rtc_get( struct rtc_time *tmp ) 585c4fa2eeSHeiko Schocher { 595c4fa2eeSHeiko Schocher int ret; 605c4fa2eeSHeiko Schocher unsigned char buf[RTC_RV3029_PAGE_LEN]; 615c4fa2eeSHeiko Schocher 625c4fa2eeSHeiko Schocher ret = i2c_read(CONFIG_SYS_I2C_RTC_ADDR, RTC_RV3029_CLOCK_PAGE, 1, buf, \ 635c4fa2eeSHeiko Schocher RTC_RV3029_PAGE_LEN); 645c4fa2eeSHeiko Schocher if (ret) { 655c4fa2eeSHeiko Schocher printf("%s: error reading RTC: %x\n", __func__, ret); 665c4fa2eeSHeiko Schocher return -1; 675c4fa2eeSHeiko Schocher } 685c4fa2eeSHeiko Schocher tmp->tm_sec = bcd2bin( buf[RV3029C2_W_SECONDS] & 0x7f); 695c4fa2eeSHeiko Schocher tmp->tm_min = bcd2bin( buf[RV3029C2_W_MINUTES] & 0x7f); 705c4fa2eeSHeiko Schocher if (buf[RV3029C2_W_HOURS] & RV3029C2_REG_HR_12_24) { 715c4fa2eeSHeiko Schocher /* 12h format */ 725c4fa2eeSHeiko Schocher tmp->tm_hour = bcd2bin(buf[RV3029C2_W_HOURS] & 0x1f); 735c4fa2eeSHeiko Schocher if (buf[RV3029C2_W_HOURS] & RV3029C2_REG_HR_PM) 745c4fa2eeSHeiko Schocher /* PM flag set */ 755c4fa2eeSHeiko Schocher tmp->tm_hour += 12; 765c4fa2eeSHeiko Schocher } else 775c4fa2eeSHeiko Schocher tmp->tm_hour = bcd2bin(buf[RV3029C2_W_HOURS] & 0x3f); 785c4fa2eeSHeiko Schocher 795c4fa2eeSHeiko Schocher tmp->tm_mday = bcd2bin( buf[RV3029C2_W_DATE] & 0x3F ); 805c4fa2eeSHeiko Schocher tmp->tm_mon = bcd2bin( buf[RV3029C2_W_MONTHS] & 0x1F ); 815c4fa2eeSHeiko Schocher tmp->tm_wday = bcd2bin( buf[RV3029C2_W_DAYS] & 0x07 ); 825c4fa2eeSHeiko Schocher /* RTC supports only years > 1999 */ 835c4fa2eeSHeiko Schocher tmp->tm_year = bcd2bin( buf[RV3029C2_W_YEARS]) + 2000; 845c4fa2eeSHeiko Schocher tmp->tm_yday = 0; 855c4fa2eeSHeiko Schocher tmp->tm_isdst = 0; 865c4fa2eeSHeiko Schocher 87*b633741bSWolfgang Denk debug( "Get DATE: %4d-%02d-%02d (wday=%d) TIME: %2d:%02d:%02d\n", 885c4fa2eeSHeiko Schocher tmp->tm_year, tmp->tm_mon, tmp->tm_mday, tmp->tm_wday, 895c4fa2eeSHeiko Schocher tmp->tm_hour, tmp->tm_min, tmp->tm_sec ); 905c4fa2eeSHeiko Schocher 915c4fa2eeSHeiko Schocher return 0; 925c4fa2eeSHeiko Schocher } 935c4fa2eeSHeiko Schocher 945c4fa2eeSHeiko Schocher int rtc_set( struct rtc_time *tmp ) 955c4fa2eeSHeiko Schocher { 965c4fa2eeSHeiko Schocher int ret; 975c4fa2eeSHeiko Schocher unsigned char buf[RTC_RV3029_PAGE_LEN]; 98*b633741bSWolfgang Denk 99*b633741bSWolfgang Denk debug( "Set DATE: %4d-%02d-%02d (wday=%d) TIME: %2d:%02d:%02d\n", 1005c4fa2eeSHeiko Schocher tmp->tm_year, tmp->tm_mon, tmp->tm_mday, tmp->tm_wday, 1015c4fa2eeSHeiko Schocher tmp->tm_hour, tmp->tm_min, tmp->tm_sec); 1025c4fa2eeSHeiko Schocher 1035c4fa2eeSHeiko Schocher if (tmp->tm_year < 2000) { 1045c4fa2eeSHeiko Schocher printf("RTC: year %d < 2000 not possible\n", tmp->tm_year); 1055c4fa2eeSHeiko Schocher return -1; 1065c4fa2eeSHeiko Schocher } 1075c4fa2eeSHeiko Schocher buf[RV3029C2_W_SECONDS] = bin2bcd(tmp->tm_sec); 1085c4fa2eeSHeiko Schocher buf[RV3029C2_W_MINUTES] = bin2bcd(tmp->tm_min); 1095c4fa2eeSHeiko Schocher buf[RV3029C2_W_HOURS] = bin2bcd(tmp->tm_hour); 1105c4fa2eeSHeiko Schocher /* set 24h format */ 1115c4fa2eeSHeiko Schocher buf[RV3029C2_W_HOURS] &= ~RV3029C2_REG_HR_12_24; 1125c4fa2eeSHeiko Schocher buf[RV3029C2_W_DATE] = bin2bcd(tmp->tm_mday); 1135c4fa2eeSHeiko Schocher buf[RV3029C2_W_DAYS] = bin2bcd(tmp->tm_wday); 1145c4fa2eeSHeiko Schocher buf[RV3029C2_W_MONTHS] = bin2bcd(tmp->tm_mon); 1155c4fa2eeSHeiko Schocher tmp->tm_year -= 2000; 1165c4fa2eeSHeiko Schocher buf[RV3029C2_W_YEARS] = bin2bcd(tmp->tm_year); 1175c4fa2eeSHeiko Schocher ret = i2c_write(CONFIG_SYS_I2C_RTC_ADDR, RTC_RV3029_CLOCK_PAGE, 1, 1185c4fa2eeSHeiko Schocher buf, RTC_RV3029_PAGE_LEN); 1195c4fa2eeSHeiko Schocher 1205c4fa2eeSHeiko Schocher /* give the RTC some time to update */ 1215c4fa2eeSHeiko Schocher udelay(1000); 122*b633741bSWolfgang Denk return ret; 1235c4fa2eeSHeiko Schocher } 1245c4fa2eeSHeiko Schocher 12571d19f30SHeiko Schocher /* sets EERE-Bit (automatic EEPROM refresh) */ 12671d19f30SHeiko Schocher static void set_eere_bit(int state) 12771d19f30SHeiko Schocher { 12871d19f30SHeiko Schocher unsigned char reg_ctrl1; 12971d19f30SHeiko Schocher 130*b633741bSWolfgang Denk (void)i2c_read(CONFIG_SYS_I2C_RTC_ADDR, RTC_RV3029_CTRL1, 1, 13171d19f30SHeiko Schocher ®_ctrl1, 1); 13271d19f30SHeiko Schocher 13371d19f30SHeiko Schocher if (state) 13471d19f30SHeiko Schocher reg_ctrl1 |= RTC_RV3029_CTRL1_EERE; 13571d19f30SHeiko Schocher else 13671d19f30SHeiko Schocher reg_ctrl1 &= (~RTC_RV3029_CTRL1_EERE); 13771d19f30SHeiko Schocher 138*b633741bSWolfgang Denk (void)i2c_write(CONFIG_SYS_I2C_RTC_ADDR, RTC_RV3029_CTRL1, 1, 13971d19f30SHeiko Schocher ®_ctrl1, 1); 14071d19f30SHeiko Schocher } 14171d19f30SHeiko Schocher 14271d19f30SHeiko Schocher /* waits until EEPROM page is no longer busy (times out after 10ms*loops) */ 14371d19f30SHeiko Schocher static int wait_eebusy(int loops) 14471d19f30SHeiko Schocher { 145*b633741bSWolfgang Denk int i; 14671d19f30SHeiko Schocher unsigned char ctrl_status; 14771d19f30SHeiko Schocher 14871d19f30SHeiko Schocher for (i = 0; i < loops; i++) { 149*b633741bSWolfgang Denk (void)i2c_read(CONFIG_SYS_I2C_RTC_ADDR, RTC_RV3029_CTRL_STATUS, 15071d19f30SHeiko Schocher 1, &ctrl_status, 1); 15171d19f30SHeiko Schocher 15271d19f30SHeiko Schocher if ((ctrl_status & RTC_RV3029_CTRLS_EEBUSY) == 0) 15371d19f30SHeiko Schocher break; 15471d19f30SHeiko Schocher udelay(10000); 15571d19f30SHeiko Schocher } 15671d19f30SHeiko Schocher return i; 15771d19f30SHeiko Schocher } 15871d19f30SHeiko Schocher 1595c4fa2eeSHeiko Schocher void rtc_reset (void) 1605c4fa2eeSHeiko Schocher { 1615c4fa2eeSHeiko Schocher unsigned char buf[RTC_RV3029_PAGE_LEN]; 1625c4fa2eeSHeiko Schocher 1635c4fa2eeSHeiko Schocher buf[0] = RTC_RV3029_CTRL_SYS_R; 164*b633741bSWolfgang Denk (void)i2c_write(CONFIG_SYS_I2C_RTC_ADDR, RTC_RV3029_CTRL_RESET, 1, 1655c4fa2eeSHeiko Schocher buf, 1); 16671d19f30SHeiko Schocher 16771d19f30SHeiko Schocher #if defined(CONFIG_SYS_RV3029_TCR) 16871d19f30SHeiko Schocher /* 16971d19f30SHeiko Schocher * because EEPROM_CTRL register is in EEPROM page it is necessary to 17071d19f30SHeiko Schocher * disable automatic EEPROM refresh and check if EEPROM is busy 17171d19f30SHeiko Schocher * before EEPORM_CTRL register may be accessed 17271d19f30SHeiko Schocher */ 17371d19f30SHeiko Schocher set_eere_bit(0); 17471d19f30SHeiko Schocher wait_eebusy(100); 17571d19f30SHeiko Schocher /* read current trickle charger setting */ 176*b633741bSWolfgang Denk (void)i2c_read(CONFIG_SYS_I2C_RTC_ADDR, RTC_RV3029_EEPROM_CTRL, 17771d19f30SHeiko Schocher 1, buf, 1); 17871d19f30SHeiko Schocher /* enable automatic EEPROM refresh again */ 17971d19f30SHeiko Schocher set_eere_bit(1); 18071d19f30SHeiko Schocher 18171d19f30SHeiko Schocher /* 18271d19f30SHeiko Schocher * to minimize EEPROM access write trickle charger setting only if it 18371d19f30SHeiko Schocher * differs from current value 18471d19f30SHeiko Schocher */ 18571d19f30SHeiko Schocher if ((buf[0] & 0xF0) != CONFIG_SYS_RV3029_TCR) { 18671d19f30SHeiko Schocher buf[0] = (buf[0] & 0x0F) | CONFIG_SYS_RV3029_TCR; 18771d19f30SHeiko Schocher /* 18871d19f30SHeiko Schocher * write trickle charger setting (disable autom. EEPROM 18971d19f30SHeiko Schocher * refresh and wait until EEPROM is idle) 19071d19f30SHeiko Schocher */ 19171d19f30SHeiko Schocher set_eere_bit(0); 19271d19f30SHeiko Schocher wait_eebusy(100); 193*b633741bSWolfgang Denk (void)i2c_write(CONFIG_SYS_I2C_RTC_ADDR, 19471d19f30SHeiko Schocher RTC_RV3029_EEPROM_CTRL, 1, buf, 1); 19571d19f30SHeiko Schocher /* 19671d19f30SHeiko Schocher * it is necessary to wait 10ms before EEBUSY-Bit may be read 19771d19f30SHeiko Schocher * (this is not documented in the data sheet yet, but the 19871d19f30SHeiko Schocher * manufacturer recommends it) 19971d19f30SHeiko Schocher */ 20071d19f30SHeiko Schocher udelay(10000); 20171d19f30SHeiko Schocher /* wait until EEPROM write access is finished */ 20271d19f30SHeiko Schocher wait_eebusy(100); 20371d19f30SHeiko Schocher set_eere_bit(1); 20471d19f30SHeiko Schocher } 20571d19f30SHeiko Schocher #endif 2065c4fa2eeSHeiko Schocher } 207