From 02e504cdb1607beec1ecb66b79afc26149e8d60b Mon Sep 17 00:00:00 2001 From: Aaron Durbin Date: Mon, 18 Jul 2016 11:53:10 -0500 Subject: drivers/intel/fsp2_0: range check stack provided to FSPM Ensure that the stack provided to FSPM doesn't overlap the current program which is loading the FSPM component. If there is a conflict that's an error since it could cause the current program to crash. BUG=chrome-os-partner:52679 Change-Id: Ifff465266e5bb3cb3cf9b616d322a46199f802c7 Signed-off-by: Aaron Durbin Reviewed-on: https://review.coreboot.org/15746 Tested-by: build bot (Jenkins) Reviewed-by: Furquan Shaikh Reviewed-by: Andrey Petrov --- src/drivers/intel/fsp2_0/memory_init.c | 84 +++++++++++++++++++++++----------- 1 file changed, 58 insertions(+), 26 deletions(-) (limited to 'src/drivers') diff --git a/src/drivers/intel/fsp2_0/memory_init.c b/src/drivers/intel/fsp2_0/memory_init.c index 70c920e321..642df460c0 100644 --- a/src/drivers/intel/fsp2_0/memory_init.c +++ b/src/drivers/intel/fsp2_0/memory_init.c @@ -128,28 +128,61 @@ static void fsp_fill_mrc_cache(struct FSPM_ARCH_UPD *arch_upd, bool s3wake, mrc_cache->size, arch_upd->BootMode); } -static void fsp_fill_common_arch_params(struct FSPM_ARCH_UPD *arch_upd, - bool s3wake, uint32_t fsp_version) +static enum cb_err check_region_overlap(const struct memranges *ranges, + const char *description, + uintptr_t begin, uintptr_t end) { + const struct range_entry *r; + + memranges_each_entry(r, ranges) { + if (end <= range_entry_base(r)) + continue; + if (begin >= range_entry_end(r)) + continue; + printk(BIOS_ERR, "'%s' overlaps currently running program: " + "[%p, %p)\n", description, (void *)begin, (void *)end); + return CB_ERR; + } + + return CB_SUCCESS; +} + +static enum cb_err fsp_fill_common_arch_params(struct FSPM_ARCH_UPD *arch_upd, + bool s3wake, uint32_t fsp_version, + const struct memranges *memmap) +{ + uintptr_t stack_begin; + uintptr_t stack_end; + /* * FSPM_UPD passed here is populated with default values provided by * the blob itself. We let FSPM use top of CAR region of the size it * requests. - * TODO: add checks to avoid overlap/conflict of CAR usage. */ - arch_upd->StackBase = _car_region_end - arch_upd->StackSize; + stack_end = (uintptr_t)_car_region_end; + stack_begin = stack_end - arch_upd->StackSize; + + if (check_region_overlap(memmap, "FSPM stack", stack_begin, + stack_end) != CB_SUCCESS) + return CB_ERR; + + arch_upd->StackBase = (void *)stack_begin; arch_upd->BootMode = FSP_BOOT_WITH_FULL_CONFIGURATION; fsp_fill_mrc_cache(arch_upd, s3wake, fsp_version); + + return CB_SUCCESS; } -static enum fsp_status do_fsp_memory_init(struct fsp_header *hdr, bool s3wake) +static enum fsp_status do_fsp_memory_init(struct fsp_header *hdr, bool s3wake, + const struct memranges *memmap) { enum fsp_status status; fsp_memory_init_fn fsp_raminit; struct FSPM_UPD fspm_upd, *upd; void *hob_list_ptr; + struct FSPM_ARCH_UPD *arch_upd; post_code(0x34); @@ -163,12 +196,15 @@ static enum fsp_status do_fsp_memory_init(struct fsp_header *hdr, bool s3wake) /* Copy the default values from the UPD area */ memcpy(&fspm_upd, upd, sizeof(fspm_upd)); + arch_upd = &fspm_upd.FspmArchUpd; + /* Reserve enough memory under TOLUD to save CBMEM header */ - fspm_upd.FspmArchUpd.BootLoaderTolumSize = cbmem_overhead_size(); + arch_upd->BootLoaderTolumSize = cbmem_overhead_size(); /* Fill common settings on behalf of chipset. */ - fsp_fill_common_arch_params(&fspm_upd.FspmArchUpd, s3wake, - hdr->fsp_revision); + if (fsp_fill_common_arch_params(arch_upd, s3wake, hdr->fsp_revision, + memmap) != CB_SUCCESS) + return FSP_NOT_FOUND; /* Give SoC and mainboard a chance to update the UPD */ platform_fsp_memory_init_params_cb(&fspm_upd); @@ -196,11 +232,9 @@ static enum fsp_status do_fsp_memory_init(struct fsp_header *hdr, bool s3wake) /* Load the binary into the memory specified by the info header. */ static enum cb_err load_fspm_mem(struct fsp_header *hdr, - const struct region_device *rdev) + const struct region_device *rdev, + const struct memranges *memmap) { - struct memranges ranges; - struct range_entry freeranges[2]; - const struct range_entry *r; uintptr_t fspm_begin; uintptr_t fspm_end; @@ -210,19 +244,9 @@ static enum cb_err load_fspm_mem(struct fsp_header *hdr, fspm_begin = hdr->image_base; fspm_end = fspm_begin + hdr->image_size; - /* Build up memory map of romstage address space including CAR. */ - memranges_init_empty(&ranges, &freeranges[0], ARRAY_SIZE(freeranges)); - memranges_insert(&ranges, (uintptr_t)_car_region_start, - _car_relocatable_data_end - _car_region_start, 0); - memranges_insert(&ranges, (uintptr_t)_program, _program_size, 0); - memranges_each_entry(r, &ranges) { - if (fspm_end <= range_entry_base(r)) - continue; - if (fspm_begin >= range_entry_end(r)) - continue; - printk(BIOS_ERR, "FSPM overlaps currently running program.\n"); + if (check_region_overlap(memmap, "FSPM", fspm_begin, fspm_end) != + CB_SUCCESS) return CB_ERR; - } /* Load binary into memory at provided address. */ if (rdev_readat(rdev, (void *)fspm_begin, 0, fspm_end - fspm_begin) < 0) @@ -261,6 +285,8 @@ enum fsp_status fsp_memory_init(bool s3wake) struct cbfsf file_desc; struct region_device file_data; const char *name = CONFIG_FSP_M_CBFS; + struct memranges memmap; + struct range_entry freeranges[2]; if (cbfs_boot_locate(&file_desc, name, NULL)) { printk(BIOS_ERR, "Could not locate %s in CBFS\n", name); @@ -269,8 +295,14 @@ enum fsp_status fsp_memory_init(bool s3wake) cbfs_file_data(&file_data, &file_desc); + /* Build up memory map of romstage address space including CAR. */ + memranges_init_empty(&memmap, &freeranges[0], ARRAY_SIZE(freeranges)); + memranges_insert(&memmap, (uintptr_t)_car_region_start, + _car_relocatable_data_end - _car_region_start, 0); + memranges_insert(&memmap, (uintptr_t)_program, _program_size, 0); + if (IS_ENABLED(CONFIG_NO_XIP_EARLY_STAGES)) - status = load_fspm_mem(&hdr, &file_data); + status = load_fspm_mem(&hdr, &file_data, &memmap); else status = load_fspm_xip(&hdr, &file_data); @@ -282,5 +314,5 @@ enum fsp_status fsp_memory_init(bool s3wake) /* Signal that FSP component has been loaded. */ prog_segment_loaded(hdr.image_base, hdr.image_size, SEG_FINAL); - return do_fsp_memory_init(&hdr, s3wake); + return do_fsp_memory_init(&hdr, s3wake, &memmap); } -- cgit v1.2.3