diff --git a/src/vendorcode/google/chromeos/vbnv_flash.c b/src/vendorcode/google/chromeos/vbnv_flash.c index ea5d9f3ff9..88f39b08f1 100644 --- a/src/vendorcode/google/chromeos/vbnv_flash.c +++ b/src/vendorcode/google/chromeos/vbnv_flash.c @@ -11,10 +11,9 @@ * 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. - * - * TODO: Make this CAR-friendly in case we use it on x86 some day. */ +#include #include #include #include @@ -27,20 +26,26 @@ #define BLOB_SIZE VB2_NVDATA_SIZE -/* FMAP descriptor of the NVRAM area */ -static struct region_device nvram_region; +struct vbnv_flash_ctx { + /* VBNV flash is initialized */ + int initialized; -/* offset of the current nvdata in SPI flash */ -static int blob_offset = -1; + /* Offset of the current nvdata in SPI flash */ + int blob_offset; -/* Offset of the topmost nvdata blob in SPI flash */ -static int top_offset; + /* Offset of the topmost nvdata blob in SPI flash */ + int top_offset; -/* cache of the current nvdata */ -static uint8_t cache[BLOB_SIZE]; + /* SPI flash handler used when saving data */ + struct spi_flash *flash; -/* spi_flash struct used when saving data */ -static struct spi_flash *spi_flash = NULL; + /* FMAP descriptor of the NVRAM area */ + struct region_device region; + + /* Cache of the current nvdata */ + uint8_t cache[BLOB_SIZE]; +}; +static struct vbnv_flash_ctx vbnv_flash CAR_GLOBAL; /* * This code assumes that flash is erased to 1-bits, and write operations can @@ -57,20 +62,16 @@ static inline int can_overwrite(uint8_t current, uint8_t new) return (current & new) == new; } -static inline int is_initialized(void) -{ - return blob_offset >= 0; -} - static int init_vbnv(void) { + struct vbnv_flash_ctx *ctx = car_get_var_ptr(&vbnv_flash); uint8_t buf[BLOB_SIZE]; uint8_t empty_blob[BLOB_SIZE]; int offset; int i; - if (vboot_named_region_device("RW_NVRAM", &nvram_region) || - region_device_sz(&nvram_region) < BLOB_SIZE) { + if (vboot_named_region_device("RW_NVRAM", &ctx->region) || + region_device_sz(&ctx->region) < BLOB_SIZE) { printk(BIOS_ERR, "%s: failed to locate NVRAM\n", __func__); return 1; } @@ -80,7 +81,7 @@ static int init_vbnv(void) empty_blob[i] = erase_value(); offset = 0; - top_offset = region_device_sz(&nvram_region) - BLOB_SIZE; + ctx->top_offset = region_device_sz(&ctx->region) - BLOB_SIZE; /* * after the loop, offset is supposed to point the blob right before @@ -88,8 +89,8 @@ static int init_vbnv(void) * empty blob, or the base of the region if the nvram has never been * used. */ - for (i = 0; i <= top_offset; i += BLOB_SIZE) { - if (rdev_readat(&nvram_region, buf, i, BLOB_SIZE) < 0) { + for (i = 0; i <= ctx->top_offset; i += BLOB_SIZE) { + if (rdev_readat(&ctx->region, buf, i, BLOB_SIZE) < 0) { printk(BIOS_ERR, "failed to read nvdata\n"); return 1; } @@ -99,12 +100,13 @@ static int init_vbnv(void) } /* reread the nvdata and write it to the cache */ - if (rdev_readat(&nvram_region, cache, offset, BLOB_SIZE) < 0) { + if (rdev_readat(&ctx->region, ctx->cache, offset, BLOB_SIZE) < 0) { printk(BIOS_ERR, "failed to read nvdata\n"); return 1; } - blob_offset = offset; + ctx->blob_offset = offset; + ctx->initialized = 1; return 0; } @@ -122,15 +124,19 @@ static void vbnv_is_erasable(void) * * TODO: Check by calling can_erase implemented by each spi flash driver */ - assert(!(region_device_offset(&nvram_region) % spi_flash->sector_size)); - assert(!(region_device_sz(&nvram_region) % spi_flash->sector_size)); + struct vbnv_flash_ctx *ctx = car_get_var_ptr(&vbnv_flash); + + assert(!(region_device_offset(&ctx->region) % ctx->flash->sector_size)); + assert(!(region_device_sz(&ctx->region) % ctx->flash->sector_size)); } static int vbnv_flash_probe(void) { - if (!spi_flash) { - spi_flash = spi_flash_probe(CONFIG_BOOT_MEDIA_SPI_BUS, 0); - if (!spi_flash) { + struct vbnv_flash_ctx *ctx = car_get_var_ptr(&vbnv_flash); + + if (!ctx->flash) { + ctx->flash = spi_flash_probe(CONFIG_BOOT_MEDIA_SPI_BUS, 0); + if (!ctx->flash) { printk(BIOS_ERR, "failed to probe spi flash\n"); return 1; } @@ -140,16 +146,25 @@ static int vbnv_flash_probe(void) */ vbnv_is_erasable(); } + + /* + * Handle the case where spi_flash_probe returns a CAR_GLOBAL + * in early execution on x86 but then later is moved to RAM. + */ + ctx->flash = car_get_var_ptr(ctx->flash); + return 0; } static int erase_nvram(void) { + struct vbnv_flash_ctx *ctx = car_get_var_ptr(&vbnv_flash); + if (vbnv_flash_probe()) return 1; - if (spi_flash->erase(spi_flash, region_device_offset(&nvram_region), - region_device_sz(&nvram_region))) { + if (ctx->flash->erase(ctx->flash, region_device_offset(&ctx->region), + region_device_sz(&ctx->region))) { printk(BIOS_ERR, "failed to erase nvram\n"); return 1; } @@ -160,33 +175,37 @@ static int erase_nvram(void) void read_vbnv_flash(uint8_t *vbnv_copy) { - if (!is_initialized()) + struct vbnv_flash_ctx *ctx = car_get_var_ptr(&vbnv_flash); + + if (!ctx->initialized) if (init_vbnv()) return; /* error */ - memcpy(vbnv_copy, cache, BLOB_SIZE); + + memcpy(vbnv_copy, ctx->cache, BLOB_SIZE); } void save_vbnv_flash(const uint8_t *vbnv_copy) { + struct vbnv_flash_ctx *ctx = car_get_var_ptr(&vbnv_flash); int new_offset; int i; - if (!is_initialized()) + if (!ctx->initialized) if (init_vbnv()) return; /* error */ /* Bail out if there have been no changes. */ - if (!memcmp(vbnv_copy, cache, BLOB_SIZE)) + if (!memcmp(vbnv_copy, ctx->cache, BLOB_SIZE)) return; - new_offset = blob_offset; + new_offset = ctx->blob_offset; /* See if we can overwrite the current blob with the new one */ for (i = 0; i < BLOB_SIZE; i++) { - if (!can_overwrite(cache[i], vbnv_copy[i])) { + if (!can_overwrite(ctx->cache[i], vbnv_copy[i])) { /* unable to overwrite. need to use the next blob */ new_offset += BLOB_SIZE; - if (new_offset > top_offset) { + if (new_offset > ctx->top_offset) { if (erase_nvram()) return; /* error */ new_offset = 0; @@ -196,12 +215,12 @@ void save_vbnv_flash(const uint8_t *vbnv_copy) } if (!vbnv_flash_probe() && - !spi_flash->write(spi_flash, - region_device_offset(&nvram_region) + new_offset, - BLOB_SIZE, vbnv_copy)) { + !ctx->flash->write(ctx->flash, + region_device_offset(&ctx->region) + new_offset, + BLOB_SIZE, vbnv_copy)) { /* write was successful. safely move pointer forward */ - blob_offset = new_offset; - memcpy(cache, vbnv_copy, BLOB_SIZE); + ctx->blob_offset = new_offset; + memcpy(ctx->cache, vbnv_copy, BLOB_SIZE); } else { printk(BIOS_ERR, "failed to save nvdata\n"); }