blob: e3491662bc5819bc837545d9ed953ff21937ad0c [file] [log] [blame]
wdenk38635852002-08-27 05:55:31 +00001/*
2 * (C) Copyright 2001
3 * Wolfgang Denk, DENX Software Engineering, wd@denx.de.
4 *
Wolfgang Denkd79de1d2013-07-08 09:37:19 +02005 * SPDX-License-Identifier: GPL-2.0+
wdenk38635852002-08-27 05:55:31 +00006 */
7
8/*
9 * RTC, Date & Time support: get and set date & time
10 */
11#include <common.h>
12#include <command.h>
13#include <rtc.h>
Stefan Roese096cc9b2007-02-20 10:51:26 +010014#include <i2c.h>
wdenk38635852002-08-27 05:55:31 +000015
Wolfgang Denk6405a152006-03-31 18:32:53 +020016DECLARE_GLOBAL_DATA_PTR;
17
Mike Frysinger59e14722010-10-20 07:17:23 -040018static const char * const weekdays[] = {
wdenk38635852002-08-27 05:55:31 +000019 "Sun", "Mon", "Tues", "Wednes", "Thurs", "Fri", "Satur",
20};
21
Wolfgang Denkd0813e52010-10-28 20:00:11 +020022#ifdef CONFIG_NEEDS_MANUAL_RELOC
wdenkb02744a2003-04-05 00:53:31 +000023#define RELOC(a) ((typeof(a))((unsigned long)(a) + gd->reloc_off))
Wolfgang Denkd0813e52010-10-28 20:00:11 +020024#else
25#define RELOC(a) a
Peter Tyser9057cbf2009-09-21 11:20:36 -050026#endif
wdenkb02744a2003-04-05 00:53:31 +000027
Mike Frysinger59e14722010-10-20 07:17:23 -040028int mk_date (const char *, struct rtc_time *);
wdenk38635852002-08-27 05:55:31 +000029
Kim Phillipsdc00a682012-10-29 13:34:31 +000030static int do_date(cmd_tbl_t *cmdtp, int flag, int argc, char * const argv[])
wdenk38635852002-08-27 05:55:31 +000031{
32 struct rtc_time tm;
33 int rcode = 0;
Stefan Roese096cc9b2007-02-20 10:51:26 +010034 int old_bus;
35
36 /* switch to correct I2C bus */
Heiko Schochere0e55bc2012-01-16 21:12:24 +000037#ifdef CONFIG_SYS_I2C
38 old_bus = i2c_get_bus_num();
39 i2c_set_bus_num(CONFIG_SYS_RTC_BUS_NUM);
40#else
Stefan Roese096cc9b2007-02-20 10:51:26 +010041 old_bus = I2C_GET_BUS();
Jean-Christophe PLAGNIOL-VILLARD03836942008-10-16 15:01:15 +020042 I2C_SET_BUS(CONFIG_SYS_RTC_BUS_NUM);
Heiko Schochere0e55bc2012-01-16 21:12:24 +000043#endif
wdenk38635852002-08-27 05:55:31 +000044
45 switch (argc) {
46 case 2: /* set date & time */
47 if (strcmp(argv[1],"reset") == 0) {
wdenk42c05472004-03-23 22:14:11 +000048 puts ("Reset RTC...\n");
wdenk38635852002-08-27 05:55:31 +000049 rtc_reset ();
50 } else {
51 /* initialize tm with current time */
Jean-Christophe PLAGNIOL-VILLARD97a2e102008-09-01 23:06:23 +020052 rcode = rtc_get (&tm);
53
54 if(!rcode) {
55 /* insert new date & time */
56 if (mk_date (argv[1], &tm) != 0) {
57 puts ("## Bad date format\n");
58 break;
59 }
60 /* and write to RTC */
61 rcode = rtc_set (&tm);
62 if(rcode)
Magnus Lilja55992362009-11-11 19:56:36 +010063 puts("## Set date failed\n");
Jean-Christophe PLAGNIOL-VILLARD97a2e102008-09-01 23:06:23 +020064 } else {
Magnus Lilja55992362009-11-11 19:56:36 +010065 puts("## Get date failed\n");
wdenk38635852002-08-27 05:55:31 +000066 }
wdenk38635852002-08-27 05:55:31 +000067 }
68 /* FALL TROUGH */
69 case 1: /* get date & time */
Jean-Christophe PLAGNIOL-VILLARD97a2e102008-09-01 23:06:23 +020070 rcode = rtc_get (&tm);
71
72 if (rcode) {
Magnus Lilja55992362009-11-11 19:56:36 +010073 puts("## Get date failed\n");
Jean-Christophe PLAGNIOL-VILLARD97a2e102008-09-01 23:06:23 +020074 break;
75 }
wdenk38635852002-08-27 05:55:31 +000076
77 printf ("Date: %4d-%02d-%02d (%sday) Time: %2d:%02d:%02d\n",
78 tm.tm_year, tm.tm_mon, tm.tm_mday,
79 (tm.tm_wday<0 || tm.tm_wday>6) ?
wdenkb02744a2003-04-05 00:53:31 +000080 "unknown " : RELOC(weekdays[tm.tm_wday]),
wdenk38635852002-08-27 05:55:31 +000081 tm.tm_hour, tm.tm_min, tm.tm_sec);
82
Stefan Roese096cc9b2007-02-20 10:51:26 +010083 break;
wdenk38635852002-08-27 05:55:31 +000084 default:
Simon Glassa06dfc72011-12-10 08:44:01 +000085 rcode = CMD_RET_USAGE;
wdenk38635852002-08-27 05:55:31 +000086 }
Stefan Roese096cc9b2007-02-20 10:51:26 +010087
88 /* switch back to original I2C bus */
Heiko Schochere0e55bc2012-01-16 21:12:24 +000089#ifdef CONFIG_SYS_I2C
90 i2c_set_bus_num(old_bus);
91#else
Stefan Roese096cc9b2007-02-20 10:51:26 +010092 I2C_SET_BUS(old_bus);
Heiko Schochere0e55bc2012-01-16 21:12:24 +000093#endif
Stefan Roese096cc9b2007-02-20 10:51:26 +010094
wdenk38635852002-08-27 05:55:31 +000095 return rcode;
96}
97
98/*
99 * simple conversion of two-digit string with error checking
100 */
Mike Frysinger59e14722010-10-20 07:17:23 -0400101static int cnvrt2 (const char *str, int *valp)
wdenk38635852002-08-27 05:55:31 +0000102{
103 int val;
104
105 if ((*str < '0') || (*str > '9'))
106 return (-1);
107
108 val = *str - '0';
109
110 ++str;
111
112 if ((*str < '0') || (*str > '9'))
113 return (-1);
114
115 *valp = 10 * val + (*str - '0');
116
117 return (0);
118}
119
120/*
121 * Convert date string: MMDDhhmm[[CC]YY][.ss]
122 *
123 * Some basic checking for valid values is done, but this will not catch
124 * all possible error conditions.
125 */
Mike Frysinger59e14722010-10-20 07:17:23 -0400126int mk_date (const char *datestr, struct rtc_time *tmp)
wdenk38635852002-08-27 05:55:31 +0000127{
128 int len, val;
129 char *ptr;
130
131 ptr = strchr (datestr,'.');
132 len = strlen (datestr);
133
134 /* Set seconds */
135 if (ptr) {
136 int sec;
137
138 *ptr++ = '\0';
139 if ((len - (ptr - datestr)) != 2)
140 return (-1);
141
142 len = strlen (datestr);
143
144 if (cnvrt2 (ptr, &sec))
145 return (-1);
146
147 tmp->tm_sec = sec;
148 } else {
149 tmp->tm_sec = 0;
150 }
151
152 if (len == 12) { /* MMDDhhmmCCYY */
153 int year, century;
154
155 if (cnvrt2 (datestr+ 8, &century) ||
156 cnvrt2 (datestr+10, &year) ) {
157 return (-1);
158 }
159 tmp->tm_year = 100 * century + year;
160 } else if (len == 10) { /* MMDDhhmmYY */
161 int year, century;
162
163 century = tmp->tm_year / 100;
164 if (cnvrt2 (datestr+ 8, &year))
165 return (-1);
166 tmp->tm_year = 100 * century + year;
167 }
168
169 switch (len) {
170 case 8: /* MMDDhhmm */
171 /* fall thru */
172 case 10: /* MMDDhhmmYY */
173 /* fall thru */
174 case 12: /* MMDDhhmmCCYY */
175 if (cnvrt2 (datestr+0, &val) ||
176 val > 12) {
177 break;
178 }
179 tmp->tm_mon = val;
180 if (cnvrt2 (datestr+2, &val) ||
181 val > ((tmp->tm_mon==2) ? 29 : 31)) {
182 break;
183 }
184 tmp->tm_mday = val;
185
186 if (cnvrt2 (datestr+4, &val) ||
187 val > 23) {
188 break;
189 }
190 tmp->tm_hour = val;
191
192 if (cnvrt2 (datestr+6, &val) ||
193 val > 59) {
194 break;
195 }
196 tmp->tm_min = val;
197
198 /* calculate day of week */
199 GregorianDay (tmp);
200
201 return (0);
202 default:
203 break;
204 }
205
206 return (-1);
207}
208
wdenk57b2d802003-06-27 21:31:46 +0000209/***************************************************/
210
wdenkf287a242003-07-01 21:06:45 +0000211U_BOOT_CMD(
212 date, 2, 1, do_date,
Peter Tyserdfb72b82009-01-27 18:03:12 -0600213 "get/set/reset date & time",
wdenk57b2d802003-06-27 21:31:46 +0000214 "[MMDDhhmm[[CC]YY][.ss]]\ndate reset\n"
215 " - without arguments: print date & time\n"
216 " - with numeric argument: set the system date & time\n"
Wolfgang Denkc54781c2009-05-24 17:06:54 +0200217 " - with 'reset' argument: reset the RTC"
wdenk57b2d802003-06-27 21:31:46 +0000218);