time.c revision 83825
1/*
2 * Copyright (c) 1999, 2000
3 * Intel Corporation.
4 * All rights reserved.
5 *
6 * Redistribution and use in source and binary forms, with or without
7 * modification, are permitted provided that the following conditions
8 * are met:
9 *
10 * 1. Redistributions of source code must retain the above copyright
11 *    notice, this list of conditions and the following disclaimer.
12 *
13 * 2. Redistributions in binary form must reproduce the above copyright
14 *    notice, this list of conditions and the following disclaimer in the
15 *    documentation and/or other materials provided with the distribution.
16 *
17 * 3. All advertising materials mentioning features or use of this software
18 *    must display the following acknowledgement:
19 *
20 *    This product includes software developed by Intel Corporation and
21 *    its contributors.
22 *
23 * 4. Neither the name of Intel Corporation or its contributors may be
24 *    used to endorse or promote products derived from this software
25 *    without specific prior written permission.
26 *
27 * THIS SOFTWARE IS PROVIDED BY INTEL CORPORATION AND CONTRIBUTORS ``AS IS''
28 * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
29 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
30 * ARE DISCLAIMED.  IN NO EVENT SHALL INTEL CORPORATION OR CONTRIBUTORS BE
31 * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
32 * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
33 * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
34 * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
35 * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
36 * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF
37 * THE POSSIBILITY OF SUCH DAMAGE.
38 *
39 */
40
41#ifndef lint
42static const char rcsid[] =
43  "$FreeBSD: head/sys/boot/efi/libefi/time.c 83825 2001-09-22 18:33:09Z dfr $";
44#endif /* not lint */
45
46#include <efi.h>
47#include <efilib.h>
48
49#include <time.h>
50#include <sys/time.h>
51
52/*
53// Accurate only for the past couple of centuries;
54// that will probably do.
55//
56// (#defines From FreeBSD 3.2 lib/libc/stdtime/tzfile.h)
57*/
58
59#define isleap(y)	(((y) % 4) == 0 && (((y) % 100) != 0 || ((y) % 400) == 0))
60#define SECSPERHOUR ( 60*60 )
61#define SECSPERDAY	(24 * SECSPERHOUR)
62
63time_t
64EfiTimeToUnixTime(EFI_TIME *ETime)
65{
66    /*
67    //  These arrays give the cumulative number of days up to the first of the
68    //  month number used as the index (1 -> 12) for regular and leap years.
69    //  The value at index 13 is for the whole year.
70    */
71    static time_t CumulativeDays[2][14] = {
72    {0,
73     0,
74     31,
75     31 + 28,
76     31 + 28 + 31,
77     31 + 28 + 31 + 30,
78     31 + 28 + 31 + 30 + 31,
79     31 + 28 + 31 + 30 + 31 + 30,
80     31 + 28 + 31 + 30 + 31 + 30 + 31,
81     31 + 28 + 31 + 30 + 31 + 30 + 31 + 31,
82     31 + 28 + 31 + 30 + 31 + 30 + 31 + 31 + 30,
83     31 + 28 + 31 + 30 + 31 + 30 + 31 + 31 + 30 + 31,
84     31 + 28 + 31 + 30 + 31 + 30 + 31 + 31 + 30 + 31 + 30,
85     31 + 28 + 31 + 30 + 31 + 30 + 31 + 31 + 30 + 31 + 30 + 31 },
86    {0,
87     0,
88     31,
89     31 + 29,
90     31 + 29 + 31,
91     31 + 29 + 31 + 30,
92     31 + 29 + 31 + 30 + 31,
93     31 + 29 + 31 + 30 + 31 + 30,
94     31 + 29 + 31 + 30 + 31 + 30 + 31,
95     31 + 29 + 31 + 30 + 31 + 30 + 31 + 31,
96     31 + 29 + 31 + 30 + 31 + 30 + 31 + 31 + 30,
97     31 + 29 + 31 + 30 + 31 + 30 + 31 + 31 + 30 + 31,
98     31 + 29 + 31 + 30 + 31 + 30 + 31 + 31 + 30 + 31 + 30,
99     31 + 29 + 31 + 30 + 31 + 30 + 31 + 31 + 30 + 31 + 30 + 31 }};
100
101    time_t  UTime;
102    int     Year;
103
104    /*
105    //  Do a santity check
106    */
107    if ( ETime->Year  <  1998 || ETime->Year   > 2099 ||
108    	 ETime->Month ==    0 || ETime->Month  >   12 ||
109    	 ETime->Day   ==    0 || ETime->Month  >   31 ||
110    	                         ETime->Hour   >   23 ||
111    	                         ETime->Minute >   59 ||
112    	                         ETime->Second >   59 ||
113    	 ETime->TimeZone  < -1440                     ||
114    	 (ETime->TimeZone >  1440 && ETime->TimeZone != 2047) ) {
115    	return (0);
116    }
117
118    /*
119    // Years
120    */
121    UTime = 0;
122    for (Year = 1970; Year != ETime->Year; ++Year) {
123        UTime += (CumulativeDays[isleap(Year)][13] * SECSPERDAY);
124    }
125
126    /*
127    // UTime should now be set to 00:00:00 on Jan 1 of the file's year.
128    //
129    // Months
130    */
131    UTime += (CumulativeDays[isleap(ETime->Year)][ETime->Month] * SECSPERDAY);
132
133    /*
134    // UTime should now be set to 00:00:00 on the first of the file's month and year
135    //
136    // Days -- Don't count the file's day
137    */
138    UTime += (((ETime->Day > 0) ? ETime->Day-1:0) * SECSPERDAY);
139
140    /*
141    // Hours
142    */
143    UTime += (ETime->Hour * SECSPERHOUR);
144
145    /*
146    // Minutes
147    */
148    UTime += (ETime->Minute * 60);
149
150    /*
151    // Seconds
152    */
153    UTime += ETime->Second;
154
155    /*
156    //  EFI time is repored in local time.  Adjust for any time zone offset to
157    //  get true UT
158    */
159    if ( ETime->TimeZone != EFI_UNSPECIFIED_TIMEZONE ) {
160    	/*
161    	//  TimeZone is kept in minues...
162    	*/
163    	UTime += (ETime->TimeZone * 60);
164    }
165
166    return UTime;
167}
168
169int
170EFI_GetTimeOfDay(
171	OUT struct timeval *tp,
172	OUT struct timezone *tzp
173	)
174{
175	EFI_TIME				EfiTime;
176	EFI_TIME_CAPABILITIES	Capabilities;
177	EFI_STATUS				Status;
178
179	/*
180	//  Get time from EFI
181	*/
182
183	Status = RS->GetTime( &EfiTime, &Capabilities );
184	if (EFI_ERROR(Status))
185		return (-1);
186
187	/*
188	//  Convert to UNIX time (ie seconds since the epoch
189	*/
190
191	tp->tv_sec  = EfiTimeToUnixTime( &EfiTime );
192	tp->tv_usec = 0; /* EfiTime.Nanosecond * 1000; */
193
194	/*
195	//  Do something with the timezone if needed
196	*/
197
198	if (tzp) {
199		tzp->tz_minuteswest =
200			EfiTime.TimeZone == EFI_UNSPECIFIED_TIMEZONE ? 0 : EfiTime.TimeZone;
201		/*
202		//  This isn't quit right since it doesn't deal with EFI_TIME_IN_DAYLIGHT
203		*/
204		tzp->tz_dsttime =
205			EfiTime.Daylight & EFI_TIME_ADJUST_DAYLIGHT ? 1 : 0;
206	}
207
208	return (0);
209}
210
211time_t
212time(time_t *tloc)
213{
214	struct timeval tv;
215	EFI_GetTimeOfDay(&tv, 0);
216
217	if (tloc)
218		*tloc = tv.tv_sec;
219	return tv.tv_sec;
220}
221
222time_t
223getsecs()
224{
225    return time(0);
226}
227