blob: 9dc0423c31aa1f9bdc0bc2238db7374f20b1e692 [file] [log] [blame]
#include "types.h"
#include "bootparam.h"
#include "bootstub.h"
#include "spi-uart.h"
#include "sfi.h"
#define bs_printk(x) { if (!*(int *)SPI_UART_SUPPRESSION) bs_spi_printk(x); }
#define SFI_BASE_ADDR 0x000E0000
#define SFI_LENGTH 0x00020000
static unsigned long sfi_search_mmap(unsigned long start, int len)
{
unsigned long i = 0;
char *pchar = (char *)start;
for (i = 0; i < len; i += 4, pchar += 4) {
if (pchar[0] == 'M'
&& pchar[1] == 'M'
&& pchar[2] == 'A'
&& pchar[3] == 'P')
return start + i;
}
return 0;
}
void sfi_setup_e820(struct boot_params *bp)
{
struct sfi_table *sb;
struct sfi_mem_entry *mentry;
unsigned long long start, end, size;
int i, num, type, total;
bp->e820_entries = 0;
total = 0;
/* search for sfi mmap table */
sb = (struct sfi_table *)sfi_search_mmap(SFI_BASE_ADDR, SFI_LENGTH);
if (!sb)
return;
bs_printk("Bootstub: will use sfi mmap table for e820 table\n");
num = SFI_GET_ENTRY_NUM(sb, sfi_mem_entry);
mentry = (struct sfi_mem_entry *)sb->pentry;
for (i = 0; i < num; i++) {
start = mentry->phy_start;
size = mentry->pages << 12;
end = start + size;
if (start > end)
continue;
/* translate SFI mmap type to E820 map type */
switch (mentry->type) {
case SFI_MEM_CONV:
type = E820_RAM;
break;
case SFI_MEM_UNUSABLE:
case SFI_RUNTIME_SERVICE_DATA:
mentry++;
continue;
default:
type = E820_RESERVED;
}
if (total == E820MAX)
break;
bp->e820_map[total].addr = start;
bp->e820_map[total].size = size;
bp->e820_map[total++].type = type;
mentry++;
}
bp->e820_entries = total;
}