Fix checksum calculation both in romstage and ramstage.
[coreboot.git] / src / pc80 / mc146818rtc_early.c
index 96f8f5d0a417638dd73aba876db738526a9a8a26..0652f27c47ff5967eb4a270e8fc37de4b2775b22 100644 (file)
@@ -1,25 +1,17 @@
+#include <stdint.h>
 #include <pc80/mc146818rtc.h>
-#include <part/fallback_boot.h>
+#include <fallback.h>
+#if CONFIG_USE_OPTION_TABLE
+#include "option_table.h"
+#endif
 
-#ifndef MAX_REBOOT_CNT
-#error "MAX_REBOOT_CNT not defined"
+#ifndef CONFIG_MAX_REBOOT_CNT
+#error "CONFIG_MAX_REBOOT_CNT not defined"
 #endif
-#if  MAX_REBOOT_CNT > 15
-#error "MAX_REBOOT_CNT too high"
+#if  CONFIG_MAX_REBOOT_CNT > 15
+#error "CONFIG_MAX_REBOOT_CNT too high"
 #endif
 
-static unsigned char cmos_read(unsigned char addr)
-{
-       outb(addr, RTC_BASE_PORT + 0);
-       return inb(RTC_BASE_PORT + 1);
-}
-
-static void cmos_write(unsigned char val, unsigned char addr)
-{
-       outb(addr, RTC_BASE_PORT + 0);
-       outb(val, RTC_BASE_PORT + 1);
-}
-
 static int cmos_error(void)
 {
        unsigned char reg_d;
@@ -30,62 +22,83 @@ static int cmos_error(void)
 
 static int cmos_chksum_valid(void)
 {
+#if CONFIG_USE_OPTION_TABLE
        unsigned char addr;
-       unsigned long sum, old_sum;
+       u16 sum, old_sum;
        sum = 0;
-       /* Comput the cmos checksum */
+       /* Compute the cmos checksum */
        for(addr = LB_CKS_RANGE_START; addr <= LB_CKS_RANGE_END; addr++) {
                sum += cmos_read(addr);
        }
-       sum = (sum & 0xffff) ^ 0xffff;
 
        /* Read the stored checksum */
        old_sum = cmos_read(LB_CKS_LOC) << 8;
        old_sum |=  cmos_read(LB_CKS_LOC+1);
 
        return sum == old_sum;
+#else
+       return 0;
+#endif
 }
 
 
-static int do_normal_boot(void)
+static inline int last_boot_normal(void)
+{
+       unsigned char byte;
+       byte = cmos_read(RTC_BOOT_BYTE);
+       return (byte & (1 << 1));
+}
+
+static inline int do_normal_boot(void)
 {
        unsigned char byte;
 
        if (cmos_error() || !cmos_chksum_valid()) {
-               unsigned char byte;
-               /* There are no impossible values, no cheksums so just
+               /* There are no impossible values, no checksums so just
                 * trust whatever value we have in the the cmos,
                 * but clear the fallback bit.
                 */
                byte = cmos_read(RTC_BOOT_BYTE);
                byte &= 0x0c;
-               byte |= MAX_REBOOT_CNT << 4;
+               byte |= CONFIG_MAX_REBOOT_CNT << 4;
                cmos_write(byte, RTC_BOOT_BYTE);
        }
 
        /* The RTC_BOOT_BYTE is now o.k. see where to go. */
        byte = cmos_read(RTC_BOOT_BYTE);
-       
+
        /* Are we in normal mode? */
        if (byte & 1) {
                byte &= 0x0f; /* yes, clear the boot count */
        }
 
+       /* Properly set the last boot flag */
+       byte &= 0xfc;
+       if ((byte >> 4) < CONFIG_MAX_REBOOT_CNT) {
+               byte |= (1<<1);
+       }
+
        /* Are we already at the max count? */
-       if ((byte >> 4) < MAX_REBOOT_CNT) {
+       if ((byte >> 4) < CONFIG_MAX_REBOOT_CNT) {
                byte += 1 << 4; /* No, add 1 to the count */
        }
        else {
                byte &= 0xfc;   /* Yes, put in fallback mode */
        }
 
-       /* Is this the first boot? */
-       if ((byte >> 4) <= 1) {
-               byte = (byte & 0xfc) | ((byte & 1) << 1); /* yes, shift the boot bits */
-       }
-
        /* Save the boot byte */
        cmos_write(byte, RTC_BOOT_BYTE);
 
-       return ((byte >> 4) < MAX_REBOOT_CNT);
+       return (byte & (1<<1));
+}
+
+unsigned read_option_lowlevel(unsigned start, unsigned size, unsigned def)
+{
+#if CONFIG_USE_OPTION_TABLE
+       unsigned byte;
+       byte = cmos_read(start/8);
+       return (byte >> (start & 7U)) & ((1U << size) - 1U);
+#else
+       return def;
+#endif
 }