| #include <stdint.h> |
| #include <lib.h> /* Prototypes */ |
| #include <console/console.h> |
| |
| static void write_phys(unsigned long addr, u32 value) |
| { |
| // Assembler in lib/ is very ugly. But we properly guarded |
| // it so let's obey this one for now |
| #if CONFIG_SSE2 |
| asm volatile( |
| "movnti %1, (%0)" |
| : /* outputs */ |
| : "r" (addr), "r" (value) /* inputs */ |
| #ifndef __GNUC__ /* GCC does not like empty clobbers? */ |
| : /* clobbers */ |
| #endif |
| ); |
| #else |
| volatile unsigned long *ptr; |
| ptr = (void *)addr; |
| *ptr = value; |
| #endif |
| } |
| |
| static u32 read_phys(unsigned long addr) |
| { |
| volatile unsigned long *ptr; |
| ptr = (void *)addr; |
| return *ptr; |
| } |
| |
| static void phys_memory_barrier(void) |
| { |
| #if CONFIG_SSE2 |
| // Needed for movnti |
| asm volatile ( |
| "sfence" |
| :: |
| #ifdef __GNUC__ /* ROMCC does not like memory clobbers */ |
| : "memory" |
| #endif |
| ); |
| #else |
| #ifdef __GNUC__ /* ROMCC does not like empty asm statements */ |
| asm volatile ("" ::: "memory"); |
| #endif |
| #endif |
| } |
| |
| /** |
| * Rotate ones test pattern that access every bit on a 128bit wide |
| * memory bus. To test most address lines, addresses are scattered |
| * using 256B, 4kB and 64kB increments. |
| * |
| * @idx Index to test pattern (0=<idx<0x400) |
| * @addr Memory to access on @idx |
| * @value Value to write or read at @addr |
| */ |
| static inline void test_pattern(unsigned short int idx, |
| unsigned long *addr, unsigned long *value) |
| { |
| uint8_t j, k; |
| |
| k = (idx >> 8) + 1; |
| j = (idx >> 4) & 0x0f; |
| *addr = idx & 0x0f; |
| *addr |= j << (4*k); |
| *value = 0x01010101 << (j & 7); |
| if (j & 8) |
| *value = ~(*value); |
| } |
| |
| /** |
| * Simple write-read-verify memory test. See console debug output for |
| * any dislocated bytes. |
| * |
| * @start System memory offset, aligned to 128bytes |
| */ |
| static int ram_bitset_nodie(unsigned long start) |
| { |
| unsigned long addr, value, value2; |
| unsigned short int idx; |
| unsigned char failed, failures; |
| uint8_t verbose = 0; |
| |
| #if !defined(__ROMCC__) |
| printk(BIOS_DEBUG, "DRAM bitset write: 0x%08lx\n", start); |
| #else |
| print_debug("DRAM bitset write: 0x"); |
| print_debug_hex32(start); |
| print_debug("\n"); |
| #endif |
| for (idx=0; idx<0x400; idx+=4) { |
| test_pattern(idx, &addr, &value); |
| write_phys(start + addr, value); |
| } |
| |
| /* Make sure we don't read before we wrote */ |
| phys_memory_barrier(); |
| |
| #if !defined(__ROMCC__) |
| printk(BIOS_DEBUG, "DRAM bitset verify: 0x%08lx\n", start); |
| #else |
| print_debug("DRAM bitset verify: 0x"); |
| print_debug_hex32(start); |
| print_debug("\n"); |
| #endif |
| failures = 0; |
| for (idx=0; idx<0x400; idx+=4) { |
| test_pattern(idx, &addr, &value); |
| value2 = read_phys(start + addr); |
| |
| failed = (value2 != value); |
| failures |= failed; |
| if (failed && !verbose) { |
| #if !defined(__ROMCC__) |
| printk(BIOS_ERR, "0x%08lx wr: 0x%08lx rd: 0x%08lx FAIL\n", |
| start + addr, value, value2); |
| #else |
| print_err_hex32(start + addr); |
| print_err(" wr: 0x"); |
| print_err_hex32(value); |
| print_err(" rd: 0x"); |
| print_err_hex32(value2); |
| print_err(" FAIL\n"); |
| #endif |
| } |
| if (verbose) { |
| #if !defined(__ROMCC__) |
| if ((addr & 0x0f) == 0) |
| printk(BIOS_DEBUG, "%08lx wr: %08lx rd:", |
| start + addr, value); |
| if (failed) |
| printk(BIOS_DEBUG, " %08lx!", value2); |
| else |
| printk(BIOS_DEBUG, " %08lx ", value2); |
| if ((addr & 0x0f) == 0xc) |
| printk(BIOS_DEBUG, "\n"); |
| #else |
| if ((addr & 0x0f) == 0) { |
| print_dbg_hex32(start + addr); |
| print_dbg(" wr: "); |
| print_dbg_hex32(value); |
| print_dbg(" rd: "); |
| } |
| print_dbg_hex32(value2); |
| if (failed) |
| print_dbg("! "); |
| else |
| print_dbg(" "); |
| if ((addr & 0x0f) == 0xc) |
| print_dbg("\n"); |
| #endif |
| } |
| } |
| if (failures) { |
| post_code(0xea); |
| #if !defined(__ROMCC__) |
| printk(BIOS_DEBUG, "\nDRAM did _NOT_ verify!\n"); |
| #else |
| print_debug("\nDRAM did _NOT_ verify!\n"); |
| #endif |
| return 1; |
| } |
| else { |
| #if !defined(__ROMCC__) |
| printk(BIOS_DEBUG, "\nDRAM range verified.\n"); |
| #else |
| print_debug("\nDRAM range verified.\n"); |
| return 0; |
| #endif |
| } |
| return 0; |
| } |
| |
| |
| void ram_check(unsigned long start, unsigned long stop) |
| { |
| /* |
| * This is much more of a "Is my DRAM properly configured?" |
| * test than a "Is my DRAM faulty?" test. Not all bits |
| * are tested. -Tyson |
| */ |
| #if !defined(__ROMCC__) |
| printk(BIOS_DEBUG, "Testing DRAM at: %08lx\n", start); |
| #else |
| print_debug("Testing DRAM at: "); |
| print_debug_hex32(start); |
| print_debug("\n"); |
| #endif |
| if (ram_bitset_nodie(start)) |
| die("DRAM ERROR"); |
| #if !defined(__ROMCC__) |
| printk(BIOS_DEBUG, "Done.\n"); |
| #else |
| print_debug("Done.\n"); |
| #endif |
| } |
| |
| |
| int ram_check_nodie(unsigned long start, unsigned long stop) |
| { |
| int ret; |
| /* |
| * This is much more of a "Is my DRAM properly configured?" |
| * test than a "Is my DRAM faulty?" test. Not all bits |
| * are tested. -Tyson |
| */ |
| #if !defined(__ROMCC__) |
| printk(BIOS_DEBUG, "Testing DRAM at : %08lx\n", start); |
| #else |
| print_debug("Testing DRAM at : "); |
| print_debug_hex32(start); |
| print_debug("\n"); |
| #endif |
| |
| ret = ram_bitset_nodie(start); |
| #if !defined(__ROMCC__) |
| printk(BIOS_DEBUG, "Done.\n"); |
| #else |
| print_debug("Done.\n"); |
| #endif |
| return ret; |
| } |
| |
| void quick_ram_check(void) |
| { |
| int fail = 0; |
| u32 backup; |
| backup = read_phys(CONFIG_RAMBASE); |
| write_phys(CONFIG_RAMBASE, 0x55555555); |
| phys_memory_barrier(); |
| if (read_phys(CONFIG_RAMBASE) != 0x55555555) |
| fail=1; |
| write_phys(CONFIG_RAMBASE, 0xaaaaaaaa); |
| phys_memory_barrier(); |
| if (read_phys(CONFIG_RAMBASE) != 0xaaaaaaaa) |
| fail=1; |
| write_phys(CONFIG_RAMBASE, 0x00000000); |
| phys_memory_barrier(); |
| if (read_phys(CONFIG_RAMBASE) != 0x00000000) |
| fail=1; |
| write_phys(CONFIG_RAMBASE, 0xffffffff); |
| phys_memory_barrier(); |
| if (read_phys(CONFIG_RAMBASE) != 0xffffffff) |
| fail=1; |
| |
| write_phys(CONFIG_RAMBASE, backup); |
| if (fail) { |
| post_code(0xea); |
| die("RAM INIT FAILURE!\n"); |
| } |
| phys_memory_barrier(); |
| } |