libpayload: Add gettimeofday() and friends
authorJordan Crouse <jordan.crouse@amd.com>
Fri, 25 Apr 2008 23:11:02 +0000 (23:11 +0000)
committerJordan Crouse <jordan.crouse@amd.com>
Fri, 25 Apr 2008 23:11:02 +0000 (23:11 +0000)
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 <jordan.crouse@amd.com>
Acked-by: Uwe Hermann <uwe@hermann-uwe.de>
git-svn-id: svn://svn.coreboot.org/coreboot/trunk@3272 2b7e53f0-3cfb-0310-b3e9-8179ed1497e1

payloads/libpayload/drivers/nvram.c
payloads/libpayload/i386/timer.c
payloads/libpayload/include/arch/rdtsc.h
payloads/libpayload/include/arch/types.h
payloads/libpayload/include/libpayload.h
payloads/libpayload/libc/Makefile.inc
payloads/libpayload/libc/time.c [new file with mode: 0644]

index 9196d80b86983e2276e017f5e67a1447537236de..b28de92b783674794671602c1393860cdc659050 100644 (file)
@@ -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;
+}
+
+
index 789bf0b150a78e080df8baa840acf7b9b5af95c5..ba3c31ae4e5e56b59162fd2250f3d5642cdef698 100644 (file)
@@ -30,7 +30,7 @@
 #include <libpayload.h>
 #include <arch/rdtsc.h>
 
-static unsigned int cpu_khz;
+unsigned int cpu_khz;
 
 /**
  * Calculate the speed of the processor for use in delays.
index da7949d808db8e5e554801d5d4596a7e35b51017..52f8c9c83ce01b72b7dba0027374d5f2e1560659 100644 (file)
@@ -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;
 }
index 641fb0aef83bb424de57cc2e479c1b4d8e890de6..1bd815bb44733d8da04da4bc34a841fe9a916b74 100644 (file)
@@ -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
index d557eeff697a65d8de7bf37435c2d83a658bac65..5bafb305ef58d179bfad3160ba07e8e52b648aeb 100644 (file)
 #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);
 
index 44975a6a143e4fe6e87ccd90cac5002f3503d6ab..938c1fe10c0e8d0a7908ceea738ed8c307123ff6 100644 (file)
@@ -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 (file)
index 0000000..b4dc9c6
--- /dev/null
@@ -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 <libpayload.h>
+#include <arch/rdtsc.h>
+
+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;
+}