2010-10-05 11:07:10 +02:00
|
|
|
#include <stdint.h>
|
2010-03-19 03:33:40 +01:00
|
|
|
#include <lib.h> /* Prototypes */
|
2010-10-05 11:07:10 +02:00
|
|
|
#include <console/console.h>
|
2010-03-19 03:33:40 +01:00
|
|
|
|
2010-03-28 23:31:30 +02:00
|
|
|
static void write_phys(unsigned long addr, u32 value)
|
2003-04-22 21:02:15 +02:00
|
|
|
{
|
2010-02-25 14:40:49 +01:00
|
|
|
// Assembler in lib/ is very ugly. But we properly guarded
|
|
|
|
// it so let's obey this one for now
|
2019-03-06 01:53:33 +01:00
|
|
|
#if CONFIG(SSE2)
|
2003-06-17 10:42:17 +02:00
|
|
|
asm volatile(
|
|
|
|
"movnti %1, (%0)"
|
|
|
|
: /* outputs */
|
|
|
|
: "r" (addr), "r" (value) /* inputs */
|
2010-03-28 23:31:30 +02:00
|
|
|
#ifndef __GNUC__ /* GCC does not like empty clobbers? */
|
2003-06-17 10:42:17 +02:00
|
|
|
: /* clobbers */
|
2004-05-27 13:13:24 +02:00
|
|
|
#endif
|
2003-06-17 10:42:17 +02:00
|
|
|
);
|
|
|
|
#else
|
2003-05-19 21:16:21 +02:00
|
|
|
volatile unsigned long *ptr;
|
2003-04-22 21:02:15 +02:00
|
|
|
ptr = (void *)addr;
|
|
|
|
*ptr = value;
|
2003-06-17 10:42:17 +02:00
|
|
|
#endif
|
2003-04-22 21:02:15 +02:00
|
|
|
}
|
|
|
|
|
2010-03-28 23:31:30 +02:00
|
|
|
static u32 read_phys(unsigned long addr)
|
2003-04-22 21:02:15 +02:00
|
|
|
{
|
2003-05-19 21:16:21 +02:00
|
|
|
volatile unsigned long *ptr;
|
2003-04-22 21:02:15 +02:00
|
|
|
ptr = (void *)addr;
|
|
|
|
return *ptr;
|
|
|
|
}
|
|
|
|
|
2010-03-28 23:31:30 +02:00
|
|
|
static void phys_memory_barrier(void)
|
|
|
|
{
|
2019-03-06 01:53:33 +01:00
|
|
|
#if CONFIG(SSE2)
|
2010-03-28 23:31:30 +02:00
|
|
|
// 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
|
|
|
|
}
|
|
|
|
|
2012-03-17 08:09:14 +01:00
|
|
|
/**
|
|
|
|
* 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.
|
|
|
|
*
|
2015-01-05 00:47:39 +01:00
|
|
|
* @param idx Index to test pattern (0=<idx<0x400)
|
|
|
|
* @param addr Memory to access on idx
|
|
|
|
* @param value Value to write or read at addr
|
2012-03-17 08:09:14 +01:00
|
|
|
*/
|
|
|
|
static inline void test_pattern(unsigned short int idx,
|
|
|
|
unsigned long *addr, unsigned long *value)
|
2003-04-22 21:02:15 +02:00
|
|
|
{
|
2012-03-17 08:09:14 +01:00
|
|
|
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);
|
2003-04-22 21:02:15 +02:00
|
|
|
}
|
|
|
|
|
2012-03-17 08:09:14 +01:00
|
|
|
/**
|
|
|
|
* Simple write-read-verify memory test. See console debug output for
|
|
|
|
* any dislocated bytes.
|
|
|
|
*
|
2017-10-27 11:49:29 +02:00
|
|
|
* Tests 1MiB of memory starting from start.
|
|
|
|
*
|
2015-01-05 00:47:39 +01:00
|
|
|
* @param start System memory offset, aligned to 128bytes
|
2012-03-17 08:09:14 +01:00
|
|
|
*/
|
|
|
|
static int ram_bitset_nodie(unsigned long start)
|
2003-04-22 21:02:15 +02:00
|
|
|
{
|
2012-03-17 08:09:14 +01:00
|
|
|
unsigned long addr, value, value2;
|
|
|
|
unsigned short int idx;
|
|
|
|
unsigned char failed, failures;
|
|
|
|
uint8_t verbose = 0;
|
|
|
|
|
|
|
|
printk(BIOS_DEBUG, "DRAM bitset write: 0x%08lx\n", start);
|
2017-03-10 02:35:28 +01:00
|
|
|
for (idx = 0; idx < 0x400; idx += 4) {
|
2012-03-17 08:09:14 +01:00
|
|
|
test_pattern(idx, &addr, &value);
|
|
|
|
write_phys(start + addr, value);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Make sure we don't read before we wrote */
|
|
|
|
phys_memory_barrier();
|
|
|
|
|
|
|
|
printk(BIOS_DEBUG, "DRAM bitset verify: 0x%08lx\n", start);
|
|
|
|
failures = 0;
|
2017-03-10 02:35:28 +01:00
|
|
|
for (idx = 0; idx < 0x400; idx += 4) {
|
2012-03-17 08:09:14 +01:00
|
|
|
test_pattern(idx, &addr, &value);
|
|
|
|
value2 = read_phys(start + addr);
|
|
|
|
|
|
|
|
failed = (value2 != value);
|
|
|
|
failures |= failed;
|
2018-05-04 16:30:39 +02:00
|
|
|
if (failed && !verbose) {
|
2012-03-17 08:09:14 +01:00
|
|
|
printk(BIOS_ERR, "0x%08lx wr: 0x%08lx rd: 0x%08lx FAIL\n",
|
|
|
|
start + addr, value, value2);
|
|
|
|
}
|
|
|
|
if (verbose) {
|
|
|
|
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");
|
2003-04-22 21:02:15 +02:00
|
|
|
}
|
|
|
|
}
|
2012-03-17 08:09:14 +01:00
|
|
|
if (failures) {
|
|
|
|
post_code(0xea);
|
2010-03-31 16:47:43 +02:00
|
|
|
printk(BIOS_DEBUG, "\nDRAM did _NOT_ verify!\n");
|
2011-12-02 16:23:06 +01:00
|
|
|
return 1;
|
2006-04-01 06:10:44 +02:00
|
|
|
}
|
2017-03-11 02:48:31 +01:00
|
|
|
printk(BIOS_DEBUG, "\nDRAM range verified.\n");
|
2011-12-02 16:23:06 +01:00
|
|
|
return 0;
|
2003-04-22 21:02:15 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2003-06-17 10:42:17 +02:00
|
|
|
void ram_check(unsigned long start, unsigned long stop)
|
2003-04-22 21:02:15 +02:00
|
|
|
{
|
|
|
|
/*
|
|
|
|
* 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
|
|
|
|
*/
|
2012-03-17 08:09:14 +01:00
|
|
|
printk(BIOS_DEBUG, "Testing DRAM at: %08lx\n", start);
|
|
|
|
if (ram_bitset_nodie(start))
|
2011-12-02 16:23:06 +01:00
|
|
|
die("DRAM ERROR");
|
|
|
|
printk(BIOS_DEBUG, "Done.\n");
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
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
|
|
|
|
*/
|
2012-03-17 08:09:14 +01:00
|
|
|
printk(BIOS_DEBUG, "Testing DRAM at : %08lx\n", start);
|
2011-12-02 16:23:06 +01:00
|
|
|
|
2012-03-17 08:09:14 +01:00
|
|
|
ret = ram_bitset_nodie(start);
|
2010-03-31 16:47:43 +02:00
|
|
|
printk(BIOS_DEBUG, "Done.\n");
|
2011-12-02 16:23:06 +01:00
|
|
|
return ret;
|
2003-04-22 21:02:15 +02:00
|
|
|
}
|
|
|
|
|
2013-06-08 18:32:36 +02:00
|
|
|
int ram_check_noprint_nodie(unsigned long start, unsigned long stop)
|
|
|
|
{
|
|
|
|
unsigned long addr, value, value2;
|
|
|
|
unsigned short int idx;
|
|
|
|
unsigned char failed, failures;
|
|
|
|
|
2017-03-10 02:35:28 +01:00
|
|
|
for (idx = 0; idx < 0x400; idx += 4) {
|
2013-06-08 18:32:36 +02:00
|
|
|
test_pattern(idx, &addr, &value);
|
|
|
|
write_phys(start + addr, value);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Make sure we don't read before we wrote */
|
|
|
|
phys_memory_barrier();
|
|
|
|
|
|
|
|
failures = 0;
|
2017-03-10 02:35:28 +01:00
|
|
|
for (idx = 0; idx < 0x400; idx += 4) {
|
2013-06-08 18:32:36 +02:00
|
|
|
test_pattern(idx, &addr, &value);
|
|
|
|
value2 = read_phys(start + addr);
|
|
|
|
|
|
|
|
failed = (value2 != value);
|
|
|
|
failures |= failed;
|
|
|
|
}
|
|
|
|
return failures;
|
|
|
|
}
|
|
|
|
|
2016-06-17 22:31:42 +02:00
|
|
|
static void __quick_ram_check(uintptr_t dst)
|
2010-03-28 23:31:30 +02:00
|
|
|
{
|
|
|
|
int fail = 0;
|
|
|
|
u32 backup;
|
2016-06-17 22:31:42 +02:00
|
|
|
backup = read_phys(dst);
|
|
|
|
write_phys(dst, 0x55555555);
|
2010-03-28 23:31:30 +02:00
|
|
|
phys_memory_barrier();
|
2016-06-17 22:31:42 +02:00
|
|
|
if (read_phys(dst) != 0x55555555)
|
2017-03-10 02:35:28 +01:00
|
|
|
fail = 1;
|
2016-06-17 22:31:42 +02:00
|
|
|
write_phys(dst, 0xaaaaaaaa);
|
2010-03-28 23:31:30 +02:00
|
|
|
phys_memory_barrier();
|
2016-06-17 22:31:42 +02:00
|
|
|
if (read_phys(dst) != 0xaaaaaaaa)
|
2017-03-10 02:35:28 +01:00
|
|
|
fail = 1;
|
2016-06-17 22:31:42 +02:00
|
|
|
write_phys(dst, 0x00000000);
|
2010-03-28 23:31:30 +02:00
|
|
|
phys_memory_barrier();
|
2016-06-17 22:31:42 +02:00
|
|
|
if (read_phys(dst) != 0x00000000)
|
2017-03-10 02:35:28 +01:00
|
|
|
fail = 1;
|
2016-06-17 22:31:42 +02:00
|
|
|
write_phys(dst, 0xffffffff);
|
2010-03-28 23:31:30 +02:00
|
|
|
phys_memory_barrier();
|
2016-06-17 22:31:42 +02:00
|
|
|
if (read_phys(dst) != 0xffffffff)
|
2017-03-10 02:35:28 +01:00
|
|
|
fail = 1;
|
2010-03-28 23:31:30 +02:00
|
|
|
|
2016-06-17 22:31:42 +02:00
|
|
|
write_phys(dst, backup);
|
2010-03-28 23:31:30 +02:00
|
|
|
if (fail) {
|
|
|
|
post_code(0xea);
|
|
|
|
die("RAM INIT FAILURE!\n");
|
|
|
|
}
|
|
|
|
phys_memory_barrier();
|
|
|
|
}
|
2016-06-17 22:31:42 +02:00
|
|
|
|
|
|
|
void quick_ram_check(void)
|
|
|
|
{
|
|
|
|
__quick_ram_check(0x100000);
|
|
|
|
}
|