2009-03-27 21:25:12 +08:00
|
|
|
/*
|
|
|
|
* Procedures for creating, accessing and interpreting the device tree.
|
|
|
|
*
|
|
|
|
* Paul Mackerras August 1996.
|
|
|
|
* Copyright (C) 1996-2005 Paul Mackerras.
|
|
|
|
*
|
|
|
|
* Adapted for 64bit PowerPC by Dave Engebretsen and Peter Bergner.
|
|
|
|
* {engebret|bergner}@us.ibm.com
|
|
|
|
*
|
|
|
|
* This program is free software; you can redistribute it and/or
|
|
|
|
* modify it under the terms of the GNU General Public License
|
|
|
|
* as published by the Free Software Foundation; either version
|
|
|
|
* 2 of the License, or (at your option) any later version.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <stdarg.h>
|
|
|
|
#include <linux/kernel.h>
|
|
|
|
#include <linux/string.h>
|
|
|
|
#include <linux/init.h>
|
|
|
|
#include <linux/threads.h>
|
|
|
|
#include <linux/spinlock.h>
|
|
|
|
#include <linux/types.h>
|
|
|
|
#include <linux/pci.h>
|
|
|
|
#include <linux/stringify.h>
|
|
|
|
#include <linux/delay.h>
|
|
|
|
#include <linux/initrd.h>
|
|
|
|
#include <linux/bitops.h>
|
|
|
|
#include <linux/module.h>
|
|
|
|
#include <linux/kexec.h>
|
|
|
|
#include <linux/debugfs.h>
|
|
|
|
#include <linux/irq.h>
|
|
|
|
#include <linux/lmb.h>
|
|
|
|
|
|
|
|
#include <asm/prom.h>
|
|
|
|
#include <asm/page.h>
|
|
|
|
#include <asm/processor.h>
|
|
|
|
#include <asm/irq.h>
|
|
|
|
#include <linux/io.h>
|
|
|
|
#include <asm/system.h>
|
|
|
|
#include <asm/mmu.h>
|
|
|
|
#include <asm/pgtable.h>
|
|
|
|
#include <asm/sections.h>
|
|
|
|
#include <asm/pci-bridge.h>
|
|
|
|
|
|
|
|
/* export that to outside world */
|
|
|
|
struct device_node *of_chosen;
|
|
|
|
|
|
|
|
#define early_init_dt_scan_drconf_memory(node) 0
|
|
|
|
|
2009-12-11 14:42:21 +08:00
|
|
|
void __init early_init_dt_scan_chosen_arch(unsigned long node)
|
2009-03-27 21:25:12 +08:00
|
|
|
{
|
2009-12-11 14:42:21 +08:00
|
|
|
/* No Microblaze specific code here */
|
2009-03-27 21:25:12 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static int __init early_init_dt_scan_memory(unsigned long node,
|
|
|
|
const char *uname, int depth, void *data)
|
|
|
|
{
|
|
|
|
char *type = of_get_flat_dt_prop(node, "device_type", NULL);
|
2009-12-11 14:42:17 +08:00
|
|
|
__be32 *reg, *endp;
|
2009-03-27 21:25:12 +08:00
|
|
|
unsigned long l;
|
|
|
|
|
|
|
|
/* Look for the ibm,dynamic-reconfiguration-memory node */
|
|
|
|
/* if (depth == 1 &&
|
|
|
|
strcmp(uname, "ibm,dynamic-reconfiguration-memory") == 0)
|
|
|
|
return early_init_dt_scan_drconf_memory(node);
|
|
|
|
*/
|
|
|
|
/* We are scanning "memory" nodes only */
|
|
|
|
if (type == NULL) {
|
|
|
|
/*
|
|
|
|
* The longtrail doesn't have a device_type on the
|
|
|
|
* /memory node, so look for the node called /memory@0.
|
|
|
|
*/
|
|
|
|
if (depth != 1 || strcmp(uname, "memory@0") != 0)
|
|
|
|
return 0;
|
|
|
|
} else if (strcmp(type, "memory") != 0)
|
|
|
|
return 0;
|
|
|
|
|
2009-12-11 14:42:17 +08:00
|
|
|
reg = (__be32 *)of_get_flat_dt_prop(node, "linux,usable-memory", &l);
|
2009-03-27 21:25:12 +08:00
|
|
|
if (reg == NULL)
|
2009-12-11 14:42:17 +08:00
|
|
|
reg = (__be32 *)of_get_flat_dt_prop(node, "reg", &l);
|
2009-03-27 21:25:12 +08:00
|
|
|
if (reg == NULL)
|
|
|
|
return 0;
|
|
|
|
|
2009-12-11 14:42:17 +08:00
|
|
|
endp = reg + (l / sizeof(__be32));
|
2009-03-27 21:25:12 +08:00
|
|
|
|
|
|
|
pr_debug("memory scan node %s, reg size %ld, data: %x %x %x %x,\n",
|
|
|
|
uname, l, reg[0], reg[1], reg[2], reg[3]);
|
|
|
|
|
|
|
|
while ((endp - reg) >= (dt_root_addr_cells + dt_root_size_cells)) {
|
|
|
|
u64 base, size;
|
|
|
|
|
|
|
|
base = dt_mem_next_cell(dt_root_addr_cells, ®);
|
|
|
|
size = dt_mem_next_cell(dt_root_size_cells, ®);
|
|
|
|
|
|
|
|
if (size == 0)
|
|
|
|
continue;
|
|
|
|
pr_debug(" - %llx , %llx\n", (unsigned long long)base,
|
|
|
|
(unsigned long long)size);
|
|
|
|
|
|
|
|
lmb_add(base, size);
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
#ifdef CONFIG_EARLY_PRINTK
|
|
|
|
/* MS this is Microblaze specifig function */
|
|
|
|
static int __init early_init_dt_scan_serial(unsigned long node,
|
|
|
|
const char *uname, int depth, void *data)
|
|
|
|
{
|
|
|
|
unsigned long l;
|
|
|
|
char *p;
|
|
|
|
int *addr;
|
|
|
|
|
|
|
|
pr_debug("search \"chosen\", depth: %d, uname: %s\n", depth, uname);
|
|
|
|
|
|
|
|
/* find all serial nodes */
|
|
|
|
if (strncmp(uname, "serial", 6) != 0)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
early_init_dt_check_for_initrd(node);
|
|
|
|
|
|
|
|
/* find compatible node with uartlite */
|
|
|
|
p = of_get_flat_dt_prop(node, "compatible", &l);
|
|
|
|
if ((strncmp(p, "xlnx,xps-uartlite", 17) != 0) &&
|
|
|
|
(strncmp(p, "xlnx,opb-uartlite", 17) != 0))
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
addr = of_get_flat_dt_prop(node, "reg", &l);
|
|
|
|
return *addr; /* return address */
|
|
|
|
}
|
|
|
|
|
|
|
|
/* this function is looking for early uartlite console - Microblaze specific */
|
|
|
|
int __init early_uartlite_console(void)
|
|
|
|
{
|
|
|
|
return of_scan_flat_dt(early_init_dt_scan_serial, NULL);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
void __init early_init_devtree(void *params)
|
|
|
|
{
|
|
|
|
pr_debug(" -> early_init_devtree(%p)\n", params);
|
|
|
|
|
|
|
|
/* Setup flat device-tree pointer */
|
|
|
|
initial_boot_params = params;
|
|
|
|
|
|
|
|
/* Retrieve various informations from the /chosen node of the
|
|
|
|
* device-tree, including the platform type, initrd location and
|
|
|
|
* size, TCE reserve, and more ...
|
|
|
|
*/
|
|
|
|
of_scan_flat_dt(early_init_dt_scan_chosen, NULL);
|
|
|
|
|
|
|
|
/* Scan memory nodes and rebuild LMBs */
|
|
|
|
lmb_init();
|
|
|
|
of_scan_flat_dt(early_init_dt_scan_root, NULL);
|
|
|
|
of_scan_flat_dt(early_init_dt_scan_memory, NULL);
|
|
|
|
|
|
|
|
/* Save command line for /proc/cmdline and then parse parameters */
|
|
|
|
strlcpy(boot_command_line, cmd_line, COMMAND_LINE_SIZE);
|
|
|
|
parse_early_param();
|
|
|
|
|
|
|
|
lmb_analyze();
|
|
|
|
|
|
|
|
pr_debug("Phys. mem: %lx\n", (unsigned long) lmb_phys_mem_size());
|
|
|
|
|
|
|
|
pr_debug(" <- early_init_devtree()\n");
|
|
|
|
}
|
|
|
|
|
|
|
|
/*******
|
|
|
|
*
|
|
|
|
* New implementation of the OF "find" APIs, return a refcounted
|
|
|
|
* object, call of_node_put() when done. The device tree and list
|
|
|
|
* are protected by a rw_lock.
|
|
|
|
*
|
|
|
|
* Note that property management will need some locking as well,
|
|
|
|
* this isn't dealt with yet.
|
|
|
|
*
|
|
|
|
*******/
|
|
|
|
|
|
|
|
#if defined(CONFIG_DEBUG_FS) && defined(DEBUG)
|
|
|
|
static struct debugfs_blob_wrapper flat_dt_blob;
|
|
|
|
|
|
|
|
static int __init export_flat_device_tree(void)
|
|
|
|
{
|
|
|
|
struct dentry *d;
|
|
|
|
|
|
|
|
flat_dt_blob.data = initial_boot_params;
|
|
|
|
flat_dt_blob.size = initial_boot_params->totalsize;
|
|
|
|
|
|
|
|
d = debugfs_create_blob("flat-device-tree", S_IFREG | S_IRUSR,
|
|
|
|
of_debugfs_root, &flat_dt_blob);
|
|
|
|
if (!d)
|
|
|
|
return 1;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
device_initcall(export_flat_device_tree);
|
|
|
|
#endif
|