blob: 44235970a7c79c1d2691c76aa60641b706c478c9 [file] [log] [blame]
Leif Lindholm5aab1b82019-01-21 12:12:57 +09001// SPDX-License-Identifier: GPL-2.0+
2/*
3 * EFI Human Interface Infrastructure ... database and packages
4 *
5 * Copyright (c) 2017 Leif Lindholm
6 * Copyright (c) 2018 AKASHI Takahiro, Linaro Limited
7 */
8
Heinrich Schuchardt955a3212025-01-16 20:26:59 +01009#define LOG_CATEGORY LOGC_EFI
10
Leif Lindholm5aab1b82019-01-21 12:12:57 +090011#include <efi_loader.h>
12#include <malloc.h>
13#include <asm/unaligned.h>
14
15const efi_guid_t efi_guid_hii_database_protocol
16 = EFI_HII_DATABASE_PROTOCOL_GUID;
17const efi_guid_t efi_guid_hii_string_protocol = EFI_HII_STRING_PROTOCOL_GUID;
18
19static LIST_HEAD(efi_package_lists);
AKASHI Takahiro976e9312019-01-21 12:12:59 +090020static LIST_HEAD(efi_keyboard_layout_list);
Leif Lindholm5aab1b82019-01-21 12:12:57 +090021
22struct efi_hii_packagelist {
23 struct list_head link;
24 // TODO should there be an associated efi_object?
25 efi_handle_t driver_handle;
26 u32 max_string_id;
27 struct list_head string_tables; /* list of efi_string_table */
AKASHI Takahiro0fa82b92019-01-21 12:12:58 +090028 struct list_head guid_list;
AKASHI Takahiro976e9312019-01-21 12:12:59 +090029 struct list_head keyboard_packages;
Leif Lindholm5aab1b82019-01-21 12:12:57 +090030
31 /* we could also track fonts, images, etc */
32};
33
34static int efi_hii_packagelist_exists(efi_hii_handle_t package_list)
35{
36 struct efi_hii_packagelist *hii;
37 int found = 0;
38
39 list_for_each_entry(hii, &efi_package_lists, link) {
40 if (hii == package_list) {
41 found = 1;
42 break;
43 }
44 }
45
46 return found;
47}
48
49static u32 efi_hii_package_type(struct efi_hii_package_header *header)
50{
51 u32 fields;
52
53 fields = get_unaligned_le32(&header->fields);
54
55 return (fields >> __EFI_HII_PACKAGE_TYPE_SHIFT)
56 & __EFI_HII_PACKAGE_TYPE_MASK;
57}
58
59static u32 efi_hii_package_len(struct efi_hii_package_header *header)
60{
61 u32 fields;
62
63 fields = get_unaligned_le32(&header->fields);
64
65 return (fields >> __EFI_HII_PACKAGE_LEN_SHIFT)
66 & __EFI_HII_PACKAGE_LEN_MASK;
67}
68
69struct efi_string_info {
70 efi_string_t string;
71 /* we could also track font info, etc */
72};
73
74struct efi_string_table {
75 struct list_head link;
76 efi_string_id_t language_name;
77 char *language;
78 u32 nstrings;
79 /*
80 * NOTE:
81 * string id starts at 1 so value is stbl->strings[id-1],
82 * and strings[] is a array of stbl->nstrings elements
83 */
84 struct efi_string_info *strings;
85};
86
AKASHI Takahiro0fa82b92019-01-21 12:12:58 +090087struct efi_guid_data {
88 struct list_head link;
89 struct efi_hii_guid_package package;
90};
91
AKASHI Takahiro976e9312019-01-21 12:12:59 +090092struct efi_keyboard_layout_data {
93 struct list_head link; /* in package */
94 struct list_head link_sys; /* in global list */
95 struct efi_hii_keyboard_layout keyboard_layout;
96};
97
98struct efi_keyboard_package_data {
99 struct list_head link; /* in package_list */
100 struct list_head keyboard_layout_list;
101};
102
Leif Lindholm5aab1b82019-01-21 12:12:57 +0900103static void free_strings_table(struct efi_string_table *stbl)
104{
105 int i;
106
107 for (i = 0; i < stbl->nstrings; i++)
108 free(stbl->strings[i].string);
109 free(stbl->strings);
110 free(stbl->language);
111 free(stbl);
112}
113
114static void remove_strings_package(struct efi_hii_packagelist *hii)
115{
116 while (!list_empty(&hii->string_tables)) {
117 struct efi_string_table *stbl;
118
119 stbl = list_first_entry(&hii->string_tables,
120 struct efi_string_table, link);
121 list_del(&stbl->link);
122 free_strings_table(stbl);
123 }
124}
125
126static efi_status_t
127add_strings_package(struct efi_hii_packagelist *hii,
128 struct efi_hii_strings_package *strings_package)
129{
130 struct efi_hii_string_block *block;
131 void *end;
132 u32 nstrings = 0, idx = 0;
133 struct efi_string_table *stbl = NULL;
134 efi_status_t ret;
135
Heinrich Schuchardt4b6f19f2019-01-23 22:55:36 +0100136 EFI_PRINT("header_size: %08x\n",
137 get_unaligned_le32(&strings_package->header_size));
138 EFI_PRINT("string_info_offset: %08x\n",
139 get_unaligned_le32(&strings_package->string_info_offset));
140 EFI_PRINT("language_name: %u\n",
141 get_unaligned_le16(&strings_package->language_name));
142 EFI_PRINT("language: %s\n", strings_package->language);
Leif Lindholm5aab1b82019-01-21 12:12:57 +0900143
144 /* count # of string entries: */
145 end = ((void *)strings_package)
146 + efi_hii_package_len(&strings_package->header);
147 block = ((void *)strings_package)
148 + get_unaligned_le32(&strings_package->string_info_offset);
149
150 while ((void *)block < end) {
151 switch (block->block_type) {
152 case EFI_HII_SIBT_STRING_UCS2: {
153 struct efi_hii_sibt_string_ucs2_block *ucs2;
154
155 ucs2 = (void *)block;
156 nstrings++;
157 block = efi_hii_sibt_string_ucs2_block_next(ucs2);
158 break;
159 }
160 case EFI_HII_SIBT_END:
161 block = end;
162 break;
163 default:
Heinrich Schuchardt4b6f19f2019-01-23 22:55:36 +0100164 EFI_PRINT("unknown HII string block type: %02x\n",
165 block->block_type);
Leif Lindholm5aab1b82019-01-21 12:12:57 +0900166 return EFI_INVALID_PARAMETER;
167 }
168 }
169
170 stbl = calloc(sizeof(*stbl), 1);
171 if (!stbl) {
172 ret = EFI_OUT_OF_RESOURCES;
173 goto error;
174 }
175 stbl->strings = calloc(sizeof(stbl->strings[0]), nstrings);
176 if (!stbl->strings) {
177 ret = EFI_OUT_OF_RESOURCES;
178 goto error;
179 }
180 stbl->language_name =
181 get_unaligned_le16(&strings_package->language_name);
182 stbl->language = strdup((char *)strings_package->language);
183 if (!stbl->language) {
184 ret = EFI_OUT_OF_RESOURCES;
185 goto error;
186 }
187 stbl->nstrings = nstrings;
188
189 /* and now parse string entries and populate efi_string_table */
190 block = ((void *)strings_package)
191 + get_unaligned_le32(&strings_package->string_info_offset);
192
193 while ((void *)block < end) {
194 switch (block->block_type) {
195 case EFI_HII_SIBT_STRING_UCS2: {
196 struct efi_hii_sibt_string_ucs2_block *ucs2;
197
198 ucs2 = (void *)block;
Heinrich Schuchardt4b6f19f2019-01-23 22:55:36 +0100199 EFI_PRINT("%4u: \"%ls\"\n", idx + 1, ucs2->string_text);
Leif Lindholm5aab1b82019-01-21 12:12:57 +0900200 stbl->strings[idx].string =
201 u16_strdup(ucs2->string_text);
202 if (!stbl->strings[idx].string) {
203 ret = EFI_OUT_OF_RESOURCES;
204 goto error;
205 }
206 idx++;
207 /* FIXME: accessing u16 * here */
208 block = efi_hii_sibt_string_ucs2_block_next(ucs2);
209 break;
210 }
211 case EFI_HII_SIBT_END:
212 goto out;
213 default:
Heinrich Schuchardt4b6f19f2019-01-23 22:55:36 +0100214 EFI_PRINT("unknown HII string block type: %02x\n",
215 block->block_type);
Leif Lindholm5aab1b82019-01-21 12:12:57 +0900216 ret = EFI_INVALID_PARAMETER;
217 goto error;
218 }
219 }
220
221out:
222 list_add(&stbl->link, &hii->string_tables);
223 if (hii->max_string_id < nstrings)
224 hii->max_string_id = nstrings;
225
226 return EFI_SUCCESS;
227
228error:
229 if (stbl) {
230 free(stbl->language);
Heinrich Schuchardtd177dbd2019-03-19 12:30:27 +0100231 while (idx > 0)
232 free(stbl->strings[--idx].string);
Leif Lindholm5aab1b82019-01-21 12:12:57 +0900233 free(stbl->strings);
234 }
235 free(stbl);
236
237 return ret;
238}
239
AKASHI Takahiro0fa82b92019-01-21 12:12:58 +0900240static void remove_guid_package(struct efi_hii_packagelist *hii)
241{
242 struct efi_guid_data *data;
243
244 while (!list_empty(&hii->guid_list)) {
245 data = list_first_entry(&hii->guid_list,
246 struct efi_guid_data, link);
247 list_del(&data->link);
248 free(data);
249 }
250}
251
252static efi_status_t
253add_guid_package(struct efi_hii_packagelist *hii,
254 struct efi_hii_guid_package *package)
255{
256 struct efi_guid_data *data;
257
258 data = calloc(sizeof(*data), 1);
259 if (!data)
260 return EFI_OUT_OF_RESOURCES;
261
262 /* TODO: we don't know any about data field */
263 memcpy(&data->package, package, sizeof(*package));
264 list_add_tail(&data->link, &hii->guid_list);
265
266 return EFI_SUCCESS;
AKASHI Takahiro976e9312019-01-21 12:12:59 +0900267}
268
269static void free_keyboard_layouts(struct efi_keyboard_package_data *package)
270{
271 struct efi_keyboard_layout_data *layout_data;
272
273 while (!list_empty(&package->keyboard_layout_list)) {
274 layout_data = list_first_entry(&package->keyboard_layout_list,
275 struct efi_keyboard_layout_data,
276 link);
277 list_del(&layout_data->link);
278 list_del(&layout_data->link_sys);
279 free(layout_data);
280 }
281}
282
283static void remove_keyboard_package(struct efi_hii_packagelist *hii)
284{
285 struct efi_keyboard_package_data *package;
286
287 while (!list_empty(&hii->keyboard_packages)) {
288 package = list_first_entry(&hii->keyboard_packages,
289 struct efi_keyboard_package_data,
290 link);
291 free_keyboard_layouts(package);
292 list_del(&package->link);
293 free(package);
294 }
295}
296
297static efi_status_t
298add_keyboard_package(struct efi_hii_packagelist *hii,
299 struct efi_hii_keyboard_package *keyboard_package)
300{
301 struct efi_keyboard_package_data *package_data;
302 struct efi_hii_keyboard_layout *layout;
303 struct efi_keyboard_layout_data *layout_data;
304 u16 layout_count, layout_length;
305 int i;
306
307 package_data = malloc(sizeof(*package_data));
308 if (!package_data)
309 return EFI_OUT_OF_RESOURCES;
310 INIT_LIST_HEAD(&package_data->link);
311 INIT_LIST_HEAD(&package_data->keyboard_layout_list);
312
313 layout = &keyboard_package->layout[0];
314 layout_count = get_unaligned_le16(&keyboard_package->layout_count);
315 for (i = 0; i < layout_count; i++) {
316 layout_length = get_unaligned_le16(&layout->layout_length);
317 layout_data = malloc(sizeof(*layout_data) + layout_length);
318 if (!layout_data)
319 goto out;
320
321 memcpy(&layout_data->keyboard_layout, layout, layout_length);
322 list_add_tail(&layout_data->link,
323 &package_data->keyboard_layout_list);
324 list_add_tail(&layout_data->link_sys,
325 &efi_keyboard_layout_list);
326
327 layout += layout_length;
328 }
329
330 list_add_tail(&package_data->link, &hii->keyboard_packages);
331
332 return EFI_SUCCESS;
333
334out:
335 free_keyboard_layouts(package_data);
336 free(package_data);
337
338 return EFI_OUT_OF_RESOURCES;
AKASHI Takahiro0fa82b92019-01-21 12:12:58 +0900339}
340
Leif Lindholm5aab1b82019-01-21 12:12:57 +0900341static struct efi_hii_packagelist *new_packagelist(void)
342{
343 struct efi_hii_packagelist *hii;
344
345 hii = malloc(sizeof(*hii));
Heinrich Schuchardt87895f62019-02-28 23:07:00 +0100346 list_add_tail(&hii->link, &efi_package_lists);
Leif Lindholm5aab1b82019-01-21 12:12:57 +0900347 hii->max_string_id = 0;
348 INIT_LIST_HEAD(&hii->string_tables);
AKASHI Takahiro0fa82b92019-01-21 12:12:58 +0900349 INIT_LIST_HEAD(&hii->guid_list);
AKASHI Takahiro976e9312019-01-21 12:12:59 +0900350 INIT_LIST_HEAD(&hii->keyboard_packages);
Leif Lindholm5aab1b82019-01-21 12:12:57 +0900351
352 return hii;
353}
354
355static void free_packagelist(struct efi_hii_packagelist *hii)
356{
357 remove_strings_package(hii);
AKASHI Takahiro0fa82b92019-01-21 12:12:58 +0900358 remove_guid_package(hii);
AKASHI Takahiro976e9312019-01-21 12:12:59 +0900359 remove_keyboard_package(hii);
Leif Lindholm5aab1b82019-01-21 12:12:57 +0900360
361 list_del(&hii->link);
362 free(hii);
363}
364
365static efi_status_t
366add_packages(struct efi_hii_packagelist *hii,
367 const struct efi_hii_package_list_header *package_list)
368{
369 struct efi_hii_package_header *package;
370 void *end;
371 efi_status_t ret = EFI_SUCCESS;
372
373 end = ((void *)package_list)
374 + get_unaligned_le32(&package_list->package_length);
375
Heinrich Schuchardt282249d2022-01-16 14:15:31 +0100376 EFI_PRINT("package_list: %pUs (%u)\n", &package_list->package_list_guid,
Heinrich Schuchardt4b6f19f2019-01-23 22:55:36 +0100377 get_unaligned_le32(&package_list->package_length));
Leif Lindholm5aab1b82019-01-21 12:12:57 +0900378
379 package = ((void *)package_list) + sizeof(*package_list);
380 while ((void *)package < end) {
Heinrich Schuchardt4b6f19f2019-01-23 22:55:36 +0100381 EFI_PRINT("package=%p, package type=%x, length=%u\n", package,
382 efi_hii_package_type(package),
383 efi_hii_package_len(package));
Leif Lindholm5aab1b82019-01-21 12:12:57 +0900384
385 switch (efi_hii_package_type(package)) {
386 case EFI_HII_PACKAGE_TYPE_GUID:
AKASHI Takahiro0fa82b92019-01-21 12:12:58 +0900387 ret = add_guid_package(hii,
388 (struct efi_hii_guid_package *)package);
Leif Lindholm5aab1b82019-01-21 12:12:57 +0900389 break;
390 case EFI_HII_PACKAGE_FORMS:
Heinrich Schuchardt196e2122019-02-28 23:56:35 +0100391 EFI_PRINT("Form package not supported\n");
Leif Lindholm5aab1b82019-01-21 12:12:57 +0900392 ret = EFI_INVALID_PARAMETER;
393 break;
394 case EFI_HII_PACKAGE_STRINGS:
395 ret = add_strings_package(hii,
396 (struct efi_hii_strings_package *)package);
397 break;
398 case EFI_HII_PACKAGE_FONTS:
Heinrich Schuchardt196e2122019-02-28 23:56:35 +0100399 EFI_PRINT("Font package not supported\n");
Leif Lindholm5aab1b82019-01-21 12:12:57 +0900400 ret = EFI_INVALID_PARAMETER;
401 break;
402 case EFI_HII_PACKAGE_IMAGES:
Heinrich Schuchardt196e2122019-02-28 23:56:35 +0100403 EFI_PRINT("Image package not supported\n");
Leif Lindholm5aab1b82019-01-21 12:12:57 +0900404 ret = EFI_INVALID_PARAMETER;
405 break;
406 case EFI_HII_PACKAGE_SIMPLE_FONTS:
Heinrich Schuchardt196e2122019-02-28 23:56:35 +0100407 EFI_PRINT("Simple font package not supported\n");
Leif Lindholm5aab1b82019-01-21 12:12:57 +0900408 ret = EFI_INVALID_PARAMETER;
409 break;
410 case EFI_HII_PACKAGE_DEVICE_PATH:
Heinrich Schuchardt196e2122019-02-28 23:56:35 +0100411 EFI_PRINT("Device path package not supported\n");
Leif Lindholm5aab1b82019-01-21 12:12:57 +0900412 ret = EFI_INVALID_PARAMETER;
413 break;
414 case EFI_HII_PACKAGE_KEYBOARD_LAYOUT:
AKASHI Takahiro976e9312019-01-21 12:12:59 +0900415 ret = add_keyboard_package(hii,
416 (struct efi_hii_keyboard_package *)package);
Leif Lindholm5aab1b82019-01-21 12:12:57 +0900417 break;
418 case EFI_HII_PACKAGE_ANIMATIONS:
Heinrich Schuchardt196e2122019-02-28 23:56:35 +0100419 EFI_PRINT("Animation package not supported\n");
Leif Lindholm5aab1b82019-01-21 12:12:57 +0900420 ret = EFI_INVALID_PARAMETER;
421 break;
422 case EFI_HII_PACKAGE_END:
423 goto out;
424 case EFI_HII_PACKAGE_TYPE_SYSTEM_BEGIN:
425 case EFI_HII_PACKAGE_TYPE_SYSTEM_END:
426 default:
427 break;
428 }
429
430 if (ret != EFI_SUCCESS)
431 return ret;
432
433 package = (void *)package + efi_hii_package_len(package);
434 }
435out:
436 // TODO in theory there is some notifications that should be sent..
437 return EFI_SUCCESS;
438}
439
440/*
441 * EFI_HII_DATABASE_PROTOCOL
442 */
443
444static efi_status_t EFIAPI
445new_package_list(const struct efi_hii_database_protocol *this,
446 const struct efi_hii_package_list_header *package_list,
447 const efi_handle_t driver_handle,
448 efi_hii_handle_t *handle)
449{
450 struct efi_hii_packagelist *hii;
451 efi_status_t ret;
452
453 EFI_ENTRY("%p, %p, %p, %p", this, package_list, driver_handle, handle);
454
455 if (!package_list || !handle)
456 return EFI_EXIT(EFI_INVALID_PARAMETER);
457
458 hii = new_packagelist();
459 if (!hii)
460 return EFI_EXIT(EFI_OUT_OF_RESOURCES);
461
462 ret = add_packages(hii, package_list);
463 if (ret != EFI_SUCCESS) {
464 free_packagelist(hii);
465 return EFI_EXIT(ret);
466 }
467
468 hii->driver_handle = driver_handle;
Leif Lindholm5aab1b82019-01-21 12:12:57 +0900469 *handle = hii;
470
471 return EFI_EXIT(EFI_SUCCESS);
472}
473
474static efi_status_t EFIAPI
475remove_package_list(const struct efi_hii_database_protocol *this,
476 efi_hii_handle_t handle)
477{
478 struct efi_hii_packagelist *hii = handle;
479
480 EFI_ENTRY("%p, %p", this, handle);
481
482 if (!handle || !efi_hii_packagelist_exists(handle))
483 return EFI_EXIT(EFI_NOT_FOUND);
484
485 free_packagelist(hii);
486
487 return EFI_EXIT(EFI_SUCCESS);
488}
489
490static efi_status_t EFIAPI
491update_package_list(const struct efi_hii_database_protocol *this,
492 efi_hii_handle_t handle,
493 const struct efi_hii_package_list_header *package_list)
494{
495 struct efi_hii_packagelist *hii = handle;
496 struct efi_hii_package_header *package;
497 void *end;
498 efi_status_t ret = EFI_SUCCESS;
499
500 EFI_ENTRY("%p, %p, %p", this, handle, package_list);
501
502 if (!handle || !efi_hii_packagelist_exists(handle))
503 return EFI_EXIT(EFI_NOT_FOUND);
504
505 if (!package_list)
506 return EFI_EXIT(EFI_INVALID_PARAMETER);
507
Heinrich Schuchardt282249d2022-01-16 14:15:31 +0100508 EFI_PRINT("package_list: %pUs (%u)\n", &package_list->package_list_guid,
Heinrich Schuchardt4b6f19f2019-01-23 22:55:36 +0100509 get_unaligned_le32(&package_list->package_length));
Leif Lindholm5aab1b82019-01-21 12:12:57 +0900510
511 package = ((void *)package_list) + sizeof(*package_list);
512 end = ((void *)package_list)
513 + get_unaligned_le32(&package_list->package_length);
514
515 while ((void *)package < end) {
Heinrich Schuchardt4b6f19f2019-01-23 22:55:36 +0100516 EFI_PRINT("package=%p, package type=%x, length=%u\n", package,
517 efi_hii_package_type(package),
518 efi_hii_package_len(package));
Leif Lindholm5aab1b82019-01-21 12:12:57 +0900519
520 switch (efi_hii_package_type(package)) {
521 case EFI_HII_PACKAGE_TYPE_GUID:
AKASHI Takahiro0fa82b92019-01-21 12:12:58 +0900522 remove_guid_package(hii);
Leif Lindholm5aab1b82019-01-21 12:12:57 +0900523 break;
524 case EFI_HII_PACKAGE_FORMS:
Heinrich Schuchardt196e2122019-02-28 23:56:35 +0100525 EFI_PRINT("Form package not supported\n");
Leif Lindholm5aab1b82019-01-21 12:12:57 +0900526 ret = EFI_INVALID_PARAMETER;
527 break;
528 case EFI_HII_PACKAGE_STRINGS:
529 remove_strings_package(hii);
530 break;
531 case EFI_HII_PACKAGE_FONTS:
Heinrich Schuchardt196e2122019-02-28 23:56:35 +0100532 EFI_PRINT("Font package not supported\n");
Leif Lindholm5aab1b82019-01-21 12:12:57 +0900533 ret = EFI_INVALID_PARAMETER;
534 break;
535 case EFI_HII_PACKAGE_IMAGES:
Heinrich Schuchardt196e2122019-02-28 23:56:35 +0100536 EFI_PRINT("Image package not supported\n");
Leif Lindholm5aab1b82019-01-21 12:12:57 +0900537 ret = EFI_INVALID_PARAMETER;
538 break;
539 case EFI_HII_PACKAGE_SIMPLE_FONTS:
Heinrich Schuchardt196e2122019-02-28 23:56:35 +0100540 EFI_PRINT("Simple font package not supported\n");
Leif Lindholm5aab1b82019-01-21 12:12:57 +0900541 ret = EFI_INVALID_PARAMETER;
542 break;
543 case EFI_HII_PACKAGE_DEVICE_PATH:
Heinrich Schuchardt196e2122019-02-28 23:56:35 +0100544 EFI_PRINT("Device path package not supported\n");
Leif Lindholm5aab1b82019-01-21 12:12:57 +0900545 ret = EFI_INVALID_PARAMETER;
546 break;
547 case EFI_HII_PACKAGE_KEYBOARD_LAYOUT:
AKASHI Takahiro976e9312019-01-21 12:12:59 +0900548 remove_keyboard_package(hii);
Leif Lindholm5aab1b82019-01-21 12:12:57 +0900549 break;
550 case EFI_HII_PACKAGE_ANIMATIONS:
Heinrich Schuchardt196e2122019-02-28 23:56:35 +0100551 EFI_PRINT("Animation package not supported\n");
Leif Lindholm5aab1b82019-01-21 12:12:57 +0900552 ret = EFI_INVALID_PARAMETER;
553 break;
554 case EFI_HII_PACKAGE_END:
555 goto out;
556 case EFI_HII_PACKAGE_TYPE_SYSTEM_BEGIN:
557 case EFI_HII_PACKAGE_TYPE_SYSTEM_END:
558 default:
559 break;
560 }
561
562 /* TODO: already removed some packages */
563 if (ret != EFI_SUCCESS)
564 return EFI_EXIT(ret);
565
566 package = ((void *)package)
567 + efi_hii_package_len(package);
568 }
569out:
570 ret = add_packages(hii, package_list);
571
572 return EFI_EXIT(ret);
573}
574
575static efi_status_t EFIAPI
576list_package_lists(const struct efi_hii_database_protocol *this,
577 u8 package_type,
578 const efi_guid_t *package_guid,
579 efi_uintn_t *handle_buffer_length,
580 efi_hii_handle_t *handle)
581{
582 struct efi_hii_packagelist *hii =
583 (struct efi_hii_packagelist *)handle;
584 int package_cnt, package_max;
Heinrich Schuchardtbd425182019-06-17 20:46:29 +0200585 efi_status_t ret = EFI_NOT_FOUND;
Leif Lindholm5aab1b82019-01-21 12:12:57 +0900586
Heinrich Schuchardt282249d2022-01-16 14:15:31 +0100587 EFI_ENTRY("%p, %u, %pUs, %p, %p", this, package_type, package_guid,
Leif Lindholm5aab1b82019-01-21 12:12:57 +0900588 handle_buffer_length, handle);
589
590 if (!handle_buffer_length ||
Heinrich Schuchardtbd425182019-06-17 20:46:29 +0200591 (*handle_buffer_length && !handle)) {
592 ret = EFI_INVALID_PARAMETER;
593 goto out;
594 }
Leif Lindholm5aab1b82019-01-21 12:12:57 +0900595
596 if ((package_type != EFI_HII_PACKAGE_TYPE_GUID && package_guid) ||
Heinrich Schuchardtbd425182019-06-17 20:46:29 +0200597 (package_type == EFI_HII_PACKAGE_TYPE_GUID && !package_guid)) {
598 ret = EFI_INVALID_PARAMETER;
599 goto out;
600 }
Leif Lindholm5aab1b82019-01-21 12:12:57 +0900601
Heinrich Schuchardt282249d2022-01-16 14:15:31 +0100602 EFI_PRINT("package type=%x, guid=%pUs, length=%zu\n", (int)package_type,
Heinrich Schuchardt4b6f19f2019-01-23 22:55:36 +0100603 package_guid, *handle_buffer_length);
Leif Lindholm5aab1b82019-01-21 12:12:57 +0900604
605 package_cnt = 0;
606 package_max = *handle_buffer_length / sizeof(*handle);
607 list_for_each_entry(hii, &efi_package_lists, link) {
608 switch (package_type) {
609 case EFI_HII_PACKAGE_TYPE_ALL:
610 break;
611 case EFI_HII_PACKAGE_TYPE_GUID:
AKASHI Takahiro0fa82b92019-01-21 12:12:58 +0900612 if (!list_empty(&hii->guid_list))
613 break;
Leif Lindholm5aab1b82019-01-21 12:12:57 +0900614 continue;
Leif Lindholm5aab1b82019-01-21 12:12:57 +0900615 case EFI_HII_PACKAGE_STRINGS:
616 if (!list_empty(&hii->string_tables))
617 break;
618 continue;
Leif Lindholm5aab1b82019-01-21 12:12:57 +0900619 case EFI_HII_PACKAGE_KEYBOARD_LAYOUT:
AKASHI Takahiro976e9312019-01-21 12:12:59 +0900620 if (!list_empty(&hii->keyboard_packages))
621 break;
Leif Lindholm5aab1b82019-01-21 12:12:57 +0900622 continue;
Leif Lindholm5aab1b82019-01-21 12:12:57 +0900623 default:
624 continue;
625 }
626
627 package_cnt++;
Heinrich Schuchardtbd425182019-06-17 20:46:29 +0200628 if (package_cnt <= package_max) {
Leif Lindholm5aab1b82019-01-21 12:12:57 +0900629 *handle++ = hii;
Heinrich Schuchardtbd425182019-06-17 20:46:29 +0200630 ret = EFI_SUCCESS;
631 } else {
Leif Lindholm5aab1b82019-01-21 12:12:57 +0900632 ret = EFI_BUFFER_TOO_SMALL;
Heinrich Schuchardtbd425182019-06-17 20:46:29 +0200633 }
Leif Lindholm5aab1b82019-01-21 12:12:57 +0900634 }
635 *handle_buffer_length = package_cnt * sizeof(*handle);
Heinrich Schuchardtbd425182019-06-17 20:46:29 +0200636out:
Leif Lindholm5aab1b82019-01-21 12:12:57 +0900637 return EFI_EXIT(ret);
638}
639
640static efi_status_t EFIAPI
641export_package_lists(const struct efi_hii_database_protocol *this,
642 efi_hii_handle_t handle,
643 efi_uintn_t *buffer_size,
644 struct efi_hii_package_list_header *buffer)
645{
646 EFI_ENTRY("%p, %p, %p, %p", this, handle, buffer_size, buffer);
647
648 if (!buffer_size || !buffer)
649 return EFI_EXIT(EFI_INVALID_PARAMETER);
650
651 return EFI_EXIT(EFI_NOT_FOUND);
652}
653
654static efi_status_t EFIAPI
655register_package_notify(const struct efi_hii_database_protocol *this,
656 u8 package_type,
657 const efi_guid_t *package_guid,
658 const void *package_notify_fn,
659 efi_uintn_t notify_type,
660 efi_handle_t *notify_handle)
661{
Heinrich Schuchardt282249d2022-01-16 14:15:31 +0100662 EFI_ENTRY("%p, %u, %pUs, %p, %zu, %p", this, package_type,
Leif Lindholm5aab1b82019-01-21 12:12:57 +0900663 package_guid, package_notify_fn, notify_type,
664 notify_handle);
665
666 if (!notify_handle)
667 return EFI_EXIT(EFI_INVALID_PARAMETER);
668
669 if ((package_type != EFI_HII_PACKAGE_TYPE_GUID && package_guid) ||
670 (package_type == EFI_HII_PACKAGE_TYPE_GUID && !package_guid))
671 return EFI_EXIT(EFI_INVALID_PARAMETER);
672
673 return EFI_EXIT(EFI_OUT_OF_RESOURCES);
674}
675
676static efi_status_t EFIAPI
677unregister_package_notify(const struct efi_hii_database_protocol *this,
678 efi_handle_t notification_handle)
679{
680 EFI_ENTRY("%p, %p", this, notification_handle);
681
682 return EFI_EXIT(EFI_NOT_FOUND);
683}
684
685static efi_status_t EFIAPI
686find_keyboard_layouts(const struct efi_hii_database_protocol *this,
687 u16 *key_guid_buffer_length,
688 efi_guid_t *key_guid_buffer)
689{
AKASHI Takahiro976e9312019-01-21 12:12:59 +0900690 struct efi_keyboard_layout_data *layout_data;
691 int package_cnt, package_max;
692 efi_status_t ret = EFI_SUCCESS;
693
Leif Lindholm5aab1b82019-01-21 12:12:57 +0900694 EFI_ENTRY("%p, %p, %p", this, key_guid_buffer_length, key_guid_buffer);
695
AKASHI Takahiro976e9312019-01-21 12:12:59 +0900696 if (!key_guid_buffer_length ||
697 (*key_guid_buffer_length && !key_guid_buffer))
698 return EFI_EXIT(EFI_INVALID_PARAMETER);
699
700 package_cnt = 0;
701 package_max = *key_guid_buffer_length / sizeof(*key_guid_buffer);
702 list_for_each_entry(layout_data, &efi_keyboard_layout_list, link_sys) {
703 package_cnt++;
704 if (package_cnt <= package_max)
705 memcpy(key_guid_buffer++,
706 &layout_data->keyboard_layout.guid,
707 sizeof(*key_guid_buffer));
708 else
709 ret = EFI_BUFFER_TOO_SMALL;
710 }
711 *key_guid_buffer_length = package_cnt * sizeof(*key_guid_buffer);
712
713 return EFI_EXIT(ret);
Leif Lindholm5aab1b82019-01-21 12:12:57 +0900714}
715
716static efi_status_t EFIAPI
717get_keyboard_layout(const struct efi_hii_database_protocol *this,
718 efi_guid_t *key_guid,
719 u16 *keyboard_layout_length,
720 struct efi_hii_keyboard_layout *keyboard_layout)
721{
AKASHI Takahiro976e9312019-01-21 12:12:59 +0900722 struct efi_keyboard_layout_data *layout_data;
723 u16 layout_length;
724
Heinrich Schuchardt282249d2022-01-16 14:15:31 +0100725 EFI_ENTRY("%p, %pUs, %p, %p", this, key_guid, keyboard_layout_length,
Leif Lindholm5aab1b82019-01-21 12:12:57 +0900726 keyboard_layout);
727
AKASHI Takahiro976e9312019-01-21 12:12:59 +0900728 if (!keyboard_layout_length ||
729 (*keyboard_layout_length && !keyboard_layout))
730 return EFI_EXIT(EFI_INVALID_PARAMETER);
731
732 /* TODO: no notion of current keyboard layout */
733 if (!key_guid)
734 return EFI_EXIT(EFI_INVALID_PARAMETER);
735
736 list_for_each_entry(layout_data, &efi_keyboard_layout_list, link_sys) {
737 if (!guidcmp(&layout_data->keyboard_layout.guid, key_guid))
738 goto found;
739 }
740
Leif Lindholm5aab1b82019-01-21 12:12:57 +0900741 return EFI_EXIT(EFI_NOT_FOUND);
AKASHI Takahiro976e9312019-01-21 12:12:59 +0900742
743found:
744 layout_length =
745 get_unaligned_le16(&layout_data->keyboard_layout.layout_length);
746 if (*keyboard_layout_length < layout_length) {
747 *keyboard_layout_length = layout_length;
748 return EFI_EXIT(EFI_BUFFER_TOO_SMALL);
749 }
750
751 memcpy(keyboard_layout, &layout_data->keyboard_layout, layout_length);
752
753 return EFI_EXIT(EFI_SUCCESS);
Leif Lindholm5aab1b82019-01-21 12:12:57 +0900754}
755
756static efi_status_t EFIAPI
757set_keyboard_layout(const struct efi_hii_database_protocol *this,
758 efi_guid_t *key_guid)
759{
Heinrich Schuchardt282249d2022-01-16 14:15:31 +0100760 EFI_ENTRY("%p, %pUs", this, key_guid);
Leif Lindholm5aab1b82019-01-21 12:12:57 +0900761
Vincent Stehlé00a3dfd2023-01-06 10:46:40 +0100762 if (!key_guid)
763 return EFI_EXIT(EFI_INVALID_PARAMETER);
764
Leif Lindholm5aab1b82019-01-21 12:12:57 +0900765 return EFI_EXIT(EFI_NOT_FOUND);
766}
767
768static efi_status_t EFIAPI
769get_package_list_handle(const struct efi_hii_database_protocol *this,
770 efi_hii_handle_t package_list_handle,
771 efi_handle_t *driver_handle)
772{
773 struct efi_hii_packagelist *hii;
774
775 EFI_ENTRY("%p, %p, %p", this, package_list_handle, driver_handle);
776
777 if (!driver_handle)
778 return EFI_EXIT(EFI_INVALID_PARAMETER);
779
780 list_for_each_entry(hii, &efi_package_lists, link) {
781 if (hii == package_list_handle) {
782 *driver_handle = hii->driver_handle;
783 return EFI_EXIT(EFI_SUCCESS);
784 }
785 }
786
Vincent Stehléd6f8e7a2022-12-13 22:39:09 +0100787 return EFI_EXIT(EFI_INVALID_PARAMETER);
Leif Lindholm5aab1b82019-01-21 12:12:57 +0900788}
789
790const struct efi_hii_database_protocol efi_hii_database = {
791 .new_package_list = new_package_list,
792 .remove_package_list = remove_package_list,
793 .update_package_list = update_package_list,
794 .list_package_lists = list_package_lists,
795 .export_package_lists = export_package_lists,
796 .register_package_notify = register_package_notify,
797 .unregister_package_notify = unregister_package_notify,
798 .find_keyboard_layouts = find_keyboard_layouts,
799 .get_keyboard_layout = get_keyboard_layout,
800 .set_keyboard_layout = set_keyboard_layout,
801 .get_package_list_handle = get_package_list_handle
802};
803
804/*
805 * EFI_HII_STRING_PROTOCOL
806 */
807
808static bool language_match(char *language, char *languages)
809{
810 size_t n;
811
812 n = strlen(language);
813 /* match primary language? */
814 if (!strncasecmp(language, languages, n) &&
815 (languages[n] == ';' || languages[n] == '\0'))
816 return true;
817
818 return false;
819}
820
821static efi_status_t EFIAPI
822new_string(const struct efi_hii_string_protocol *this,
823 efi_hii_handle_t package_list,
824 efi_string_id_t *string_id,
825 const u8 *language,
826 const u16 *language_name,
827 const efi_string_t string,
828 const struct efi_font_info *string_font_info)
829{
830 struct efi_hii_packagelist *hii = package_list;
831 struct efi_string_table *stbl;
832
833 EFI_ENTRY("%p, %p, %p, \"%s\", %p, \"%ls\", %p", this, package_list,
834 string_id, language, language_name, string,
835 string_font_info);
836
837 if (!package_list || !efi_hii_packagelist_exists(package_list))
838 return EFI_EXIT(EFI_NOT_FOUND);
839
840 if (!string_id || !language || !string)
841 return EFI_EXIT(EFI_INVALID_PARAMETER);
842
843 list_for_each_entry(stbl, &hii->string_tables, link) {
844 if (language_match((char *)language, stbl->language)) {
845 efi_string_id_t new_id;
846 void *buf;
847 efi_string_t str;
848
849 new_id = ++hii->max_string_id;
850 if (stbl->nstrings < new_id) {
851 buf = realloc(stbl->strings,
852 sizeof(stbl->strings[0])
853 * new_id);
854 if (!buf)
855 return EFI_EXIT(EFI_OUT_OF_RESOURCES);
856
857 memset(&stbl->strings[stbl->nstrings], 0,
858 (new_id - stbl->nstrings)
859 * sizeof(stbl->strings[0]));
860 stbl->strings = buf;
861 stbl->nstrings = new_id;
862 }
863
864 str = u16_strdup(string);
865 if (!str)
866 return EFI_EXIT(EFI_OUT_OF_RESOURCES);
867
868 stbl->strings[new_id - 1].string = str;
869 *string_id = new_id;
870
871 return EFI_EXIT(EFI_SUCCESS);
872 }
873 }
874
875 return EFI_EXIT(EFI_NOT_FOUND);
876}
877
878static efi_status_t EFIAPI
879get_string(const struct efi_hii_string_protocol *this,
880 const u8 *language,
881 efi_hii_handle_t package_list,
882 efi_string_id_t string_id,
883 efi_string_t string,
884 efi_uintn_t *string_size,
885 struct efi_font_info **string_font_info)
886{
887 struct efi_hii_packagelist *hii = package_list;
888 struct efi_string_table *stbl;
889
890 EFI_ENTRY("%p, \"%s\", %p, %u, %p, %p, %p", this, language,
891 package_list, string_id, string, string_size,
892 string_font_info);
893
894 if (!package_list || !efi_hii_packagelist_exists(package_list))
895 return EFI_EXIT(EFI_NOT_FOUND);
896
897 list_for_each_entry(stbl, &hii->string_tables, link) {
898 if (language_match((char *)language, stbl->language)) {
899 efi_string_t str;
900 size_t len;
901
902 if (stbl->nstrings < string_id)
903 return EFI_EXIT(EFI_NOT_FOUND);
904
905 str = stbl->strings[string_id - 1].string;
906 if (str) {
Heinrich Schuchardt28b06b22022-04-02 11:47:00 +0200907 len = u16_strsize(str);
Leif Lindholm5aab1b82019-01-21 12:12:57 +0900908 if (*string_size < len) {
909 *string_size = len;
910
911 return EFI_EXIT(EFI_BUFFER_TOO_SMALL);
912 }
913 memcpy(string, str, len);
914 *string_size = len;
915 } else {
916 return EFI_EXIT(EFI_NOT_FOUND);
917 }
918
919 return EFI_EXIT(EFI_SUCCESS);
920 }
921 }
922
923 return EFI_EXIT(EFI_NOT_FOUND);
924}
925
926static efi_status_t EFIAPI
927set_string(const struct efi_hii_string_protocol *this,
928 efi_hii_handle_t package_list,
929 efi_string_id_t string_id,
930 const u8 *language,
931 const efi_string_t string,
932 const struct efi_font_info *string_font_info)
933{
934 struct efi_hii_packagelist *hii = package_list;
935 struct efi_string_table *stbl;
936
937 EFI_ENTRY("%p, %p, %u, \"%s\", \"%ls\", %p", this, package_list,
938 string_id, language, string, string_font_info);
939
940 if (!package_list || !efi_hii_packagelist_exists(package_list))
941 return EFI_EXIT(EFI_NOT_FOUND);
942
943 if (string_id > hii->max_string_id)
944 return EFI_EXIT(EFI_NOT_FOUND);
945
946 if (!string || !language)
947 return EFI_EXIT(EFI_INVALID_PARAMETER);
948
949 list_for_each_entry(stbl, &hii->string_tables, link) {
950 if (language_match((char *)language, stbl->language)) {
951 efi_string_t str;
952
953 if (hii->max_string_id < string_id)
954 return EFI_EXIT(EFI_NOT_FOUND);
955
956 if (stbl->nstrings < string_id) {
957 void *buf;
958
959 buf = realloc(stbl->strings,
960 string_id
961 * sizeof(stbl->strings[0]));
962 if (!buf)
963 return EFI_EXIT(EFI_OUT_OF_RESOURCES);
964
965 memset(&stbl->strings[string_id - 1], 0,
966 (string_id - stbl->nstrings)
967 * sizeof(stbl->strings[0]));
968 stbl->strings = buf;
969 }
970
971 str = u16_strdup(string);
972 if (!str)
973 return EFI_EXIT(EFI_OUT_OF_RESOURCES);
974
975 free(stbl->strings[string_id - 1].string);
976 stbl->strings[string_id - 1].string = str;
977
978 return EFI_EXIT(EFI_SUCCESS);
979 }
980 }
981
982 return EFI_EXIT(EFI_NOT_FOUND);
983}
984
985static efi_status_t EFIAPI
986get_languages(const struct efi_hii_string_protocol *this,
987 efi_hii_handle_t package_list,
988 u8 *languages,
989 efi_uintn_t *languages_size)
990{
991 struct efi_hii_packagelist *hii = package_list;
992 struct efi_string_table *stbl;
993 size_t len = 0;
994 char *p;
995
996 EFI_ENTRY("%p, %p, %p, %p", this, package_list, languages,
997 languages_size);
998
999 if (!package_list || !efi_hii_packagelist_exists(package_list))
1000 return EFI_EXIT(EFI_NOT_FOUND);
1001
1002 if (!languages_size ||
1003 (*languages_size && !languages))
1004 return EFI_EXIT(EFI_INVALID_PARAMETER);
1005
1006 /* figure out required size: */
1007 list_for_each_entry(stbl, &hii->string_tables, link) {
1008 len += strlen((char *)stbl->language) + 1;
1009 }
1010
1011 if (*languages_size < len) {
1012 *languages_size = len;
1013
1014 return EFI_EXIT(EFI_BUFFER_TOO_SMALL);
1015 }
1016
1017 p = (char *)languages;
1018 list_for_each_entry(stbl, &hii->string_tables, link) {
1019 if (p != (char *)languages)
1020 *p++ = ';';
1021 strcpy(p, stbl->language);
1022 p += strlen((char *)stbl->language);
1023 }
1024 *p = '\0';
1025
Heinrich Schuchardt4b6f19f2019-01-23 22:55:36 +01001026 EFI_PRINT("languages: %s\n", languages);
Leif Lindholm5aab1b82019-01-21 12:12:57 +09001027
1028 return EFI_EXIT(EFI_SUCCESS);
1029}
1030
1031static efi_status_t EFIAPI
1032get_secondary_languages(const struct efi_hii_string_protocol *this,
1033 efi_hii_handle_t package_list,
1034 const u8 *primary_language,
1035 u8 *secondary_languages,
1036 efi_uintn_t *secondary_languages_size)
1037{
1038 struct efi_hii_packagelist *hii = package_list;
1039 struct efi_string_table *stbl;
1040 bool found = false;
1041
1042 EFI_ENTRY("%p, %p, \"%s\", %p, %p", this, package_list,
1043 primary_language, secondary_languages,
1044 secondary_languages_size);
1045
1046 if (!package_list || !efi_hii_packagelist_exists(package_list))
1047 return EFI_EXIT(EFI_NOT_FOUND);
1048
1049 if (!secondary_languages_size ||
1050 (*secondary_languages_size && !secondary_languages))
1051 return EFI_EXIT(EFI_INVALID_PARAMETER);
1052
1053 list_for_each_entry(stbl, &hii->string_tables, link) {
1054 if (language_match((char *)primary_language, stbl->language)) {
1055 found = true;
1056 break;
1057 }
1058 }
1059 if (!found)
1060 return EFI_EXIT(EFI_INVALID_LANGUAGE);
1061
1062 /*
1063 * TODO: What is secondary language?
1064 * *secondary_languages = '\0';
1065 * *secondary_languages_size = 0;
1066 */
1067
1068 return EFI_EXIT(EFI_NOT_FOUND);
1069}
1070
1071const struct efi_hii_string_protocol efi_hii_string = {
1072 .new_string = new_string,
1073 .get_string = get_string,
1074 .set_string = set_string,
1075 .get_languages = get_languages,
1076 .get_secondary_languages = get_secondary_languages
1077};