blob: 8a4fd09981521d5fb72b70e253ba52974a7e0472 [file] [log] [blame]
Angel Pons986d50e2020-04-02 23:48:53 +02001/* SPDX-License-Identifier: GPL-2.0-only */
Daisuke Nojiri477dd182014-10-08 11:32:23 -07002
Furquan Shaikh2a12e2e2016-07-25 11:48:03 -07003#include <commonlib/region.h>
Daisuke Nojiri2b59a832014-10-29 11:18:11 -07004#include <console/console.h>
Yu-Ping Wu29c8fa42019-11-18 11:25:47 +08005#include <fmap.h>
Arthur Heymans61ef0e42023-01-12 11:29:57 +01006#include <fmap_config.h>
Daisuke Nojiri2b59a832014-10-29 11:18:11 -07007#include <string.h>
Randall Spangler144c2282014-12-03 17:35:53 -08008#include <vb2_api.h>
Philipp Deppenwiesefea24292017-10-17 17:02:29 +02009#include <security/vboot/vboot_common.h>
10#include <security/vboot/vbnv.h>
11#include <security/vboot/vbnv_layout.h>
Daisuke Nojiri477dd182014-10-08 11:32:23 -070012
Daisuke Nojiri2b59a832014-10-29 11:18:11 -070013#define BLOB_SIZE VB2_NVDATA_SIZE
Daisuke Nojiri2b59a832014-10-29 11:18:11 -070014
Duncan Laurie01650382016-01-26 16:22:53 -080015struct vbnv_flash_ctx {
16 /* VBNV flash is initialized */
17 int initialized;
Daisuke Nojiri2b59a832014-10-29 11:18:11 -070018
Aaron Durbinb18a6662016-08-12 12:48:58 -050019 /* Offset of the current nvdata in flash */
Duncan Laurie01650382016-01-26 16:22:53 -080020 int blob_offset;
Daisuke Nojiri2b59a832014-10-29 11:18:11 -070021
Aaron Durbinb18a6662016-08-12 12:48:58 -050022 /* Offset of the topmost nvdata blob in flash */
Duncan Laurie01650382016-01-26 16:22:53 -080023 int top_offset;
Vadim Bendebury9e381a12015-01-10 23:11:43 -080024
Aaron Durbinb18a6662016-08-12 12:48:58 -050025 /* Region to store and retrieve the VBNV contents. */
26 struct region_device vbnv_dev;
Duncan Laurie01650382016-01-26 16:22:53 -080027
28 /* Cache of the current nvdata */
29 uint8_t cache[BLOB_SIZE];
30};
Arthur Heymans344e86b2019-11-20 19:47:10 +010031static struct vbnv_flash_ctx vbnv_flash;
Daisuke Nojiri2b59a832014-10-29 11:18:11 -070032
33/*
34 * This code assumes that flash is erased to 1-bits, and write operations can
35 * only change 1-bits to 0-bits. So if the new contents only change 1-bits to
36 * 0-bits, we can reuse the current blob.
37 */
38static inline uint8_t erase_value(void)
39{
40 return 0xff;
41}
42
43static inline int can_overwrite(uint8_t current, uint8_t new)
44{
45 return (current & new) == new;
46}
47
Arthur Heymans61ef0e42023-01-12 11:29:57 +010048_Static_assert(FMAP_SECTION_RW_NVRAM_SIZE >= BLOB_SIZE,
49 "RW_NVRAM FMAP section not present or too small");
50
Daisuke Nojiri2b59a832014-10-29 11:18:11 -070051static int init_vbnv(void)
52{
Arthur Heymans344e86b2019-11-20 19:47:10 +010053 struct vbnv_flash_ctx *ctx = &vbnv_flash;
Aaron Durbinb18a6662016-08-12 12:48:58 -050054 struct region_device *rdev = &ctx->vbnv_dev;
Daisuke Nojiri2b59a832014-10-29 11:18:11 -070055 uint8_t buf[BLOB_SIZE];
56 uint8_t empty_blob[BLOB_SIZE];
Randall Spangleraa1d3142016-08-26 16:01:16 -070057 int used_below, empty_above;
Daisuke Nojiri2b59a832014-10-29 11:18:11 -070058 int offset;
59 int i;
60
Yu-Ping Wu29c8fa42019-11-18 11:25:47 +080061 if (fmap_locate_area_as_rdev_rw("RW_NVRAM", rdev) ||
Aaron Durbinb18a6662016-08-12 12:48:58 -050062 region_device_sz(rdev) < BLOB_SIZE) {
Daisuke Nojiri2b59a832014-10-29 11:18:11 -070063 printk(BIOS_ERR, "%s: failed to locate NVRAM\n", __func__);
64 return 1;
65 }
66
67 /* Prepare an empty blob to compare against. */
68 for (i = 0; i < BLOB_SIZE; i++)
69 empty_blob[i] = erase_value();
70
Aaron Durbinb18a6662016-08-12 12:48:58 -050071 ctx->top_offset = region_device_sz(rdev) - BLOB_SIZE;
Vadim Bendebury9e381a12015-01-10 23:11:43 -080072
Randall Spangleraa1d3142016-08-26 16:01:16 -070073 /* Binary search for the border between used and empty */
74 used_below = 0;
75 empty_above = region_device_sz(rdev) / BLOB_SIZE;
76
77 while (used_below + 1 < empty_above) {
78 int guess = (used_below + empty_above) / 2;
79 if (rdev_readat(rdev, buf, guess * BLOB_SIZE, BLOB_SIZE) < 0) {
Daisuke Nojiri2b59a832014-10-29 11:18:11 -070080 printk(BIOS_ERR, "failed to read nvdata\n");
81 return 1;
82 }
83 if (!memcmp(buf, empty_blob, BLOB_SIZE))
Randall Spangleraa1d3142016-08-26 16:01:16 -070084 empty_above = guess;
85 else
86 used_below = guess;
Daisuke Nojiri2b59a832014-10-29 11:18:11 -070087 }
88
Randall Spangleraa1d3142016-08-26 16:01:16 -070089 /*
90 * Offset points to the last non-empty blob. Or if all blobs are empty
91 * (nvram is totally erased), point to the first blob.
92 */
93 offset = used_below * BLOB_SIZE;
94
Daisuke Nojiri2b59a832014-10-29 11:18:11 -070095 /* reread the nvdata and write it to the cache */
Aaron Durbinb18a6662016-08-12 12:48:58 -050096 if (rdev_readat(rdev, ctx->cache, offset, BLOB_SIZE) < 0) {
Daisuke Nojiri2b59a832014-10-29 11:18:11 -070097 printk(BIOS_ERR, "failed to read nvdata\n");
98 return 1;
99 }
100
Duncan Laurie01650382016-01-26 16:22:53 -0800101 ctx->blob_offset = offset;
102 ctx->initialized = 1;
Daisuke Nojiri2b59a832014-10-29 11:18:11 -0700103
104 return 0;
105}
106
Daisuke Nojiri2b59a832014-10-29 11:18:11 -0700107static int erase_nvram(void)
108{
Arthur Heymans344e86b2019-11-20 19:47:10 +0100109 struct vbnv_flash_ctx *ctx = &vbnv_flash;
Aaron Durbinb18a6662016-08-12 12:48:58 -0500110 const struct region_device *rdev = &ctx->vbnv_dev;
Duncan Laurie01650382016-01-26 16:22:53 -0800111
Aaron Durbinb18a6662016-08-12 12:48:58 -0500112 if (rdev_eraseat(rdev, 0, region_device_sz(rdev)) < 0) {
Daisuke Nojiri2b59a832014-10-29 11:18:11 -0700113 printk(BIOS_ERR, "failed to erase nvram\n");
114 return 1;
115 }
116
117 printk(BIOS_INFO, "nvram is cleared\n");
118 return 0;
119}
120
Duncan Laurie88b28ad2016-01-25 17:13:27 -0800121void read_vbnv_flash(uint8_t *vbnv_copy)
Daisuke Nojiri477dd182014-10-08 11:32:23 -0700122{
Arthur Heymans344e86b2019-11-20 19:47:10 +0100123 struct vbnv_flash_ctx *ctx = &vbnv_flash;
Duncan Laurie01650382016-01-26 16:22:53 -0800124
125 if (!ctx->initialized)
Daisuke Nojiri2b59a832014-10-29 11:18:11 -0700126 if (init_vbnv())
127 return; /* error */
Duncan Laurie01650382016-01-26 16:22:53 -0800128
129 memcpy(vbnv_copy, ctx->cache, BLOB_SIZE);
Daisuke Nojiri477dd182014-10-08 11:32:23 -0700130}
131
Duncan Laurie88b28ad2016-01-25 17:13:27 -0800132void save_vbnv_flash(const uint8_t *vbnv_copy)
Daisuke Nojiri477dd182014-10-08 11:32:23 -0700133{
Arthur Heymans344e86b2019-11-20 19:47:10 +0100134 struct vbnv_flash_ctx *ctx = &vbnv_flash;
Vadim Bendebury9e381a12015-01-10 23:11:43 -0800135 int new_offset;
Daisuke Nojiri2b59a832014-10-29 11:18:11 -0700136 int i;
Aaron Durbinb18a6662016-08-12 12:48:58 -0500137 const struct region_device *rdev = &ctx->vbnv_dev;
Daisuke Nojiri2b59a832014-10-29 11:18:11 -0700138
Duncan Laurie01650382016-01-26 16:22:53 -0800139 if (!ctx->initialized)
Daisuke Nojiri2b59a832014-10-29 11:18:11 -0700140 if (init_vbnv())
141 return; /* error */
142
143 /* Bail out if there have been no changes. */
Duncan Laurie01650382016-01-26 16:22:53 -0800144 if (!memcmp(vbnv_copy, ctx->cache, BLOB_SIZE))
Daisuke Nojiri2b59a832014-10-29 11:18:11 -0700145 return;
146
Duncan Laurie01650382016-01-26 16:22:53 -0800147 new_offset = ctx->blob_offset;
Vadim Bendebury9e381a12015-01-10 23:11:43 -0800148
Daisuke Nojiri2b59a832014-10-29 11:18:11 -0700149 /* See if we can overwrite the current blob with the new one */
150 for (i = 0; i < BLOB_SIZE; i++) {
Duncan Laurie01650382016-01-26 16:22:53 -0800151 if (!can_overwrite(ctx->cache[i], vbnv_copy[i])) {
Daisuke Nojiri2b59a832014-10-29 11:18:11 -0700152 /* unable to overwrite. need to use the next blob */
153 new_offset += BLOB_SIZE;
Duncan Laurie01650382016-01-26 16:22:53 -0800154 if (new_offset > ctx->top_offset) {
Daisuke Nojiri2b59a832014-10-29 11:18:11 -0700155 if (erase_nvram())
156 return; /* error */
Aaron Durbin4e50cdd2015-05-15 23:25:46 -0500157 new_offset = 0;
Daisuke Nojiri2b59a832014-10-29 11:18:11 -0700158 }
159 break;
160 }
161 }
162
Aaron Durbin1b5581c2016-08-23 13:22:17 -0500163 if (rdev_writeat(rdev, vbnv_copy, new_offset, BLOB_SIZE) == BLOB_SIZE) {
Vadim Bendebury9e381a12015-01-10 23:11:43 -0800164 /* write was successful. safely move pointer forward */
Duncan Laurie01650382016-01-26 16:22:53 -0800165 ctx->blob_offset = new_offset;
166 memcpy(ctx->cache, vbnv_copy, BLOB_SIZE);
Vadim Bendebury9e381a12015-01-10 23:11:43 -0800167 } else {
168 printk(BIOS_ERR, "failed to save nvdata\n");
Daisuke Nojiri2b59a832014-10-29 11:18:11 -0700169 }
Daisuke Nojiri477dd182014-10-08 11:32:23 -0700170}