| /* |
| * |
| * Copyright (C) 2015 Google Inc. |
| * |
| * Redistribution and use in source and binary forms, with or without |
| * modification, are permitted provided that the following conditions |
| * are met: |
| * 1. Redistributions of source code must retain the above copyright |
| * notice, this list of conditions and the following disclaimer. |
| * 2. Redistributions in binary form must reproduce the above copyright |
| * notice, this list of conditions and the following disclaimer in the |
| * documentation and/or other materials provided with the distribution. |
| * 3. The name of the author may not be used to endorse or promote products |
| * derived from this software without specific prior written permission. |
| * |
| * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND |
| * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE |
| * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE |
| * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE |
| * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL |
| * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS |
| * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) |
| * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT |
| * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY |
| * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF |
| * SUCH DAMAGE. |
| */ |
| |
| #include <libpayload-config.h> |
| #include <libpayload.h> |
| #include <commonlib/bsd/fmap_serialized.h> |
| #include <coreboot_tables.h> |
| #include <cbfs.h> |
| #include <boot_device.h> |
| #include <stdint.h> |
| #include <arch/virtual.h> |
| |
| /* Private fmap cache. */ |
| static struct fmap *_fmap_cache; |
| |
| static enum cb_err fmap_find_area(struct fmap *fmap, const char *name, size_t *offset, |
| size_t *size) |
| { |
| for (size_t i = 0; i < le32toh(fmap->nareas); ++i) { |
| if (strncmp((const char *)fmap->areas[i].name, name, FMAP_STRLEN) != 0) |
| continue; |
| if (offset) |
| *offset = le32toh(fmap->areas[i].offset); |
| if (size) |
| *size = le32toh(fmap->areas[i].size); |
| return CB_SUCCESS; |
| } |
| |
| return CB_ERR; |
| } |
| |
| static bool fmap_is_signature_valid(struct fmap *fmap) |
| { |
| return memcmp(fmap->signature, FMAP_SIGNATURE, sizeof(fmap->signature)) == 0; |
| } |
| |
| static bool fmap_setup_cache(void) |
| { |
| /* Use FMAP cache if available */ |
| if (lib_sysinfo.fmap_cache |
| && fmap_is_signature_valid((struct fmap *)phys_to_virt(lib_sysinfo.fmap_cache))) { |
| _fmap_cache = (struct fmap *)phys_to_virt(lib_sysinfo.fmap_cache); |
| return true; |
| } |
| |
| return false; |
| } |
| |
| enum cb_err fmap_locate_area(const char *name, size_t *offset, size_t *size) |
| { |
| if (!_fmap_cache && !fmap_setup_cache()) |
| return CB_ERR; |
| |
| return fmap_find_area(_fmap_cache, name, offset, size); |
| } |
| |
| /*********************************************************************************************** |
| * LEGACY CODE * |
| **********************************************************************************************/ |
| |
| int fmap_region_by_name(const uint32_t fmap_offset, const char * const name, |
| uint32_t * const offset, uint32_t * const size) |
| { |
| int i; |
| |
| struct fmap *fmap; |
| struct fmap fmap_head; |
| struct cbfs_media default_media; |
| struct cbfs_media *media = &default_media; |
| |
| if (init_default_cbfs_media(media) != 0) |
| return -1; |
| |
| media->open(media); |
| |
| if (!media->read(media, &fmap_head, fmap_offset, sizeof(fmap_head))) |
| return -1; |
| |
| if (memcmp(fmap_head.signature, FMAP_SIGNATURE, sizeof(fmap_head.signature))) { |
| return -1; |
| } |
| |
| int fmap_size = sizeof(*fmap) + |
| fmap_head.nareas * sizeof(struct fmap_area); |
| |
| fmap = malloc(fmap_size); |
| if (!fmap) |
| return -1; |
| |
| if (!media->read(media, fmap, fmap_offset, fmap_size)) |
| goto err; |
| |
| media->close(media); |
| |
| for (i = 0; i < fmap->nareas; i++) { |
| if (strcmp((const char *)fmap->areas[i].name, name) != 0) |
| continue; |
| if (offset) |
| *offset = fmap->areas[i].offset; |
| if (size) |
| *size = fmap->areas[i].size; |
| free(fmap); |
| return 0; |
| } |
| err: |
| free(fmap); |
| return -1; |
| } |