Move cmos.default handling to bootblock
authorPatrick Georgi <patrick.georgi@secunet.com>
Tue, 8 Mar 2011 07:50:43 +0000 (07:50 +0000)
committerPatrick Georgi <patrick.georgi@coresystems.de>
Tue, 8 Mar 2011 07:50:43 +0000 (07:50 +0000)
The cmos.default code wasn't actually used so far, due to an oversight
when forward-porting this feature from an old branch.

- Extend walkcbfs' use by factoring out the stage handling into C code.
- New sanitize_cmos() function that looks if CMOS data is invalid and
  cmos.default exists and if so overwrites CMOS with cmos.default data.
- Use sanitize_cmos() in both bootblock implementations.
- Drop the need to reboot after writing CMOS: CMOS wasn't used so far,
  so we can go on without a reboot.
- Remove the restriction that cmos.default only works on CAR boards.
- Always build in cmos.default support on boards that
  USE_OPTION_TABLE.

Signed-off-by: Patrick Georgi <patrick.georgi@secunet.com>
Acked-by: Stefan Reinauer <stefan.reinauer@coreboot.org>
git-svn-id: svn://svn.coreboot.org/coreboot/trunk@6436 2b7e53f0-3cfb-0310-b3e9-8179ed1497e1

src/arch/x86/Kconfig
src/arch/x86/Makefile.bootblock.inc
src/arch/x86/include/bootblock_common.h
src/arch/x86/init/bootblock_normal.c
src/arch/x86/init/bootblock_simple.c
src/arch/x86/lib/walkcbfs.S
src/pc80/mc146818rtc_early.c

index 3684c69e27b663698db6f4036e0657a8f06c1fc4..aacc0982a6c01b8646ef9f12222a02f954ac4de3 100644 (file)
@@ -91,14 +91,8 @@ config PC80_SYSTEM
 config BOOTBLOCK_NORTHBRIDGE_INIT
        string
 
-config USE_CMOS_RECOVERY
-       bool
-       default n if ROMCC
-       default y
-
 config HAVE_CMOS_DEFAULT
        def_bool n
-       depends on USE_CMOS_RECOVERY
 
 config CMOS_DEFAULT_FILE
        string
index 31db2249159190d8bcb788df82f5e7a86bf6a4e3..4958a77cb7a60cfce63ec05f2c155d7489b67c19 100644 (file)
@@ -57,7 +57,7 @@ $(obj)/mainboard/$(MAINBOARDDIR)/bootblock.s: $(obj)/bootblock/bootblock.S
        @printf "    CC         $(subst $(obj)/,,$(@))\n"
        $(CC) -MMD -DASSEMBLY -E -I$(src)/include -I$(src)/arch/x86/include -I$(obj) -I$(obj)/bootblock -include $(obj)/config.h -I. -I$(src) $< -o $@
 
-$(obj)/mainboard/$(MAINBOARDDIR)/bootblock.inc: $(src)/arch/x86/init/$(subst ",,$(CONFIG_BOOTBLOCK_SOURCE)) $(objutil)/romcc/romcc
+$(obj)/mainboard/$(MAINBOARDDIR)/bootblock.inc: $(src)/arch/x86/init/$(subst ",,$(CONFIG_BOOTBLOCK_SOURCE)) $(objutil)/romcc/romcc $(OPTION_TABLE_H)
        @printf "    ROMCC      $(subst $(obj)/,,$(@))\n"
        $(CC) -MM -MT$(obj)/mainboard/$(MAINBOARDDIR)/bootblock.inc \
                $< > $(obj)/mainboard/$(MAINBOARDDIR)/bootblock.inc.d
index 895a185c6f1ececcb58e63dab940342479a3ba17..a808cec7a5d2f4ba01884024c7c29d8ec1ede1ac 100644 (file)
@@ -17,17 +17,45 @@ static void bootblock_northbridge_init(void) { }
 static void bootblock_southbridge_init(void) { }
 #endif
 
-static unsigned long findstage(char* target)
+static void *walkcbfs(char *target)
 {
-       unsigned long entry;
+       void *entry;
        asm volatile (
                "mov $1f, %%esp\n\t"
-               "jmp walkcbfs\n\t"
+               "jmp walkcbfs_asm\n\t"
                "1:\n\t" : "=a" (entry) : "S" (target) : "ebx", "ecx", "edi", "esp");
        return entry;
 }
 
+/* just enough to support findstage. copied because the original version doesn't easily pass through romcc */
+struct cbfs_stage {
+       unsigned long compression;
+       unsigned long entry; // this is really 64bit, but properly endianized
+};
+
+static unsigned long findstage(char* target)
+{
+       return ((struct cbfs_stage*)walkcbfs(target))->entry;
+}
+
 static void call(unsigned long addr, unsigned long bist)
 {
        asm volatile ("jmp *%0\n\t" : : "r" (addr), "a" (bist));
 }
+
+#if CONFIG_USE_OPTION_TABLE
+#include <pc80/mc146818rtc.h>
+
+static void sanitize_cmos(void)
+{
+       if (cmos_error() || !cmos_chksum_valid()) {
+               unsigned char *cmos_default = (unsigned char*)walkcbfs("cmos.default");
+               if (cmos_default) {
+                       int i;
+                       for (i = 14; i < 128; i++) {
+                               cmos_write(cmos_default[i], i);
+                       }
+               }
+       }
+}
+#endif
index 08651c32bbfd86efbdeb91106b9b1c3d35144ba0..99551484aaadf0f2d680e349e29b718532313e8a 100644 (file)
@@ -8,6 +8,10 @@ static void main(unsigned long bist)
                bootblock_southbridge_init();
        }
 
+#if CONFIG_USE_OPTION_TABLE
+       sanitize_cmos();
+#endif
+
        unsigned long entry;
        if (do_normal_boot())
                entry = findstage("normal/romstage");
index e8994ee092463ee589d75b16061c19510d0aa229..3887b97c58a5cd48d771d890833a8b2179c64adb 100644 (file)
@@ -6,6 +6,11 @@ static void main(unsigned long bist)
                bootblock_northbridge_init();
                bootblock_southbridge_init();
        }
