blob: 282d542a096c72859ae77616d1bacbcac7e5c88b [file] [log] [blame]
Tom Rini70df9d62018-05-07 17:02:21 -04001// SPDX-License-Identifier: GPL-2.0+
Rob Clark15f3d742017-09-13 18:05:37 -04002/*
Heinrich Schuchardtc69addd2020-03-19 17:15:18 +00003 * UEFI runtime variable services
Rob Clark15f3d742017-09-13 18:05:37 -04004 *
Heinrich Schuchardtc69addd2020-03-19 17:15:18 +00005 * Copyright (c) 2017 Rob Clark
Rob Clark15f3d742017-09-13 18:05:37 -04006 */
7
Heinrich Schuchardt4b7d5c12020-07-14 21:25:28 +02008#define LOG_CATEGORY LOGC_EFI
9
Heinrich Schuchardt147d5d32019-10-26 23:53:48 +020010#include <common.h>
Rob Clark15f3d742017-09-13 18:05:37 -040011#include <efi_loader.h>
Heinrich Schuchardt9827e842020-06-22 18:10:27 +020012#include <efi_variable.h>
Simon Glassed38aef2020-05-10 11:40:03 -060013#include <env.h>
Simon Glass9d1f6192019-08-02 09:44:25 -060014#include <env_internal.h>
Heinrich Schuchardt147d5d32019-10-26 23:53:48 +020015#include <hexdump.h>
Heinrich Schuchardt4b7d5c12020-07-14 21:25:28 +020016#include <log.h>
Heinrich Schuchardt147d5d32019-10-26 23:53:48 +020017#include <malloc.h>
AKASHI Takahirob0f49ee2020-04-14 11:51:41 +090018#include <rtc.h>
AKASHI Takahiroad3f36c2019-01-21 12:43:13 +010019#include <search.h>
Simon Glass6f044982020-05-10 11:39:52 -060020#include <uuid.h>
AKASHI Takahiro6ec67672020-04-21 09:38:17 +090021#include <crypto/pkcs7_parser.h>
AKASHI Takahirob0f49ee2020-04-14 11:51:41 +090022#include <linux/compat.h>
Simon Glass48b6c6b2019-11-14 12:57:16 -070023#include <u-boot/crc.h>
Heinrich Schuchardt4b7d5c12020-07-14 21:25:28 +020024#include <asm/sections.h>
AKASHI Takahirob0f49ee2020-04-14 11:51:41 +090025
26#ifdef CONFIG_EFI_SECURE_BOOT
27static u8 pkcs7_hdr[] = {
28 /* SEQUENCE */
29 0x30, 0x82, 0x05, 0xc7,
30 /* OID: pkcs7-signedData */
31 0x06, 0x09, 0x2a, 0x86, 0x48, 0x86, 0xf7, 0x0d, 0x01, 0x07, 0x02,
32 /* Context Structured? */
33 0xa0, 0x82, 0x05, 0xb8,
34};
35
36/**
37 * efi_variable_parse_signature - parse a signature in variable
38 * @buf: Pointer to variable's value
39 * @buflen: Length of @buf
AKASHI Takahiro1bb6edf2020-08-12 09:37:50 +090040 * @tmpbuf: Pointer to temporary buffer
Heinrich Schuchardt0ffda472019-01-18 19:52:05 +010041 *
AKASHI Takahirob0f49ee2020-04-14 11:51:41 +090042 * Parse a signature embedded in variable's value and instantiate
43 * a pkcs7_message structure. Since pkcs7_parse_message() accepts only
44 * pkcs7's signedData, some header needed be prepended for correctly
45 * parsing authentication data, particularly for variable's.
AKASHI Takahiro1bb6edf2020-08-12 09:37:50 +090046 * A temporary buffer will be allocated if needed, and it should be
47 * kept valid during the authentication because some data in the buffer
48 * will be referenced by efi_signature_verify().
Heinrich Schuchardt0ffda472019-01-18 19:52:05 +010049 *
AKASHI Takahirob0f49ee2020-04-14 11:51:41 +090050 * Return: Pointer to pkcs7_message structure on success, NULL on error
Heinrich Schuchardt0ffda472019-01-18 19:52:05 +010051 */
AKASHI Takahirob0f49ee2020-04-14 11:51:41 +090052static struct pkcs7_message *efi_variable_parse_signature(const void *buf,
AKASHI Takahiro1bb6edf2020-08-12 09:37:50 +090053 size_t buflen,
54 u8 **tmpbuf)
AKASHI Takahirob0f49ee2020-04-14 11:51:41 +090055{
56 u8 *ebuf;
57 size_t ebuflen, len;
58 struct pkcs7_message *msg;
59
60 /*
61 * This is the best assumption to check if the binary is
62 * already in a form of pkcs7's signedData.
63 */
64 if (buflen > sizeof(pkcs7_hdr) &&
65 !memcmp(&((u8 *)buf)[4], &pkcs7_hdr[4], 11)) {
66 msg = pkcs7_parse_message(buf, buflen);
AKASHI Takahiro1bb6edf2020-08-12 09:37:50 +090067 if (IS_ERR(msg))
68 return NULL;
69 return msg;
AKASHI Takahirob0f49ee2020-04-14 11:51:41 +090070 }
71
72 /*
73 * Otherwise, we should add a dummy prefix sequence for pkcs7
74 * message parser to be able to process.
75 * NOTE: EDK2 also uses similar hack in WrapPkcs7Data()
76 * in CryptoPkg/Library/BaseCryptLib/Pk/CryptPkcs7VerifyCommon.c
77 * TODO:
78 * The header should be composed in a more refined manner.
79 */
AKASHI Takahiro37123732020-06-09 14:09:34 +090080 EFI_PRINT("Makeshift prefix added to authentication data\n");
AKASHI Takahirob0f49ee2020-04-14 11:51:41 +090081 ebuflen = sizeof(pkcs7_hdr) + buflen;
82 if (ebuflen <= 0x7f) {
AKASHI Takahiro37123732020-06-09 14:09:34 +090083 EFI_PRINT("Data is too short\n");
AKASHI Takahirob0f49ee2020-04-14 11:51:41 +090084 return NULL;
85 }
86
87 ebuf = malloc(ebuflen);
88 if (!ebuf) {
AKASHI Takahiro37123732020-06-09 14:09:34 +090089 EFI_PRINT("Out of memory\n");
AKASHI Takahirob0f49ee2020-04-14 11:51:41 +090090 return NULL;
91 }
92
93 memcpy(ebuf, pkcs7_hdr, sizeof(pkcs7_hdr));
94 memcpy(ebuf + sizeof(pkcs7_hdr), buf, buflen);
95 len = ebuflen - 4;
96 ebuf[2] = (len >> 8) & 0xff;
97 ebuf[3] = len & 0xff;
98 len = ebuflen - 0x13;
99 ebuf[0x11] = (len >> 8) & 0xff;
100 ebuf[0x12] = len & 0xff;
101
102 msg = pkcs7_parse_message(ebuf, ebuflen);
103
AKASHI Takahiro1bb6edf2020-08-12 09:37:50 +0900104 if (IS_ERR(msg)) {
105 free(ebuf);
AKASHI Takahirob0f49ee2020-04-14 11:51:41 +0900106 return NULL;
AKASHI Takahiro1bb6edf2020-08-12 09:37:50 +0900107 }
AKASHI Takahirob0f49ee2020-04-14 11:51:41 +0900108
AKASHI Takahiro1bb6edf2020-08-12 09:37:50 +0900109 *tmpbuf = ebuf;
AKASHI Takahirob0f49ee2020-04-14 11:51:41 +0900110 return msg;
111}
112
113/**
114 * efi_variable_authenticate - authenticate a variable
115 * @variable: Variable name in u16
116 * @vendor: Guid of variable
117 * @data_size: Size of @data
118 * @data: Pointer to variable's value
119 * @given_attr: Attributes to be given at SetVariable()
120 * @env_attr: Attributes that an existing variable holds
121 * @time: signed time that an existing variable holds
122 *
123 * Called by efi_set_variable() to verify that the input is correct.
124 * Will replace the given data pointer with another that points to
125 * the actual data to store in the internal memory.
126 * On success, @data and @data_size will be replaced with variable's
127 * actual data, excluding authentication data, and its size, and variable's
128 * attributes and signed time will also be returned in @env_attr and @time,
129 * respectively.
130 *
Heinrich Schuchardte2c43da2020-05-03 16:29:00 +0200131 * Return: status code
AKASHI Takahirob0f49ee2020-04-14 11:51:41 +0900132 */
133static efi_status_t efi_variable_authenticate(u16 *variable,
134 const efi_guid_t *vendor,
135 efi_uintn_t *data_size,
136 const void **data, u32 given_attr,
137 u32 *env_attr, u64 *time)
138{
139 const struct efi_variable_authentication_2 *auth;
140 struct efi_signature_store *truststore, *truststore2;
141 struct pkcs7_message *var_sig;
142 struct efi_image_regions *regs;
143 struct efi_time timestamp;
144 struct rtc_time tm;
145 u64 new_time;
AKASHI Takahiro1bb6edf2020-08-12 09:37:50 +0900146 u8 *ebuf;
Heinrich Schuchardt3a280332020-07-15 12:40:35 +0200147 enum efi_auth_var_type var_type;
AKASHI Takahirob0f49ee2020-04-14 11:51:41 +0900148 efi_status_t ret;
149
150 var_sig = NULL;
151 truststore = NULL;
152 truststore2 = NULL;
153 regs = NULL;
AKASHI Takahiro1bb6edf2020-08-12 09:37:50 +0900154 ebuf = NULL;
AKASHI Takahirob0f49ee2020-04-14 11:51:41 +0900155 ret = EFI_SECURITY_VIOLATION;
156
157 if (*data_size < sizeof(struct efi_variable_authentication_2))
158 goto err;
159
160 /* authentication data */
161 auth = *data;
162 if (*data_size < (sizeof(auth->time_stamp)
163 + auth->auth_info.hdr.dwLength))
164 goto err;
165
166 if (guidcmp(&auth->auth_info.cert_type, &efi_guid_cert_type_pkcs7))
167 goto err;
168
Heinrich Schuchardt7a76a3f2020-07-01 12:44:00 +0200169 memcpy(&timestamp, &auth->time_stamp, sizeof(timestamp));
170 if (timestamp.pad1 || timestamp.nanosecond || timestamp.timezone ||
171 timestamp.daylight || timestamp.pad2)
172 goto err;
173
AKASHI Takahirob0f49ee2020-04-14 11:51:41 +0900174 *data += sizeof(auth->time_stamp) + auth->auth_info.hdr.dwLength;
175 *data_size -= (sizeof(auth->time_stamp)
176 + auth->auth_info.hdr.dwLength);
177
AKASHI Takahirob0f49ee2020-04-14 11:51:41 +0900178 memset(&tm, 0, sizeof(tm));
179 tm.tm_year = timestamp.year;
180 tm.tm_mon = timestamp.month;
181 tm.tm_mday = timestamp.day;
182 tm.tm_hour = timestamp.hour;
183 tm.tm_min = timestamp.minute;
184 tm.tm_sec = timestamp.second;
185 new_time = rtc_mktime(&tm);
186
187 if (!efi_secure_boot_enabled()) {
188 /* finished checking */
189 *time = new_time;
190 return EFI_SUCCESS;
191 }
192
193 if (new_time <= *time)
194 goto err;
195
196 /* data to be digested */
197 regs = calloc(sizeof(*regs) + sizeof(struct image_region) * 5, 1);
198 if (!regs)
199 goto err;
200 regs->max = 5;
201 efi_image_region_add(regs, (uint8_t *)variable,
202 (uint8_t *)variable
203 + u16_strlen(variable) * sizeof(u16), 1);
204 efi_image_region_add(regs, (uint8_t *)vendor,
205 (uint8_t *)vendor + sizeof(*vendor), 1);
206 efi_image_region_add(regs, (uint8_t *)&given_attr,
207 (uint8_t *)&given_attr + sizeof(given_attr), 1);
208 efi_image_region_add(regs, (uint8_t *)&timestamp,
209 (uint8_t *)&timestamp + sizeof(timestamp), 1);
210 efi_image_region_add(regs, (uint8_t *)*data,
211 (uint8_t *)*data + *data_size, 1);
212
213 /* variable's signature list */
214 if (auth->auth_info.hdr.dwLength < sizeof(auth->auth_info))
215 goto err;
AKASHI Takahiro1bb6edf2020-08-12 09:37:50 +0900216
217 /* ebuf should be kept valid during the authentication */
AKASHI Takahirob0f49ee2020-04-14 11:51:41 +0900218 var_sig = efi_variable_parse_signature(auth->auth_info.cert_data,
219 auth->auth_info.hdr.dwLength
AKASHI Takahiro1bb6edf2020-08-12 09:37:50 +0900220 - sizeof(auth->auth_info),
221 &ebuf);
Patrick Wildtcd8a55b2020-05-07 02:13:18 +0200222 if (!var_sig) {
AKASHI Takahiro37123732020-06-09 14:09:34 +0900223 EFI_PRINT("Parsing variable's signature failed\n");
AKASHI Takahirob0f49ee2020-04-14 11:51:41 +0900224 goto err;
225 }
226
227 /* signature database used for authentication */
Heinrich Schuchardt3a280332020-07-15 12:40:35 +0200228 var_type = efi_auth_var_get_type(variable, vendor);
229 switch (var_type) {
230 case EFI_AUTH_VAR_PK:
231 case EFI_AUTH_VAR_KEK:
AKASHI Takahirob0f49ee2020-04-14 11:51:41 +0900232 /* with PK */
233 truststore = efi_sigstore_parse_sigdb(L"PK");
234 if (!truststore)
235 goto err;
Heinrich Schuchardt3a280332020-07-15 12:40:35 +0200236 break;
237 case EFI_AUTH_VAR_DB:
238 case EFI_AUTH_VAR_DBX:
AKASHI Takahirob0f49ee2020-04-14 11:51:41 +0900239 /* with PK and KEK */
240 truststore = efi_sigstore_parse_sigdb(L"KEK");
241 truststore2 = efi_sigstore_parse_sigdb(L"PK");
AKASHI Takahirob0f49ee2020-04-14 11:51:41 +0900242 if (!truststore) {
243 if (!truststore2)
244 goto err;
245
246 truststore = truststore2;
247 truststore2 = NULL;
248 }
Heinrich Schuchardt3a280332020-07-15 12:40:35 +0200249 break;
250 default:
AKASHI Takahirob0f49ee2020-04-14 11:51:41 +0900251 /* TODO: support private authenticated variables */
252 goto err;
253 }
254
255 /* verify signature */
AKASHI Takahiro14afd062020-07-21 19:35:22 +0900256 if (efi_signature_verify(regs, var_sig, truststore, NULL)) {
AKASHI Takahiro37123732020-06-09 14:09:34 +0900257 EFI_PRINT("Verified\n");
AKASHI Takahirob0f49ee2020-04-14 11:51:41 +0900258 } else {
259 if (truststore2 &&
AKASHI Takahiro14afd062020-07-21 19:35:22 +0900260 efi_signature_verify(regs, var_sig, truststore2, NULL)) {
AKASHI Takahiro37123732020-06-09 14:09:34 +0900261 EFI_PRINT("Verified\n");
AKASHI Takahirob0f49ee2020-04-14 11:51:41 +0900262 } else {
AKASHI Takahiro37123732020-06-09 14:09:34 +0900263 EFI_PRINT("Verifying variable's signature failed\n");
AKASHI Takahirob0f49ee2020-04-14 11:51:41 +0900264 goto err;
265 }
266 }
267
268 /* finished checking */
Heinrich Schuchardt5106dc72020-06-30 12:02:14 +0200269 *time = new_time;
AKASHI Takahirob0f49ee2020-04-14 11:51:41 +0900270 ret = EFI_SUCCESS;
271
272err:
273 efi_sigstore_free(truststore);
274 efi_sigstore_free(truststore2);
275 pkcs7_free_message(var_sig);
AKASHI Takahiro1bb6edf2020-08-12 09:37:50 +0900276 free(ebuf);
AKASHI Takahirob0f49ee2020-04-14 11:51:41 +0900277 free(regs);
278
279 return ret;
280}
281#else
282static efi_status_t efi_variable_authenticate(u16 *variable,
283 const efi_guid_t *vendor,
284 efi_uintn_t *data_size,
285 const void **data, u32 given_attr,
286 u32 *env_attr, u64 *time)
287{
288 return EFI_SUCCESS;
289}
290#endif /* CONFIG_EFI_SECURE_BOOT */
291
Heinrich Schuchardt047276c2020-07-04 18:34:15 +0200292efi_status_t __efi_runtime
293efi_get_variable_int(u16 *variable_name, const efi_guid_t *vendor,
294 u32 *attributes, efi_uintn_t *data_size, void *data,
295 u64 *timep)
Rob Clark15f3d742017-09-13 18:05:37 -0400296{
Ilias Apalodimasa4d1b1b2020-07-23 15:49:49 +0300297 return efi_get_variable_mem(variable_name, vendor, attributes, data_size, data, timep);
AKASHI Takahiroad3f36c2019-01-21 12:43:13 +0100298}
299
Heinrich Schuchardt047276c2020-07-04 18:34:15 +0200300efi_status_t __efi_runtime
301efi_get_next_variable_name_int(efi_uintn_t *variable_name_size,
302 u16 *variable_name, efi_guid_t *vendor)
Rob Clark15f3d742017-09-13 18:05:37 -0400303{
Ilias Apalodimasa4d1b1b2020-07-23 15:49:49 +0300304 return efi_get_next_variable_name_mem(variable_name_size, variable_name, vendor);
Rob Clark15f3d742017-09-13 18:05:37 -0400305}
306
Heinrich Schuchardt9827e842020-06-22 18:10:27 +0200307efi_status_t efi_set_variable_int(u16 *variable_name, const efi_guid_t *vendor,
308 u32 attributes, efi_uintn_t data_size,
309 const void *data, bool ro_check)
Rob Clark15f3d742017-09-13 18:05:37 -0400310{
Heinrich Schuchardt047276c2020-07-04 18:34:15 +0200311 struct efi_var_entry *var;
312 efi_uintn_t ret;
AKASHI Takahirob0f49ee2020-04-14 11:51:41 +0900313 bool append, delete;
314 u64 time = 0;
Heinrich Schuchardt3a280332020-07-15 12:40:35 +0200315 enum efi_auth_var_type var_type;
Rob Clark15f3d742017-09-13 18:05:37 -0400316
Heinrich Schuchardtadba3962019-06-12 23:28:42 +0200317 if (!variable_name || !*variable_name || !vendor ||
318 ((attributes & EFI_VARIABLE_RUNTIME_ACCESS) &&
Heinrich Schuchardt047276c2020-07-04 18:34:15 +0200319 !(attributes & EFI_VARIABLE_BOOTSERVICE_ACCESS)))
320 return EFI_INVALID_PARAMETER;
Rob Clark15f3d742017-09-13 18:05:37 -0400321
AKASHI Takahirob0f49ee2020-04-14 11:51:41 +0900322 /* check if a variable exists */
Heinrich Schuchardt047276c2020-07-04 18:34:15 +0200323 var = efi_var_mem_find(vendor, variable_name, NULL);
AKASHI Takahirob0f49ee2020-04-14 11:51:41 +0900324 append = !!(attributes & EFI_VARIABLE_APPEND_WRITE);
325 attributes &= ~(u32)EFI_VARIABLE_APPEND_WRITE;
326 delete = !append && (!data_size || !attributes);
327
328 /* check attributes */
Heinrich Schuchardt4b7d5c12020-07-14 21:25:28 +0200329 var_type = efi_auth_var_get_type(variable_name, vendor);
Heinrich Schuchardt047276c2020-07-04 18:34:15 +0200330 if (var) {
331 if (ro_check && (var->attr & EFI_VARIABLE_READ_ONLY))
332 return EFI_WRITE_PROTECTED;
AKASHI Takahiroe5023b72019-09-06 15:09:52 +0900333
Heinrich Schuchardt4b7d5c12020-07-14 21:25:28 +0200334 if (IS_ENABLED(CONFIG_EFI_VARIABLES_PRESEED)) {
335 if (var_type != EFI_AUTH_VAR_NONE)
336 return EFI_WRITE_PROTECTED;
337 }
338
AKASHI Takahiroe5023b72019-09-06 15:09:52 +0900339 /* attributes won't be changed */
AKASHI Takahirob0f49ee2020-04-14 11:51:41 +0900340 if (!delete &&
Heinrich Schuchardt047276c2020-07-04 18:34:15 +0200341 ((ro_check && var->attr != attributes) ||
342 (!ro_check && ((var->attr & ~(u32)EFI_VARIABLE_READ_ONLY)
Heinrich Schuchardt9827e842020-06-22 18:10:27 +0200343 != (attributes & ~(u32)EFI_VARIABLE_READ_ONLY))))) {
Heinrich Schuchardt047276c2020-07-04 18:34:15 +0200344 return EFI_INVALID_PARAMETER;
AKASHI Takahiroe5023b72019-09-06 15:09:52 +0900345 }
Heinrich Schuchardt047276c2020-07-04 18:34:15 +0200346 time = var->time;
AKASHI Takahiroe5023b72019-09-06 15:09:52 +0900347 } else {
Heinrich Schuchardt047276c2020-07-04 18:34:15 +0200348 if (delete || append)
Heinrich Schuchardt8021bbe2019-09-26 21:40:18 +0200349 /*
350 * Trying to delete or to update a non-existent
351 * variable.
352 */
Heinrich Schuchardt047276c2020-07-04 18:34:15 +0200353 return EFI_NOT_FOUND;
AKASHI Takahirob0f49ee2020-04-14 11:51:41 +0900354 }
AKASHI Takahiroe5023b72019-09-06 15:09:52 +0900355
Heinrich Schuchardt3a280332020-07-15 12:40:35 +0200356 if (var_type != EFI_AUTH_VAR_NONE) {
AKASHI Takahirob0f49ee2020-04-14 11:51:41 +0900357 /* authentication is mandatory */
358 if (!(attributes &
359 EFI_VARIABLE_TIME_BASED_AUTHENTICATED_WRITE_ACCESS)) {
Heinrich Schuchardt047276c2020-07-04 18:34:15 +0200360 EFI_PRINT("%ls: TIME_BASED_AUTHENTICATED_WRITE_ACCESS required\n",
AKASHI Takahiro37123732020-06-09 14:09:34 +0900361 variable_name);
Heinrich Schuchardt047276c2020-07-04 18:34:15 +0200362 return EFI_INVALID_PARAMETER;
AKASHI Takahirob0f49ee2020-04-14 11:51:41 +0900363 }
364 }
365
366 /* authenticate a variable */
367 if (IS_ENABLED(CONFIG_EFI_SECURE_BOOT)) {
Heinrich Schuchardt047276c2020-07-04 18:34:15 +0200368 if (attributes & EFI_VARIABLE_AUTHENTICATED_WRITE_ACCESS)
369 return EFI_INVALID_PARAMETER;
AKASHI Takahirob0f49ee2020-04-14 11:51:41 +0900370 if (attributes &
371 EFI_VARIABLE_TIME_BASED_AUTHENTICATED_WRITE_ACCESS) {
Heinrich Schuchardt047276c2020-07-04 18:34:15 +0200372 u32 env_attr;
373
AKASHI Takahirob0f49ee2020-04-14 11:51:41 +0900374 ret = efi_variable_authenticate(variable_name, vendor,
375 &data_size, &data,
Heinrich Schuchardt047276c2020-07-04 18:34:15 +0200376 attributes, &env_attr,
AKASHI Takahirob0f49ee2020-04-14 11:51:41 +0900377 &time);
378 if (ret != EFI_SUCCESS)
Heinrich Schuchardt047276c2020-07-04 18:34:15 +0200379 return ret;
AKASHI Takahirob0f49ee2020-04-14 11:51:41 +0900380
381 /* last chance to check for delete */
382 if (!data_size)
383 delete = true;
384 }
385 } else {
386 if (attributes &
387 (EFI_VARIABLE_AUTHENTICATED_WRITE_ACCESS |
388 EFI_VARIABLE_TIME_BASED_AUTHENTICATED_WRITE_ACCESS)) {
AKASHI Takahiro37123732020-06-09 14:09:34 +0900389 EFI_PRINT("Secure boot is not configured\n");
Heinrich Schuchardt047276c2020-07-04 18:34:15 +0200390 return EFI_INVALID_PARAMETER;
AKASHI Takahirob0f49ee2020-04-14 11:51:41 +0900391 }
392 }
393
AKASHI Takahirob0f49ee2020-04-14 11:51:41 +0900394 if (delete) {
Heinrich Schuchardt047276c2020-07-04 18:34:15 +0200395 /* EFI_NOT_FOUND has been handled before */
AKASHI Takahirob0f49ee2020-04-14 11:51:41 +0900396 ret = EFI_SUCCESS;
Heinrich Schuchardt047276c2020-07-04 18:34:15 +0200397 } else if (append) {
398 u16 *old_data = var->name;
AKASHI Takahirob0f49ee2020-04-14 11:51:41 +0900399
Heinrich Schuchardt047276c2020-07-04 18:34:15 +0200400 for (; *old_data; ++old_data)
401 ;
402 ++old_data;
403 ret = efi_var_mem_ins(variable_name, vendor, attributes,
404 var->length, old_data, data_size, data,
405 time);
AKASHI Takahirob0f49ee2020-04-14 11:51:41 +0900406 } else {
Heinrich Schuchardt047276c2020-07-04 18:34:15 +0200407 ret = efi_var_mem_ins(variable_name, vendor, attributes,
408 data_size, data, 0, NULL, time);
Rob Clark15f3d742017-09-13 18:05:37 -0400409 }
Heinrich Schuchardt047276c2020-07-04 18:34:15 +0200410 efi_var_mem_del(var);
Rob Clark15f3d742017-09-13 18:05:37 -0400411
Heinrich Schuchardt047276c2020-07-04 18:34:15 +0200412 if (ret != EFI_SUCCESS)
413 return ret;
Rob Clark15f3d742017-09-13 18:05:37 -0400414
Heinrich Schuchardt3a280332020-07-15 12:40:35 +0200415 if (var_type == EFI_AUTH_VAR_PK)
Heinrich Schuchardt047276c2020-07-04 18:34:15 +0200416 ret = efi_init_secure_state();
417 else
418 ret = EFI_SUCCESS;
Rob Clark15f3d742017-09-13 18:05:37 -0400419
Heinrich Schuchardt09a8d502020-03-19 18:21:58 +0000420 /* Write non-volatile EFI variables to file */
421 if (attributes & EFI_VARIABLE_NON_VOLATILE &&
422 ret == EFI_SUCCESS && efi_obj_list_initialized == EFI_SUCCESS)
423 efi_var_to_file();
424
Heinrich Schuchardt047276c2020-07-04 18:34:15 +0200425 return EFI_SUCCESS;
AKASHI Takahirob0f49ee2020-04-14 11:51:41 +0900426}
427
Heinrich Schuchardt276c61d2020-06-26 17:57:48 +0200428efi_status_t efi_query_variable_info_int(u32 attributes,
429 u64 *maximum_variable_storage_size,
430 u64 *remaining_variable_storage_size,
431 u64 *maximum_variable_size)
432{
Heinrich Schuchardt047276c2020-07-04 18:34:15 +0200433 *maximum_variable_storage_size = EFI_VAR_BUF_SIZE -
434 sizeof(struct efi_var_file);
435 *remaining_variable_storage_size = efi_var_mem_free();
436 *maximum_variable_size = EFI_VAR_BUF_SIZE -
437 sizeof(struct efi_var_file) -
438 sizeof(struct efi_var_entry);
439 return EFI_SUCCESS;
Heinrich Schuchardt276c61d2020-06-26 17:57:48 +0200440}
441
AKASHI Takahirob0f49ee2020-04-14 11:51:41 +0900442/**
Heinrich Schuchardt276c61d2020-06-26 17:57:48 +0200443 * efi_query_variable_info_runtime() - runtime implementation of
444 * QueryVariableInfo()
Heinrich Schuchardt9b19dae2019-06-20 12:13:05 +0200445 *
446 * @attributes: bitmask to select variables to be
447 * queried
448 * @maximum_variable_storage_size: maximum size of storage area for the
449 * selected variable types
450 * @remaining_variable_storage_size: remaining size of storage are for the
451 * selected variable types
452 * @maximum_variable_size: maximum size of a variable of the
453 * selected type
454 * Returns: status code
455 */
Heinrich Schuchardt276c61d2020-06-26 17:57:48 +0200456efi_status_t __efi_runtime EFIAPI efi_query_variable_info_runtime(
Heinrich Schuchardt9b19dae2019-06-20 12:13:05 +0200457 u32 attributes,
458 u64 *maximum_variable_storage_size,
459 u64 *remaining_variable_storage_size,
460 u64 *maximum_variable_size)
461{
462 return EFI_UNSUPPORTED;
463}
Heinrich Schuchardtcf3b1182019-06-20 13:52:16 +0200464
465/**
Heinrich Schuchardt2ac62582019-06-20 15:25:48 +0200466 * efi_set_variable_runtime() - runtime implementation of SetVariable()
Heinrich Schuchardtee1f0e42019-07-14 12:11:16 +0200467 *
468 * @variable_name: name of the variable
469 * @vendor: vendor GUID
470 * @attributes: attributes of the variable
471 * @data_size: size of the buffer with the variable value
472 * @data: buffer with the variable value
473 * Return: status code
Heinrich Schuchardt2ac62582019-06-20 15:25:48 +0200474 */
475static efi_status_t __efi_runtime EFIAPI
476efi_set_variable_runtime(u16 *variable_name, const efi_guid_t *vendor,
477 u32 attributes, efi_uintn_t data_size,
478 const void *data)
479{
480 return EFI_UNSUPPORTED;
481}
482
483/**
484 * efi_variables_boot_exit_notify() - notify ExitBootServices() is called
485 */
486void efi_variables_boot_exit_notify(void)
487{
Heinrich Schuchardt09a8d502020-03-19 18:21:58 +0000488 /* Switch variable services functions to runtime version */
Heinrich Schuchardt2ac62582019-06-20 15:25:48 +0200489 efi_runtime_services.get_variable = efi_get_variable_runtime;
490 efi_runtime_services.get_next_variable_name =
491 efi_get_next_variable_name_runtime;
492 efi_runtime_services.set_variable = efi_set_variable_runtime;
Heinrich Schuchardt276c61d2020-06-26 17:57:48 +0200493 efi_runtime_services.query_variable_info =
494 efi_query_variable_info_runtime;
Heinrich Schuchardt2ac62582019-06-20 15:25:48 +0200495 efi_update_table_header_crc32(&efi_runtime_services.hdr);
496}
497
498/**
Heinrich Schuchardtcf3b1182019-06-20 13:52:16 +0200499 * efi_init_variables() - initialize variable services
500 *
501 * Return: status code
502 */
503efi_status_t efi_init_variables(void)
504{
AKASHI Takahiroc78dc192020-04-14 11:51:42 +0900505 efi_status_t ret;
506
Heinrich Schuchardt047276c2020-07-04 18:34:15 +0200507 ret = efi_var_mem_init();
508 if (ret != EFI_SUCCESS)
509 return ret;
510
AKASHI Takahiroc78dc192020-04-14 11:51:42 +0900511 ret = efi_init_secure_state();
Heinrich Schuchardt09a8d502020-03-19 18:21:58 +0000512 if (ret != EFI_SUCCESS)
513 return ret;
AKASHI Takahiroc78dc192020-04-14 11:51:42 +0900514
Heinrich Schuchardt4b7d5c12020-07-14 21:25:28 +0200515 if (IS_ENABLED(CONFIG_EFI_VARIABLES_PRESEED)) {
516 ret = efi_var_restore((struct efi_var_file *)
517 __efi_var_file_begin);
518 if (ret != EFI_SUCCESS)
519 log_err("Invalid EFI variable seed\n");
520 }
521
Heinrich Schuchardt09a8d502020-03-19 18:21:58 +0000522 return efi_var_from_file();
Heinrich Schuchardtcf3b1182019-06-20 13:52:16 +0200523}