1b30be4dcSRob Herring // SPDX-License-Identifier: GPL-2.0-only
2b30be4dcSRob Herring /*
3b30be4dcSRob Herring * Copyright (C) 2020 Arm Limited
4b30be4dcSRob Herring *
5b30be4dcSRob Herring * Based on arch/arm64/kernel/machine_kexec_file.c:
6b30be4dcSRob Herring * Copyright (C) 2018 Linaro Limited
7b30be4dcSRob Herring *
8b30be4dcSRob Herring * And arch/powerpc/kexec/file_load.c:
9b30be4dcSRob Herring * Copyright (C) 2016 IBM Corporation
10b30be4dcSRob Herring */
11b30be4dcSRob Herring
12b69a2afdSJonathan McDowell #include <linux/ima.h>
13b30be4dcSRob Herring #include <linux/kernel.h>
14b30be4dcSRob Herring #include <linux/kexec.h>
15fee3ff99SLakshmi Ramasubramanian #include <linux/memblock.h>
16b30be4dcSRob Herring #include <linux/libfdt.h>
17b30be4dcSRob Herring #include <linux/of.h>
18b30be4dcSRob Herring #include <linux/of_fdt.h>
19b30be4dcSRob Herring #include <linux/random.h>
208587ca6fSMatthew Wilcox (Oracle) #include <linux/slab.h>
21b30be4dcSRob Herring #include <linux/types.h>
22b30be4dcSRob Herring
23b30be4dcSRob Herring #define RNG_SEED_SIZE 128
24b30be4dcSRob Herring
25b30be4dcSRob Herring /*
26b30be4dcSRob Herring * Additional space needed for the FDT buffer so that we can add initrd,
27b30be4dcSRob Herring * bootargs, kaslr-seed, rng-seed, useable-memory-range and elfcorehdr.
28b30be4dcSRob Herring */
29b30be4dcSRob Herring #define FDT_EXTRA_SPACE 0x1000
30b30be4dcSRob Herring
31b30be4dcSRob Herring /**
32b30be4dcSRob Herring * fdt_find_and_del_mem_rsv - delete memory reservation with given address and size
33b30be4dcSRob Herring *
34b30be4dcSRob Herring * @fdt: Flattened device tree for the current kernel.
35b30be4dcSRob Herring * @start: Starting address of the reserved memory.
36b30be4dcSRob Herring * @size: Size of the reserved memory.
37b30be4dcSRob Herring *
38b30be4dcSRob Herring * Return: 0 on success, or negative errno on error.
39b30be4dcSRob Herring */
fdt_find_and_del_mem_rsv(void * fdt,unsigned long start,unsigned long size)40b30be4dcSRob Herring static int fdt_find_and_del_mem_rsv(void *fdt, unsigned long start, unsigned long size)
41b30be4dcSRob Herring {
42b30be4dcSRob Herring int i, ret, num_rsvs = fdt_num_mem_rsv(fdt);
43b30be4dcSRob Herring
44b30be4dcSRob Herring for (i = 0; i < num_rsvs; i++) {
45b30be4dcSRob Herring u64 rsv_start, rsv_size;
46b30be4dcSRob Herring
47b30be4dcSRob Herring ret = fdt_get_mem_rsv(fdt, i, &rsv_start, &rsv_size);
48b30be4dcSRob Herring if (ret) {
49b30be4dcSRob Herring pr_err("Malformed device tree.\n");
50b30be4dcSRob Herring return -EINVAL;
51b30be4dcSRob Herring }
52b30be4dcSRob Herring
53b30be4dcSRob Herring if (rsv_start == start && rsv_size == size) {
54b30be4dcSRob Herring ret = fdt_del_mem_rsv(fdt, i);
55b30be4dcSRob Herring if (ret) {
56b30be4dcSRob Herring pr_err("Error deleting device tree reservation.\n");
57b30be4dcSRob Herring return -EINVAL;
58b30be4dcSRob Herring }
59b30be4dcSRob Herring
60b30be4dcSRob Herring return 0;
61b30be4dcSRob Herring }
62b30be4dcSRob Herring }
63b30be4dcSRob Herring
64b30be4dcSRob Herring return -ENOENT;
65b30be4dcSRob Herring }
66b30be4dcSRob Herring
67fee3ff99SLakshmi Ramasubramanian /**
68fee3ff99SLakshmi Ramasubramanian * get_addr_size_cells - Get address and size of root node
69fee3ff99SLakshmi Ramasubramanian *
70fee3ff99SLakshmi Ramasubramanian * @addr_cells: Return address of the root node
71fee3ff99SLakshmi Ramasubramanian * @size_cells: Return size of the root node
72fee3ff99SLakshmi Ramasubramanian *
73fee3ff99SLakshmi Ramasubramanian * Return: 0 on success, or negative errno on error.
74fee3ff99SLakshmi Ramasubramanian */
get_addr_size_cells(int * addr_cells,int * size_cells)75fee3ff99SLakshmi Ramasubramanian static int get_addr_size_cells(int *addr_cells, int *size_cells)
76fee3ff99SLakshmi Ramasubramanian {
77fee3ff99SLakshmi Ramasubramanian struct device_node *root;
78fee3ff99SLakshmi Ramasubramanian
79fee3ff99SLakshmi Ramasubramanian root = of_find_node_by_path("/");
80fee3ff99SLakshmi Ramasubramanian if (!root)
81fee3ff99SLakshmi Ramasubramanian return -EINVAL;
82fee3ff99SLakshmi Ramasubramanian
83fee3ff99SLakshmi Ramasubramanian *addr_cells = of_n_addr_cells(root);
84fee3ff99SLakshmi Ramasubramanian *size_cells = of_n_size_cells(root);
85fee3ff99SLakshmi Ramasubramanian
86fee3ff99SLakshmi Ramasubramanian of_node_put(root);
87fee3ff99SLakshmi Ramasubramanian
88fee3ff99SLakshmi Ramasubramanian return 0;
89fee3ff99SLakshmi Ramasubramanian }
90fee3ff99SLakshmi Ramasubramanian
91fee3ff99SLakshmi Ramasubramanian /**
92fee3ff99SLakshmi Ramasubramanian * do_get_kexec_buffer - Get address and size of device tree property
93fee3ff99SLakshmi Ramasubramanian *
94fee3ff99SLakshmi Ramasubramanian * @prop: Device tree property
95fee3ff99SLakshmi Ramasubramanian * @len: Size of @prop
96fee3ff99SLakshmi Ramasubramanian * @addr: Return address of the node
97fee3ff99SLakshmi Ramasubramanian * @size: Return size of the node
98fee3ff99SLakshmi Ramasubramanian *
99fee3ff99SLakshmi Ramasubramanian * Return: 0 on success, or negative errno on error.
100fee3ff99SLakshmi Ramasubramanian */
do_get_kexec_buffer(const void * prop,int len,unsigned long * addr,size_t * size)101fee3ff99SLakshmi Ramasubramanian static int do_get_kexec_buffer(const void *prop, int len, unsigned long *addr,
102fee3ff99SLakshmi Ramasubramanian size_t *size)
103fee3ff99SLakshmi Ramasubramanian {
104fee3ff99SLakshmi Ramasubramanian int ret, addr_cells, size_cells;
105fee3ff99SLakshmi Ramasubramanian
106fee3ff99SLakshmi Ramasubramanian ret = get_addr_size_cells(&addr_cells, &size_cells);
107fee3ff99SLakshmi Ramasubramanian if (ret)
108fee3ff99SLakshmi Ramasubramanian return ret;
109fee3ff99SLakshmi Ramasubramanian
110fee3ff99SLakshmi Ramasubramanian if (len < 4 * (addr_cells + size_cells))
111fee3ff99SLakshmi Ramasubramanian return -ENOENT;
112fee3ff99SLakshmi Ramasubramanian
113fee3ff99SLakshmi Ramasubramanian *addr = of_read_number(prop, addr_cells);
114fee3ff99SLakshmi Ramasubramanian *size = of_read_number(prop + 4 * addr_cells, size_cells);
115fee3ff99SLakshmi Ramasubramanian
116fee3ff99SLakshmi Ramasubramanian return 0;
117fee3ff99SLakshmi Ramasubramanian }
118fee3ff99SLakshmi Ramasubramanian
119b69a2afdSJonathan McDowell #ifdef CONFIG_HAVE_IMA_KEXEC
120fee3ff99SLakshmi Ramasubramanian /**
121fee3ff99SLakshmi Ramasubramanian * ima_get_kexec_buffer - get IMA buffer from the previous kernel
122fee3ff99SLakshmi Ramasubramanian * @addr: On successful return, set to point to the buffer contents.
123fee3ff99SLakshmi Ramasubramanian * @size: On successful return, set to the buffer size.
124fee3ff99SLakshmi Ramasubramanian *
125fee3ff99SLakshmi Ramasubramanian * Return: 0 on success, negative errno on error.
126fee3ff99SLakshmi Ramasubramanian */
ima_get_kexec_buffer(void ** addr,size_t * size)127b69a2afdSJonathan McDowell int __init ima_get_kexec_buffer(void **addr, size_t *size)
128fee3ff99SLakshmi Ramasubramanian {
129fee3ff99SLakshmi Ramasubramanian int ret, len;
130fee3ff99SLakshmi Ramasubramanian unsigned long tmp_addr;
131cbf9c4b9SVaibhav Jain unsigned long start_pfn, end_pfn;
132fee3ff99SLakshmi Ramasubramanian size_t tmp_size;
133fee3ff99SLakshmi Ramasubramanian const void *prop;
134fee3ff99SLakshmi Ramasubramanian
135fee3ff99SLakshmi Ramasubramanian prop = of_get_property(of_chosen, "linux,ima-kexec-buffer", &len);
136fee3ff99SLakshmi Ramasubramanian if (!prop)
137fee3ff99SLakshmi Ramasubramanian return -ENOENT;
138fee3ff99SLakshmi Ramasubramanian
139fee3ff99SLakshmi Ramasubramanian ret = do_get_kexec_buffer(prop, len, &tmp_addr, &tmp_size);
140fee3ff99SLakshmi Ramasubramanian if (ret)
141fee3ff99SLakshmi Ramasubramanian return ret;
142fee3ff99SLakshmi Ramasubramanian
143cbf9c4b9SVaibhav Jain /* Do some sanity on the returned size for the ima-kexec buffer */
144cbf9c4b9SVaibhav Jain if (!tmp_size)
145cbf9c4b9SVaibhav Jain return -ENOENT;
146cbf9c4b9SVaibhav Jain
147cbf9c4b9SVaibhav Jain /*
148cbf9c4b9SVaibhav Jain * Calculate the PFNs for the buffer and ensure
149cbf9c4b9SVaibhav Jain * they are with in addressable memory.
150cbf9c4b9SVaibhav Jain */
151cbf9c4b9SVaibhav Jain start_pfn = PHYS_PFN(tmp_addr);
152cbf9c4b9SVaibhav Jain end_pfn = PHYS_PFN(tmp_addr + tmp_size - 1);
153cbf9c4b9SVaibhav Jain if (!page_is_ram(start_pfn) || !page_is_ram(end_pfn)) {
154cbf9c4b9SVaibhav Jain pr_warn("IMA buffer at 0x%lx, size = 0x%zx beyond memory\n",
155cbf9c4b9SVaibhav Jain tmp_addr, tmp_size);
156cbf9c4b9SVaibhav Jain return -EINVAL;
157cbf9c4b9SVaibhav Jain }
158cbf9c4b9SVaibhav Jain
159fee3ff99SLakshmi Ramasubramanian *addr = __va(tmp_addr);
160fee3ff99SLakshmi Ramasubramanian *size = tmp_size;
161fee3ff99SLakshmi Ramasubramanian
162fee3ff99SLakshmi Ramasubramanian return 0;
163fee3ff99SLakshmi Ramasubramanian }
164fee3ff99SLakshmi Ramasubramanian
165fee3ff99SLakshmi Ramasubramanian /**
166fee3ff99SLakshmi Ramasubramanian * ima_free_kexec_buffer - free memory used by the IMA buffer
167fee3ff99SLakshmi Ramasubramanian */
ima_free_kexec_buffer(void)168b69a2afdSJonathan McDowell int __init ima_free_kexec_buffer(void)
169fee3ff99SLakshmi Ramasubramanian {
170fee3ff99SLakshmi Ramasubramanian int ret;
171fee3ff99SLakshmi Ramasubramanian unsigned long addr;
172fee3ff99SLakshmi Ramasubramanian size_t size;
173fee3ff99SLakshmi Ramasubramanian struct property *prop;
174fee3ff99SLakshmi Ramasubramanian
175fee3ff99SLakshmi Ramasubramanian prop = of_find_property(of_chosen, "linux,ima-kexec-buffer", NULL);
176fee3ff99SLakshmi Ramasubramanian if (!prop)
177fee3ff99SLakshmi Ramasubramanian return -ENOENT;
178fee3ff99SLakshmi Ramasubramanian
179fee3ff99SLakshmi Ramasubramanian ret = do_get_kexec_buffer(prop->value, prop->length, &addr, &size);
180fee3ff99SLakshmi Ramasubramanian if (ret)
181fee3ff99SLakshmi Ramasubramanian return ret;
182fee3ff99SLakshmi Ramasubramanian
183fee3ff99SLakshmi Ramasubramanian ret = of_remove_property(of_chosen, prop);
184fee3ff99SLakshmi Ramasubramanian if (ret)
185fee3ff99SLakshmi Ramasubramanian return ret;
186fee3ff99SLakshmi Ramasubramanian
187f0362a25SRik van Riel memblock_free_late(addr, size);
188f0362a25SRik van Riel return 0;
189fee3ff99SLakshmi Ramasubramanian }
190b69a2afdSJonathan McDowell #endif
191fee3ff99SLakshmi Ramasubramanian
192fee3ff99SLakshmi Ramasubramanian /**
193fee3ff99SLakshmi Ramasubramanian * remove_ima_buffer - remove the IMA buffer property and reservation from @fdt
194fee3ff99SLakshmi Ramasubramanian *
195fee3ff99SLakshmi Ramasubramanian * @fdt: Flattened Device Tree to update
196fee3ff99SLakshmi Ramasubramanian * @chosen_node: Offset to the chosen node in the device tree
197fee3ff99SLakshmi Ramasubramanian *
198fee3ff99SLakshmi Ramasubramanian * The IMA measurement buffer is of no use to a subsequent kernel, so we always
199fee3ff99SLakshmi Ramasubramanian * remove it from the device tree.
200fee3ff99SLakshmi Ramasubramanian */
remove_ima_buffer(void * fdt,int chosen_node)201fee3ff99SLakshmi Ramasubramanian static void remove_ima_buffer(void *fdt, int chosen_node)
202fee3ff99SLakshmi Ramasubramanian {
203fee3ff99SLakshmi Ramasubramanian int ret, len;
204fee3ff99SLakshmi Ramasubramanian unsigned long addr;
205fee3ff99SLakshmi Ramasubramanian size_t size;
206fee3ff99SLakshmi Ramasubramanian const void *prop;
207fee3ff99SLakshmi Ramasubramanian
208fee3ff99SLakshmi Ramasubramanian if (!IS_ENABLED(CONFIG_HAVE_IMA_KEXEC))
209fee3ff99SLakshmi Ramasubramanian return;
210fee3ff99SLakshmi Ramasubramanian
211fee3ff99SLakshmi Ramasubramanian prop = fdt_getprop(fdt, chosen_node, "linux,ima-kexec-buffer", &len);
212fee3ff99SLakshmi Ramasubramanian if (!prop)
213fee3ff99SLakshmi Ramasubramanian return;
214fee3ff99SLakshmi Ramasubramanian
215fee3ff99SLakshmi Ramasubramanian ret = do_get_kexec_buffer(prop, len, &addr, &size);
216fee3ff99SLakshmi Ramasubramanian fdt_delprop(fdt, chosen_node, "linux,ima-kexec-buffer");
217fee3ff99SLakshmi Ramasubramanian if (ret)
218fee3ff99SLakshmi Ramasubramanian return;
219fee3ff99SLakshmi Ramasubramanian
220fee3ff99SLakshmi Ramasubramanian ret = fdt_find_and_del_mem_rsv(fdt, addr, size);
221fee3ff99SLakshmi Ramasubramanian if (!ret)
222fee3ff99SLakshmi Ramasubramanian pr_debug("Removed old IMA buffer reservation.\n");
223fee3ff99SLakshmi Ramasubramanian }
224fee3ff99SLakshmi Ramasubramanian
225fee3ff99SLakshmi Ramasubramanian #ifdef CONFIG_IMA_KEXEC
226fee3ff99SLakshmi Ramasubramanian /**
227fee3ff99SLakshmi Ramasubramanian * setup_ima_buffer - add IMA buffer information to the fdt
228fee3ff99SLakshmi Ramasubramanian * @image: kexec image being loaded.
229fee3ff99SLakshmi Ramasubramanian * @fdt: Flattened device tree for the next kernel.
230fee3ff99SLakshmi Ramasubramanian * @chosen_node: Offset to the chosen node.
231fee3ff99SLakshmi Ramasubramanian *
232fee3ff99SLakshmi Ramasubramanian * Return: 0 on success, or negative errno on error.
233fee3ff99SLakshmi Ramasubramanian */
setup_ima_buffer(const struct kimage * image,void * fdt,int chosen_node)234fee3ff99SLakshmi Ramasubramanian static int setup_ima_buffer(const struct kimage *image, void *fdt,
235fee3ff99SLakshmi Ramasubramanian int chosen_node)
236fee3ff99SLakshmi Ramasubramanian {
23728db15d4SLakshmi Ramasubramanian int ret;
238fee3ff99SLakshmi Ramasubramanian
239fee3ff99SLakshmi Ramasubramanian if (!image->ima_buffer_size)
240fee3ff99SLakshmi Ramasubramanian return 0;
241fee3ff99SLakshmi Ramasubramanian
24228db15d4SLakshmi Ramasubramanian ret = fdt_appendprop_addrrange(fdt, 0, chosen_node,
24328db15d4SLakshmi Ramasubramanian "linux,ima-kexec-buffer",
24428db15d4SLakshmi Ramasubramanian image->ima_buffer_addr,
24528db15d4SLakshmi Ramasubramanian image->ima_buffer_size);
246fee3ff99SLakshmi Ramasubramanian if (ret < 0)
247fee3ff99SLakshmi Ramasubramanian return -EINVAL;
248fee3ff99SLakshmi Ramasubramanian
249fee3ff99SLakshmi Ramasubramanian ret = fdt_add_mem_rsv(fdt, image->ima_buffer_addr,
250fee3ff99SLakshmi Ramasubramanian image->ima_buffer_size);
251fee3ff99SLakshmi Ramasubramanian if (ret)
252fee3ff99SLakshmi Ramasubramanian return -EINVAL;
253fee3ff99SLakshmi Ramasubramanian
254fee3ff99SLakshmi Ramasubramanian pr_debug("IMA buffer at 0x%llx, size = 0x%zx\n",
255fee3ff99SLakshmi Ramasubramanian image->ima_buffer_addr, image->ima_buffer_size);
256fee3ff99SLakshmi Ramasubramanian
257fee3ff99SLakshmi Ramasubramanian return 0;
258fee3ff99SLakshmi Ramasubramanian }
259fee3ff99SLakshmi Ramasubramanian #else /* CONFIG_IMA_KEXEC */
setup_ima_buffer(const struct kimage * image,void * fdt,int chosen_node)260fee3ff99SLakshmi Ramasubramanian static inline int setup_ima_buffer(const struct kimage *image, void *fdt,
261fee3ff99SLakshmi Ramasubramanian int chosen_node)
262fee3ff99SLakshmi Ramasubramanian {
263fee3ff99SLakshmi Ramasubramanian return 0;
264fee3ff99SLakshmi Ramasubramanian }
265fee3ff99SLakshmi Ramasubramanian #endif /* CONFIG_IMA_KEXEC */
266fee3ff99SLakshmi Ramasubramanian
267b30be4dcSRob Herring /*
268b30be4dcSRob Herring * of_kexec_alloc_and_setup_fdt - Alloc and setup a new Flattened Device Tree
269b30be4dcSRob Herring *
270b30be4dcSRob Herring * @image: kexec image being loaded.
271b30be4dcSRob Herring * @initrd_load_addr: Address where the next initrd will be loaded.
272b30be4dcSRob Herring * @initrd_len: Size of the next initrd, or 0 if there will be none.
273b30be4dcSRob Herring * @cmdline: Command line for the next kernel, or NULL if there will
274b30be4dcSRob Herring * be none.
275b30be4dcSRob Herring * @extra_fdt_size: Additional size for the new FDT buffer.
276b30be4dcSRob Herring *
277b30be4dcSRob Herring * Return: fdt on success, or NULL errno on error.
278b30be4dcSRob Herring */
of_kexec_alloc_and_setup_fdt(const struct kimage * image,unsigned long initrd_load_addr,unsigned long initrd_len,const char * cmdline,size_t extra_fdt_size)279b30be4dcSRob Herring void *of_kexec_alloc_and_setup_fdt(const struct kimage *image,
280b30be4dcSRob Herring unsigned long initrd_load_addr,
281b30be4dcSRob Herring unsigned long initrd_len,
282b30be4dcSRob Herring const char *cmdline, size_t extra_fdt_size)
283b30be4dcSRob Herring {
284b30be4dcSRob Herring void *fdt;
285e553ad8dSRob Herring int ret, chosen_node, len;
286b30be4dcSRob Herring const void *prop;
287b30be4dcSRob Herring size_t fdt_size;
288b30be4dcSRob Herring
289b30be4dcSRob Herring fdt_size = fdt_totalsize(initial_boot_params) +
290b30be4dcSRob Herring (cmdline ? strlen(cmdline) : 0) +
291b30be4dcSRob Herring FDT_EXTRA_SPACE +
292b30be4dcSRob Herring extra_fdt_size;
293b30be4dcSRob Herring fdt = kvmalloc(fdt_size, GFP_KERNEL);
294b30be4dcSRob Herring if (!fdt)
295b30be4dcSRob Herring return NULL;
296b30be4dcSRob Herring
297b30be4dcSRob Herring ret = fdt_open_into(initial_boot_params, fdt, fdt_size);
298b30be4dcSRob Herring if (ret < 0) {
299b30be4dcSRob Herring pr_err("Error %d setting up the new device tree.\n", ret);
300b30be4dcSRob Herring goto out;
301b30be4dcSRob Herring }
302b30be4dcSRob Herring
303b30be4dcSRob Herring /* Remove memory reservation for the current device tree. */
304*1103d3b5SUsama Arif ret = fdt_find_and_del_mem_rsv(fdt, initial_boot_params_pa,
305b30be4dcSRob Herring fdt_totalsize(initial_boot_params));
306b30be4dcSRob Herring if (ret == -EINVAL) {
307b30be4dcSRob Herring pr_err("Error removing memory reservation.\n");
308b30be4dcSRob Herring goto out;
309b30be4dcSRob Herring }
310b30be4dcSRob Herring
311b30be4dcSRob Herring chosen_node = fdt_path_offset(fdt, "/chosen");
312b30be4dcSRob Herring if (chosen_node == -FDT_ERR_NOTFOUND)
313b30be4dcSRob Herring chosen_node = fdt_add_subnode(fdt, fdt_path_offset(fdt, "/"),
314b30be4dcSRob Herring "chosen");
315b30be4dcSRob Herring if (chosen_node < 0) {
316b30be4dcSRob Herring ret = chosen_node;
317b30be4dcSRob Herring goto out;
318b30be4dcSRob Herring }
319b30be4dcSRob Herring
320cc6ef3d1SGeert Uytterhoeven ret = fdt_delprop(fdt, chosen_node, "linux,elfcorehdr");
321b30be4dcSRob Herring if (ret && ret != -FDT_ERR_NOTFOUND)
322b30be4dcSRob Herring goto out;
323cc6ef3d1SGeert Uytterhoeven ret = fdt_delprop(fdt, chosen_node, "linux,usable-memory-range");
324b30be4dcSRob Herring if (ret && ret != -FDT_ERR_NOTFOUND)
325b30be4dcSRob Herring goto out;
326b30be4dcSRob Herring
327b30be4dcSRob Herring /* Did we boot using an initrd? */
328e553ad8dSRob Herring prop = fdt_getprop(fdt, chosen_node, "linux,initrd-start", &len);
329b30be4dcSRob Herring if (prop) {
330b30be4dcSRob Herring u64 tmp_start, tmp_end, tmp_size;
331b30be4dcSRob Herring
332e553ad8dSRob Herring tmp_start = of_read_number(prop, len / 4);
333b30be4dcSRob Herring
334e553ad8dSRob Herring prop = fdt_getprop(fdt, chosen_node, "linux,initrd-end", &len);
335b30be4dcSRob Herring if (!prop) {
336b30be4dcSRob Herring ret = -EINVAL;
337b30be4dcSRob Herring goto out;
338b30be4dcSRob Herring }
339b30be4dcSRob Herring
340e553ad8dSRob Herring tmp_end = of_read_number(prop, len / 4);
341b30be4dcSRob Herring
342b30be4dcSRob Herring /*
343b30be4dcSRob Herring * kexec reserves exact initrd size, while firmware may
344b30be4dcSRob Herring * reserve a multiple of PAGE_SIZE, so check for both.
345b30be4dcSRob Herring */
346b30be4dcSRob Herring tmp_size = tmp_end - tmp_start;
347b30be4dcSRob Herring ret = fdt_find_and_del_mem_rsv(fdt, tmp_start, tmp_size);
348b30be4dcSRob Herring if (ret == -ENOENT)
349b30be4dcSRob Herring ret = fdt_find_and_del_mem_rsv(fdt, tmp_start,
350b30be4dcSRob Herring round_up(tmp_size, PAGE_SIZE));
351b30be4dcSRob Herring if (ret == -EINVAL)
352b30be4dcSRob Herring goto out;
353b30be4dcSRob Herring }
354b30be4dcSRob Herring
355b30be4dcSRob Herring /* add initrd-* */
356b30be4dcSRob Herring if (initrd_load_addr) {
357cc6ef3d1SGeert Uytterhoeven ret = fdt_setprop_u64(fdt, chosen_node, "linux,initrd-start",
358b30be4dcSRob Herring initrd_load_addr);
359b30be4dcSRob Herring if (ret)
360b30be4dcSRob Herring goto out;
361b30be4dcSRob Herring
362cc6ef3d1SGeert Uytterhoeven ret = fdt_setprop_u64(fdt, chosen_node, "linux,initrd-end",
363b30be4dcSRob Herring initrd_load_addr + initrd_len);
364b30be4dcSRob Herring if (ret)
365b30be4dcSRob Herring goto out;
366b30be4dcSRob Herring
367b30be4dcSRob Herring ret = fdt_add_mem_rsv(fdt, initrd_load_addr, initrd_len);
368b30be4dcSRob Herring if (ret)
369b30be4dcSRob Herring goto out;
370b30be4dcSRob Herring
371b30be4dcSRob Herring } else {
372cc6ef3d1SGeert Uytterhoeven ret = fdt_delprop(fdt, chosen_node, "linux,initrd-start");
373b30be4dcSRob Herring if (ret && (ret != -FDT_ERR_NOTFOUND))
374b30be4dcSRob Herring goto out;
375b30be4dcSRob Herring
376cc6ef3d1SGeert Uytterhoeven ret = fdt_delprop(fdt, chosen_node, "linux,initrd-end");
377b30be4dcSRob Herring if (ret && (ret != -FDT_ERR_NOTFOUND))
378b30be4dcSRob Herring goto out;
379b30be4dcSRob Herring }
380b30be4dcSRob Herring
381b30be4dcSRob Herring if (image->type == KEXEC_TYPE_CRASH) {
382b30be4dcSRob Herring /* add linux,elfcorehdr */
383b30be4dcSRob Herring ret = fdt_appendprop_addrrange(fdt, 0, chosen_node,
384cc6ef3d1SGeert Uytterhoeven "linux,elfcorehdr", image->elf_load_addr,
385b30be4dcSRob Herring image->elf_headers_sz);
386b30be4dcSRob Herring if (ret)
387b30be4dcSRob Herring goto out;
388b30be4dcSRob Herring
389b30be4dcSRob Herring /*
390b30be4dcSRob Herring * Avoid elfcorehdr from being stomped on in kdump kernel by
391b30be4dcSRob Herring * setting up memory reserve map.
392b30be4dcSRob Herring */
393b30be4dcSRob Herring ret = fdt_add_mem_rsv(fdt, image->elf_load_addr,
394b30be4dcSRob Herring image->elf_headers_sz);
395b30be4dcSRob Herring if (ret)
396b30be4dcSRob Herring goto out;
397b30be4dcSRob Herring
398b30be4dcSRob Herring /* add linux,usable-memory-range */
399b30be4dcSRob Herring ret = fdt_appendprop_addrrange(fdt, 0, chosen_node,
400cc6ef3d1SGeert Uytterhoeven "linux,usable-memory-range", crashk_res.start,
401b30be4dcSRob Herring crashk_res.end - crashk_res.start + 1);
402b30be4dcSRob Herring if (ret)
403b30be4dcSRob Herring goto out;
4048af6b91fSZhen Lei
4058af6b91fSZhen Lei if (crashk_low_res.end) {
4068af6b91fSZhen Lei ret = fdt_appendprop_addrrange(fdt, 0, chosen_node,
4078af6b91fSZhen Lei "linux,usable-memory-range",
4088af6b91fSZhen Lei crashk_low_res.start,
4098af6b91fSZhen Lei crashk_low_res.end - crashk_low_res.start + 1);
4108af6b91fSZhen Lei if (ret)
4118af6b91fSZhen Lei goto out;
4128af6b91fSZhen Lei }
413b30be4dcSRob Herring }
414b30be4dcSRob Herring
415b30be4dcSRob Herring /* add bootargs */
416b30be4dcSRob Herring if (cmdline) {
417cc6ef3d1SGeert Uytterhoeven ret = fdt_setprop_string(fdt, chosen_node, "bootargs", cmdline);
418b30be4dcSRob Herring if (ret)
419b30be4dcSRob Herring goto out;
420b30be4dcSRob Herring } else {
421cc6ef3d1SGeert Uytterhoeven ret = fdt_delprop(fdt, chosen_node, "bootargs");
422b30be4dcSRob Herring if (ret && (ret != -FDT_ERR_NOTFOUND))
423b30be4dcSRob Herring goto out;
424b30be4dcSRob Herring }
425b30be4dcSRob Herring
426b30be4dcSRob Herring /* add kaslr-seed */
427cc6ef3d1SGeert Uytterhoeven ret = fdt_delprop(fdt, chosen_node, "kaslr-seed");
428b30be4dcSRob Herring if (ret == -FDT_ERR_NOTFOUND)
429b30be4dcSRob Herring ret = 0;
430b30be4dcSRob Herring else if (ret)
431b30be4dcSRob Herring goto out;
432b30be4dcSRob Herring
433b30be4dcSRob Herring if (rng_is_initialized()) {
434b30be4dcSRob Herring u64 seed = get_random_u64();
435b30be4dcSRob Herring
436cc6ef3d1SGeert Uytterhoeven ret = fdt_setprop_u64(fdt, chosen_node, "kaslr-seed", seed);
437b30be4dcSRob Herring if (ret)
438b30be4dcSRob Herring goto out;
439b30be4dcSRob Herring } else {
440b30be4dcSRob Herring pr_notice("RNG is not initialised: omitting \"%s\" property\n",
441cc6ef3d1SGeert Uytterhoeven "kaslr-seed");
442b30be4dcSRob Herring }
443b30be4dcSRob Herring
444b30be4dcSRob Herring /* add rng-seed */
445b30be4dcSRob Herring if (rng_is_initialized()) {
446b30be4dcSRob Herring void *rng_seed;
447b30be4dcSRob Herring
448cc6ef3d1SGeert Uytterhoeven ret = fdt_setprop_placeholder(fdt, chosen_node, "rng-seed",
449b30be4dcSRob Herring RNG_SEED_SIZE, &rng_seed);
450b30be4dcSRob Herring if (ret)
451b30be4dcSRob Herring goto out;
452b30be4dcSRob Herring get_random_bytes(rng_seed, RNG_SEED_SIZE);
453b30be4dcSRob Herring } else {
454b30be4dcSRob Herring pr_notice("RNG is not initialised: omitting \"%s\" property\n",
455cc6ef3d1SGeert Uytterhoeven "rng-seed");
456b30be4dcSRob Herring }
457b30be4dcSRob Herring
458b30be4dcSRob Herring ret = fdt_setprop(fdt, chosen_node, "linux,booted-from-kexec", NULL, 0);
459fee3ff99SLakshmi Ramasubramanian if (ret)
460fee3ff99SLakshmi Ramasubramanian goto out;
461fee3ff99SLakshmi Ramasubramanian
462fee3ff99SLakshmi Ramasubramanian remove_ima_buffer(fdt, chosen_node);
463fee3ff99SLakshmi Ramasubramanian ret = setup_ima_buffer(image, fdt, fdt_path_offset(fdt, "/chosen"));
464b30be4dcSRob Herring
465b30be4dcSRob Herring out:
466b30be4dcSRob Herring if (ret) {
467b30be4dcSRob Herring kvfree(fdt);
468b30be4dcSRob Herring fdt = NULL;
469b30be4dcSRob Herring }
470b30be4dcSRob Herring
471b30be4dcSRob Herring return fdt;
472b30be4dcSRob Herring }
473