| /* |
| * sfi.c - driver for parsing sfi mmap table and build e820 table |
| * |
| * Copyright (c) 2009, Intel Corporation. |
| * |
| * This program is free software; you can redistribute it and/or modify it |
| * under the terms and conditions of the GNU General Public License, |
| * version 2, as published by the Free Software Foundation. |
| * |
| * This program is distributed in the hope it will be useful, but WITHOUT |
| * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or |
| * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for |
| * more details. |
| * |
| * You should have received a copy of the GNU General Public License along with |
| * this program; if not, write to the Free Software Foundation, Inc., |
| * 51 Franklin St - Fifth Floor, Boston, MA 02110-1301 USA. |
| */ |
| |
| #include "types.h" |
| #include "bootparam.h" |
| #include "bootstub.h" |
| #include "mb.h" |
| #include "sfi.h" |
| |
| #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++, pchar++) { |
| if (pchar[0] == 'M' |
| && pchar[1] == 'M' |
| && pchar[2] == 'A' |
| && pchar[3] == 'P') |
| return start + i; |
| } |
| return 0; |
| } |
| |
| int sfi_add_e820_entry(struct boot_params *bp, memory_map_t *mb_mmap, u64 start, u64 size, int type) |
| { |
| struct e820entry * e820_entry; |
| memory_map_t *mb_mmap_entry; |
| int i; |
| |
| if (!bp || !mb_mmap) { |
| bs_printk("Bootstub: sfi_add_e820_entry failed\n"); |
| return -1; |
| } |
| |
| for (i=0; i < bp->e820_entries; i++) { |
| e820_entry = &(bp->e820_map[i]); |
| mb_mmap_entry = &(mb_mmap[i]); |
| if (e820_entry->addr == start) { |
| /* Override size and type */ |
| e820_entry->size = size; |
| e820_entry->type = type; |
| mb_mmap_entry->length_low = size; |
| mb_mmap_entry->length_high = 0; |
| mb_mmap_entry->type = (type == E820_RAM)?1:0; |
| return 0; |
| } |
| } |
| |
| /* ASSERT: no duplicate start address found */ |
| if (bp->e820_entries == E820MAX) |
| return -1; |
| |
| e820_entry = &(bp->e820_map[bp->e820_entries]); |
| mb_mmap_entry = &(mb_mmap[bp->e820_entries]); |
| |
| e820_entry->addr = start; |
| e820_entry->size = size; |
| e820_entry->type = type; |
| |
| mb_mmap_entry->size = 20; |
| mb_mmap_entry->base_addr_low = start; |
| mb_mmap_entry->base_addr_high = 0; |
| mb_mmap_entry->length_low = size; |
| mb_mmap_entry->length_high = 0; |
| mb_mmap_entry->type = (type == E820_RAM)?1:0; |
| |
| bp->e820_entries++; |
| |
| return 0; |
| } |
| |
| void sfi_setup_mmap(struct boot_params *bp, memory_map_t *mb_mmap) |
| { |
| struct sfi_table *sb; |
| struct sfi_mem_entry *mentry; |
| unsigned long long start, end, size; |
| int i, num, type; |
| |
| if (!bp || !mb_mmap) { |
| bs_printk("Bootstub: sfi_setup_mmap failed\n"); |
| return; |
| } |
| |
| bp->e820_entries = 0; |
| |
| /* search for sfi mmap table */ |
| sb = (struct sfi_table *)sfi_search_mmap(SFI_BASE_ADDR, SFI_LENGTH); |
| if (!sb) { |
| bs_printk("Bootstub: SFI MMAP table not found\n"); |
| return; |
| } |
| bs_printk("Bootstub: map SFI MMAP to 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 (sfi_add_e820_entry(bp, mb_mmap, start, size, type) != 0) |
| break; |
| |
| mentry++; |
| } |
| |
| } |