xref: /rk3399_rockchip-uboot/drivers/rtc/davinci.c (revision 41d272d1efa2c54ed81bfb701d1960691bb4eb0d)
1*41d272d1SHeiko Schocher /*
2*41d272d1SHeiko Schocher  * (C) Copyright 2011 DENX Software Engineering GmbH
3*41d272d1SHeiko Schocher  * Heiko Schocher <hs@denx.de>
4*41d272d1SHeiko Schocher  *
5*41d272d1SHeiko Schocher  * See file CREDITS for list of people who contributed to this
6*41d272d1SHeiko Schocher  * project.
7*41d272d1SHeiko Schocher  *
8*41d272d1SHeiko Schocher  * This program is free software; you can redistribute it and/or
9*41d272d1SHeiko Schocher  * modify it under the terms of the GNU General Public License as
10*41d272d1SHeiko Schocher  * published by the Free Software Foundation; either version 2 of
11*41d272d1SHeiko Schocher  * the License, or (at your option) any later version.
12*41d272d1SHeiko Schocher  *
13*41d272d1SHeiko Schocher  * This program is distributed in the hope that it will be useful,
14*41d272d1SHeiko Schocher  * but WITHOUT ANY WARRANTY; without even the implied warranty of
15*41d272d1SHeiko Schocher  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
16*41d272d1SHeiko Schocher  * GNU General Public License for more details.
17*41d272d1SHeiko Schocher  *
18*41d272d1SHeiko Schocher  * You should have received a copy of the GNU General Public License
19*41d272d1SHeiko Schocher  * along with this program; if not, write to the Free Software
20*41d272d1SHeiko Schocher  * Foundation, Inc., 59 Temple Place, Suite 330, Boston,
21*41d272d1SHeiko Schocher  * MA 02111-1307 USA
22*41d272d1SHeiko Schocher  */
23*41d272d1SHeiko Schocher #include <common.h>
24*41d272d1SHeiko Schocher #include <command.h>
25*41d272d1SHeiko Schocher #include <rtc.h>
26*41d272d1SHeiko Schocher #include <asm/io.h>
27*41d272d1SHeiko Schocher #include <asm/arch/hardware.h>
28*41d272d1SHeiko Schocher 
29*41d272d1SHeiko Schocher #if defined(CONFIG_CMD_DATE)
30*41d272d1SHeiko Schocher struct davinci_rtc {
31*41d272d1SHeiko Schocher 	u_int32_t	second;
32*41d272d1SHeiko Schocher 	u_int32_t	minutes;
33*41d272d1SHeiko Schocher 	u_int32_t	hours;
34*41d272d1SHeiko Schocher 	u_int32_t	day;
35*41d272d1SHeiko Schocher 	u_int32_t	month; /* 0x10 */
36*41d272d1SHeiko Schocher 	u_int32_t	year;
37*41d272d1SHeiko Schocher 	u_int32_t	dotw;
38*41d272d1SHeiko Schocher 	u_int32_t	resv1;
39*41d272d1SHeiko Schocher 	u_int32_t	alarmsecond; /* 0x20 */
40*41d272d1SHeiko Schocher 	u_int32_t	alarmminute;
41*41d272d1SHeiko Schocher 	u_int32_t	alarmhour;
42*41d272d1SHeiko Schocher 	u_int32_t	alarmday;
43*41d272d1SHeiko Schocher 	u_int32_t	alarmmonth; /* 0x30 */
44*41d272d1SHeiko Schocher 	u_int32_t	alarmyear;
45*41d272d1SHeiko Schocher 	u_int32_t	resv2[2];
46*41d272d1SHeiko Schocher 	u_int32_t	ctrl; /* 0x40 */
47*41d272d1SHeiko Schocher 	u_int32_t	status;
48*41d272d1SHeiko Schocher 	u_int32_t	irq;
49*41d272d1SHeiko Schocher };
50*41d272d1SHeiko Schocher 
51*41d272d1SHeiko Schocher #define RTC_STATE_BUSY	0x01
52*41d272d1SHeiko Schocher #define RTC_STATE_RUN	0x02
53*41d272d1SHeiko Schocher 
54*41d272d1SHeiko Schocher #define davinci_rtc_base ((struct davinci_rtc *)DAVINCI_RTC_BASE)
55*41d272d1SHeiko Schocher 
56*41d272d1SHeiko Schocher int rtc_get(struct rtc_time *tmp)
57*41d272d1SHeiko Schocher {
58*41d272d1SHeiko Schocher 	struct davinci_rtc *rtc = davinci_rtc_base;
59*41d272d1SHeiko Schocher 	unsigned long sec, min, hour, mday, wday, mon_cent, year;
60*41d272d1SHeiko Schocher 	unsigned long status;
61*41d272d1SHeiko Schocher 
62*41d272d1SHeiko Schocher 	status = readl(&rtc->status);
63*41d272d1SHeiko Schocher 	if ((status & RTC_STATE_RUN) != RTC_STATE_RUN) {
64*41d272d1SHeiko Schocher 		printf("RTC doesn't run\n");
65*41d272d1SHeiko Schocher 		return -1;
66*41d272d1SHeiko Schocher 	}
67*41d272d1SHeiko Schocher 	if ((status & RTC_STATE_BUSY) == RTC_STATE_BUSY)
68*41d272d1SHeiko Schocher 		udelay(20);
69*41d272d1SHeiko Schocher 
70*41d272d1SHeiko Schocher 	sec	= readl(&rtc->second);
71*41d272d1SHeiko Schocher 	min	= readl(&rtc->minutes);
72*41d272d1SHeiko Schocher 	hour	= readl(&rtc->hours);
73*41d272d1SHeiko Schocher 	mday	= readl(&rtc->day);
74*41d272d1SHeiko Schocher 	wday	= readl(&rtc->dotw);
75*41d272d1SHeiko Schocher 	mon_cent = readl(&rtc->month);
76*41d272d1SHeiko Schocher 	year	= readl(&rtc->year);
77*41d272d1SHeiko Schocher 
78*41d272d1SHeiko Schocher 	debug("Get RTC year: %02lx mon/cent: %02lx mday: %02lx wday: %02lx "
79*41d272d1SHeiko Schocher 		"hr: %02lx min: %02lx sec: %02lx\n",
80*41d272d1SHeiko Schocher 		year, mon_cent, mday, wday,
81*41d272d1SHeiko Schocher 		hour, min, sec);
82*41d272d1SHeiko Schocher 
83*41d272d1SHeiko Schocher 	tmp->tm_sec  = bcd2bin(sec  & 0x7F);
84*41d272d1SHeiko Schocher 	tmp->tm_min  = bcd2bin(min  & 0x7F);
85*41d272d1SHeiko Schocher 	tmp->tm_hour = bcd2bin(hour & 0x3F);
86*41d272d1SHeiko Schocher 	tmp->tm_mday = bcd2bin(mday & 0x3F);
87*41d272d1SHeiko Schocher 	tmp->tm_mon  = bcd2bin(mon_cent & 0x1F);
88*41d272d1SHeiko Schocher 	tmp->tm_year = bcd2bin(year) + 2000;
89*41d272d1SHeiko Schocher 	tmp->tm_wday = bcd2bin(wday & 0x07);
90*41d272d1SHeiko Schocher 	tmp->tm_yday = 0;
91*41d272d1SHeiko Schocher 	tmp->tm_isdst = 0;
92*41d272d1SHeiko Schocher 
93*41d272d1SHeiko Schocher 	debug("Get DATE: %4d-%02d-%02d (wday=%d)  TIME: %2d:%02d:%02d\n",
94*41d272d1SHeiko Schocher 		tmp->tm_year, tmp->tm_mon, tmp->tm_mday, tmp->tm_wday,
95*41d272d1SHeiko Schocher 		tmp->tm_hour, tmp->tm_min, tmp->tm_sec);
96*41d272d1SHeiko Schocher 
97*41d272d1SHeiko Schocher 	return 0;
98*41d272d1SHeiko Schocher }
99*41d272d1SHeiko Schocher 
100*41d272d1SHeiko Schocher int rtc_set(struct rtc_time *tmp)
101*41d272d1SHeiko Schocher {
102*41d272d1SHeiko Schocher 	struct davinci_rtc *rtc = davinci_rtc_base;
103*41d272d1SHeiko Schocher 
104*41d272d1SHeiko Schocher 	debug("Set DATE: %4d-%02d-%02d (wday=%d)  TIME: %2d:%02d:%02d\n",
105*41d272d1SHeiko Schocher 		tmp->tm_year, tmp->tm_mon, tmp->tm_mday, tmp->tm_wday,
106*41d272d1SHeiko Schocher 		tmp->tm_hour, tmp->tm_min, tmp->tm_sec);
107*41d272d1SHeiko Schocher 	writel(bin2bcd(tmp->tm_year % 100), &rtc->year);
108*41d272d1SHeiko Schocher 	writel(bin2bcd(tmp->tm_mon), &rtc->month);
109*41d272d1SHeiko Schocher 
110*41d272d1SHeiko Schocher 	writel(bin2bcd(tmp->tm_wday), &rtc->dotw);
111*41d272d1SHeiko Schocher 	writel(bin2bcd(tmp->tm_mday), &rtc->day);
112*41d272d1SHeiko Schocher 	writel(bin2bcd(tmp->tm_hour), &rtc->hours);
113*41d272d1SHeiko Schocher 	writel(bin2bcd(tmp->tm_min), &rtc->minutes);
114*41d272d1SHeiko Schocher 	writel(bin2bcd(tmp->tm_sec), &rtc->second);
115*41d272d1SHeiko Schocher 	return 0;
116*41d272d1SHeiko Schocher }
117*41d272d1SHeiko Schocher 
118*41d272d1SHeiko Schocher void rtc_reset(void)
119*41d272d1SHeiko Schocher {
120*41d272d1SHeiko Schocher 	struct davinci_rtc *rtc = davinci_rtc_base;
121*41d272d1SHeiko Schocher 
122*41d272d1SHeiko Schocher 	/* run RTC counter */
123*41d272d1SHeiko Schocher 	writel(0x01, &rtc->ctrl);
124*41d272d1SHeiko Schocher }
125*41d272d1SHeiko Schocher #endif
126