Home | History | Annotate | Download | only in bootstub
      1 /*
      2  * sfi.c - driver for parsing sfi mmap table and build e820 table
      3  *
      4  * Copyright (c) 2009, Intel Corporation.
      5  *
      6  * This program is free software; you can redistribute it and/or modify it
      7  * under the terms and conditions of the GNU General Public License,
      8  * version 2, as published by the Free Software Foundation.
      9  *
     10  * This program is distributed in the hope it will be useful, but WITHOUT
     11  * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
     12  * FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License for
     13  * more details.
     14  *
     15  * You should have received a copy of the GNU General Public License along with
     16  * this program; if not, write to the Free Software Foundation, Inc.,
     17  * 51 Franklin St - Fifth Floor, Boston, MA 02110-1301 USA.
     18  */
     19 
     20 #include "types.h"
     21 #include "bootparam.h"
     22 #include "bootstub.h"
     23 #include "mb.h"
     24 #include "sfi.h"
     25 
     26 #define SFI_BASE_ADDR		0x000E0000
     27 #define SFI_LENGTH		0x00020000
     28 
     29 static unsigned long sfi_search_mmap(unsigned long start, int len)
     30 {
     31 	unsigned long i = 0;
     32 	char *pchar = (char *)start;
     33 
     34 	for (i = 0; i < len; i++, pchar++) {
     35 		if (pchar[0] == 'M'
     36 			&& pchar[1] == 'M'
     37 			&& pchar[2] == 'A'
     38 			&& pchar[3] == 'P')
     39 			return start + i;
     40 	}
     41 	return 0;
     42 }
     43 
     44 int sfi_add_e820_entry(struct boot_params *bp, memory_map_t *mb_mmap, u64 start, u64 size, int type)
     45 {
     46         struct e820entry * e820_entry;
     47 	memory_map_t	*mb_mmap_entry;
     48 	int	i;
     49 
     50 	if (!bp || !mb_mmap) {
     51 		bs_printk("Bootstub: sfi_add_e820_entry failed\n");
     52 		return -1;
     53 	}
     54 
     55 	for (i=0; i < bp->e820_entries; i++) {
     56 		e820_entry = &(bp->e820_map[i]);
     57 		mb_mmap_entry = &(mb_mmap[i]);
     58 		if (e820_entry->addr == start) {
     59 			/* Override size and type */
     60 			e820_entry->size = size;
     61 			e820_entry->type = type;
     62 			mb_mmap_entry->length_low = size;
     63 			mb_mmap_entry->length_high = 0;
     64 			mb_mmap_entry->type = (type == E820_RAM)?1:0;
     65 			return 0;
     66 		}
     67 	}
     68 
     69 	/* ASSERT: no duplicate start address found */
     70 	if (bp->e820_entries == E820MAX)
     71 		return -1;
     72 
     73 	e820_entry = &(bp->e820_map[bp->e820_entries]);
     74 	mb_mmap_entry = &(mb_mmap[bp->e820_entries]);
     75 
     76 	e820_entry->addr = start;
     77 	e820_entry->size = size;
     78 	e820_entry->type = type;
     79 
     80 	mb_mmap_entry->size = 20;
     81 	mb_mmap_entry->base_addr_low = start;
     82 	mb_mmap_entry->base_addr_high = 0;
     83 	mb_mmap_entry->length_low = size;
     84 	mb_mmap_entry->length_high = 0;
     85 	mb_mmap_entry->type = (type == E820_RAM)?1:0;
     86 
     87 	bp->e820_entries++;
     88 
     89 	return 0;
     90 }
     91 
     92 void sfi_setup_mmap(struct boot_params *bp, memory_map_t *mb_mmap)
     93 {
     94 	struct sfi_table *sb;
     95 	struct sfi_mem_entry *mentry;
     96 	unsigned long long start, end, size;
     97 	int i, num, type;
     98 
     99 	if (!bp || !mb_mmap) {
    100 		bs_printk("Bootstub: sfi_setup_mmap failed\n");
    101 		return;
    102 	}
    103 
    104 	bp->e820_entries = 0;
    105 
    106 	/* search for sfi mmap table */
    107 	sb = (struct sfi_table *)sfi_search_mmap(SFI_BASE_ADDR, SFI_LENGTH);
    108 	if (!sb) {
    109 		bs_printk("Bootstub: SFI MMAP table not found\n");
    110 		return;
    111 	}
    112 	bs_printk("Bootstub: map SFI MMAP to e820 table\n");
    113 	num = SFI_GET_ENTRY_NUM(sb, sfi_mem_entry);
    114 	mentry = (struct sfi_mem_entry *)sb->pentry;
    115 
    116 	for (i = 0; i < num; i++) {
    117 		start = mentry->phy_start;
    118 		size = mentry->pages << 12;
    119 		end = start + size;
    120 
    121 		if (start > end)
    122 			continue;
    123 
    124 		/* translate SFI mmap type to E820 map type */
    125 		switch (mentry->type) {
    126 		case SFI_MEM_CONV:
    127 			type = E820_RAM;
    128 			break;
    129 		case SFI_MEM_UNUSABLE:
    130 		case SFI_RUNTIME_SERVICE_DATA:
    131 			mentry++;
    132 			continue;
    133 		default:
    134 			type = E820_RESERVED;
    135 		}
    136 
    137 		if (sfi_add_e820_entry(bp, mb_mmap, start, size, type) != 0)
    138 			break;
    139 
    140 		mentry++;
    141 	}
    142 
    143 }
    144