blob: 0c06931135e32fe71d3167916903b519ea32977e [file] [log] [blame]
Tom Rinif739fcd2018-05-07 17:02:21 -04001// SPDX-License-Identifier: GPL-2.0+
Rob Clarkad644e72017-09-13 18:05:37 -04002/*
Heinrich Schuchardtf0b0f7f2020-03-19 17:15:18 +00003 * UEFI runtime variable services
Rob Clarkad644e72017-09-13 18:05:37 -04004 *
Heinrich Schuchardtf0b0f7f2020-03-19 17:15:18 +00005 * Copyright (c) 2017 Rob Clark
Rob Clarkad644e72017-09-13 18:05:37 -04006 */
7
Heinrich Schuchardt7dda1632020-07-14 21:25:28 +02008#define LOG_CATEGORY LOGC_EFI
9
Heinrich Schuchardte731af42019-10-26 23:53:48 +020010#include <common.h>
Rob Clarkad644e72017-09-13 18:05:37 -040011#include <efi_loader.h>
Heinrich Schuchardtf2d2b3a2020-06-22 18:10:27 +020012#include <efi_variable.h>
Simon Glass09140112020-05-10 11:40:03 -060013#include <env.h>
Simon Glassf3998fd2019-08-02 09:44:25 -060014#include <env_internal.h>
Heinrich Schuchardte731af42019-10-26 23:53:48 +020015#include <hexdump.h>
Heinrich Schuchardt7dda1632020-07-14 21:25:28 +020016#include <log.h>
Heinrich Schuchardte731af42019-10-26 23:53:48 +020017#include <malloc.h>
AKASHI Takahiro767f6ee2020-04-14 11:51:41 +090018#include <rtc.h>
AKASHI Takahirod99a87f2019-01-21 12:43:13 +010019#include <search.h>
Simon Glassba06b3c2020-05-10 11:39:52 -060020#include <uuid.h>
AKASHI Takahiroe3f5c9c2020-04-21 09:38:17 +090021#include <crypto/pkcs7_parser.h>
AKASHI Takahiro767f6ee2020-04-14 11:51:41 +090022#include <linux/compat.h>
Simon Glass3db71102019-11-14 12:57:16 -070023#include <u-boot/crc.h>
Heinrich Schuchardt7dda1632020-07-14 21:25:28 +020024#include <asm/sections.h>
AKASHI Takahiro767f6ee2020-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 Takahiro0658bb22020-08-12 09:37:50 +090040 * @tmpbuf: Pointer to temporary buffer
AKASHI Takahiro767f6ee2020-04-14 11:51:41 +090041 *
42 * 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 Takahiro0658bb22020-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().
AKASHI Takahiro767f6ee2020-04-14 11:51:41 +090049 *
50 * Return: Pointer to pkcs7_message structure on success, NULL on error
51 */
52static struct pkcs7_message *efi_variable_parse_signature(const void *buf,
AKASHI Takahiro0658bb22020-08-12 09:37:50 +090053 size_t buflen,
54 u8 **tmpbuf)
AKASHI Takahiro767f6ee2020-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 Takahiro0658bb22020-08-12 09:37:50 +090067 if (IS_ERR(msg))
68 return NULL;
69 return msg;
AKASHI Takahiro767f6ee2020-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 Takahiroce3c3862020-06-09 14:09:34 +090080 EFI_PRINT("Makeshift prefix added to authentication data\n");
AKASHI Takahiro767f6ee2020-04-14 11:51:41 +090081 ebuflen = sizeof(pkcs7_hdr) + buflen;
82 if (ebuflen <= 0x7f) {
AKASHI Takahiroce3c3862020-06-09 14:09:34 +090083 EFI_PRINT("Data is too short\n");
AKASHI Takahiro767f6ee2020-04-14 11:51:41 +090084 return NULL;
85 }
86
87 ebuf = malloc(ebuflen);
88 if (!ebuf) {
AKASHI Takahiroce3c3862020-06-09 14:09:34 +090089 EFI_PRINT("Out of memory\n");
AKASHI Takahiro767f6ee2020-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 Takahiro0658bb22020-08-12 09:37:50 +0900104 if (IS_ERR(msg)) {
105 free(ebuf);
AKASHI Takahiro767f6ee2020-04-14 11:51:41 +0900106 return NULL;
AKASHI Takahiro0658bb22020-08-12 09:37:50 +0900107 }
AKASHI Takahiro767f6ee2020-04-14 11:51:41 +0900108
AKASHI Takahiro0658bb22020-08-12 09:37:50 +0900109 *tmpbuf = ebuf;
AKASHI Takahiro767f6ee2020-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 Schuchardt30f92ce2020-05-03 16:29:00 +0200131 * Return: status code
AKASHI Takahiro767f6ee2020-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 Takahiro0658bb22020-08-12 09:37:50 +0900146 u8 *ebuf;
Heinrich Schuchardt99bfab82020-07-15 12:40:35 +0200147 enum efi_auth_var_type var_type;
AKASHI Takahiro767f6ee2020-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 Takahiro0658bb22020-08-12 09:37:50 +0900154 ebuf = NULL;
AKASHI Takahiro767f6ee2020-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 Schuchardt33f183f2020-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 Takahiro767f6ee2020-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 Takahiro767f6ee2020-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 Takahiro0658bb22020-08-12 09:37:50 +0900216
217 /* ebuf should be kept valid during the authentication */
AKASHI Takahiro767f6ee2020-04-14 11:51:41 +0900218 var_sig = efi_variable_parse_signature(auth->auth_info.cert_data,
219 auth->auth_info.hdr.dwLength
AKASHI Takahiro0658bb22020-08-12 09:37:50 +0900220 - sizeof(auth->auth_info),
221 &ebuf);
Patrick Wildt9ad15222020-05-07 02:13:18 +0200222 if (!var_sig) {
AKASHI Takahiroce3c3862020-06-09 14:09:34 +0900223 EFI_PRINT("Parsing variable's signature failed\n");
AKASHI Takahiro767f6ee2020-04-14 11:51:41 +0900224 goto err;
225 }
226
227 /* signature database used for authentication */
Heinrich Schuchardt99bfab82020-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 Takahiro767f6ee2020-04-14 11:51:41 +0900232 /* with PK */
233 truststore = efi_sigstore_parse_sigdb(L"PK");
234 if (!truststore)
235 goto err;
Heinrich Schuchardt99bfab82020-07-15 12:40:35 +0200236 break;
237 case EFI_AUTH_VAR_DB:
238 case EFI_AUTH_VAR_DBX:
AKASHI Takahiro767f6ee2020-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 Takahiro767f6ee2020-04-14 11:51:41 +0900242 if (!truststore) {
243 if (!truststore2)
244 goto err;
245
246 truststore = truststore2;
247 truststore2 = NULL;
248 }
Heinrich Schuchardt99bfab82020-07-15 12:40:35 +0200249 break;
250 default:
AKASHI Takahiro767f6ee2020-04-14 11:51:41 +0900251 /* TODO: support private authenticated variables */
252 goto err;
253 }
254
255 /* verify signature */
256 if (efi_signature_verify_with_sigdb(regs, var_sig, truststore, NULL)) {
AKASHI Takahiroce3c3862020-06-09 14:09:34 +0900257 EFI_PRINT("Verified\n");
AKASHI Takahiro767f6ee2020-04-14 11:51:41 +0900258 } else {
259 if (truststore2 &&
260 efi_signature_verify_with_sigdb(regs, var_sig,
261 truststore2, NULL)) {
AKASHI Takahiroce3c3862020-06-09 14:09:34 +0900262 EFI_PRINT("Verified\n");
AKASHI Takahiro767f6ee2020-04-14 11:51:41 +0900263 } else {
AKASHI Takahiroce3c3862020-06-09 14:09:34 +0900264 EFI_PRINT("Verifying variable's signature failed\n");
AKASHI Takahiro767f6ee2020-04-14 11:51:41 +0900265 goto err;
266 }
267 }
268
269 /* finished checking */
Heinrich Schuchardt3a92f852020-06-30 12:02:14 +0200270 *time = new_time;
AKASHI Takahiro767f6ee2020-04-14 11:51:41 +0900271 ret = EFI_SUCCESS;
272
273err:
274 efi_sigstore_free(truststore);
275 efi_sigstore_free(truststore2);
276 pkcs7_free_message(var_sig);
AKASHI Takahiro0658bb22020-08-12 09:37:50 +0900277 free(ebuf);
AKASHI Takahiro767f6ee2020-04-14 11:51:41 +0900278 free(regs);
279
280 return ret;
281}
282#else
283static efi_status_t efi_variable_authenticate(u16 *variable,
284 const efi_guid_t *vendor,
285 efi_uintn_t *data_size,
286 const void **data, u32 given_attr,
287 u32 *env_attr, u64 *time)
288{
289 return EFI_SUCCESS;
290}
291#endif /* CONFIG_EFI_SECURE_BOOT */
292
Heinrich Schuchardtab7296c2020-07-04 18:34:15 +0200293efi_status_t __efi_runtime
294efi_get_variable_int(u16 *variable_name, const efi_guid_t *vendor,
295 u32 *attributes, efi_uintn_t *data_size, void *data,
296 u64 *timep)
AKASHI Takahiro767f6ee2020-04-14 11:51:41 +0900297{
Ilias Apalodimase01aed42020-07-23 15:49:49 +0300298 return efi_get_variable_mem(variable_name, vendor, attributes, data_size, data, timep);
AKASHI Takahirod99a87f2019-01-21 12:43:13 +0100299}
300
Heinrich Schuchardtab7296c2020-07-04 18:34:15 +0200301efi_status_t __efi_runtime
302efi_get_next_variable_name_int(efi_uintn_t *variable_name_size,
303 u16 *variable_name, efi_guid_t *vendor)
Rob Clarkad644e72017-09-13 18:05:37 -0400304{
Ilias Apalodimase01aed42020-07-23 15:49:49 +0300305 return efi_get_next_variable_name_mem(variable_name_size, variable_name, vendor);
Rob Clarkad644e72017-09-13 18:05:37 -0400306}
307
Heinrich Schuchardtf2d2b3a2020-06-22 18:10:27 +0200308efi_status_t efi_set_variable_int(u16 *variable_name, const efi_guid_t *vendor,
309 u32 attributes, efi_uintn_t data_size,
310 const void *data, bool ro_check)
AKASHI Takahiro767f6ee2020-04-14 11:51:41 +0900311{
Heinrich Schuchardtab7296c2020-07-04 18:34:15 +0200312 struct efi_var_entry *var;
313 efi_uintn_t ret;
AKASHI Takahiro767f6ee2020-04-14 11:51:41 +0900314 bool append, delete;
315 u64 time = 0;
Heinrich Schuchardt99bfab82020-07-15 12:40:35 +0200316 enum efi_auth_var_type var_type;
AKASHI Takahiro767f6ee2020-04-14 11:51:41 +0900317
AKASHI Takahiro767f6ee2020-04-14 11:51:41 +0900318 if (!variable_name || !*variable_name || !vendor ||
319 ((attributes & EFI_VARIABLE_RUNTIME_ACCESS) &&
Heinrich Schuchardtab7296c2020-07-04 18:34:15 +0200320 !(attributes & EFI_VARIABLE_BOOTSERVICE_ACCESS)))
321 return EFI_INVALID_PARAMETER;
AKASHI Takahiro767f6ee2020-04-14 11:51:41 +0900322
323 /* check if a variable exists */
Heinrich Schuchardtab7296c2020-07-04 18:34:15 +0200324 var = efi_var_mem_find(vendor, variable_name, NULL);
AKASHI Takahiro767f6ee2020-04-14 11:51:41 +0900325 append = !!(attributes & EFI_VARIABLE_APPEND_WRITE);
326 attributes &= ~(u32)EFI_VARIABLE_APPEND_WRITE;
327 delete = !append && (!data_size || !attributes);
328
329 /* check attributes */
Heinrich Schuchardt7dda1632020-07-14 21:25:28 +0200330 var_type = efi_auth_var_get_type(variable_name, vendor);
Heinrich Schuchardtab7296c2020-07-04 18:34:15 +0200331 if (var) {
332 if (ro_check && (var->attr & EFI_VARIABLE_READ_ONLY))
333 return EFI_WRITE_PROTECTED;
AKASHI Takahiro767f6ee2020-04-14 11:51:41 +0900334
Heinrich Schuchardt7dda1632020-07-14 21:25:28 +0200335 if (IS_ENABLED(CONFIG_EFI_VARIABLES_PRESEED)) {
336 if (var_type != EFI_AUTH_VAR_NONE)
337 return EFI_WRITE_PROTECTED;
338 }
339
AKASHI Takahiro767f6ee2020-04-14 11:51:41 +0900340 /* attributes won't be changed */
341 if (!delete &&
Heinrich Schuchardtab7296c2020-07-04 18:34:15 +0200342 ((ro_check && var->attr != attributes) ||
343 (!ro_check && ((var->attr & ~(u32)EFI_VARIABLE_READ_ONLY)
Heinrich Schuchardtf2d2b3a2020-06-22 18:10:27 +0200344 != (attributes & ~(u32)EFI_VARIABLE_READ_ONLY))))) {
Heinrich Schuchardtab7296c2020-07-04 18:34:15 +0200345 return EFI_INVALID_PARAMETER;
AKASHI Takahiro767f6ee2020-04-14 11:51:41 +0900346 }
Heinrich Schuchardtab7296c2020-07-04 18:34:15 +0200347 time = var->time;
AKASHI Takahiro767f6ee2020-04-14 11:51:41 +0900348 } else {
Heinrich Schuchardtab7296c2020-07-04 18:34:15 +0200349 if (delete || append)
AKASHI Takahiro767f6ee2020-04-14 11:51:41 +0900350 /*
351 * Trying to delete or to update a non-existent
352 * variable.
353 */
Heinrich Schuchardtab7296c2020-07-04 18:34:15 +0200354 return EFI_NOT_FOUND;
AKASHI Takahiro767f6ee2020-04-14 11:51:41 +0900355 }
356
Heinrich Schuchardt99bfab82020-07-15 12:40:35 +0200357 if (var_type != EFI_AUTH_VAR_NONE) {
AKASHI Takahiro767f6ee2020-04-14 11:51:41 +0900358 /* authentication is mandatory */
359 if (!(attributes &
360 EFI_VARIABLE_TIME_BASED_AUTHENTICATED_WRITE_ACCESS)) {
Heinrich Schuchardtab7296c2020-07-04 18:34:15 +0200361 EFI_PRINT("%ls: TIME_BASED_AUTHENTICATED_WRITE_ACCESS required\n",
AKASHI Takahiroce3c3862020-06-09 14:09:34 +0900362 variable_name);
Heinrich Schuchardtab7296c2020-07-04 18:34:15 +0200363 return EFI_INVALID_PARAMETER;
AKASHI Takahiro767f6ee2020-04-14 11:51:41 +0900364 }
365 }
366
367 /* authenticate a variable */
368 if (IS_ENABLED(CONFIG_EFI_SECURE_BOOT)) {
Heinrich Schuchardtab7296c2020-07-04 18:34:15 +0200369 if (attributes & EFI_VARIABLE_AUTHENTICATED_WRITE_ACCESS)
370 return EFI_INVALID_PARAMETER;
AKASHI Takahiro767f6ee2020-04-14 11:51:41 +0900371 if (attributes &
372 EFI_VARIABLE_TIME_BASED_AUTHENTICATED_WRITE_ACCESS) {
Heinrich Schuchardtab7296c2020-07-04 18:34:15 +0200373 u32 env_attr;
374
AKASHI Takahiro767f6ee2020-04-14 11:51:41 +0900375 ret = efi_variable_authenticate(variable_name, vendor,
376 &data_size, &data,
Heinrich Schuchardtab7296c2020-07-04 18:34:15 +0200377 attributes, &env_attr,
AKASHI Takahiro767f6ee2020-04-14 11:51:41 +0900378 &time);
379 if (ret != EFI_SUCCESS)
Heinrich Schuchardtab7296c2020-07-04 18:34:15 +0200380 return ret;
AKASHI Takahiro767f6ee2020-04-14 11:51:41 +0900381
382 /* last chance to check for delete */
383 if (!data_size)
384 delete = true;
385 }
386 } else {
387 if (attributes &
388 (EFI_VARIABLE_AUTHENTICATED_WRITE_ACCESS |
389 EFI_VARIABLE_TIME_BASED_AUTHENTICATED_WRITE_ACCESS)) {
AKASHI Takahiroce3c3862020-06-09 14:09:34 +0900390 EFI_PRINT("Secure boot is not configured\n");
Heinrich Schuchardtab7296c2020-07-04 18:34:15 +0200391 return EFI_INVALID_PARAMETER;
AKASHI Takahiro767f6ee2020-04-14 11:51:41 +0900392 }
393 }
394
AKASHI Takahiro767f6ee2020-04-14 11:51:41 +0900395 if (delete) {
Heinrich Schuchardtab7296c2020-07-04 18:34:15 +0200396 /* EFI_NOT_FOUND has been handled before */
AKASHI Takahiro767f6ee2020-04-14 11:51:41 +0900397 ret = EFI_SUCCESS;
Heinrich Schuchardtab7296c2020-07-04 18:34:15 +0200398 } else if (append) {
399 u16 *old_data = var->name;
AKASHI Takahiro767f6ee2020-04-14 11:51:41 +0900400
Heinrich Schuchardtab7296c2020-07-04 18:34:15 +0200401 for (; *old_data; ++old_data)
402 ;
403 ++old_data;
404 ret = efi_var_mem_ins(variable_name, vendor, attributes,
405 var->length, old_data, data_size, data,
406 time);
AKASHI Takahiro767f6ee2020-04-14 11:51:41 +0900407 } else {
Heinrich Schuchardtab7296c2020-07-04 18:34:15 +0200408 ret = efi_var_mem_ins(variable_name, vendor, attributes,
409 data_size, data, 0, NULL, time);
AKASHI Takahiro767f6ee2020-04-14 11:51:41 +0900410 }
Heinrich Schuchardtab7296c2020-07-04 18:34:15 +0200411 efi_var_mem_del(var);
AKASHI Takahiro767f6ee2020-04-14 11:51:41 +0900412
Heinrich Schuchardtab7296c2020-07-04 18:34:15 +0200413 if (ret != EFI_SUCCESS)
414 return ret;
AKASHI Takahiro767f6ee2020-04-14 11:51:41 +0900415
Heinrich Schuchardt99bfab82020-07-15 12:40:35 +0200416 if (var_type == EFI_AUTH_VAR_PK)
Heinrich Schuchardtab7296c2020-07-04 18:34:15 +0200417 ret = efi_init_secure_state();
418 else
419 ret = EFI_SUCCESS;
AKASHI Takahiro767f6ee2020-04-14 11:51:41 +0900420
Heinrich Schuchardt5f7dcf02020-03-19 18:21:58 +0000421 /* Write non-volatile EFI variables to file */
422 if (attributes & EFI_VARIABLE_NON_VOLATILE &&
423 ret == EFI_SUCCESS && efi_obj_list_initialized == EFI_SUCCESS)
424 efi_var_to_file();
425
Heinrich Schuchardtab7296c2020-07-04 18:34:15 +0200426 return EFI_SUCCESS;
AKASHI Takahiro767f6ee2020-04-14 11:51:41 +0900427}
428
Heinrich Schuchardt01df8cf2020-06-26 17:57:48 +0200429efi_status_t efi_query_variable_info_int(u32 attributes,
430 u64 *maximum_variable_storage_size,
431 u64 *remaining_variable_storage_size,
432 u64 *maximum_variable_size)
433{
Heinrich Schuchardtab7296c2020-07-04 18:34:15 +0200434 *maximum_variable_storage_size = EFI_VAR_BUF_SIZE -
435 sizeof(struct efi_var_file);
436 *remaining_variable_storage_size = efi_var_mem_free();
437 *maximum_variable_size = EFI_VAR_BUF_SIZE -
438 sizeof(struct efi_var_file) -
439 sizeof(struct efi_var_entry);
440 return EFI_SUCCESS;
Heinrich Schuchardt01df8cf2020-06-26 17:57:48 +0200441}
442
Heinrich Schuchardt77d4d392019-01-18 19:52:05 +0100443/**
Heinrich Schuchardt01df8cf2020-06-26 17:57:48 +0200444 * efi_query_variable_info_runtime() - runtime implementation of
445 * QueryVariableInfo()
Heinrich Schuchardtce435282019-06-20 12:13:05 +0200446 *
447 * @attributes: bitmask to select variables to be
448 * queried
449 * @maximum_variable_storage_size: maximum size of storage area for the
450 * selected variable types
451 * @remaining_variable_storage_size: remaining size of storage are for the
452 * selected variable types
453 * @maximum_variable_size: maximum size of a variable of the
454 * selected type
455 * Returns: status code
456 */
Heinrich Schuchardt01df8cf2020-06-26 17:57:48 +0200457efi_status_t __efi_runtime EFIAPI efi_query_variable_info_runtime(
Heinrich Schuchardtce435282019-06-20 12:13:05 +0200458 u32 attributes,
459 u64 *maximum_variable_storage_size,
460 u64 *remaining_variable_storage_size,
461 u64 *maximum_variable_size)
462{
463 return EFI_UNSUPPORTED;
464}
Heinrich Schuchardt88192092019-06-20 13:52:16 +0200465
466/**
Heinrich Schuchardt29018ab2019-06-20 15:25:48 +0200467 * efi_set_variable_runtime() - runtime implementation of SetVariable()
Heinrich Schuchardte5b44622019-07-14 12:11:16 +0200468 *
469 * @variable_name: name of the variable
470 * @vendor: vendor GUID
471 * @attributes: attributes of the variable
472 * @data_size: size of the buffer with the variable value
473 * @data: buffer with the variable value
474 * Return: status code
Heinrich Schuchardt29018ab2019-06-20 15:25:48 +0200475 */
476static efi_status_t __efi_runtime EFIAPI
477efi_set_variable_runtime(u16 *variable_name, const efi_guid_t *vendor,
478 u32 attributes, efi_uintn_t data_size,
479 const void *data)
480{
481 return EFI_UNSUPPORTED;
482}
483
484/**
485 * efi_variables_boot_exit_notify() - notify ExitBootServices() is called
486 */
487void efi_variables_boot_exit_notify(void)
488{
Heinrich Schuchardt5f7dcf02020-03-19 18:21:58 +0000489 /* Switch variable services functions to runtime version */
Heinrich Schuchardt29018ab2019-06-20 15:25:48 +0200490 efi_runtime_services.get_variable = efi_get_variable_runtime;
491 efi_runtime_services.get_next_variable_name =
492 efi_get_next_variable_name_runtime;
493 efi_runtime_services.set_variable = efi_set_variable_runtime;
Heinrich Schuchardt01df8cf2020-06-26 17:57:48 +0200494 efi_runtime_services.query_variable_info =
495 efi_query_variable_info_runtime;
Heinrich Schuchardt29018ab2019-06-20 15:25:48 +0200496 efi_update_table_header_crc32(&efi_runtime_services.hdr);
497}
498
499/**
Heinrich Schuchardt88192092019-06-20 13:52:16 +0200500 * efi_init_variables() - initialize variable services
501 *
502 * Return: status code
503 */
504efi_status_t efi_init_variables(void)
505{
AKASHI Takahiro434ffb62020-04-14 11:51:42 +0900506 efi_status_t ret;
507
Heinrich Schuchardtab7296c2020-07-04 18:34:15 +0200508 ret = efi_var_mem_init();
509 if (ret != EFI_SUCCESS)
510 return ret;
511
AKASHI Takahiro434ffb62020-04-14 11:51:42 +0900512 ret = efi_init_secure_state();
Heinrich Schuchardt5f7dcf02020-03-19 18:21:58 +0000513 if (ret != EFI_SUCCESS)
514 return ret;
AKASHI Takahiro434ffb62020-04-14 11:51:42 +0900515
Heinrich Schuchardt7dda1632020-07-14 21:25:28 +0200516 if (IS_ENABLED(CONFIG_EFI_VARIABLES_PRESEED)) {
517 ret = efi_var_restore((struct efi_var_file *)
518 __efi_var_file_begin);
519 if (ret != EFI_SUCCESS)
520 log_err("Invalid EFI variable seed\n");
521 }
522
Heinrich Schuchardt5f7dcf02020-03-19 18:21:58 +0000523 return efi_var_from_file();
Heinrich Schuchardt88192092019-06-20 13:52:16 +0200524}