2017-12-27 12:55:14 -06:00
|
|
|
// SPDX-License-Identifier: GPL-2.0
|
2009-11-23 14:53:09 -07:00
|
|
|
/*
|
|
|
|
* Functions for working with the Flattened Device Tree data format
|
|
|
|
*
|
|
|
|
* Copyright 2009 Benjamin Herrenschmidt, IBM Corp
|
|
|
|
* benh@kernel.crashing.org
|
|
|
|
*/
|
|
|
|
|
2016-10-17 12:21:23 -07:00
|
|
|
#define pr_fmt(fmt) "OF: fdt: " fmt
|
2016-06-15 08:32:18 -05:00
|
|
|
|
2021-08-11 10:51:01 +02:00
|
|
|
#include <linux/crash_dump.h>
|
2014-11-14 18:05:35 +01:00
|
|
|
#include <linux/crc32.h>
|
2009-11-23 20:07:01 -07:00
|
|
|
#include <linux/kernel.h>
|
2009-11-24 03:26:58 -07:00
|
|
|
#include <linux/initrd.h>
|
2013-08-28 21:18:32 +01:00
|
|
|
#include <linux/memblock.h>
|
2015-12-05 16:13:53 -08:00
|
|
|
#include <linux/mutex.h>
|
2009-11-23 14:53:09 -07:00
|
|
|
#include <linux/of.h>
|
|
|
|
#include <linux/of_fdt.h>
|
2014-02-28 14:42:47 +01:00
|
|
|
#include <linux/sizes.h>
|
2010-02-14 07:13:47 -07:00
|
|
|
#include <linux/string.h>
|
|
|
|
#include <linux/errno.h>
|
2010-11-18 15:55:02 -08:00
|
|
|
#include <linux/slab.h>
|
2014-04-02 15:10:14 -05:00
|
|
|
#include <linux/libfdt.h>
|
2014-04-02 16:56:48 -05:00
|
|
|
#include <linux/debugfs.h>
|
2014-03-27 08:07:01 -05:00
|
|
|
#include <linux/serial_core.h>
|
2014-11-14 18:05:35 +01:00
|
|
|
#include <linux/sysfs.h>
|
2019-08-23 14:24:51 +08:00
|
|
|
#include <linux/random.h>
|
2025-05-09 00:46:27 -07:00
|
|
|
#include <linux/kexec_handover.h>
|
2010-02-01 21:34:14 -07:00
|
|
|
|
ARM: prom.h: Fix build error by removing unneeded header file
Fix the following build error:
CC [M] fs/udf/balloc.o
In file included from /home/fabio/next/linux-next/arch/arm/include/asm/prom.h:16,
from include/linux/of.h:140,
from include/asm-generic/gpio.h:7,
from arch/arm/plat-mxc/include/mach/irqs.h:14,
from /home/fabio/next/linux-next/arch/arm/include/asm/irq.h:4,
from /home/fabio/next/linux-next/arch/arm/include/asm/hardirq.h:6,
from include/linux/hardirq.h:7,
from include/linux/highmem.h:8,
from include/linux/pagemap.h:10,
from include/linux/buffer_head.h:13,
from fs/udf/udfdecl.h:11,
from fs/udf/balloc.c:22:
/home/fabio/next/linux-next/arch/arm/include/asm/setup.h:146: error: redefinition of 'struct tag'
Signed-off-by: Fabio Estevam <fabio.estevam@freescale.com>
[grant.likely: fix build failure on drivers/of/fdt.c]
Signed-off-by: Grant Likely <grant.likely@secretlab.ca>
2012-01-02 14:19:03 -02:00
|
|
|
#include <asm/setup.h> /* for COMMAND_LINE_SIZE */
|
2010-02-14 07:13:47 -07:00
|
|
|
#include <asm/page.h>
|
|
|
|
|
2017-04-25 17:09:54 -07:00
|
|
|
#include "of_private.h"
|
|
|
|
|
2024-02-16 17:05:51 -08:00
|
|
|
/*
|
|
|
|
* __dtb_empty_root_begin[] and __dtb_empty_root_end[] magically created by
|
2024-09-05 08:47:38 +09:00
|
|
|
* cmd_wrap_S_dtb in scripts/Makefile.dtbs
|
2024-02-16 17:05:51 -08:00
|
|
|
*/
|
|
|
|
extern uint8_t __dtb_empty_root_begin[];
|
|
|
|
extern uint8_t __dtb_empty_root_end[];
|
|
|
|
|
2014-07-15 10:03:35 -07:00
|
|
|
/*
|
|
|
|
* of_fdt_limit_memory - limit the number of regions in the /memory node
|
|
|
|
* @limit: maximum entries
|
|
|
|
*
|
|
|
|
* Adjust the flattened device tree to have at most 'limit' number of
|
|
|
|
* memory entries in the /memory node. This function may be called
|
|
|
|
* any time after initial_boot_param is set.
|
|
|
|
*/
|
2019-05-14 13:40:52 -07:00
|
|
|
void __init of_fdt_limit_memory(int limit)
|
2014-07-15 10:03:35 -07:00
|
|
|
{
|
|
|
|
int memory;
|
|
|
|
int len;
|
|
|
|
const void *val;
|
2018-08-30 11:20:20 -05:00
|
|
|
int cell_size = sizeof(uint32_t)*(dt_root_addr_cells + dt_root_size_cells);
|
2014-07-15 10:03:35 -07:00
|
|
|
|
|
|
|
memory = fdt_path_offset(initial_boot_params, "/memory");
|
|
|
|
if (memory > 0) {
|
|
|
|
val = fdt_getprop(initial_boot_params, memory, "reg", &len);
|
|
|
|
if (len > limit*cell_size) {
|
|
|
|
len = limit*cell_size;
|
|
|
|
pr_debug("Limiting number of entries to %d\n", limit);
|
|
|
|
fdt_setprop(initial_boot_params, memory, "reg", val,
|
|
|
|
len);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-03-11 12:13:03 -06:00
|
|
|
bool of_fdt_device_is_available(const void *blob, unsigned long node)
|
2017-09-28 19:20:32 -05:00
|
|
|
{
|
|
|
|
const char *status = fdt_getprop(blob, node, "status", NULL);
|
|
|
|
|
|
|
|
if (!status)
|
|
|
|
return true;
|
|
|
|
|
|
|
|
if (!strcmp(status, "ok") || !strcmp(status, "okay"))
|
|
|
|
return true;
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2013-08-29 13:30:35 +01:00
|
|
|
static void *unflatten_dt_alloc(void **mem, unsigned long size,
|
2009-11-23 20:07:00 -07:00
|
|
|
unsigned long align)
|
|
|
|
{
|
|
|
|
void *res;
|
|
|
|
|
2013-08-29 13:30:35 +01:00
|
|
|
*mem = PTR_ALIGN(*mem, align);
|
|
|
|
res = *mem;
|
2009-11-23 20:07:00 -07:00
|
|
|
*mem += size;
|
|
|
|
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
2016-05-03 23:22:47 +10:00
|
|
|
static void populate_properties(const void *blob,
|
|
|
|
int offset,
|
|
|
|
void **mem,
|
|
|
|
struct device_node *np,
|
|
|
|
const char *nodename,
|
2014-10-03 16:28:27 +01:00
|
|
|
bool dryrun)
|
2009-11-23 20:07:00 -07:00
|
|
|
{
|
2016-05-03 23:22:47 +10:00
|
|
|
struct property *pp, **pprev = NULL;
|
|
|
|
int cur;
|
|
|
|
bool has_name = false;
|
|
|
|
|
|
|
|
pprev = &np->properties;
|
|
|
|
for (cur = fdt_first_property_offset(blob, offset);
|
|
|
|
cur >= 0;
|
|
|
|
cur = fdt_next_property_offset(blob, cur)) {
|
|
|
|
const __be32 *val;
|
|
|
|
const char *pname;
|
|
|
|
u32 sz;
|
|
|
|
|
|
|
|
val = fdt_getprop_by_offset(blob, cur, &pname, &sz);
|
|
|
|
if (!val) {
|
2016-06-15 08:32:18 -05:00
|
|
|
pr_warn("Cannot locate property at 0x%x\n", cur);
|
2016-05-03 23:22:47 +10:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!pname) {
|
2016-06-15 08:32:18 -05:00
|
|
|
pr_warn("Cannot find property name at 0x%x\n", cur);
|
2016-05-03 23:22:47 +10:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!strcmp(pname, "name"))
|
|
|
|
has_name = true;
|
|
|
|
|
|
|
|
pp = unflatten_dt_alloc(mem, sizeof(struct property),
|
|
|
|
__alignof__(struct property));
|
|
|
|
if (dryrun)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
/* We accept flattened tree phandles either in
|
|
|
|
* ePAPR-style "phandle" properties, or the
|
|
|
|
* legacy "linux,phandle" properties. If both
|
|
|
|
* appear and have different values, things
|
|
|
|
* will get weird. Don't do that.
|
|
|
|
*/
|
|
|
|
if (!strcmp(pname, "phandle") ||
|
|
|
|
!strcmp(pname, "linux,phandle")) {
|
|
|
|
if (!np->phandle)
|
|
|
|
np->phandle = be32_to_cpup(val);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* And we process the "ibm,phandle" property
|
|
|
|
* used in pSeries dynamic device tree
|
|
|
|
* stuff
|
|
|
|
*/
|
|
|
|
if (!strcmp(pname, "ibm,phandle"))
|
|
|
|
np->phandle = be32_to_cpup(val);
|
|
|
|
|
|
|
|
pp->name = (char *)pname;
|
|
|
|
pp->length = sz;
|
|
|
|
pp->value = (__be32 *)val;
|
|
|
|
*pprev = pp;
|
|
|
|
pprev = &pp->next;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* With version 0x10 we may not have the name property,
|
|
|
|
* recreate it here from the unit name if absent
|
|
|
|
*/
|
|
|
|
if (!has_name) {
|
|
|
|
const char *p = nodename, *ps = p, *pa = NULL;
|
|
|
|
int len;
|
|
|
|
|
|
|
|
while (*p) {
|
|
|
|
if ((*p) == '@')
|
|
|
|
pa = p;
|
|
|
|
else if ((*p) == '/')
|
|
|
|
ps = p + 1;
|
|
|
|
p++;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (pa < ps)
|
|
|
|
pa = p;
|
|
|
|
len = (pa - ps) + 1;
|
|
|
|
pp = unflatten_dt_alloc(mem, sizeof(struct property) + len,
|
|
|
|
__alignof__(struct property));
|
|
|
|
if (!dryrun) {
|
|
|
|
pp->name = "name";
|
|
|
|
pp->length = len;
|
|
|
|
pp->value = pp + 1;
|
|
|
|
*pprev = pp;
|
|
|
|
memcpy(pp->value, ps, len - 1);
|
|
|
|
((char *)pp->value)[len - 1] = 0;
|
|
|
|
pr_debug("fixed up name for %s -> %s\n",
|
|
|
|
nodename, (char *)pp->value);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-04-04 22:28:45 -05:00
|
|
|
static int populate_node(const void *blob,
|
2017-06-01 18:01:47 -05:00
|
|
|
int offset,
|
|
|
|
void **mem,
|
|
|
|
struct device_node *dad,
|
|
|
|
struct device_node **pnp,
|
|
|
|
bool dryrun)
|
2016-05-03 23:22:47 +10:00
|
|
|
{
|
2009-11-23 20:07:00 -07:00
|
|
|
struct device_node *np;
|
2014-04-02 15:10:14 -05:00
|
|
|
const char *pathp;
|
2021-04-04 22:28:45 -05:00
|
|
|
int len;
|
2009-11-23 20:07:00 -07:00
|
|
|
|
2021-04-04 22:28:45 -05:00
|
|
|
pathp = fdt_get_name(blob, offset, &len);
|
2016-05-03 23:22:47 +10:00
|
|
|
if (!pathp) {
|
|
|
|
*pnp = NULL;
|
2021-04-04 22:28:45 -05:00
|
|
|
return len;
|
2016-05-03 23:22:47 +10:00
|
|
|
}
|
2014-04-02 15:10:14 -05:00
|
|
|
|
2021-04-04 22:28:45 -05:00
|
|
|
len++;
|
2009-11-23 20:07:00 -07:00
|
|
|
|
2021-04-04 22:28:45 -05:00
|
|
|
np = unflatten_dt_alloc(mem, sizeof(struct device_node) + len,
|
2009-11-23 20:07:00 -07:00
|
|
|
__alignof__(struct device_node));
|
2014-10-03 16:28:27 +01:00
|
|
|
if (!dryrun) {
|
2012-11-14 22:37:12 +00:00
|
|
|
char *fn;
|
2013-12-13 20:08:59 +02:00
|
|
|
of_node_init(np);
|
2012-11-14 22:37:12 +00:00
|
|
|
np->full_name = fn = ((char *)np) + sizeof(*np);
|
2017-06-01 18:01:47 -05:00
|
|
|
|
2021-04-04 22:28:45 -05:00
|
|
|
memcpy(fn, pathp, len);
|
2012-11-14 22:37:12 +00:00
|
|
|
|
2009-11-23 20:07:00 -07:00
|
|
|
if (dad != NULL) {
|
|
|
|
np->parent = dad;
|
2014-11-28 16:03:33 +00:00
|
|
|
np->sibling = dad->child;
|
|
|
|
dad->child = np;
|
2009-11-23 20:07:00 -07:00
|
|
|
}
|
|
|
|
}
|
2014-04-02 15:10:14 -05:00
|
|
|
|
2016-05-03 23:22:47 +10:00
|
|
|
populate_properties(blob, offset, mem, np, pathp, dryrun);
|
2014-10-03 16:28:27 +01:00
|
|
|
if (!dryrun) {
|
2009-11-23 20:07:00 -07:00
|
|
|
np->name = of_get_property(np, "name", NULL);
|
|
|
|
if (!np->name)
|
|
|
|
np->name = "<NULL>";
|
|
|
|
}
|
2014-04-02 15:10:14 -05:00
|
|
|
|
2016-05-03 23:22:47 +10:00
|
|
|
*pnp = np;
|
2022-08-01 12:05:06 +00:00
|
|
|
return 0;
|
2016-05-03 23:22:47 +10:00
|
|
|
}
|
|
|
|
|
2016-05-03 23:22:48 +10:00
|
|
|
static void reverse_nodes(struct device_node *parent)
|
|
|
|
{
|
|
|
|
struct device_node *child, *next;
|
|
|
|
|
|
|
|
/* In-depth first */
|
|
|
|
child = parent->child;
|
|
|
|
while (child) {
|
|
|
|
reverse_nodes(child);
|
|
|
|
|
|
|
|
child = child->sibling;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Reverse the nodes in the child list */
|
|
|
|
child = parent->child;
|
|
|
|
parent->child = NULL;
|
|
|
|
while (child) {
|
|
|
|
next = child->sibling;
|
|
|
|
|
|
|
|
child->sibling = parent->child;
|
|
|
|
parent->child = child;
|
|
|
|
child = next;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-05-03 23:22:47 +10:00
|
|
|
/**
|
2016-05-03 23:22:49 +10:00
|
|
|
* unflatten_dt_nodes - Alloc and populate a device_node from the flat tree
|
2016-05-03 23:22:47 +10:00
|
|
|
* @blob: The parent device tree blob
|
|
|
|
* @mem: Memory chunk to use for allocating device nodes and properties
|
|
|
|
* @dad: Parent struct device_node
|
|
|
|
* @nodepp: The device_node tree created by the call
|
2016-05-03 23:22:48 +10:00
|
|
|
*
|
2021-03-25 10:47:12 -06:00
|
|
|
* Return: The size of unflattened device tree or error code
|
2016-05-03 23:22:47 +10:00
|
|
|
*/
|
2016-05-03 23:22:49 +10:00
|
|
|
static int unflatten_dt_nodes(const void *blob,
|
|
|
|
void *mem,
|
|
|
|
struct device_node *dad,
|
|
|
|
struct device_node **nodepp)
|
2016-05-03 23:22:47 +10:00
|
|
|
{
|
2016-05-03 23:22:48 +10:00
|
|
|
struct device_node *root;
|
2016-06-09 15:50:49 +10:00
|
|
|
int offset = 0, depth = 0, initial_depth = 0;
|
2016-05-03 23:22:48 +10:00
|
|
|
#define FDT_MAX_DEPTH 64
|
|
|
|
struct device_node *nps[FDT_MAX_DEPTH];
|
|
|
|
void *base = mem;
|
|
|
|
bool dryrun = !base;
|
2021-04-04 22:28:45 -05:00
|
|
|
int ret;
|
2016-05-03 23:22:47 +10:00
|
|
|
|
2016-05-03 23:22:48 +10:00
|
|
|
if (nodepp)
|
|
|
|
*nodepp = NULL;
|
|
|
|
|
2016-06-09 15:50:49 +10:00
|
|
|
/*
|
|
|
|
* We're unflattening device sub-tree if @dad is valid. There are
|
|
|
|
* possibly multiple nodes in the first level of depth. We need
|
|
|
|
* set @depth to 1 to make fdt_next_node() happy as it bails
|
|
|
|
* immediately when negative @depth is found. Otherwise, the device
|
|
|
|
* nodes except the first one won't be unflattened successfully.
|
|
|
|
*/
|
|
|
|
if (dad)
|
|
|
|
depth = initial_depth = 1;
|
|
|
|
|
2016-05-03 23:22:48 +10:00
|
|
|
root = dad;
|
2016-05-11 13:36:57 -04:00
|
|
|
nps[depth] = dad;
|
2016-06-09 15:50:49 +10:00
|
|
|
|
2016-05-03 23:22:48 +10:00
|
|
|
for (offset = 0;
|
2016-06-09 15:50:49 +10:00
|
|
|
offset >= 0 && depth >= initial_depth;
|
2016-05-03 23:22:48 +10:00
|
|
|
offset = fdt_next_node(blob, offset, &depth)) {
|
2022-08-13 23:34:16 +03:00
|
|
|
if (WARN_ON_ONCE(depth >= FDT_MAX_DEPTH - 1))
|
2016-05-03 23:22:48 +10:00
|
|
|
continue;
|
2016-05-03 23:22:47 +10:00
|
|
|
|
2017-10-03 11:07:55 -05:00
|
|
|
if (!IS_ENABLED(CONFIG_OF_KOBJ) &&
|
|
|
|
!of_fdt_device_is_available(blob, offset))
|
|
|
|
continue;
|
|
|
|
|
2021-04-04 22:28:45 -05:00
|
|
|
ret = populate_node(blob, offset, &mem, nps[depth],
|
|
|
|
&nps[depth+1], dryrun);
|
|
|
|
if (ret < 0)
|
|
|
|
return ret;
|
2016-05-03 23:22:48 +10:00
|
|
|
|
|
|
|
if (!dryrun && nodepp && !*nodepp)
|
2016-05-11 13:36:57 -04:00
|
|
|
*nodepp = nps[depth+1];
|
2016-05-03 23:22:48 +10:00
|
|
|
if (!dryrun && !root)
|
2016-05-11 13:36:57 -04:00
|
|
|
root = nps[depth+1];
|
2016-05-03 23:22:48 +10:00
|
|
|
}
|
2014-04-02 15:10:14 -05:00
|
|
|
|
2016-05-03 23:22:48 +10:00
|
|
|
if (offset < 0 && offset != -FDT_ERR_NOTFOUND) {
|
2016-06-15 08:32:18 -05:00
|
|
|
pr_err("Error %d processing FDT\n", offset);
|
2016-05-03 23:22:48 +10:00
|
|
|
return -EINVAL;
|
|
|
|
}
|
2014-04-02 15:10:14 -05:00
|
|
|
|
2014-11-28 16:03:33 +00:00
|
|
|
/*
|
|
|
|
* Reverse the child list. Some drivers assumes node order matches .dts
|
|
|
|
* node order
|
|
|
|
*/
|
2016-05-03 23:22:48 +10:00
|
|
|
if (!dryrun)
|
|
|
|
reverse_nodes(root);
|
2014-04-02 15:10:14 -05:00
|
|
|
|
2016-05-03 23:22:48 +10:00
|
|
|
return mem - base;
|
2009-11-23 20:07:00 -07:00
|
|
|
}
|
2009-11-23 20:07:01 -07:00
|
|
|
|
2010-11-18 15:55:02 -08:00
|
|
|
/**
|
|
|
|
* __unflatten_device_tree - create tree of device_nodes from flat blob
|
|
|
|
* @blob: The blob to expand
|
2016-05-03 23:22:50 +10:00
|
|
|
* @dad: Parent device node
|
2010-11-18 15:55:02 -08:00
|
|
|
* @mynodes: The device_node tree created by the call
|
|
|
|
* @dt_alloc: An allocator that provides a virtual address to memory
|
|
|
|
* for the resulting tree
|
2017-10-13 00:41:29 -07:00
|
|
|
* @detached: if true set OF_DETACHED on @mynodes
|
2016-05-03 23:22:51 +10:00
|
|
|
*
|
2021-03-26 13:26:06 -06:00
|
|
|
* unflattens a device-tree, creating the tree of struct device_node. It also
|
|
|
|
* fills the "name" and "type" pointers of the nodes so the normal device-tree
|
|
|
|
* walking functions can be used.
|
|
|
|
*
|
2021-03-25 10:47:12 -06:00
|
|
|
* Return: NULL on failure or the memory chunk containing the unflattened
|
2016-05-03 23:22:51 +10:00
|
|
|
* device tree on success.
|
2010-11-18 15:55:02 -08:00
|
|
|
*/
|
2017-04-25 17:09:54 -07:00
|
|
|
void *__unflatten_device_tree(const void *blob,
|
|
|
|
struct device_node *dad,
|
|
|
|
struct device_node **mynodes,
|
|
|
|
void *(*dt_alloc)(u64 size, u64 align),
|
|
|
|
bool detached)
|
2010-11-18 15:55:02 -08:00
|
|
|
{
|
2016-05-03 23:22:48 +10:00
|
|
|
int size;
|
2014-04-02 15:10:14 -05:00
|
|
|
void *mem;
|
2021-04-04 22:28:45 -05:00
|
|
|
int ret;
|
|
|
|
|
|
|
|
if (mynodes)
|
|
|
|
*mynodes = NULL;
|
2010-11-18 15:55:02 -08:00
|
|
|
|
|
|
|
pr_debug(" -> unflatten_device_tree()\n");
|
|
|
|
|
|
|
|
if (!blob) {
|
|
|
|
pr_debug("No device tree pointer\n");
|
2016-05-03 23:22:51 +10:00
|
|
|
return NULL;
|
2010-11-18 15:55:02 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
pr_debug("Unflattening device tree:\n");
|
2014-04-01 22:48:01 -05:00
|
|
|
pr_debug("magic: %08x\n", fdt_magic(blob));
|
|
|
|
pr_debug("size: %08x\n", fdt_totalsize(blob));
|
|
|
|
pr_debug("version: %08x\n", fdt_version(blob));
|
2010-11-18 15:55:02 -08:00
|
|
|
|
2014-04-01 22:48:01 -05:00
|
|
|
if (fdt_check_header(blob)) {
|
2010-11-18 15:55:02 -08:00
|
|
|
pr_err("Invalid device tree blob header\n");
|
2016-05-03 23:22:51 +10:00
|
|
|
return NULL;
|
2010-11-18 15:55:02 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
/* First pass, scan for size */
|
2016-05-03 23:22:50 +10:00
|
|
|
size = unflatten_dt_nodes(blob, NULL, dad, NULL);
|
2021-04-04 22:28:45 -05:00
|
|
|
if (size <= 0)
|
2016-05-03 23:22:51 +10:00
|
|
|
return NULL;
|
2010-11-18 15:55:02 -08:00
|
|
|
|
2016-05-03 23:22:48 +10:00
|
|
|
size = ALIGN(size, 4);
|
|
|
|
pr_debug(" size is %d, allocating...\n", size);
|
2010-11-18 15:55:02 -08:00
|
|
|
|
|
|
|
/* Allocate memory for the expanded device tree */
|
2013-08-29 13:30:35 +01:00
|
|
|
mem = dt_alloc(size + 4, __alignof__(struct device_node));
|
2017-05-17 17:29:09 +02:00
|
|
|
if (!mem)
|
|
|
|
return NULL;
|
|
|
|
|
2013-08-29 13:30:35 +01:00
|
|
|
memset(mem, 0, size);
|
2010-11-18 15:55:02 -08:00
|
|
|
|
2013-08-29 13:30:35 +01:00
|
|
|
*(__be32 *)(mem + size) = cpu_to_be32(0xdeadbeef);
|
2013-08-12 13:06:53 +02:00
|
|
|
|
2013-08-29 13:30:35 +01:00
|
|
|
pr_debug(" unflattening %p...\n", mem);
|
2010-11-18 15:55:02 -08:00
|
|
|
|
|
|
|
/* Second pass, do actual unflattening */
|
2021-04-04 22:28:45 -05:00
|
|
|
ret = unflatten_dt_nodes(blob, mem, dad, mynodes);
|
|
|
|
|
2013-08-29 13:30:35 +01:00
|
|
|
if (be32_to_cpup(mem + size) != 0xdeadbeef)
|
2019-10-18 11:18:33 +08:00
|
|
|
pr_warn("End of tree marker overwritten: %08x\n",
|
|
|
|
be32_to_cpup(mem + size));
|
2010-11-18 15:55:02 -08:00
|
|
|
|
2021-04-04 22:28:45 -05:00
|
|
|
if (ret <= 0)
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
if (detached && mynodes && *mynodes) {
|
2016-07-19 00:01:12 +02:00
|
|
|
of_node_set_flag(*mynodes, OF_DETACHED);
|
|
|
|
pr_debug("unflattened tree is detached\n");
|
|
|
|
}
|
|
|
|
|
2010-11-18 15:55:02 -08:00
|
|
|
pr_debug(" <- unflatten_device_tree()\n");
|
2016-05-03 23:22:51 +10:00
|
|
|
return mem;
|
2010-11-18 15:55:02 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
static void *kernel_tree_alloc(u64 size, u64 align)
|
|
|
|
{
|
|
|
|
return kzalloc(size, GFP_KERNEL);
|
|
|
|
}
|
|
|
|
|
2015-12-05 16:13:53 -08:00
|
|
|
static DEFINE_MUTEX(of_fdt_unflatten_mutex);
|
|
|
|
|
2010-11-18 15:55:02 -08:00
|
|
|
/**
|
|
|
|
* of_fdt_unflatten_tree - create tree of device_nodes from flat blob
|
2016-05-03 23:22:50 +10:00
|
|
|
* @blob: Flat device tree blob
|
|
|
|
* @dad: Parent device node
|
|
|
|
* @mynodes: The device tree created by the call
|
2010-11-18 15:55:02 -08:00
|
|
|
*
|
|
|
|
* unflattens the device-tree passed by the firmware, creating the
|
|
|
|
* tree of struct device_node. It also fills the "name" and "type"
|
|
|
|
* pointers of the nodes so the normal device-tree walking functions
|
|
|
|
* can be used.
|
2016-05-03 23:22:51 +10:00
|
|
|
*
|
2021-03-25 10:47:12 -06:00
|
|
|
* Return: NULL on failure or the memory chunk containing the unflattened
|
2016-05-03 23:22:51 +10:00
|
|
|
* device tree on success.
|
2010-11-18 15:55:02 -08:00
|
|
|
*/
|
2016-05-03 23:22:51 +10:00
|
|
|
void *of_fdt_unflatten_tree(const unsigned long *blob,
|
|
|
|
struct device_node *dad,
|
|
|
|
struct device_node **mynodes)
|
2010-11-18 15:55:02 -08:00
|
|
|
{
|
2016-05-03 23:22:51 +10:00
|
|
|
void *mem;
|
|
|
|
|
2015-12-05 16:13:53 -08:00
|
|
|
mutex_lock(&of_fdt_unflatten_mutex);
|
2016-07-19 00:01:12 +02:00
|
|
|
mem = __unflatten_device_tree(blob, dad, mynodes, &kernel_tree_alloc,
|
|
|
|
true);
|
2015-12-05 16:13:53 -08:00
|
|
|
mutex_unlock(&of_fdt_unflatten_mutex);
|
2016-05-03 23:22:51 +10:00
|
|
|
|
|
|
|
return mem;
|
2010-11-18 15:55:02 -08:00
|
|
|
}
|
|
|
|
EXPORT_SYMBOL_GPL(of_fdt_unflatten_tree);
|
|
|
|
|
2010-11-18 15:55:01 -08:00
|
|
|
/* Everything below here references initial_boot_params directly. */
|
|
|
|
int __initdata dt_root_addr_cells;
|
|
|
|
int __initdata dt_root_size_cells;
|
|
|
|
|
2019-05-14 13:40:53 -07:00
|
|
|
void *initial_boot_params __ro_after_init;
|
2024-10-23 18:14:26 +01:00
|
|
|
phys_addr_t initial_boot_params_pa __ro_after_init;
|
2010-11-18 15:55:01 -08:00
|
|
|
|
|
|
|
#ifdef CONFIG_OF_EARLY_FLATTREE
|
|
|
|
|
2014-11-14 18:05:35 +01:00
|
|
|
static u32 of_fdt_crc32;
|
|
|
|
|
2021-08-11 10:51:01 +02:00
|
|
|
/*
|
2021-08-25 11:40:40 +02:00
|
|
|
* fdt_reserve_elfcorehdr() - reserves memory for elf core header
|
2021-08-11 10:51:01 +02:00
|
|
|
*
|
|
|
|
* This function reserves the memory occupied by an elf core header
|
|
|
|
* described in the device tree. This region contains all the
|
|
|
|
* information about primary kernel's core image and is used by a dump
|
|
|
|
* capture kernel to access the system memory on primary kernel.
|
|
|
|
*/
|
2021-08-25 11:40:40 +02:00
|
|
|
static void __init fdt_reserve_elfcorehdr(void)
|
2021-08-11 10:51:01 +02:00
|
|
|
{
|
|
|
|
if (!IS_ENABLED(CONFIG_CRASH_DUMP) || !elfcorehdr_size)
|
|
|
|
return;
|
|
|
|
|
|
|
|
if (memblock_is_region_reserved(elfcorehdr_addr, elfcorehdr_size)) {
|
|
|
|
pr_warn("elfcorehdr is overlapped\n");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
memblock_reserve(elfcorehdr_addr, elfcorehdr_size);
|
|
|
|
|
|
|
|
pr_info("Reserving %llu KiB of memory at 0x%llx for elfcorehdr\n",
|
|
|
|
elfcorehdr_size >> 10, elfcorehdr_addr);
|
|
|
|
}
|
|
|
|
|
2014-02-28 14:42:47 +01:00
|
|
|
/**
|
|
|
|
* early_init_fdt_scan_reserved_mem() - create reserved memory regions
|
|
|
|
*
|
|
|
|
* This function grabs memory from early allocator for device exclusive use
|
|
|
|
* defined in device tree structures. It should be called by arch specific code
|
|
|
|
* once the early allocator (i.e. memblock) has been fully activated.
|
|
|
|
*/
|
|
|
|
void __init early_init_fdt_scan_reserved_mem(void)
|
|
|
|
{
|
2014-04-01 22:46:48 -05:00
|
|
|
int n;
|
2025-01-09 21:27:04 +08:00
|
|
|
int res;
|
2014-04-01 22:46:48 -05:00
|
|
|
u64 base, size;
|
|
|
|
|
2014-03-13 16:36:36 -05:00
|
|
|
if (!initial_boot_params)
|
|
|
|
return;
|
|
|
|
|
2023-04-24 12:38:46 +01:00
|
|
|
fdt_scan_reserved_mem();
|
|
|
|
fdt_reserve_elfcorehdr();
|
|
|
|
|
2014-04-01 22:46:48 -05:00
|
|
|
/* Process header /memreserve/ fields */
|
|
|
|
for (n = 0; ; n++) {
|
2025-01-09 21:27:04 +08:00
|
|
|
res = fdt_get_mem_rsv(initial_boot_params, n, &base, &size);
|
|
|
|
if (res) {
|
|
|
|
pr_err("Invalid memory reservation block index %d\n", n);
|
|
|
|
break;
|
|
|
|
}
|
2014-04-01 22:46:48 -05:00
|
|
|
if (!size)
|
|
|
|
break;
|
2022-07-22 18:53:31 -07:00
|
|
|
memblock_reserve(base, size);
|
2014-04-01 22:46:48 -05:00
|
|
|
}
|
2014-02-28 14:42:47 +01:00
|
|
|
}
|
|
|
|
|
2015-06-01 13:40:31 +02:00
|
|
|
/**
|
|
|
|
* early_init_fdt_reserve_self() - reserve the memory used by the FDT blob
|
|
|
|
*/
|
|
|
|
void __init early_init_fdt_reserve_self(void)
|
|
|
|
{
|
|
|
|
if (!initial_boot_params)
|
|
|
|
return;
|
|
|
|
|
|
|
|
/* Reserve the dtb region */
|
2022-07-22 18:53:31 -07:00
|
|
|
memblock_reserve(__pa(initial_boot_params),
|
|
|
|
fdt_totalsize(initial_boot_params));
|
2015-06-01 13:40:31 +02:00
|
|
|
}
|
|
|
|
|
2010-11-18 15:55:01 -08:00
|
|
|
/**
|
|
|
|
* of_scan_flat_dt - scan flattened tree blob and call callback on each.
|
|
|
|
* @it: callback function
|
|
|
|
* @data: context data pointer
|
|
|
|
*
|
|
|
|
* This function is used to scan the flattened device-tree, it is
|
|
|
|
* used to extract the memory information at boot before we can
|
|
|
|
* unflatten the tree
|
|
|
|
*/
|
|
|
|
int __init of_scan_flat_dt(int (*it)(unsigned long node,
|
|
|
|
const char *uname, int depth,
|
|
|
|
void *data),
|
|
|
|
void *data)
|
|
|
|
{
|
2014-04-02 15:10:14 -05:00
|
|
|
const void *blob = initial_boot_params;
|
|
|
|
const char *pathp;
|
|
|
|
int offset, rc = 0, depth = -1;
|
|
|
|
|
2016-11-23 10:40:07 +01:00
|
|
|
if (!blob)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
for (offset = fdt_next_node(blob, -1, &depth);
|
|
|
|
offset >= 0 && depth >= 0 && !rc;
|
|
|
|
offset = fdt_next_node(blob, offset, &depth)) {
|
2014-04-02 15:10:14 -05:00
|
|
|
|
|
|
|
pathp = fdt_get_name(blob, offset, NULL);
|
|
|
|
rc = it(offset, pathp, depth, data);
|
|
|
|
}
|
2010-11-18 15:55:01 -08:00
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
|
2017-04-19 05:12:18 +10:00
|
|
|
/**
|
|
|
|
* of_scan_flat_dt_subnodes - scan sub-nodes of a node call callback on each.
|
2021-03-18 10:40:33 +00:00
|
|
|
* @parent: parent node
|
2017-04-19 05:12:18 +10:00
|
|
|
* @it: callback function
|
|
|
|
* @data: context data pointer
|
|
|
|
*
|
|
|
|
* This function is used to scan sub-nodes of a node.
|
|
|
|
*/
|
|
|
|
int __init of_scan_flat_dt_subnodes(unsigned long parent,
|
|
|
|
int (*it)(unsigned long node,
|
|
|
|
const char *uname,
|
|
|
|
void *data),
|
|
|
|
void *data)
|
|
|
|
{
|
|
|
|
const void *blob = initial_boot_params;
|
|
|
|
int node;
|
|
|
|
|
|
|
|
fdt_for_each_subnode(node, blob, parent) {
|
|
|
|
const char *pathp;
|
|
|
|
int rc;
|
|
|
|
|
|
|
|
pathp = fdt_get_name(blob, node, NULL);
|
|
|
|
rc = it(node, pathp, data);
|
|
|
|
if (rc)
|
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2016-04-07 20:03:33 +08:00
|
|
|
/**
|
|
|
|
* of_get_flat_dt_subnode_by_name - get the subnode by given name
|
|
|
|
*
|
|
|
|
* @node: the parent node
|
|
|
|
* @uname: the name of subnode
|
|
|
|
* @return offset of the subnode, or -FDT_ERR_NOTFOUND if there is none
|
|
|
|
*/
|
|
|
|
|
2019-05-14 13:40:52 -07:00
|
|
|
int __init of_get_flat_dt_subnode_by_name(unsigned long node, const char *uname)
|
2016-04-07 20:03:33 +08:00
|
|
|
{
|
|
|
|
return fdt_subnode_offset(initial_boot_params, node, uname);
|
|
|
|
}
|
|
|
|
|
2021-03-18 10:40:33 +00:00
|
|
|
/*
|
2010-11-18 15:55:01 -08:00
|
|
|
* of_get_flat_dt_root - find the root node in the flat blob
|
|
|
|
*/
|
|
|
|
unsigned long __init of_get_flat_dt_root(void)
|
|
|
|
{
|
2014-04-02 15:10:14 -05:00
|
|
|
return 0;
|
2010-11-18 15:55:01 -08:00
|
|
|
}
|
|
|
|
|
2021-03-18 10:40:33 +00:00
|
|
|
/*
|
2010-11-18 15:55:01 -08:00
|
|
|
* of_get_flat_dt_prop - Given a node in the flat blob, return the property ptr
|
|
|
|
*
|
|
|
|
* This function can be used within scan_flattened_dt callback to get
|
|
|
|
* access to properties
|
|
|
|
*/
|
2014-04-01 23:49:03 -05:00
|
|
|
const void *__init of_get_flat_dt_prop(unsigned long node, const char *name,
|
|
|
|
int *size)
|
2010-11-18 15:55:01 -08:00
|
|
|
{
|
2014-04-02 15:10:14 -05:00
|
|
|
return fdt_getprop(initial_boot_params, node, name, size);
|
2010-11-18 15:55:01 -08:00
|
|
|
}
|
|
|
|
|
2019-06-15 11:03:43 +08:00
|
|
|
/**
|
|
|
|
* of_fdt_is_compatible - Return true if given node from the given blob has
|
|
|
|
* compat in its compatible list
|
|
|
|
* @blob: A device tree blob
|
|
|
|
* @node: node to test
|
|
|
|
* @compat: compatible string to compare with compatible list.
|
|
|
|
*
|
2021-03-25 10:47:12 -06:00
|
|
|
* Return: a non-zero value on match with smaller values returned for more
|
2019-06-15 11:03:43 +08:00
|
|
|
* specific compatible values.
|
|
|
|
*/
|
|
|
|
static int of_fdt_is_compatible(const void *blob,
|
|
|
|
unsigned long node, const char *compat)
|
|
|
|
{
|
|
|
|
const char *cp;
|
|
|
|
int cplen;
|
|
|
|
unsigned long l, score = 0;
|
|
|
|
|
|
|
|
cp = fdt_getprop(blob, node, "compatible", &cplen);
|
|
|
|
if (cp == NULL)
|
|
|
|
return 0;
|
|
|
|
while (cplen > 0) {
|
|
|
|
score++;
|
|
|
|
if (of_compat_cmp(cp, compat, strlen(compat)) == 0)
|
|
|
|
return score;
|
|
|
|
l = strlen(cp) + 1;
|
|
|
|
cp += l;
|
|
|
|
cplen -= l;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2010-11-18 15:55:01 -08:00
|
|
|
/**
|
|
|
|
* of_flat_dt_is_compatible - Return true if given node has compat in compatible list
|
|
|
|
* @node: node to test
|
|
|
|
* @compat: compatible string to compare with compatible list.
|
|
|
|
*/
|
|
|
|
int __init of_flat_dt_is_compatible(unsigned long node, const char *compat)
|
|
|
|
{
|
|
|
|
return of_fdt_is_compatible(initial_boot_params, node, compat);
|
|
|
|
}
|
|
|
|
|
2021-03-18 10:40:33 +00:00
|
|
|
/*
|
2010-10-30 11:49:09 -04:00
|
|
|
* of_flat_dt_match - Return true if node matches a list of compatible values
|
|
|
|
*/
|
2019-05-14 13:40:52 -07:00
|
|
|
static int __init of_flat_dt_match(unsigned long node, const char *const *compat)
|
2010-10-30 11:49:09 -04:00
|
|
|
{
|
2019-06-15 11:03:43 +08:00
|
|
|
unsigned int tmp, score = 0;
|
|
|
|
|
|
|
|
if (!compat)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
while (*compat) {
|
|
|
|
tmp = of_fdt_is_compatible(initial_boot_params, node, *compat);
|
|
|
|
if (tmp && (score == 0 || (tmp < score)))
|
|
|
|
score = tmp;
|
|
|
|
compat++;
|
|
|
|
}
|
|
|
|
|
|
|
|
return score;
|
2010-10-30 11:49:09 -04:00
|
|
|
}
|
|
|
|
|
2021-03-18 10:40:33 +00:00
|
|
|
/*
|
|
|
|
* of_get_flat_dt_phandle - Given a node in the flat blob, return the phandle
|
2017-04-19 05:12:18 +10:00
|
|
|
*/
|
|
|
|
uint32_t __init of_get_flat_dt_phandle(unsigned long node)
|
|
|
|
{
|
|
|
|
return fdt_get_phandle(initial_boot_params, node);
|
|
|
|
}
|
|
|
|
|
2013-08-27 21:41:56 -05:00
|
|
|
const char * __init of_flat_dt_get_machine_name(void)
|
|
|
|
{
|
|
|
|
const char *name;
|
|
|
|
unsigned long dt_root = of_get_flat_dt_root();
|
|
|
|
|
|
|
|
name = of_get_flat_dt_prop(dt_root, "model", NULL);
|
|
|
|
if (!name)
|
|
|
|
name = of_get_flat_dt_prop(dt_root, "compatible", NULL);
|
|
|
|
return name;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* of_flat_dt_match_machine - Iterate match tables to find matching machine.
|
|
|
|
*
|
|
|
|
* @default_match: A machine specific ptr to return in case of no match.
|
|
|
|
* @get_next_compat: callback function to return next compatible match table.
|
|
|
|
*
|
|
|
|
* Iterate through machine match tables to find the best match for the machine
|
|
|
|
* compatible string in the FDT.
|
|
|
|
*/
|
|
|
|
const void * __init of_flat_dt_match_machine(const void *default_match,
|
|
|
|
const void * (*get_next_compat)(const char * const**))
|
|
|
|
{
|
|
|
|
const void *data = NULL;
|
|
|
|
const void *best_data = default_match;
|
|
|
|
const char *const *compat;
|
|
|
|
unsigned long dt_root;
|
|
|
|
unsigned int best_score = ~1, score = 0;
|
|
|
|
|
|
|
|
dt_root = of_get_flat_dt_root();
|
|
|
|
while ((data = get_next_compat(&compat))) {
|
|
|
|
score = of_flat_dt_match(dt_root, compat);
|
|
|
|
if (score > 0 && score < best_score) {
|
|
|
|
best_data = data;
|
|
|
|
best_score = score;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (!best_data) {
|
|
|
|
const char *prop;
|
2014-04-01 23:49:03 -05:00
|
|
|
int size;
|
2013-08-27 21:41:56 -05:00
|
|
|
|
|
|
|
pr_err("\n unrecognized device tree list:\n[ ");
|
|
|
|
|
|
|
|
prop = of_get_flat_dt_prop(dt_root, "compatible", &size);
|
|
|
|
if (prop) {
|
|
|
|
while (size > 0) {
|
|
|
|
printk("'%s' ", prop);
|
|
|
|
size -= strlen(prop) + 1;
|
|
|
|
prop += strlen(prop) + 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
printk("]\n\n");
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
pr_info("Machine model: %s\n", of_flat_dt_get_machine_name());
|
|
|
|
|
|
|
|
return best_data;
|
|
|
|
}
|
|
|
|
|
2016-02-16 13:52:33 +01:00
|
|
|
static void __early_init_dt_declare_initrd(unsigned long start,
|
|
|
|
unsigned long end)
|
|
|
|
{
|
2023-03-24 16:54:21 +01:00
|
|
|
/*
|
|
|
|
* __va() is not yet available this early on some platforms. In that
|
|
|
|
* case, the platform uses phys_initrd_start/phys_initrd_size instead
|
|
|
|
* and does the VA conversion itself.
|
2018-11-05 14:54:30 -08:00
|
|
|
*/
|
2023-03-24 16:54:21 +01:00
|
|
|
if (!IS_ENABLED(CONFIG_ARM64) &&
|
|
|
|
!(IS_ENABLED(CONFIG_RISCV) && IS_ENABLED(CONFIG_64BIT))) {
|
2018-11-05 14:54:30 -08:00
|
|
|
initrd_start = (unsigned long)__va(start);
|
|
|
|
initrd_end = (unsigned long)__va(end);
|
|
|
|
initrd_below_start_ok = 1;
|
|
|
|
}
|
2016-02-16 13:52:33 +01:00
|
|
|
}
|
|
|
|
|
2009-11-24 03:26:58 -07:00
|
|
|
/**
|
|
|
|
* early_init_dt_check_for_initrd - Decode initrd location from flat tree
|
|
|
|
* @node: reference to node containing initrd location ('chosen')
|
|
|
|
*/
|
2013-08-30 17:06:53 -05:00
|
|
|
static void __init early_init_dt_check_for_initrd(unsigned long node)
|
2009-11-24 03:26:58 -07:00
|
|
|
{
|
2013-07-01 14:20:35 -04:00
|
|
|
u64 start, end;
|
2014-04-01 23:49:03 -05:00
|
|
|
int len;
|
|
|
|
const __be32 *prop;
|
2009-11-24 03:26:58 -07:00
|
|
|
|
2021-08-11 10:51:03 +02:00
|
|
|
if (!IS_ENABLED(CONFIG_BLK_DEV_INITRD))
|
|
|
|
return;
|
|
|
|
|
2009-11-24 03:26:58 -07:00
|
|
|
pr_debug("Looking for initrd properties... ");
|
|
|
|
|
|
|
|
prop = of_get_flat_dt_prop(node, "linux,initrd-start", &len);
|
2010-01-30 01:31:21 -07:00
|
|
|
if (!prop)
|
|
|
|
return;
|
2013-07-01 14:20:35 -04:00
|
|
|
start = of_read_number(prop, len/4);
|
2010-01-30 01:31:21 -07:00
|
|
|
|
|
|
|
prop = of_get_flat_dt_prop(node, "linux,initrd-end", &len);
|
|
|
|
if (!prop)
|
|
|
|
return;
|
2013-07-01 14:20:35 -04:00
|
|
|
end = of_read_number(prop, len/4);
|
of/fdt: Don't calculate initrd size from DT if start > end
If the properties 'linux,initrd-start' and 'linux,initrd-end' of
the chosen node populated from the bootloader, eg. U-Boot, are so that
start > end, then the phys_initrd_size calculated from end - start is
negative that subsequently gets converted to a high positive value for
being unsigned long long. Then, the memory region with the (invalid)
size is added to the bootmem and attempted being paged in paging_init()
that results in the kernel fault.
For example, on the FVP ARM64 system I'm running, the U-Boot populates
the 'linux,initrd-start' with 8800_0000 and 'linux,initrd-end' with 0.
The phys_initrd_size calculated is then ffff_ffff_7800_0000
(= 0 - 8800_0000 = -8800_0000 + ULLONG_MAX + 1). paging_init() then
attempts to map the address 8800_0000 + ffff_ffff_7800_0000 and oops'es
as below.
It should be stressed, it is generally a fault of the bootloader's with
the kernel relying on it, however we should not allow the bootloader's
misconfiguration to lead to the kernel oops. Not only the kernel should be
bullet proof against it but also finding the root cause of the paging
fault spanning over the bootloader, DT, and kernel may happen is not so
easy.
Unable to handle kernel paging request at virtual address fffffffefe43c000
Mem abort info:
ESR = 0x96000007
EC = 0x25: DABT (current EL), IL = 32 bits
SET = 0, FnV = 0
EA = 0, S1PTW = 0
Data abort info:
ISV = 0, ISS = 0x00000007
CM = 0, WnR = 0
swapper pgtable: 4k pages, 39-bit VAs, pgdp=0000000080e3d000
[fffffffefe43c000] pgd=0000000080de9003, pud=0000000080de9003
Unable to handle kernel paging request at virtual address ffffff8000de9f90
Mem abort info:
ESR = 0x96000005
EC = 0x25: DABT (current EL), IL = 32 bits
SET = 0, FnV = 0
EA = 0, S1PTW = 0
Data abort info:
ISV = 0, ISS = 0x00000005
CM = 0, WnR = 0
swapper pgtable: 4k pages, 39-bit VAs, pgdp=0000000080e3d000
[ffffff8000de9f90] pgd=0000000000000000, pud=0000000000000000
Internal error: Oops: 96000005 [#1] PREEMPT SMP
Modules linked in:
CPU: 0 PID: 0 Comm: swapper Not tainted 5.4.51-yocto-standard #1
Hardware name: FVP Base (DT)
pstate: 60000085 (nZCv daIf -PAN -UAO)
pc : show_pte+0x12c/0x1b4
lr : show_pte+0x100/0x1b4
sp : ffffffc010ce3b30
x29: ffffffc010ce3b30 x28: ffffffc010ceed80
x27: fffffffefe43c000 x26: fffffffefe43a028
x25: 0000000080bf0000 x24: 0000000000000025
x23: ffffffc010b8d000 x22: ffffffc010e3d000
x23: ffffffc010b8d000 x22: ffffffc010e3d000
x21: 0000000080de9000 x20: ffffff7f80000f90
x19: fffffffefe43c000 x18: 0000000000000030
x17: 0000000000001400 x16: 0000000000001c00
x15: ffffffc010cef1b8 x14: ffffffffffffffff
x13: ffffffc010df1f40 x12: ffffffc010df1b70
x11: ffffffc010ce3b30 x10: ffffffc010ce3b30
x9 : 00000000ffffffc8 x8 : 0000000000000000
x7 : 000000000000000f x6 : ffffffc010df16e8
x5 : 0000000000000000 x4 : 0000000000000000
x3 : 00000000ffffffff x2 : 0000000000000000
x1 : 0000008080000000 x0 : ffffffc010af1d68
Call trace:
show_pte+0x12c/0x1b4
die_kernel_fault+0x54/0x78
__do_kernel_fault+0x11c/0x128
do_translation_fault+0x58/0xac
do_mem_abort+0x50/0xb0
el1_da+0x1c/0x90
__create_pgd_mapping+0x348/0x598
paging_init+0x3f0/0x70d0
setup_arch+0x2c0/0x5d4
start_kernel+0x94/0x49c
Code: 92748eb5 900052a0 9135a000 cb010294 (f8756a96)
Signed-off-by: Marek Bykowski <marek.bykowski@gmail.com>
Link: https://lore.kernel.org/r/20220909023358.76881-1-marek.bykowski@gmail.com
Signed-off-by: Rob Herring <robh@kernel.org>
2022-09-09 02:33:57 +00:00
|
|
|
if (start > end)
|
|
|
|
return;
|
2009-11-24 03:26:58 -07:00
|
|
|
|
2016-02-16 13:52:33 +01:00
|
|
|
__early_init_dt_declare_initrd(start, end);
|
2018-11-05 14:54:28 -08:00
|
|
|
phys_initrd_start = start;
|
|
|
|
phys_initrd_size = end - start;
|
2013-08-30 17:06:53 -05:00
|
|
|
|
2021-06-16 11:27:45 +02:00
|
|
|
pr_debug("initrd_start=0x%llx initrd_end=0x%llx\n", start, end);
|
2009-11-24 03:26:58 -07:00
|
|
|
}
|
|
|
|
|
2021-08-11 10:51:01 +02:00
|
|
|
/**
|
|
|
|
* early_init_dt_check_for_elfcorehdr - Decode elfcorehdr location from flat
|
|
|
|
* tree
|
|
|
|
* @node: reference to node containing elfcorehdr location ('chosen')
|
|
|
|
*/
|
|
|
|
static void __init early_init_dt_check_for_elfcorehdr(unsigned long node)
|
|
|
|
{
|
|
|
|
const __be32 *prop;
|
|
|
|
int len;
|
|
|
|
|
|
|
|
if (!IS_ENABLED(CONFIG_CRASH_DUMP))
|
|
|
|
return;
|
|
|
|
|
|
|
|
pr_debug("Looking for elfcorehdr property... ");
|
|
|
|
|
|
|
|
prop = of_get_flat_dt_prop(node, "linux,elfcorehdr", &len);
|
|
|
|
if (!prop || (len < (dt_root_addr_cells + dt_root_size_cells)))
|
|
|
|
return;
|
|
|
|
|
|
|
|
elfcorehdr_addr = dt_mem_next_cell(dt_root_addr_cells, &prop);
|
|
|
|
elfcorehdr_size = dt_mem_next_cell(dt_root_size_cells, &prop);
|
|
|
|
|
|
|
|
pr_debug("elfcorehdr_start=0x%llx elfcorehdr_size=0x%llx\n",
|
|
|
|
elfcorehdr_addr, elfcorehdr_size);
|
|
|
|
}
|
|
|
|
|
2021-12-14 12:01:56 +08:00
|
|
|
static unsigned long chosen_node_offset = -FDT_ERR_NOTFOUND;
|
2021-08-11 10:51:02 +02:00
|
|
|
|
2022-05-06 19:44:00 +08:00
|
|
|
/*
|
|
|
|
* The main usage of linux,usable-memory-range is for crash dump kernel.
|
|
|
|
* Originally, the number of usable-memory regions is one. Now there may
|
|
|
|
* be two regions, low region and high region.
|
|
|
|
* To make compatibility with existing user-space and older kdump, the low
|
|
|
|
* region is always the last range of linux,usable-memory-range if exist.
|
|
|
|
*/
|
|
|
|
#define MAX_USABLE_RANGES 2
|
|
|
|
|
2021-08-11 10:51:02 +02:00
|
|
|
/**
|
|
|
|
* early_init_dt_check_for_usable_mem_range - Decode usable memory range
|
|
|
|
* location from flat tree
|
|
|
|
*/
|
efi: apply memblock cap after memblock_add()
On arm64, during kdump kernel saves vmcore, it runs into the following bug:
...
[ 15.148919] usercopy: Kernel memory exposure attempt detected from SLUB object 'kmem_cache_node' (offset 0, size 4096)!
[ 15.159707] ------------[ cut here ]------------
[ 15.164311] kernel BUG at mm/usercopy.c:99!
[ 15.168482] Internal error: Oops - BUG: 0 [#1] SMP
[ 15.173261] Modules linked in: xfs libcrc32c crct10dif_ce ghash_ce sha2_ce sha256_arm64 sha1_ce sbsa_gwdt ast i2c_algo_bit drm_vram_helper drm_kms_helper syscopyarea sysfillrect sysimgblt fb_sys_fops cec drm_ttm_helper ttm drm nvme nvme_core xgene_hwmon i2c_designware_platform i2c_designware_core dm_mirror dm_region_hash dm_log dm_mod overlay squashfs zstd_decompress loop
[ 15.206186] CPU: 0 PID: 542 Comm: cp Not tainted 5.16.0-rc4 #1
[ 15.212006] Hardware name: GIGABYTE R272-P30-JG/MP32-AR0-JG, BIOS F12 (SCP: 1.5.20210426) 05/13/2021
[ 15.221125] pstate: 60400009 (nZCv daif +PAN -UAO -TCO -DIT -SSBS BTYPE=--)
[ 15.228073] pc : usercopy_abort+0x9c/0xa0
[ 15.232074] lr : usercopy_abort+0x9c/0xa0
[ 15.236070] sp : ffff8000121abba0
[ 15.239371] x29: ffff8000121abbb0 x28: 0000000000003000 x27: 0000000000000000
[ 15.246494] x26: 0000000080000400 x25: 0000ffff885c7000 x24: 0000000000000000
[ 15.253617] x23: 000007ff80400000 x22: ffff07ff80401000 x21: 0000000000000001
[ 15.260739] x20: 0000000000001000 x19: ffff07ff80400000 x18: ffffffffffffffff
[ 15.267861] x17: 656a626f2042554c x16: 53206d6f72662064 x15: 6574636574656420
[ 15.274983] x14: 74706d6574746120 x13: 2129363930342065 x12: 7a6973202c302074
[ 15.282105] x11: ffffc8b041d1b148 x10: 00000000ffff8000 x9 : ffffc8b04012812c
[ 15.289228] x8 : 00000000ffff7fff x7 : ffffc8b041d1b148 x6 : 0000000000000000
[ 15.296349] x5 : 0000000000000000 x4 : 0000000000007fff x3 : 0000000000000000
[ 15.303471] x2 : 0000000000000000 x1 : ffff07ff8c064800 x0 : 000000000000006b
[ 15.310593] Call trace:
[ 15.313027] usercopy_abort+0x9c/0xa0
[ 15.316677] __check_heap_object+0xd4/0xf0
[ 15.320762] __check_object_size.part.0+0x160/0x1e0
[ 15.325628] __check_object_size+0x2c/0x40
[ 15.329711] copy_oldmem_page+0x7c/0x140
[ 15.333623] read_from_oldmem.part.0+0xfc/0x1c0
[ 15.338142] __read_vmcore.constprop.0+0x23c/0x350
[ 15.342920] read_vmcore+0x28/0x34
[ 15.346309] proc_reg_read+0xb4/0xf0
[ 15.349871] vfs_read+0xb8/0x1f0
[ 15.353088] ksys_read+0x74/0x100
[ 15.356390] __arm64_sys_read+0x28/0x34
...
This bug introduced by commit b261dba2fdb2 ("arm64: kdump: Remove custom
linux,usable-memory-range handling"), which moves
memblock_cap_memory_range() to fdt, but it breaches the rules that
memblock_cap_memory_range() should come after memblock_add() etc as said
in commit e888fa7bb882 ("memblock: Check memory add/cap ordering").
As a consequence, the virtual address set up by copy_oldmem_page() does
not bail out from the test of virt_addr_valid() in check_heap_object(),
and finally hits the BUG_ON().
Since memblock allocator has no idea about when the memblock is fully
populated, while efi_init() is aware, so tackling this issue by calling the
interface early_init_dt_check_for_usable_mem_range() exposed by of/fdt.
Fixes: b261dba2fdb2 ("arm64: kdump: Remove custom linux,usable-memory-range handling")
Signed-off-by: Pingfan Liu <kernelfans@gmail.com>
Cc: Rob Herring <robh+dt@kernel.org>
Cc: Zhen Lei <thunder.leizhen@huawei.com>
Cc: Catalin Marinas <catalin.marinas@arm.com>
Cc: Will Deacon <will@kernel.org>
Cc: Andrew Morton <akpm@linux-foundation.org>
Cc: Mike Rapoport <rppt@kernel.org>
Cc: Geert Uytterhoeven <geert+renesas@glider.be>
Cc: Frank Rowand <frowand.list@gmail.com>
Cc: Ard Biesheuvel <ardb@kernel.org>
Cc: Nick Terrell <terrelln@fb.com>
Cc: linux-arm-kernel@lists.infradead.org
To: devicetree@vger.kernel.org
To: linux-efi@vger.kernel.org
Acked-by: Ard Biesheuvel <ardb@kernel.org>
Signed-off-by: Rob Herring <robh@kernel.org>
Link: https://lore.kernel.org/r/20211215021348.8766-1-kernelfans@gmail.com
2021-12-15 10:13:48 +08:00
|
|
|
void __init early_init_dt_check_for_usable_mem_range(void)
|
2021-08-11 10:51:02 +02:00
|
|
|
{
|
2022-05-06 19:44:00 +08:00
|
|
|
struct memblock_region rgn[MAX_USABLE_RANGES] = {0};
|
|
|
|
const __be32 *prop, *endp;
|
|
|
|
int len, i;
|
2021-12-14 12:01:56 +08:00
|
|
|
unsigned long node = chosen_node_offset;
|
|
|
|
|
|
|
|
if ((long)node < 0)
|
|
|
|
return;
|
2021-08-11 10:51:02 +02:00
|
|
|
|
|
|
|
pr_debug("Looking for usable-memory-range property... ");
|
|
|
|
|
|
|
|
prop = of_get_flat_dt_prop(node, "linux,usable-memory-range", &len);
|
2022-05-06 19:44:00 +08:00
|
|
|
if (!prop || (len % (dt_root_addr_cells + dt_root_size_cells)))
|
2021-08-11 10:51:02 +02:00
|
|
|
return;
|
|
|
|
|
2022-05-06 19:44:00 +08:00
|
|
|
endp = prop + (len / sizeof(__be32));
|
|
|
|
for (i = 0; i < MAX_USABLE_RANGES && prop < endp; i++) {
|
|
|
|
rgn[i].base = dt_mem_next_cell(dt_root_addr_cells, &prop);
|
|
|
|
rgn[i].size = dt_mem_next_cell(dt_root_size_cells, &prop);
|
2021-08-11 10:51:02 +02:00
|
|
|
|
2022-05-06 19:44:00 +08:00
|
|
|
pr_debug("cap_mem_regions[%d]: base=%pa, size=%pa\n",
|
|
|
|
i, &rgn[i].base, &rgn[i].size);
|
|
|
|
}
|
2021-12-14 12:01:56 +08:00
|
|
|
|
2022-05-06 19:44:00 +08:00
|
|
|
memblock_cap_memory_range(rgn[0].base, rgn[0].size);
|
|
|
|
for (i = 1; i < MAX_USABLE_RANGES && rgn[i].size; i++)
|
|
|
|
memblock_add(rgn[i].base, rgn[i].size);
|
2021-08-11 10:51:02 +02:00
|
|
|
}
|
|
|
|
|
2025-05-09 00:46:27 -07:00
|
|
|
/**
|
|
|
|
* early_init_dt_check_kho - Decode info required for kexec handover from DT
|
|
|
|
*/
|
|
|
|
static void __init early_init_dt_check_kho(void)
|
|
|
|
{
|
|
|
|
unsigned long node = chosen_node_offset;
|
|
|
|
u64 fdt_start, fdt_size, scratch_start, scratch_size;
|
|
|
|
const __be32 *p;
|
|
|
|
int l;
|
|
|
|
|
|
|
|
if (!IS_ENABLED(CONFIG_KEXEC_HANDOVER) || (long)node < 0)
|
|
|
|
return;
|
|
|
|
|
|
|
|
p = of_get_flat_dt_prop(node, "linux,kho-fdt", &l);
|
|
|
|
if (l != (dt_root_addr_cells + dt_root_size_cells) * sizeof(__be32))
|
|
|
|
return;
|
|
|
|
|
|
|
|
fdt_start = dt_mem_next_cell(dt_root_addr_cells, &p);
|
|
|
|
fdt_size = dt_mem_next_cell(dt_root_addr_cells, &p);
|
|
|
|
|
|
|
|
p = of_get_flat_dt_prop(node, "linux,kho-scratch", &l);
|
|
|
|
if (l != (dt_root_addr_cells + dt_root_size_cells) * sizeof(__be32))
|
|
|
|
return;
|
|
|
|
|
|
|
|
scratch_start = dt_mem_next_cell(dt_root_addr_cells, &p);
|
|
|
|
scratch_size = dt_mem_next_cell(dt_root_addr_cells, &p);
|
|
|
|
|
|
|
|
kho_populate(fdt_start, fdt_size, scratch_start, scratch_size);
|
|
|
|
}
|
|
|
|
|
2014-03-27 08:07:01 -05:00
|
|
|
#ifdef CONFIG_SERIAL_EARLYCON
|
|
|
|
|
2016-09-27 23:54:12 +03:00
|
|
|
int __init early_init_dt_scan_chosen_stdout(void)
|
2014-03-27 08:07:01 -05:00
|
|
|
{
|
|
|
|
int offset;
|
2016-01-16 15:23:42 -08:00
|
|
|
const char *p, *q, *options = NULL;
|
2014-03-27 08:07:01 -05:00
|
|
|
int l;
|
2020-11-23 11:23:13 +01:00
|
|
|
const struct earlycon_id *match;
|
2014-03-27 08:07:01 -05:00
|
|
|
const void *fdt = initial_boot_params;
|
2022-06-28 14:07:05 +02:00
|
|
|
int ret;
|
2014-03-27 08:07:01 -05:00
|
|
|
|
|
|
|
offset = fdt_path_offset(fdt, "/chosen");
|
|
|
|
if (offset < 0)
|
|
|
|
offset = fdt_path_offset(fdt, "/chosen@0");
|
|
|
|
if (offset < 0)
|
|
|
|
return -ENOENT;
|
|
|
|
|
|
|
|
p = fdt_getprop(fdt, offset, "stdout-path", &l);
|
|
|
|
if (!p)
|
|
|
|
p = fdt_getprop(fdt, offset, "linux,stdout-path", &l);
|
|
|
|
if (!p || !l)
|
|
|
|
return -ENOENT;
|
|
|
|
|
2016-01-16 15:23:42 -08:00
|
|
|
q = strchrnul(p, ':');
|
|
|
|
if (*q != '\0')
|
|
|
|
options = q + 1;
|
2016-01-16 15:23:48 -08:00
|
|
|
l = q - p;
|
2015-10-10 01:29:30 -07:00
|
|
|
|
2014-03-27 08:07:01 -05:00
|
|
|
/* Get the node specified by stdout-path */
|
2016-01-16 15:23:48 -08:00
|
|
|
offset = fdt_path_offset_namelen(fdt, p, l);
|
|
|
|
if (offset < 0) {
|
|
|
|
pr_warn("earlycon: stdout-path %.*s not found\n", l, p);
|
|
|
|
return 0;
|
|
|
|
}
|
2014-03-27 08:07:01 -05:00
|
|
|
|
2020-11-23 11:23:13 +01:00
|
|
|
for (match = __earlycon_table; match < __earlycon_table_end; match++) {
|
2016-01-16 15:23:39 -08:00
|
|
|
if (!match->compatible[0])
|
|
|
|
continue;
|
|
|
|
|
|
|
|
if (fdt_node_check_compatible(fdt, offset, match->compatible))
|
2014-03-27 08:07:01 -05:00
|
|
|
continue;
|
|
|
|
|
2022-06-28 14:07:05 +02:00
|
|
|
ret = of_setup_earlycon(match, offset, options);
|
|
|
|
if (!ret || ret == -EALREADY)
|
2019-09-10 07:58:33 +02:00
|
|
|
return 0;
|
2014-03-27 08:07:01 -05:00
|
|
|
}
|
|
|
|
return -ENODEV;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2021-03-18 10:40:33 +00:00
|
|
|
/*
|
2009-11-24 03:27:10 -07:00
|
|
|
* early_init_dt_scan_root - fetch the top level address and size cells
|
|
|
|
*/
|
2021-11-18 12:12:11 -06:00
|
|
|
int __init early_init_dt_scan_root(void)
|
2009-11-24 03:27:10 -07:00
|
|
|
{
|
2014-04-01 23:49:03 -05:00
|
|
|
const __be32 *prop;
|
2021-11-18 12:12:11 -06:00
|
|
|
const void *fdt = initial_boot_params;
|
|
|
|
int node = fdt_path_offset(fdt, "/");
|
2009-11-24 03:27:10 -07:00
|
|
|
|
2021-11-18 12:12:11 -06:00
|
|
|
if (node < 0)
|
|
|
|
return -ENODEV;
|
2009-11-24 03:27:10 -07:00
|
|
|
|
2010-01-30 01:45:26 -07:00
|
|
|
dt_root_size_cells = OF_ROOT_NODE_SIZE_CELLS_DEFAULT;
|
|
|
|
dt_root_addr_cells = OF_ROOT_NODE_ADDR_CELLS_DEFAULT;
|
|
|
|
|
2009-11-24 03:27:10 -07:00
|
|
|
prop = of_get_flat_dt_prop(node, "#size-cells", NULL);
|
2024-11-06 11:10:27 -06:00
|
|
|
if (!WARN(!prop, "No '#size-cells' in root node\n"))
|
2010-01-30 01:45:26 -07:00
|
|
|
dt_root_size_cells = be32_to_cpup(prop);
|
2009-11-24 03:27:10 -07:00
|
|
|
pr_debug("dt_root_size_cells = %x\n", dt_root_size_cells);
|
|
|
|
|
|
|
|
prop = of_get_flat_dt_prop(node, "#address-cells", NULL);
|
2024-11-06 11:10:27 -06:00
|
|
|
if (!WARN(!prop, "No '#address-cells' in root node\n"))
|
2010-01-30 01:45:26 -07:00
|
|
|
dt_root_addr_cells = be32_to_cpup(prop);
|
2009-11-24 03:27:10 -07:00
|
|
|
pr_debug("dt_root_addr_cells = %x\n", dt_root_addr_cells);
|
|
|
|
|
2021-11-18 12:12:11 -06:00
|
|
|
return 0;
|
2009-11-24 03:27:10 -07:00
|
|
|
}
|
|
|
|
|
2014-04-01 23:49:03 -05:00
|
|
|
u64 __init dt_mem_next_cell(int s, const __be32 **cellp)
|
2009-11-24 03:37:56 -07:00
|
|
|
{
|
2014-04-01 23:49:03 -05:00
|
|
|
const __be32 *p = *cellp;
|
2009-11-24 03:37:56 -07:00
|
|
|
|
|
|
|
*cellp = p + s;
|
|
|
|
return of_read_number(p, s);
|
|
|
|
}
|
|
|
|
|
2021-03-18 10:40:33 +00:00
|
|
|
/*
|
2017-06-20 16:46:28 -07:00
|
|
|
* early_init_dt_scan_memory - Look for and parse memory nodes
|
2010-02-01 21:34:14 -07:00
|
|
|
*/
|
2021-12-15 09:01:02 -06:00
|
|
|
int __init early_init_dt_scan_memory(void)
|
2010-02-01 21:34:14 -07:00
|
|
|
{
|
2022-12-23 12:27:47 +01:00
|
|
|
int node, found_memory = 0;
|
2021-12-15 09:01:02 -06:00
|
|
|
const void *fdt = initial_boot_params;
|
2010-02-01 21:34:14 -07:00
|
|
|
|
2021-12-15 09:01:02 -06:00
|
|
|
fdt_for_each_subnode(node, fdt, 0) {
|
|
|
|
const char *type = of_get_flat_dt_prop(node, "device_type", NULL);
|
|
|
|
const __be32 *reg, *endp;
|
|
|
|
int l;
|
|
|
|
bool hotpluggable;
|
2010-02-01 21:34:14 -07:00
|
|
|
|
2021-12-15 09:01:02 -06:00
|
|
|
/* We are scanning "memory" nodes only */
|
|
|
|
if (type == NULL || strcmp(type, "memory") != 0)
|
|
|
|
continue;
|
2010-02-01 21:34:14 -07:00
|
|
|
|
of/fdt: Ignore disabled memory nodes
When we boot a machine using a devicetree, the generic DT code goes
through all nodes with a 'device_type = "memory"' property, and collects
all memory banks mentioned there. However it does not check for the
status property, so any nodes which are explicitly "disabled" will still
be added as a memblock.
This ends up badly for QEMU, when booting with secure firmware on
arm/arm64 machines, because QEMU adds a node describing secure-only
memory:
===================
secram@e000000 {
secure-status = "okay";
status = "disabled";
reg = <0x00 0xe000000 0x00 0x1000000>;
device_type = "memory";
};
===================
The kernel will eventually use that memory block (which is located below
the main DRAM bank), but accesses to that will be answered with an
SError:
===================
[ 0.000000] Internal error: synchronous external abort: 96000050 [#1] PREEMPT SMP
[ 0.000000] Modules linked in:
[ 0.000000] CPU: 0 PID: 0 Comm: swapper Not tainted 5.18.0-rc6-00014-g10c8acb8b679 #524
[ 0.000000] Hardware name: linux,dummy-virt (DT)
[ 0.000000] pstate: 200000c5 (nzCv daIF -PAN -UAO -TCO -DIT -SSBS BTYPE=--)
[ 0.000000] pc : new_slab+0x190/0x340
[ 0.000000] lr : new_slab+0x184/0x340
[ 0.000000] sp : ffff80000a4b3d10
....
==================
The actual crash location and call stack will be somewhat random, and
depend on the specific allocation of that physical memory range.
As the DT spec[1] explicitly mentions standard properties, add a simple
check to skip over disabled memory nodes, so that we only use memory
that is meant for non-secure code to use.
That fixes booting a QEMU arm64 VM with EL3 enabled ("secure=on"), when
not using UEFI. In this case the QEMU generated DT will be handed on
to the kernel, which will see the secram node.
This issue is reproducible when using TF-A together with U-Boot as
firmware, then booting with the "booti" command.
When using U-Boot as an UEFI provider, the code there [2] explicitly
filters for disabled nodes when generating the UEFI memory map, so we
are safe.
EDK/2 only reads the first bank of the first DT memory node [3] to learn
about memory, so we got lucky there.
[1] https://github.com/devicetree-org/devicetree-specification/blob/main/source/chapter3-devicenodes.rst#memory-node (after the table)
[2] https://source.denx.de/u-boot/u-boot/-/blob/master/lib/fdtdec.c#L1061-1063
[3] https://github.com/tianocore/edk2/blob/master/ArmVirtPkg/PrePi/FdtParser.c
Reported-by: Ross Burton <ross.burton@arm.com>
Signed-off-by: Andre Przywara <andre.przywara@arm.com>
Signed-off-by: Rob Herring <robh@kernel.org>
Link: https://lore.kernel.org/r/20220517101410.3493781-1-andre.przywara@arm.com
2022-05-17 11:14:10 +01:00
|
|
|
if (!of_fdt_device_is_available(fdt, node))
|
|
|
|
continue;
|
|
|
|
|
2021-12-15 09:01:02 -06:00
|
|
|
reg = of_get_flat_dt_prop(node, "linux,usable-memory", &l);
|
|
|
|
if (reg == NULL)
|
|
|
|
reg = of_get_flat_dt_prop(node, "reg", &l);
|
|
|
|
if (reg == NULL)
|
|
|
|
continue;
|
2010-02-01 21:34:14 -07:00
|
|
|
|
2021-12-15 09:01:02 -06:00
|
|
|
endp = reg + (l / sizeof(__be32));
|
|
|
|
hotpluggable = of_get_flat_dt_prop(node, "hotpluggable", NULL);
|
2010-02-01 21:34:14 -07:00
|
|
|
|
2021-12-15 09:01:02 -06:00
|
|
|
pr_debug("memory scan node %s, reg size %d,\n",
|
|
|
|
fdt_get_name(fdt, node, NULL), l);
|
2010-02-01 21:34:14 -07:00
|
|
|
|
2021-12-15 09:01:02 -06:00
|
|
|
while ((endp - reg) >= (dt_root_addr_cells + dt_root_size_cells)) {
|
|
|
|
u64 base, size;
|
2010-02-01 21:34:14 -07:00
|
|
|
|
2021-12-15 09:01:02 -06:00
|
|
|
base = dt_mem_next_cell(dt_root_addr_cells, ®);
|
|
|
|
size = dt_mem_next_cell(dt_root_size_cells, ®);
|
2010-02-01 21:34:14 -07:00
|
|
|
|
2021-12-15 09:01:02 -06:00
|
|
|
if (size == 0)
|
|
|
|
continue;
|
|
|
|
pr_debug(" - %llx, %llx\n", base, size);
|
2016-12-12 16:43:02 -08:00
|
|
|
|
2021-12-15 09:01:02 -06:00
|
|
|
early_init_dt_add_memory_arch(base, size);
|
2016-12-12 16:43:02 -08:00
|
|
|
|
2022-12-23 12:27:47 +01:00
|
|
|
found_memory = 1;
|
|
|
|
|
2021-12-15 09:01:02 -06:00
|
|
|
if (!hotpluggable)
|
|
|
|
continue;
|
2010-02-01 21:34:14 -07:00
|
|
|
|
2021-12-15 09:01:02 -06:00
|
|
|
if (memblock_mark_hotplug(base, size))
|
|
|
|
pr_warn("failed to mark hotplug range 0x%llx - 0x%llx\n",
|
|
|
|
base, base + size);
|
|
|
|
}
|
|
|
|
}
|
2022-12-23 12:27:47 +01:00
|
|
|
return found_memory;
|
2010-02-01 21:34:14 -07:00
|
|
|
}
|
|
|
|
|
2021-11-18 12:12:10 -06:00
|
|
|
int __init early_init_dt_scan_chosen(char *cmdline)
|
2009-12-10 23:42:21 -07:00
|
|
|
{
|
2021-11-18 12:12:10 -06:00
|
|
|
int l, node;
|
2014-04-01 23:49:03 -05:00
|
|
|
const char *p;
|
2019-08-23 14:24:51 +08:00
|
|
|
const void *rng_seed;
|
2021-11-18 12:12:10 -06:00
|
|
|
const void *fdt = initial_boot_params;
|
2009-12-10 23:42:21 -07:00
|
|
|
|
2021-11-18 12:12:10 -06:00
|
|
|
node = fdt_path_offset(fdt, "/chosen");
|
|
|
|
if (node < 0)
|
|
|
|
node = fdt_path_offset(fdt, "/chosen@0");
|
|
|
|
if (node < 0)
|
2023-01-03 12:00:32 -06:00
|
|
|
/* Handle the cmdline config options even if no /chosen node */
|
|
|
|
goto handle_cmdline;
|
2009-12-10 23:42:21 -07:00
|
|
|
|
2021-12-14 12:01:56 +08:00
|
|
|
chosen_node_offset = node;
|
|
|
|
|
2009-12-10 23:42:21 -07:00
|
|
|
early_init_dt_check_for_initrd(node);
|
2021-08-11 10:51:01 +02:00
|
|
|
early_init_dt_check_for_elfcorehdr(node);
|
2009-12-10 23:42:21 -07:00
|
|
|
|
2023-01-03 12:00:32 -06:00
|
|
|
rng_seed = of_get_flat_dt_prop(node, "rng-seed", &l);
|
|
|
|
if (rng_seed && l > 0) {
|
|
|
|
add_bootloader_randomness(rng_seed, l);
|
|
|
|
|
|
|
|
/* try to clear seed so it won't be found. */
|
|
|
|
fdt_nop_property(initial_boot_params, node, "rng-seed");
|
|
|
|
|
|
|
|
/* update CRC check value */
|
|
|
|
of_fdt_crc32 = crc32_be(~0, initial_boot_params,
|
|
|
|
fdt_totalsize(initial_boot_params));
|
|
|
|
}
|
|
|
|
|
2011-03-30 22:57:33 -03:00
|
|
|
/* Retrieve command line */
|
2009-12-10 23:42:21 -07:00
|
|
|
p = of_get_flat_dt_prop(node, "bootargs", &l);
|
|
|
|
if (p != NULL && l > 0)
|
2022-08-18 23:00:53 +02:00
|
|
|
strscpy(cmdline, p, min(l, COMMAND_LINE_SIZE));
|
2009-12-10 23:42:21 -07:00
|
|
|
|
2023-01-03 12:00:32 -06:00
|
|
|
handle_cmdline:
|
2023-01-03 12:00:31 -06:00
|
|
|
/*
|
|
|
|
* CONFIG_CMDLINE is meant to be a default in case nothing else
|
|
|
|
* managed to set the command line, unless CONFIG_CMDLINE_FORCE
|
|
|
|
* is set in which case we override whatever was found earlier.
|
|
|
|
*/
|
|
|
|
#ifdef CONFIG_CMDLINE
|
|
|
|
#if defined(CONFIG_CMDLINE_EXTEND)
|
|
|
|
strlcat(cmdline, " ", COMMAND_LINE_SIZE);
|
|
|
|
strlcat(cmdline, CONFIG_CMDLINE, COMMAND_LINE_SIZE);
|
|
|
|
#elif defined(CONFIG_CMDLINE_FORCE)
|
|
|
|
strscpy(cmdline, CONFIG_CMDLINE, COMMAND_LINE_SIZE);
|
|
|
|
#else
|
|
|
|
/* No arguments from boot loader, use kernel's cmdl*/
|
|
|
|
if (!((char *)cmdline)[0])
|
|
|
|
strscpy(cmdline, CONFIG_CMDLINE, COMMAND_LINE_SIZE);
|
|
|
|
#endif
|
|
|
|
#endif /* CONFIG_CMDLINE */
|
|
|
|
|
|
|
|
pr_debug("Command line is: %s\n", (char *)cmdline);
|
|
|
|
|
2021-11-18 12:12:10 -06:00
|
|
|
return 0;
|
2009-12-10 23:42:21 -07:00
|
|
|
}
|
|
|
|
|
2016-02-16 13:52:32 +01:00
|
|
|
#ifndef MIN_MEMBLOCK_ADDR
|
|
|
|
#define MIN_MEMBLOCK_ADDR __pa(PAGE_OFFSET)
|
|
|
|
#endif
|
2015-08-18 10:34:41 +01:00
|
|
|
#ifndef MAX_MEMBLOCK_ADDR
|
|
|
|
#define MAX_MEMBLOCK_ADDR ((phys_addr_t)~0)
|
|
|
|
#endif
|
2014-07-07 17:45:43 -07:00
|
|
|
|
2013-09-24 22:20:01 -05:00
|
|
|
void __init __weak early_init_dt_add_memory_arch(u64 base, u64 size)
|
|
|
|
{
|
2016-02-16 13:52:32 +01:00
|
|
|
const u64 phys_offset = MIN_MEMBLOCK_ADDR;
|
2014-08-20 17:10:31 +02:00
|
|
|
|
2018-10-26 15:04:48 -07:00
|
|
|
if (size < PAGE_SIZE - (base & ~PAGE_MASK)) {
|
|
|
|
pr_warn("Ignoring memory block 0x%llx - 0x%llx\n",
|
|
|
|
base, base + size);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2014-08-20 17:10:31 +02:00
|
|
|
if (!PAGE_ALIGNED(base)) {
|
|
|
|
size -= PAGE_SIZE - (base & ~PAGE_MASK);
|
|
|
|
base = PAGE_ALIGN(base);
|
|
|
|
}
|
2013-09-24 22:20:01 -05:00
|
|
|
size &= PAGE_MASK;
|
2014-06-19 20:13:38 -07:00
|
|
|
|
2015-08-18 10:34:41 +01:00
|
|
|
if (base > MAX_MEMBLOCK_ADDR) {
|
2019-10-18 11:18:33 +08:00
|
|
|
pr_warn("Ignoring memory block 0x%llx - 0x%llx\n",
|
|
|
|
base, base + size);
|
2014-07-07 17:45:43 -07:00
|
|
|
return;
|
|
|
|
}
|
2014-06-19 20:13:38 -07:00
|
|
|
|
2015-08-18 10:34:41 +01:00
|
|
|
if (base + size - 1 > MAX_MEMBLOCK_ADDR) {
|
2019-10-18 11:18:33 +08:00
|
|
|
pr_warn("Ignoring memory range 0x%llx - 0x%llx\n",
|
|
|
|
((u64)MAX_MEMBLOCK_ADDR) + 1, base + size);
|
2015-08-18 10:34:41 +01:00
|
|
|
size = MAX_MEMBLOCK_ADDR - base + 1;
|
2014-06-19 20:13:38 -07:00
|
|
|
}
|
|
|
|
|
2013-09-24 22:20:01 -05:00
|
|
|
if (base + size < phys_offset) {
|
2019-10-18 11:18:33 +08:00
|
|
|
pr_warn("Ignoring memory block 0x%llx - 0x%llx\n",
|
|
|
|
base, base + size);
|
2013-09-24 22:20:01 -05:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (base < phys_offset) {
|
2019-10-18 11:18:33 +08:00
|
|
|
pr_warn("Ignoring memory range 0x%llx - 0x%llx\n",
|
|
|
|
base, phys_offset);
|
2013-09-24 22:20:01 -05:00
|
|
|
size -= phys_offset - base;
|
|
|
|
base = phys_offset;
|
|
|
|
}
|
|
|
|
memblock_add(base, size);
|
|
|
|
}
|
|
|
|
|
2018-01-05 15:32:33 -06:00
|
|
|
static void * __init early_init_dt_alloc_memory_arch(u64 size, u64 align)
|
2015-06-18 20:35:46 -05:00
|
|
|
{
|
2025-01-02 15:25:28 +08:00
|
|
|
return memblock_alloc_or_panic(size, align);
|
2015-06-18 20:35:46 -05:00
|
|
|
}
|
2013-08-28 21:18:32 +01:00
|
|
|
|
2024-10-23 18:14:26 +01:00
|
|
|
bool __init early_init_dt_verify(void *dt_virt, phys_addr_t dt_phys)
|
2013-08-26 09:47:40 -05:00
|
|
|
{
|
2024-10-23 18:14:26 +01:00
|
|
|
if (!dt_virt)
|
2013-08-26 09:47:40 -05:00
|
|
|
return false;
|
|
|
|
|
|
|
|
/* check device tree validity */
|
2024-10-23 18:14:26 +01:00
|
|
|
if (fdt_check_header(dt_virt))
|
2013-08-26 09:47:40 -05:00
|
|
|
return false;
|
|
|
|
|
2014-10-29 12:15:00 -06:00
|
|
|
/* Setup flat device-tree pointer */
|
2024-10-23 18:14:26 +01:00
|
|
|
initial_boot_params = dt_virt;
|
|
|
|
initial_boot_params_pa = dt_phys;
|
2019-08-27 18:33:53 +08:00
|
|
|
of_fdt_crc32 = crc32_be(~0, initial_boot_params,
|
|
|
|
fdt_totalsize(initial_boot_params));
|
2018-08-29 17:20:46 -05:00
|
|
|
|
|
|
|
/* Initialize {size,address}-cells info */
|
|
|
|
early_init_dt_scan_root();
|
|
|
|
|
2014-07-15 10:03:34 -07:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void __init early_init_dt_scan_nodes(void)
|
|
|
|
{
|
2021-11-18 12:12:10 -06:00
|
|
|
int rc;
|
2018-11-10 02:53:17 +02:00
|
|
|
|
2013-08-26 09:47:40 -05:00
|
|
|
/* Retrieve various information from the /chosen node */
|
2021-11-18 12:12:10 -06:00
|
|
|
rc = early_init_dt_scan_chosen(boot_command_line);
|
|
|
|
if (rc)
|
2018-11-10 02:53:17 +02:00
|
|
|
pr_warn("No chosen node found, continuing without\n");
|
2013-08-26 09:47:40 -05:00
|
|
|
|
|
|
|
/* Setup memory, calling early_init_dt_add_memory_arch */
|
2021-12-15 09:01:02 -06:00
|
|
|
early_init_dt_scan_memory();
|
2021-08-11 10:51:02 +02:00
|
|
|
|
|
|
|
/* Handle linux,usable-memory-range property */
|
2021-12-14 12:01:56 +08:00
|
|
|
early_init_dt_check_for_usable_mem_range();
|
2025-05-09 00:46:27 -07:00
|
|
|
|
|
|
|
/* Handle kexec handover */
|
|
|
|
early_init_dt_check_kho();
|
2014-07-15 10:03:34 -07:00
|
|
|
}
|
|
|
|
|
2024-10-23 18:14:26 +01:00
|
|
|
bool __init early_init_dt_scan(void *dt_virt, phys_addr_t dt_phys)
|
2014-07-15 10:03:34 -07:00
|
|
|
{
|
|
|
|
bool status;
|
|
|
|
|
2024-10-23 18:14:26 +01:00
|
|
|
status = early_init_dt_verify(dt_virt, dt_phys);
|
2014-07-15 10:03:34 -07:00
|
|
|
if (!status)
|
|
|
|
return false;
|
2013-08-26 09:47:40 -05:00
|
|
|
|
2014-07-15 10:03:34 -07:00
|
|
|
early_init_dt_scan_nodes();
|
2013-08-26 09:47:40 -05:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2024-02-16 17:05:50 -08:00
|
|
|
static void *__init copy_device_tree(void *fdt)
|
|
|
|
{
|
|
|
|
int size;
|
|
|
|
void *dt;
|
|
|
|
|
|
|
|
size = fdt_totalsize(fdt);
|
|
|
|
dt = early_init_dt_alloc_memory_arch(size,
|
|
|
|
roundup_pow_of_two(FDT_V17_SIZE));
|
|
|
|
|
|
|
|
if (dt)
|
|
|
|
memcpy(dt, fdt, size);
|
|
|
|
|
|
|
|
return dt;
|
|
|
|
}
|
|
|
|
|
2009-11-23 20:07:01 -07:00
|
|
|
/**
|
|
|
|
* unflatten_device_tree - create tree of device_nodes from flat blob
|
|
|
|
*
|
|
|
|
* unflattens the device-tree passed by the firmware, creating the
|
|
|
|
* tree of struct device_node. It also fills the "name" and "type"
|
|
|
|
* pointers of the nodes so the normal device-tree walking functions
|
|
|
|
* can be used.
|
|
|
|
*/
|
|
|
|
void __init unflatten_device_tree(void)
|
|
|
|
{
|
2024-02-16 17:05:51 -08:00
|
|
|
void *fdt = initial_boot_params;
|
|
|
|
|
of: reserved_mem: Restructure how the reserved memory regions are processed
Reserved memory regions defined in the devicetree can be broken up into
two groups:
i) Statically-placed reserved memory regions
i.e. regions defined with a static start address and size using the
"reg" property.
ii) Dynamically-placed reserved memory regions.
i.e. regions defined by specifying an address range where they can be
placed in memory using the "alloc_ranges" and "size" properties.
These regions are processed and set aside at boot time.
This is done in two stages as seen below:
Stage 1:
At this stage, fdt_scan_reserved_mem() scans through the child nodes of
the reserved_memory node using the flattened devicetree and does the
following:
1) If the node represents a statically-placed reserved memory region,
i.e. if it is defined using the "reg" property:
- Call memblock_reserve() or memblock_mark_nomap() as needed.
- Add the information for that region into the reserved_mem array
using fdt_reserved_mem_save_node().
i.e. fdt_reserved_mem_save_node(node, name, base, size).
2) If the node represents a dynamically-placed reserved memory region,
i.e. if it is defined using "alloc-ranges" and "size" properties:
- Add the information for that region to the reserved_mem array with
the starting address and size set to 0.
i.e. fdt_reserved_mem_save_node(node, name, 0, 0).
Note: This region is saved to the array with a starting address of 0
because a starting address is not yet allocated for it.
Stage 2:
After iterating through all the reserved memory nodes and storing their
relevant information in the reserved_mem array,fdt_init_reserved_mem() is
called and does the following:
1) For statically-placed reserved memory regions:
- Call the region specific init function using
__reserved_mem_init_node().
2) For dynamically-placed reserved memory regions:
- Call __reserved_mem_alloc_size() which is used to allocate memory
for each of these regions, and mark them as nomap if they have the
nomap property specified in the DT.
- Call the region specific init function.
The current size of the resvered_mem array is 64 as is defined by
MAX_RESERVED_REGIONS. This means that there is a limitation of 64 for
how many reserved memory regions can be specified on a system.
As systems continue to grow more and more complex, the number of
reserved memory regions needed are also growing and are starting to hit
this 64 count limit, hence the need to make the reserved_mem array
dynamically sized (i.e. dynamically allocating memory for the
reserved_mem array using membock_alloc_*).
On architectures such as arm64, memory allocated using memblock is
writable only after the page tables have been setup. This means that if
the reserved_mem array is going to be dynamically allocated, it needs to
happen after the page tables have been setup, not before.
Since the reserved memory regions are currently being processed and
added to the array before the page tables are setup, there is a need to
change the order in which some of the processing is done to allow for
the reserved_mem array to be dynamically sized.
It is possible to process the statically-placed reserved memory regions
without needing to store them in the reserved_mem array until after the
page tables have been setup because all the information stored in the
array is readily available in the devicetree and can be referenced at
any time.
Dynamically-placed reserved memory regions on the other hand get
assigned a start address only at runtime, and hence need a place to be
stored once they are allocated since there is no other referrence to the
start address for these regions.
Hence this patch changes the processing order of the reserved memory
regions in the following ways:
Step 1:
fdt_scan_reserved_mem() scans through the child nodes of
the reserved_memory node using the flattened devicetree and does the
following:
1) If the node represents a statically-placed reserved memory region,
i.e. if it is defined using the "reg" property:
- Call memblock_reserve() or memblock_mark_nomap() as needed.
2) If the node represents a dynamically-placed reserved memory region,
i.e. if it is defined using "alloc-ranges" and "size" properties:
- Call __reserved_mem_alloc_size() which will:
i) Allocate memory for the reserved region and call
memblock_mark_nomap() as needed.
ii) Call the region specific initialization function using
fdt_init_reserved_mem_node().
iii) Save the region information in the reserved_mem array using
fdt_reserved_mem_save_node().
Step 2:
1) This stage of the reserved memory processing is now only used to add
the statically-placed reserved memory regions into the reserved_mem
array using fdt_scan_reserved_mem_reg_nodes(), as well as call their
region specific initialization functions.
2) This step has also been moved to be after the page tables are
setup. Moving this will allow us to replace the reserved_mem
array with a dynamically sized array before storing the rest of
these regions.
Signed-off-by: Oreoluwa Babatunde <quic_obabatun@quicinc.com>
Link: https://lore.kernel.org/r/20241008220624.551309-2-quic_obabatun@quicinc.com
Signed-off-by: Rob Herring (Arm) <robh@kernel.org>
2024-10-08 15:06:23 -07:00
|
|
|
/* Save the statically-placed regions in the reserved_mem array */
|
|
|
|
fdt_scan_reserved_mem_reg_nodes();
|
|
|
|
|
2025-01-05 17:27:41 +00:00
|
|
|
/* Populate an empty root node when bootloader doesn't provide one */
|
2024-02-16 17:05:51 -08:00
|
|
|
if (!fdt) {
|
|
|
|
fdt = (void *) __dtb_empty_root_begin;
|
|
|
|
/* fdt_totalsize() will be used for copy size */
|
|
|
|
if (fdt_totalsize(fdt) >
|
|
|
|
__dtb_empty_root_end - __dtb_empty_root_begin) {
|
|
|
|
pr_err("invalid size in dtb_empty_root\n");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
of_fdt_crc32 = crc32_be(~0, fdt, fdt_totalsize(fdt));
|
|
|
|
fdt = copy_device_tree(fdt);
|
|
|
|
}
|
|
|
|
|
|
|
|
__unflatten_device_tree(fdt, NULL, &of_root,
|
2016-07-19 00:01:12 +02:00
|
|
|
early_init_dt_alloc_memory_arch, false);
|
2009-11-23 20:07:01 -07:00
|
|
|
|
2013-05-30 05:38:08 -04:00
|
|
|
/* Get pointer to "/chosen" and "/aliases" nodes for use everywhere */
|
2011-08-15 15:28:14 +08:00
|
|
|
of_alias_scan(early_init_dt_alloc_memory_arch);
|
2017-04-25 17:09:54 -07:00
|
|
|
|
|
|
|
unittest_unflatten_overlay_base();
|
2009-11-23 20:07:01 -07:00
|
|
|
}
|
2010-11-18 15:54:56 -08:00
|
|
|
|
2013-08-26 11:22:45 -05:00
|
|
|
/**
|
|
|
|
* unflatten_and_copy_device_tree - copy and create tree of device_nodes from flat blob
|
|
|
|
*
|
|
|
|
* Copies and unflattens the device-tree passed by the firmware, creating the
|
|
|
|
* tree of struct device_node. It also fills the "name" and "type"
|
|
|
|
* pointers of the nodes so the normal device-tree walking functions
|
|
|
|
* can be used. This should only be used when the FDT memory has not been
|
|
|
|
* reserved such is the case when the FDT is built-in to the kernel init
|
|
|
|
* section. If the FDT memory is reserved already then unflatten_device_tree
|
|
|
|
* should be used instead.
|
|
|
|
*/
|
|
|
|
void __init unflatten_and_copy_device_tree(void)
|
|
|
|
{
|
2024-02-16 17:05:50 -08:00
|
|
|
if (initial_boot_params)
|
|
|
|
initial_boot_params = copy_device_tree(initial_boot_params);
|
2013-11-21 13:44:14 +00:00
|
|
|
|
2013-08-26 11:22:45 -05:00
|
|
|
unflatten_device_tree();
|
|
|
|
}
|
|
|
|
|
2014-11-14 18:05:35 +01:00
|
|
|
#ifdef CONFIG_SYSFS
|
|
|
|
static int __init of_fdt_raw_init(void)
|
|
|
|
{
|
2024-11-22 13:14:00 +01:00
|
|
|
static __ro_after_init BIN_ATTR_SIMPLE_ADMIN_RO(fdt);
|
2014-04-02 16:56:48 -05:00
|
|
|
|
2014-11-14 18:05:35 +01:00
|
|
|
if (!initial_boot_params)
|
|
|
|
return 0;
|
2014-04-02 16:56:48 -05:00
|
|
|
|
2014-11-14 18:05:35 +01:00
|
|
|
if (of_fdt_crc32 != crc32_be(~0, initial_boot_params,
|
|
|
|
fdt_totalsize(initial_boot_params))) {
|
2016-06-15 08:32:18 -05:00
|
|
|
pr_warn("not creating '/sys/firmware/fdt': CRC check failed\n");
|
2014-11-14 18:05:35 +01:00
|
|
|
return 0;
|
|
|
|
}
|
2024-11-22 13:14:00 +01:00
|
|
|
bin_attr_fdt.private = initial_boot_params;
|
|
|
|
bin_attr_fdt.size = fdt_totalsize(initial_boot_params);
|
|
|
|
return sysfs_create_bin_file(firmware_kobj, &bin_attr_fdt);
|
2014-04-02 16:56:48 -05:00
|
|
|
}
|
2014-11-14 18:05:35 +01:00
|
|
|
late_initcall(of_fdt_raw_init);
|
2014-04-02 16:56:48 -05:00
|
|
|
#endif
|
|
|
|
|
2010-11-18 15:54:56 -08:00
|
|
|
#endif /* CONFIG_OF_EARLY_FLATTREE */
|