blob: 5f0ae5c3383e874d5693202e13918ded9cb84496 [file] [log] [blame]
Heinrich Schuchardt2c1e2242020-12-13 10:30:24 +01001// SPDX-License-Identifier: GPL-2.0+
2/*
3 * EFI_DT_FIXUP_PROTOCOL
4 *
5 * Copyright (c) 2020 Heinrich Schuchardt
6 */
7
8#include <common.h>
9#include <efi_dt_fixup.h>
10#include <efi_loader.h>
11#include <mapmem.h>
12
13static efi_status_t EFIAPI efi_dt_fixup(struct efi_dt_fixup_protocol *this,
14 void *dtb,
15 efi_uintn_t *buffer_size,
16 u32 flags);
17
18struct efi_dt_fixup_protocol efi_dt_fixup_prot = {
19 .revision = EFI_DT_FIXUP_PROTOCOL_REVISION,
20 .fixup = efi_dt_fixup
21};
22
23const efi_guid_t efi_guid_dt_fixup_protocol = EFI_DT_FIXUP_PROTOCOL_GUID;
24
25/**
26 * efi_reserve_memory() - add reserved memory to memory map
27 *
28 * @addr: start address of the reserved memory range
29 * @size: size of the reserved memory range
30 * @nomap: indicates that the memory range shall not be accessed by the
31 * UEFI payload
32 */
33static void efi_reserve_memory(u64 addr, u64 size, bool nomap)
34{
35 int type;
36 efi_uintn_t ret;
37
38 /* Convert from sandbox address space. */
39 addr = (uintptr_t)map_sysmem(addr, 0);
40
41 if (nomap)
42 type = EFI_RESERVED_MEMORY_TYPE;
43 else
44 type = EFI_BOOT_SERVICES_DATA;
45
46 ret = efi_add_memory_map(addr, size, type);
47 if (ret != EFI_SUCCESS)
48 log_err("Reserved memory mapping failed addr %llx size %llx\n",
49 addr, size);
50}
51
52/**
53 * efi_carve_out_dt_rsv() - Carve out DT reserved memory ranges
54 *
55 * The mem_rsv entries of the FDT are added to the memory map. Any failures are
56 * ignored because this is not critical and we would rather continue to try to
57 * boot.
58 *
59 * @fdt: Pointer to device tree
60 */
61void efi_carve_out_dt_rsv(void *fdt)
62{
63 int nr_rsv, i;
64 u64 addr, size;
65 int nodeoffset, subnode;
66
67 nr_rsv = fdt_num_mem_rsv(fdt);
68
69 /* Look for an existing entry and add it to the efi mem map. */
70 for (i = 0; i < nr_rsv; i++) {
71 if (fdt_get_mem_rsv(fdt, i, &addr, &size) != 0)
72 continue;
73 efi_reserve_memory(addr, size, false);
74 }
75
76 /* process reserved-memory */
77 nodeoffset = fdt_subnode_offset(fdt, 0, "reserved-memory");
78 if (nodeoffset >= 0) {
79 subnode = fdt_first_subnode(fdt, nodeoffset);
80 while (subnode >= 0) {
81 fdt_addr_t fdt_addr;
82 fdt_size_t fdt_size;
83
84 /* check if this subnode has a reg property */
85 fdt_addr = fdtdec_get_addr_size_auto_parent(
86 fdt, nodeoffset, subnode,
87 "reg", 0, &fdt_size, false);
88 /*
89 * The /reserved-memory node may have children with
90 * a size instead of a reg property.
91 */
92 if (fdt_addr != FDT_ADDR_T_NONE &&
93 fdtdec_get_is_enabled(fdt, subnode)) {
94 bool nomap;
95
96 nomap = !!fdt_getprop(fdt, subnode, "no-map",
97 NULL);
98 efi_reserve_memory(fdt_addr, fdt_size, nomap);
99 }
100 subnode = fdt_next_subnode(fdt, subnode);
101 }
102 }
103}
104
105static efi_status_t EFIAPI efi_dt_fixup(struct efi_dt_fixup_protocol *this,
106 void *dtb,
107 efi_uintn_t *buffer_size,
108 u32 flags)
109{
110 efi_status_t ret;
111 size_t required_size;
112 bootm_headers_t img = { 0 };
113
114 EFI_ENTRY("%p, %p, %p, %d", this, dtb, buffer_size, flags);
115
116 if (this != &efi_dt_fixup_prot || !dtb || !buffer_size ||
117 !flags || (flags & ~EFI_DT_ALL)) {
118 ret = EFI_INVALID_PARAMETER;
119 goto out;
120 }
121 if (fdt_check_header(dtb)) {
122 ret = EFI_INVALID_PARAMETER;
123 goto out;
124 }
125 if (flags & EFI_DT_APPLY_FIXUPS) {
126 required_size = fdt_off_dt_strings(dtb) +
127 fdt_size_dt_strings(dtb) +
128 0x3000;
129 } else {
130 required_size = fdt_totalsize(dtb);
131 }
132 if (required_size > *buffer_size) {
133 *buffer_size = required_size;
134 ret = EFI_BUFFER_TOO_SMALL;
135 goto out;
136 }
137 fdt_set_totalsize(dtb, *buffer_size);
138
139 if (flags & EFI_DT_APPLY_FIXUPS) {
140 if (image_setup_libfdt(&img, dtb, 0, NULL)) {
141 log_err("failed to process device tree\n");
142 ret = EFI_INVALID_PARAMETER;
143 goto out;
144 }
145 }
146 if (flags & EFI_DT_RESERVE_MEMORY)
147 efi_carve_out_dt_rsv(dtb);
148
149 if (EFI_DT_INSTALL_TABLE) {
150 ret = efi_install_configuration_table(&efi_guid_fdt, dtb);
151 if (ret != EFI_SUCCESS) {
152 log_err("ERROR: failed to install device tree\n");
153 goto out;
154 }
155 }
156
157 ret = EFI_SUCCESS;
158out:
159 return EFI_EXIT(ret);
160}