+
+#if CONFIG_USE_OPTION_TABLE
+       sanitize_cmos();
+#endif
+
        const char* target1 = "fallback/romstage";
        unsigned long entry;
        entry = findstage(target1);
index 395c46e20c3790686d8a9475330893aa45948288..27d82ae74221a532a501f6d292f40a2d2fb8f875 100644 (file)
 
 #define CBFS_FILE_STRUCTSIZE (CBFS_FILE_OFFSET + 4)
 
-#define CBFS_STAGE_COMPRESSION 0
-#define CBFS_STAGE_ENTRY (CBFS_STAGE_COMPRESSION + 4)
-#define CBFS_STAGE_LOAD (CBFS_STAGE_ENTRY + 8)
-#define CBFS_STAGE_LEN (CBFS_STAGE_LOAD + 8)
-#define CBFS_STAGE_MEMLEN (CBFS_STAGE_LEN + 4)
-
 /*
   input %esi: filename
   input %esp: return address (not pointer to return address!)
   output %eax: entry point
   clobbers %ebx, %ecx, %edi
 */
-walkcbfs:
+walkcbfs_asm:
        cld
 
        mov CBFS_HEADER_PTR, %eax
@@ -67,8 +61,6 @@ walker:
        mov CBFS_FILE_OFFSET(%ebx), %eax
        bswap %eax
        add %ebx, %eax
-       add $CBFS_STAGE_ENTRY, %eax /* eax = ((cbfs_stage* (cbfs_file* ebx)->offset)->entry) */
-       mov 0(%eax), %eax
        jmp *%esp
 
 tryharder:
index 920dedace218b0ae2fa8c621408a1bba0a6ec320..d09d6b9df0cae7d5824b7c26a0eee3d4c23302a3 100644 (file)
 #error "CONFIG_MAX_REBOOT_CNT too high"
 #endif
 
-#if CONFIG_USE_CMOS_RECOVERY
-#include <cbfs.h>
-#include <console/loglevel.h>
-
-int do_printk(int msg_level, const char *fmt, ...) __attribute__((format(printf, 2, 3)));
-#define printk_warning(fmt, arg...) do_printk(BIOS_WARNING ,fmt, ##arg)
-#define printk_debug(fmt, arg...) do_printk(BIOS_DEBUG ,fmt, ##arg)
-#endif
-
 static int cmos_error(void)
 {
        unsigned char reg_d;
@@ -63,25 +54,6 @@ static inline int do_normal_boot(void)
        unsigned char byte;
 
        if (cmos_error() || !cmos_chksum_valid()) {
-#if CONFIG_USE_CMOS_RECOVERY
-               char *cmos_default = cbfs_find_file("cmos.default", CBFS_COMPONENT_CMOS_DEFAULT);
-               if (cmos_default) {
-                       int i;
-                       printk_warning("WARNING - CMOS CORRUPTED. RESTORING DEFAULTS.\n");
-                       /* First 14 bytes are reserved for
-                          RTC and ignored by nvramtool, too.
-                          Only 128 bytes: 128+ requires cmos configuration and
-                          contains only suspend-to-ram data, which isn't part
-                          of the recovery procedure. */
-                       for (i = 14; i < 128; i++) {
-                               cmos_write(cmos_default[i], i);
-                       }
-                       /* Now reboot to run with default cmos. */
-                       outb(0x06, 0xcf9);
-                       for (;;) asm("hlt"); /* Wait for reset! */
-               }
-#endif
-
                /* There are no impossible values, no checksums so just
                 * trust whatever value we have in the the cmos,
                 * but clear the fallback bit.