1*09344530SPriyanka Jain /* 2*09344530SPriyanka Jain * Copyright 2010 Freescale Semiconductor, Inc. 3*09344530SPriyanka Jain * 4*09344530SPriyanka Jain * Author: Priyanka Jain <Priyanka.Jain@freescale.com> 5*09344530SPriyanka Jain * 6*09344530SPriyanka Jain * See file CREDITS for list of people who contributed to this 7*09344530SPriyanka Jain * project. 8*09344530SPriyanka Jain * 9*09344530SPriyanka Jain * This program is free software; you can redistribute it and/or 10*09344530SPriyanka Jain * modify it under the terms of the GNU General Public License as 11*09344530SPriyanka Jain * published by the Free Software Foundation; either version 2 of 12*09344530SPriyanka Jain * the License, or (at your option) any later version. 13*09344530SPriyanka Jain * 14*09344530SPriyanka Jain * This program is distributed in the hope that it will be useful, 15*09344530SPriyanka Jain * but WITHOUT ANY WARRANTY; without even the implied warranty of 16*09344530SPriyanka Jain * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.See the 17*09344530SPriyanka Jain * GNU General Public License for more details. 18*09344530SPriyanka Jain * 19*09344530SPriyanka Jain * You should have received a copy of the GNU General Public License 20*09344530SPriyanka Jain * along with this program; if not, write to the Free Software 21*09344530SPriyanka Jain * Foundation, Inc., 59 Temple Place, Suite 330, Boston, 22*09344530SPriyanka Jain * MA 02111-1307 USA 23*09344530SPriyanka Jain */ 24*09344530SPriyanka Jain 25*09344530SPriyanka Jain /* 26*09344530SPriyanka Jain * This file provides Date & Time support (no alarms) for PT7C4338 chip. 27*09344530SPriyanka Jain * 28*09344530SPriyanka Jain * This file is based on drivers/rtc/ds1337.c 29*09344530SPriyanka Jain * 30*09344530SPriyanka Jain * PT7C4338 chip is manufactured by Pericom Technology Inc. 31*09344530SPriyanka Jain * It is a serial real-time clock which provides 32*09344530SPriyanka Jain * 1)Low-power clock/calendar. 33*09344530SPriyanka Jain * 2)Programmable square-wave output. 34*09344530SPriyanka Jain * It has 56 bytes of nonvolatile RAM. 35*09344530SPriyanka Jain */ 36*09344530SPriyanka Jain 37*09344530SPriyanka Jain #include <common.h> 38*09344530SPriyanka Jain #include <command.h> 39*09344530SPriyanka Jain #include <rtc.h> 40*09344530SPriyanka Jain #include <i2c.h> 41*09344530SPriyanka Jain 42*09344530SPriyanka Jain /* RTC register addresses */ 43*09344530SPriyanka Jain #define RTC_SEC_REG_ADDR 0x0 44*09344530SPriyanka Jain #define RTC_MIN_REG_ADDR 0x1 45*09344530SPriyanka Jain #define RTC_HR_REG_ADDR 0x2 46*09344530SPriyanka Jain #define RTC_DAY_REG_ADDR 0x3 47*09344530SPriyanka Jain #define RTC_DATE_REG_ADDR 0x4 48*09344530SPriyanka Jain #define RTC_MON_REG_ADDR 0x5 49*09344530SPriyanka Jain #define RTC_YR_REG_ADDR 0x6 50*09344530SPriyanka Jain #define RTC_CTL_STAT_REG_ADDR 0x7 51*09344530SPriyanka Jain 52*09344530SPriyanka Jain /* RTC second register address bit */ 53*09344530SPriyanka Jain #define RTC_SEC_BIT_CH 0x80 /* Clock Halt (in Register 0) */ 54*09344530SPriyanka Jain 55*09344530SPriyanka Jain /* RTC control and status register bits */ 56*09344530SPriyanka Jain #define RTC_CTL_STAT_BIT_RS0 0x1 /* Rate select 0 */ 57*09344530SPriyanka Jain #define RTC_CTL_STAT_BIT_RS1 0x2 /* Rate select 1 */ 58*09344530SPriyanka Jain #define RTC_CTL_STAT_BIT_SQWE 0x10 /* Square Wave Enable */ 59*09344530SPriyanka Jain #define RTC_CTL_STAT_BIT_OSF 0x20 /* Oscillator Stop Flag */ 60*09344530SPriyanka Jain #define RTC_CTL_STAT_BIT_OUT 0x80 /* Output Level Control */ 61*09344530SPriyanka Jain 62*09344530SPriyanka Jain /* RTC reset value */ 63*09344530SPriyanka Jain #define RTC_PT7C4338_RESET_VAL \ 64*09344530SPriyanka Jain (RTC_CTL_STAT_BIT_RS0 | RTC_CTL_STAT_BIT_RS1 | RTC_CTL_STAT_BIT_OUT) 65*09344530SPriyanka Jain 66*09344530SPriyanka Jain /****** Helper functions ****************************************/ 67*09344530SPriyanka Jain static u8 rtc_read(u8 reg) 68*09344530SPriyanka Jain { 69*09344530SPriyanka Jain return i2c_reg_read(CONFIG_SYS_I2C_RTC_ADDR, reg); 70*09344530SPriyanka Jain } 71*09344530SPriyanka Jain 72*09344530SPriyanka Jain static void rtc_write(u8 reg, u8 val) 73*09344530SPriyanka Jain { 74*09344530SPriyanka Jain i2c_reg_write(CONFIG_SYS_I2C_RTC_ADDR, reg, val); 75*09344530SPriyanka Jain } 76*09344530SPriyanka Jain /****************************************************************/ 77*09344530SPriyanka Jain 78*09344530SPriyanka Jain /* Get the current time from the RTC */ 79*09344530SPriyanka Jain int rtc_get(struct rtc_time *tmp) 80*09344530SPriyanka Jain { 81*09344530SPriyanka Jain int ret = 0; 82*09344530SPriyanka Jain u8 sec, min, hour, mday, wday, mon, year, ctl_stat; 83*09344530SPriyanka Jain 84*09344530SPriyanka Jain ctl_stat = rtc_read(RTC_CTL_STAT_REG_ADDR); 85*09344530SPriyanka Jain sec = rtc_read(RTC_SEC_REG_ADDR); 86*09344530SPriyanka Jain min = rtc_read(RTC_MIN_REG_ADDR); 87*09344530SPriyanka Jain hour = rtc_read(RTC_HR_REG_ADDR); 88*09344530SPriyanka Jain wday = rtc_read(RTC_DAY_REG_ADDR); 89*09344530SPriyanka Jain mday = rtc_read(RTC_DATE_REG_ADDR); 90*09344530SPriyanka Jain mon = rtc_read(RTC_MON_REG_ADDR); 91*09344530SPriyanka Jain year = rtc_read(RTC_YR_REG_ADDR); 92*09344530SPriyanka Jain debug("Get RTC year: %02x mon: %02x mday: %02x wday: %02x " 93*09344530SPriyanka Jain "hr: %02x min: %02x sec: %02x control_status: %02x\n", 94*09344530SPriyanka Jain year, mon, mday, wday, hour, min, sec, ctl_stat); 95*09344530SPriyanka Jain 96*09344530SPriyanka Jain if (ctl_stat & RTC_CTL_STAT_BIT_OSF) { 97*09344530SPriyanka Jain printf("### Warning: RTC oscillator has stopped\n"); 98*09344530SPriyanka Jain /* clear the OSF flag */ 99*09344530SPriyanka Jain rtc_write(RTC_CTL_STAT_REG_ADDR, 100*09344530SPriyanka Jain rtc_read(RTC_CTL_STAT_REG_ADDR)\ 101*09344530SPriyanka Jain & ~RTC_CTL_STAT_BIT_OSF); 102*09344530SPriyanka Jain ret = -1; 103*09344530SPriyanka Jain } 104*09344530SPriyanka Jain 105*09344530SPriyanka Jain tmp->tm_sec = bcd2bin(sec & 0x7F); 106*09344530SPriyanka Jain tmp->tm_min = bcd2bin(min & 0x7F); 107*09344530SPriyanka Jain tmp->tm_hour = bcd2bin(hour & 0x3F); 108*09344530SPriyanka Jain tmp->tm_mday = bcd2bin(mday & 0x3F); 109*09344530SPriyanka Jain tmp->tm_mon = bcd2bin(mon & 0x1F); 110*09344530SPriyanka Jain tmp->tm_year = bcd2bin(year) + 2000; 111*09344530SPriyanka Jain tmp->tm_wday = bcd2bin((wday - 1) & 0x07); 112*09344530SPriyanka Jain tmp->tm_yday = 0; 113*09344530SPriyanka Jain tmp->tm_isdst = 0; 114*09344530SPriyanka Jain debug("Get DATE: %4d-%02d-%02d (wday=%d) TIME: %2d:%02d:%02d\n", 115*09344530SPriyanka Jain tmp->tm_year, tmp->tm_mon, tmp->tm_mday, tmp->tm_wday, 116*09344530SPriyanka Jain tmp->tm_hour, tmp->tm_min, tmp->tm_sec); 117*09344530SPriyanka Jain 118*09344530SPriyanka Jain return ret; 119*09344530SPriyanka Jain } 120*09344530SPriyanka Jain 121*09344530SPriyanka Jain /* Set the RTC */ 122*09344530SPriyanka Jain int rtc_set(struct rtc_time *tmp) 123*09344530SPriyanka Jain { 124*09344530SPriyanka Jain debug("Set DATE: %4d-%02d-%02d (wday=%d) TIME: %2d:%02d:%02d\n", 125*09344530SPriyanka Jain tmp->tm_year, tmp->tm_mon, tmp->tm_mday, tmp->tm_wday, 126*09344530SPriyanka Jain tmp->tm_hour, tmp->tm_min, tmp->tm_sec); 127*09344530SPriyanka Jain 128*09344530SPriyanka Jain rtc_write(RTC_YR_REG_ADDR, bin2bcd(tmp->tm_year % 100)); 129*09344530SPriyanka Jain rtc_write(RTC_MON_REG_ADDR, bin2bcd(tmp->tm_mon)); 130*09344530SPriyanka Jain rtc_write(RTC_DAY_REG_ADDR, bin2bcd(tmp->tm_wday + 1)); 131*09344530SPriyanka Jain rtc_write(RTC_DATE_REG_ADDR, bin2bcd(tmp->tm_mday)); 132*09344530SPriyanka Jain rtc_write(RTC_HR_REG_ADDR, bin2bcd(tmp->tm_hour)); 133*09344530SPriyanka Jain rtc_write(RTC_MIN_REG_ADDR, bin2bcd(tmp->tm_min)); 134*09344530SPriyanka Jain rtc_write(RTC_SEC_REG_ADDR, bin2bcd(tmp->tm_sec)); 135*09344530SPriyanka Jain 136*09344530SPriyanka Jain return 0; 137*09344530SPriyanka Jain } 138*09344530SPriyanka Jain 139*09344530SPriyanka Jain /* Reset the RTC */ 140*09344530SPriyanka Jain void rtc_reset(void) 141*09344530SPriyanka Jain { 142*09344530SPriyanka Jain rtc_write(RTC_SEC_REG_ADDR, 0x00); /* clearing Clock Halt */ 143*09344530SPriyanka Jain rtc_write(RTC_CTL_STAT_REG_ADDR, RTC_PT7C4338_RESET_VAL); 144*09344530SPriyanka Jain } 145