xref: /openbmc/linux/drivers/of/kexec.c (revision f0362a25)
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 
187*f0362a25SRik van Riel 	memblock_free_late(addr, size);
188*f0362a25SRik 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. */
304b30be4dcSRob Herring 	ret = fdt_find_and_del_mem_rsv(fdt, __pa(initial_boot_params),
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