blob: 4a653e5bcfab8e24c35608aa256c8c8161b2b291 [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
Marek Vasutd45c1b82015-04-10 01:11:54 +020030static struct rtc_time default_tm = { 0, 0, 0, 1, 1, 2000, 6, 0, 0 };
31
Kim Phillipsdc00a682012-10-29 13:34:31 +000032static int do_date(cmd_tbl_t *cmdtp, int flag, int argc, char * const argv[])
wdenk38635852002-08-27 05:55:31 +000033{
34 struct rtc_time tm;
35 int rcode = 0;
Stefan Roese096cc9b2007-02-20 10:51:26 +010036 int old_bus;
37
38 /* switch to correct I2C bus */
Heiko Schochere0e55bc2012-01-16 21:12:24 +000039#ifdef CONFIG_SYS_I2C
40 old_bus = i2c_get_bus_num();
41 i2c_set_bus_num(CONFIG_SYS_RTC_BUS_NUM);
42#else
Stefan Roese096cc9b2007-02-20 10:51:26 +010043 old_bus = I2C_GET_BUS();
Jean-Christophe PLAGNIOL-VILLARD03836942008-10-16 15:01:15 +020044 I2C_SET_BUS(CONFIG_SYS_RTC_BUS_NUM);
Heiko Schochere0e55bc2012-01-16 21:12:24 +000045#endif
wdenk38635852002-08-27 05:55:31 +000046
47 switch (argc) {
48 case 2: /* set date & time */
49 if (strcmp(argv[1],"reset") == 0) {
wdenk42c05472004-03-23 22:14:11 +000050 puts ("Reset RTC...\n");
wdenk38635852002-08-27 05:55:31 +000051 rtc_reset ();
Marek Vasutd45c1b82015-04-10 01:11:54 +020052 rcode = rtc_set(&default_tm);
53 if (rcode)
54 puts("## Failed to set date after RTC reset\n");
wdenk38635852002-08-27 05:55:31 +000055 } else {
56 /* initialize tm with current time */
Jean-Christophe PLAGNIOL-VILLARD97a2e102008-09-01 23:06:23 +020057 rcode = rtc_get (&tm);
58
59 if(!rcode) {
60 /* insert new date & time */
61 if (mk_date (argv[1], &tm) != 0) {
62 puts ("## Bad date format\n");
63 break;
64 }
65 /* and write to RTC */
66 rcode = rtc_set (&tm);
67 if(rcode)
Magnus Lilja55992362009-11-11 19:56:36 +010068 puts("## Set date failed\n");
Jean-Christophe PLAGNIOL-VILLARD97a2e102008-09-01 23:06:23 +020069 } else {
Magnus Lilja55992362009-11-11 19:56:36 +010070 puts("## Get date failed\n");
wdenk38635852002-08-27 05:55:31 +000071 }
wdenk38635852002-08-27 05:55:31 +000072 }
73 /* FALL TROUGH */
74 case 1: /* get date & time */
Jean-Christophe PLAGNIOL-VILLARD97a2e102008-09-01 23:06:23 +020075 rcode = rtc_get (&tm);
76
77 if (rcode) {
Magnus Lilja55992362009-11-11 19:56:36 +010078 puts("## Get date failed\n");
Jean-Christophe PLAGNIOL-VILLARD97a2e102008-09-01 23:06:23 +020079 break;
80 }
wdenk38635852002-08-27 05:55:31 +000081
82 printf ("Date: %4d-%02d-%02d (%sday) Time: %2d:%02d:%02d\n",
83 tm.tm_year, tm.tm_mon, tm.tm_mday,
84 (tm.tm_wday<0 || tm.tm_wday>6) ?
wdenkb02744a2003-04-05 00:53:31 +000085 "unknown " : RELOC(weekdays[tm.tm_wday]),
wdenk38635852002-08-27 05:55:31 +000086 tm.tm_hour, tm.tm_min, tm.tm_sec);
87
Stefan Roese096cc9b2007-02-20 10:51:26 +010088 break;
wdenk38635852002-08-27 05:55:31 +000089 default:
Simon Glassa06dfc72011-12-10 08:44:01 +000090 rcode = CMD_RET_USAGE;
wdenk38635852002-08-27 05:55:31 +000091 }
Stefan Roese096cc9b2007-02-20 10:51:26 +010092
93 /* switch back to original I2C bus */
Heiko Schochere0e55bc2012-01-16 21:12:24 +000094#ifdef CONFIG_SYS_I2C
95 i2c_set_bus_num(old_bus);
96#else
Stefan Roese096cc9b2007-02-20 10:51:26 +010097 I2C_SET_BUS(old_bus);
Heiko Schochere0e55bc2012-01-16 21:12:24 +000098#endif
Stefan Roese096cc9b2007-02-20 10:51:26 +010099
wdenk38635852002-08-27 05:55:31 +0000100 return rcode;
101}
102
103/*
104 * simple conversion of two-digit string with error checking
105 */
Mike Frysinger59e14722010-10-20 07:17:23 -0400106static int cnvrt2 (const char *str, int *valp)
wdenk38635852002-08-27 05:55:31 +0000107{
108 int val;
109
110 if ((*str < '0') || (*str > '9'))
111 return (-1);
112
113 val = *str - '0';
114
115 ++str;
116
117 if ((*str < '0') || (*str > '9'))
118 return (-1);
119
120 *valp = 10 * val + (*str - '0');
121
122 return (0);
123}
124
125/*
126 * Convert date string: MMDDhhmm[[CC]YY][.ss]
127 *
128 * Some basic checking for valid values is done, but this will not catch
129 * all possible error conditions.
130 */
Mike Frysinger59e14722010-10-20 07:17:23 -0400131int mk_date (const char *datestr, struct rtc_time *tmp)
wdenk38635852002-08-27 05:55:31 +0000132{
133 int len, val;
134 char *ptr;
135
136 ptr = strchr (datestr,'.');
137 len = strlen (datestr);
138
139 /* Set seconds */
140 if (ptr) {
141 int sec;
142
143 *ptr++ = '\0';
144 if ((len - (ptr - datestr)) != 2)
145 return (-1);
146
147 len = strlen (datestr);
148
149 if (cnvrt2 (ptr, &sec))
150 return (-1);
151
152 tmp->tm_sec = sec;
153 } else {
154 tmp->tm_sec = 0;
155 }
156
157 if (len == 12) { /* MMDDhhmmCCYY */
158 int year, century;
159
160 if (cnvrt2 (datestr+ 8, &century) ||
161 cnvrt2 (datestr+10, &year) ) {
162 return (-1);
163 }
164 tmp->tm_year = 100 * century + year;
165 } else if (len == 10) { /* MMDDhhmmYY */
166 int year, century;
167
168 century = tmp->tm_year / 100;
169 if (cnvrt2 (datestr+ 8, &year))
170 return (-1);
171 tmp->tm_year = 100 * century + year;
172 }
173
174 switch (len) {
175 case 8: /* MMDDhhmm */
176 /* fall thru */
177 case 10: /* MMDDhhmmYY */
178 /* fall thru */
179 case 12: /* MMDDhhmmCCYY */
180 if (cnvrt2 (datestr+0, &val) ||
181 val > 12) {
182 break;
183 }
184 tmp->tm_mon = val;
185 if (cnvrt2 (datestr+2, &val) ||
186 val > ((tmp->tm_mon==2) ? 29 : 31)) {
187 break;
188 }
189 tmp->tm_mday = val;
190
191 if (cnvrt2 (datestr+4, &val) ||
192 val > 23) {
193 break;
194 }
195 tmp->tm_hour = val;
196
197 if (cnvrt2 (datestr+6, &val) ||
198 val > 59) {
199 break;
200 }
201 tmp->tm_min = val;
202
203 /* calculate day of week */
204 GregorianDay (tmp);
205
206 return (0);
207 default:
208 break;
209 }
210
211 return (-1);
212}
213
wdenk57b2d802003-06-27 21:31:46 +0000214/***************************************************/
215
wdenkf287a242003-07-01 21:06:45 +0000216U_BOOT_CMD(
217 date, 2, 1, do_date,
Peter Tyserdfb72b82009-01-27 18:03:12 -0600218 "get/set/reset date & time",
wdenk57b2d802003-06-27 21:31:46 +0000219 "[MMDDhhmm[[CC]YY][.ss]]\ndate reset\n"
220 " - without arguments: print date & time\n"
221 " - with numeric argument: set the system date & time\n"
Wolfgang Denkc54781c2009-05-24 17:06:54 +0200222 " - with 'reset' argument: reset the RTC"
wdenk57b2d802003-06-27 21:31:46 +0000223);