From e2271430c53d24976ec3b0869dd8993cfba6d768 Mon Sep 17 00:00:00 2001 From: Jordan Crouse Date: Fri, 25 Apr 2008 23:11:02 +0000 Subject: libpayload: Add gettimeofday() and friends Add a gettimeofday() implementation - it works pretty well, but it drifts a little bit so its not very suitable for keeping time. It works best to track changes in time over small periods of time. Signed-off-by: Jordan Crouse Acked-by: Uwe Hermann git-svn-id: svn://svn.coreboot.org/coreboot/trunk@3272 2b7e53f0-3cfb-0310-b3e9-8179ed1497e1 --- payloads/libpayload/drivers/nvram.c | 40 ++++++++++ payloads/libpayload/i386/timer.c | 2 +- payloads/libpayload/include/arch/rdtsc.h | 4 +- payloads/libpayload/include/arch/types.h | 3 + payloads/libpayload/include/libpayload.h | 25 ++++++ payloads/libpayload/libc/Makefile.inc | 2 +- payloads/libpayload/libc/time.c | 128 +++++++++++++++++++++++++++++++ 7 files changed, 200 insertions(+), 4 deletions(-) create mode 100644 payloads/libpayload/libc/time.c diff --git a/payloads/libpayload/drivers/nvram.c b/payloads/libpayload/drivers/nvram.c index 9196d80b86..b28de92b78 100644 --- a/payloads/libpayload/drivers/nvram.c +++ b/payloads/libpayload/drivers/nvram.c @@ -93,3 +93,43 @@ void nvram_write(u8 val, u8 addr) outb(addr, rtc_port); outb(val, rtc_port + 1); } + +/** + * Return 1 if the NVRAM is currently updating and a 0 otherwise + * @return A 1 if the NVRAM is updating and 0 otherwise + */ + +int nvram_updating(void) +{ + return (nvram_read(NVRAM_RTC_FREQ_SELECT) & NVRAM_RTC_UIP) ? 1 : 0; +} + +/** + * Get the current time and date from the RTC + * + * @param time A pointer to a broken-down time structure + */ +void rtc_read_clock(struct tm *time) +{ + memset(time, 0, sizeof(*time)); + + while(nvram_updating()); + + time->tm_mon = bcd2dec(nvram_read(NVRAM_RTC_MONTH)) - 1; + time->tm_sec = bcd2dec(nvram_read(NVRAM_RTC_SECONDS)); + time->tm_min = bcd2dec(nvram_read(NVRAM_RTC_MINUTES)); + time->tm_mday = bcd2dec(nvram_read(NVRAM_RTC_DAY)); + time->tm_hour = bcd2dec(nvram_read(NVRAM_RTC_HOURS)); + + /* Instead of finding the century register, + we just make an assumption that if the year value is + less then 80, then it is 2000+ + */ + + time->tm_year = bcd2dec(nvram_read(NVRAM_RTC_YEAR)); + + if (time->tm_year < 80) + time->tm_year += 100; +} + + diff --git a/payloads/libpayload/i386/timer.c b/payloads/libpayload/i386/timer.c index 789bf0b150..ba3c31ae4e 100644 --- a/payloads/libpayload/i386/timer.c +++ b/payloads/libpayload/i386/timer.c @@ -30,7 +30,7 @@ #include #include -static unsigned int cpu_khz; +unsigned int cpu_khz; /** * Calculate the speed of the processor for use in delays. diff --git a/payloads/libpayload/include/arch/rdtsc.h b/payloads/libpayload/include/arch/rdtsc.h index da7949d808..52f8c9c83c 100644 --- a/payloads/libpayload/include/arch/rdtsc.h +++ b/payloads/libpayload/include/arch/rdtsc.h @@ -30,9 +30,9 @@ #ifndef _ARCH_RDTSC_H #define _ARCH_RDTSC_H -static inline unsigned long long rdtsc(void) +static u64 rdtsc(void) { - unsigned long long val; + u64 val; __asm__ __volatile__ ("rdtsc" : "=A" (val)); return val; } diff --git a/payloads/libpayload/include/arch/types.h b/payloads/libpayload/include/arch/types.h index 641fb0aef8..1bd815bb44 100644 --- a/payloads/libpayload/include/arch/types.h +++ b/payloads/libpayload/include/arch/types.h @@ -50,6 +50,9 @@ typedef unsigned long long u64; typedef signed long long int64_t; typedef signed long long s64; +typedef long time_t; +typedef long suseconds_t; + #ifndef NULL #define NULL ((void *)0) #endif diff --git a/payloads/libpayload/include/libpayload.h b/payloads/libpayload/include/libpayload.h index d557eeff69..5bafb305ef 100644 --- a/payloads/libpayload/include/libpayload.h +++ b/payloads/libpayload/include/libpayload.h @@ -61,10 +61,26 @@ #define NVRAM_RTC_DAY 7 #define NVRAM_RTC_MONTH 8 #define NVRAM_RTC_YEAR 9 +#define NVRAM_RTC_FREQ_SELECT 10 +#define NVRAM_RTC_UIP 0x80 + +struct tm { + int tm_sec; + int tm_min; + int tm_hour; + int tm_mday; + int tm_mon; + int tm_year; + int tm_wday; + int tm_yday; + int tm_isdst; +}; /* drivers/nvram.c */ u8 nvram_read(u8 addr); void nvram_write(u8 val, u8 addr); +int nvram_updating(void); +void rtc_read_clock(struct tm *tm); /* drivers/keyboard.c */ void keyboard_init(void); @@ -182,6 +198,15 @@ char *strchr(const char *s, int c); char *strdup(const char *s); char *strstr(const char *h, const char *n); +/* libc/time.c */ + +struct timeval { + time_t tv_sec; + suseconds_t tv_usec; +}; + +int gettimeofday(struct timeval *tv, void *tz); + /* i386/coreboot.c */ int get_coreboot_info(struct sysinfo_t *info); diff --git a/payloads/libpayload/libc/Makefile.inc b/payloads/libpayload/libc/Makefile.inc index 44975a6a14..938c1fe10c 100644 --- a/payloads/libpayload/libc/Makefile.inc +++ b/payloads/libpayload/libc/Makefile.inc @@ -29,4 +29,4 @@ TARGETS-y += libc/malloc.o libc/printf.o libc/console.o libc/string.o TARGETS-y += libc/memory.o libc/ctype.o libc/ipchecksum.o libc/lib.o -TARGETS-y += libc/rand.o +TARGETS-y += libc/rand.o libc/time.o diff --git a/payloads/libpayload/libc/time.c b/payloads/libpayload/libc/time.c new file mode 100644 index 0000000000..b4dc9c60a9 --- /dev/null +++ b/payloads/libpayload/libc/time.c @@ -0,0 +1,128 @@ +/* + * This file is part of the libpayload project. + * + * Copyright (C) 2008 Advanced Micro Devices, Inc. + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions + * are met: + * 1. Redistributions of source code must retain the above copyright + * notice, this list of conditions and the following disclaimer. + * 2. Redistributions in binary form must reproduce the above copyright + * notice, this list of conditions and the following disclaimer in the + * documentation and/or other materials provided with the distribution. + * 3. The name of the author may not be used to endorse or promote products + * derived from this software without specific prior written permission. + * + * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND + * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE + * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE + * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE + * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL + * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS + * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) + * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT + * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY + * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF + * SUCH DAMAGE. + */ + +#include +#include + +extern u32 cpu_khz; + +static struct { + u64 ticks; + time_t secs; + suseconds_t usecs; +} clock; + +#define TICKS_PER_SEC (cpu_khz * 1000) +#define TICKS_PER_USEC (cpu_khz / 1000) + +static void update_clock(void) +{ + u64 delta = rdtsc() - clock.ticks; + int secs; + + clock.ticks += delta; + + secs = (int) (delta / TICKS_PER_SEC); + clock.secs += secs; + delta -= (secs * TICKS_PER_SEC); + clock.usecs += (int) (delta / TICKS_PER_USEC); + + if (clock.usecs > 1000000) { + clock.usecs -= 1000000; + clock.secs++; + } +} + +#ifdef CONFIG_NVRAM + +static unsigned int day_of_year(int mon, int day, int year) +{ + static u8 mdays[12] = { 31, 28, 31, 30, 31, 30, + 31, 31, 30, 31, 30, 31 }; + + int i, ret = 0; + + for(i = 0; i < mon; i++) { + ret += mdays[i]; + + if (i == 1 && (year % 4)) + ret++; + } + + return (ret + day); +} + +static void gettimeofday_init(void) +{ + int days, delta; + struct tm tm; + + rtc_read_clock(&tm); + clock.ticks = rdtsc(); + + /* Calculate the number of days in the year so far */ + days = day_of_year(tm.tm_mon, tm.tm_mday, tm.tm_year + 1900); + + delta = tm.tm_year - 70; + + days += (delta * 365); + + /* Figure leap years */ + + if (delta > 2) + days += (delta - 2) / 4; + + clock.secs = (days * 86400) + (tm.tm_hour * 3600) + + (tm.tm_min * 60) + tm.tm_sec; +} +#else +static void gettimeofday_init(void) +{ + /* Record the number of ticks */ + clock.ticks = rdtsc(); +} +#endif + +int gettimeofday(struct timeval *tv, void *tz) +{ + /* Call the gtod init when we need it - this keeps + the code from being included in the binary if we don't + need it + */ + + if (!clock.ticks) + gettimeofday_init(); + + update_clock(); + + tv->tv_sec = clock.secs; + tv->tv_usec = clock.usecs; + + return 0; +} -- cgit v1.2.3