19b6b563cSPaul Mackerras /* 29b6b563cSPaul Mackerras * Procedures for creating, accessing and interpreting the device tree. 39b6b563cSPaul Mackerras * 49b6b563cSPaul Mackerras * Paul Mackerras August 1996. 59b6b563cSPaul Mackerras * Copyright (C) 1996-2005 Paul Mackerras. 69b6b563cSPaul Mackerras * 79b6b563cSPaul Mackerras * Adapted for 64bit PowerPC by Dave Engebretsen and Peter Bergner. 89b6b563cSPaul Mackerras * {engebret|bergner}@us.ibm.com 99b6b563cSPaul Mackerras * 109b6b563cSPaul Mackerras * This program is free software; you can redistribute it and/or 119b6b563cSPaul Mackerras * modify it under the terms of the GNU General Public License 129b6b563cSPaul Mackerras * as published by the Free Software Foundation; either version 139b6b563cSPaul Mackerras * 2 of the License, or (at your option) any later version. 149b6b563cSPaul Mackerras */ 159b6b563cSPaul Mackerras 169b6b563cSPaul Mackerras #undef DEBUG 179b6b563cSPaul Mackerras 189b6b563cSPaul Mackerras #include <stdarg.h> 199b6b563cSPaul Mackerras #include <linux/kernel.h> 209b6b563cSPaul Mackerras #include <linux/string.h> 219b6b563cSPaul Mackerras #include <linux/init.h> 229b6b563cSPaul Mackerras #include <linux/threads.h> 239b6b563cSPaul Mackerras #include <linux/spinlock.h> 249b6b563cSPaul Mackerras #include <linux/types.h> 259b6b563cSPaul Mackerras #include <linux/pci.h> 269b6b563cSPaul Mackerras #include <linux/stringify.h> 279b6b563cSPaul Mackerras #include <linux/delay.h> 289b6b563cSPaul Mackerras #include <linux/initrd.h> 299b6b563cSPaul Mackerras #include <linux/bitops.h> 309b6b563cSPaul Mackerras #include <linux/module.h> 31dcee3036SMichael Ellerman #include <linux/kexec.h> 327a4571aeSMichael Ellerman #include <linux/debugfs.h> 33*0ebfff14SBenjamin Herrenschmidt #include <linux/irq.h> 349b6b563cSPaul Mackerras 359b6b563cSPaul Mackerras #include <asm/prom.h> 369b6b563cSPaul Mackerras #include <asm/rtas.h> 379b6b563cSPaul Mackerras #include <asm/lmb.h> 389b6b563cSPaul Mackerras #include <asm/page.h> 399b6b563cSPaul Mackerras #include <asm/processor.h> 409b6b563cSPaul Mackerras #include <asm/irq.h> 419b6b563cSPaul Mackerras #include <asm/io.h> 420cc4746cSMichael Ellerman #include <asm/kdump.h> 439b6b563cSPaul Mackerras #include <asm/smp.h> 449b6b563cSPaul Mackerras #include <asm/system.h> 459b6b563cSPaul Mackerras #include <asm/mmu.h> 469b6b563cSPaul Mackerras #include <asm/pgtable.h> 479b6b563cSPaul Mackerras #include <asm/pci.h> 489b6b563cSPaul Mackerras #include <asm/iommu.h> 499b6b563cSPaul Mackerras #include <asm/btext.h> 509b6b563cSPaul Mackerras #include <asm/sections.h> 519b6b563cSPaul Mackerras #include <asm/machdep.h> 529b6b563cSPaul Mackerras #include <asm/pSeries_reconfig.h> 5340ef8cbcSPaul Mackerras #include <asm/pci-bridge.h> 542babf5c2SMichael Ellerman #include <asm/kexec.h> 559b6b563cSPaul Mackerras 569b6b563cSPaul Mackerras #ifdef DEBUG 579b6b563cSPaul Mackerras #define DBG(fmt...) printk(KERN_ERR fmt) 589b6b563cSPaul Mackerras #else 599b6b563cSPaul Mackerras #define DBG(fmt...) 609b6b563cSPaul Mackerras #endif 619b6b563cSPaul Mackerras 629b6b563cSPaul Mackerras 639b6b563cSPaul Mackerras static int __initdata dt_root_addr_cells; 649b6b563cSPaul Mackerras static int __initdata dt_root_size_cells; 659b6b563cSPaul Mackerras 669b6b563cSPaul Mackerras #ifdef CONFIG_PPC64 6728897731SOlof Johansson int __initdata iommu_is_off; 689b6b563cSPaul Mackerras int __initdata iommu_force_on; 69cf00a8d1SPaul Mackerras unsigned long tce_alloc_start, tce_alloc_end; 709b6b563cSPaul Mackerras #endif 719b6b563cSPaul Mackerras 729b6b563cSPaul Mackerras typedef u32 cell_t; 739b6b563cSPaul Mackerras 749b6b563cSPaul Mackerras #if 0 759b6b563cSPaul Mackerras static struct boot_param_header *initial_boot_params __initdata; 769b6b563cSPaul Mackerras #else 779b6b563cSPaul Mackerras struct boot_param_header *initial_boot_params; 789b6b563cSPaul Mackerras #endif 799b6b563cSPaul Mackerras 809b6b563cSPaul Mackerras static struct device_node *allnodes = NULL; 819b6b563cSPaul Mackerras 829b6b563cSPaul Mackerras /* use when traversing tree through the allnext, child, sibling, 839b6b563cSPaul Mackerras * or parent members of struct device_node. 849b6b563cSPaul Mackerras */ 859b6b563cSPaul Mackerras static DEFINE_RWLOCK(devtree_lock); 869b6b563cSPaul Mackerras 879b6b563cSPaul Mackerras /* export that to outside world */ 889b6b563cSPaul Mackerras struct device_node *of_chosen; 899b6b563cSPaul Mackerras 909b6b563cSPaul Mackerras static inline char *find_flat_dt_string(u32 offset) 919b6b563cSPaul Mackerras { 929b6b563cSPaul Mackerras return ((char *)initial_boot_params) + 939b6b563cSPaul Mackerras initial_boot_params->off_dt_strings + offset; 949b6b563cSPaul Mackerras } 959b6b563cSPaul Mackerras 969b6b563cSPaul Mackerras /** 979b6b563cSPaul Mackerras * This function is used to scan the flattened device-tree, it is 989b6b563cSPaul Mackerras * used to extract the memory informations at boot before we can 999b6b563cSPaul Mackerras * unflatten the tree 1009b6b563cSPaul Mackerras */ 1013c726f8dSBenjamin Herrenschmidt int __init of_scan_flat_dt(int (*it)(unsigned long node, 1029b6b563cSPaul Mackerras const char *uname, int depth, 1039b6b563cSPaul Mackerras void *data), 1049b6b563cSPaul Mackerras void *data) 1059b6b563cSPaul Mackerras { 1069b6b563cSPaul Mackerras unsigned long p = ((unsigned long)initial_boot_params) + 1079b6b563cSPaul Mackerras initial_boot_params->off_dt_struct; 1089b6b563cSPaul Mackerras int rc = 0; 1099b6b563cSPaul Mackerras int depth = -1; 1109b6b563cSPaul Mackerras 1119b6b563cSPaul Mackerras do { 1129b6b563cSPaul Mackerras u32 tag = *((u32 *)p); 1139b6b563cSPaul Mackerras char *pathp; 1149b6b563cSPaul Mackerras 1159b6b563cSPaul Mackerras p += 4; 1169b6b563cSPaul Mackerras if (tag == OF_DT_END_NODE) { 1179b6b563cSPaul Mackerras depth --; 1189b6b563cSPaul Mackerras continue; 1199b6b563cSPaul Mackerras } 1209b6b563cSPaul Mackerras if (tag == OF_DT_NOP) 1219b6b563cSPaul Mackerras continue; 1229b6b563cSPaul Mackerras if (tag == OF_DT_END) 1239b6b563cSPaul Mackerras break; 1249b6b563cSPaul Mackerras if (tag == OF_DT_PROP) { 1259b6b563cSPaul Mackerras u32 sz = *((u32 *)p); 1269b6b563cSPaul Mackerras p += 8; 1279b6b563cSPaul Mackerras if (initial_boot_params->version < 0x10) 1289b6b563cSPaul Mackerras p = _ALIGN(p, sz >= 8 ? 8 : 4); 1299b6b563cSPaul Mackerras p += sz; 1309b6b563cSPaul Mackerras p = _ALIGN(p, 4); 1319b6b563cSPaul Mackerras continue; 1329b6b563cSPaul Mackerras } 1339b6b563cSPaul Mackerras if (tag != OF_DT_BEGIN_NODE) { 1349b6b563cSPaul Mackerras printk(KERN_WARNING "Invalid tag %x scanning flattened" 1359b6b563cSPaul Mackerras " device tree !\n", tag); 1369b6b563cSPaul Mackerras return -EINVAL; 1379b6b563cSPaul Mackerras } 1389b6b563cSPaul Mackerras depth++; 1399b6b563cSPaul Mackerras pathp = (char *)p; 1409b6b563cSPaul Mackerras p = _ALIGN(p + strlen(pathp) + 1, 4); 1419b6b563cSPaul Mackerras if ((*pathp) == '/') { 1429b6b563cSPaul Mackerras char *lp, *np; 1439b6b563cSPaul Mackerras for (lp = NULL, np = pathp; *np; np++) 1449b6b563cSPaul Mackerras if ((*np) == '/') 1459b6b563cSPaul Mackerras lp = np+1; 1469b6b563cSPaul Mackerras if (lp != NULL) 1479b6b563cSPaul Mackerras pathp = lp; 1489b6b563cSPaul Mackerras } 1499b6b563cSPaul Mackerras rc = it(p, pathp, depth, data); 1509b6b563cSPaul Mackerras if (rc != 0) 1519b6b563cSPaul Mackerras break; 1529b6b563cSPaul Mackerras } while(1); 1539b6b563cSPaul Mackerras 1549b6b563cSPaul Mackerras return rc; 1559b6b563cSPaul Mackerras } 1569b6b563cSPaul Mackerras 157e8222502SBenjamin Herrenschmidt unsigned long __init of_get_flat_dt_root(void) 158e8222502SBenjamin Herrenschmidt { 159e8222502SBenjamin Herrenschmidt unsigned long p = ((unsigned long)initial_boot_params) + 160e8222502SBenjamin Herrenschmidt initial_boot_params->off_dt_struct; 161e8222502SBenjamin Herrenschmidt 162e8222502SBenjamin Herrenschmidt while(*((u32 *)p) == OF_DT_NOP) 163e8222502SBenjamin Herrenschmidt p += 4; 164e8222502SBenjamin Herrenschmidt BUG_ON (*((u32 *)p) != OF_DT_BEGIN_NODE); 165e8222502SBenjamin Herrenschmidt p += 4; 166e8222502SBenjamin Herrenschmidt return _ALIGN(p + strlen((char *)p) + 1, 4); 167e8222502SBenjamin Herrenschmidt } 168e8222502SBenjamin Herrenschmidt 1699b6b563cSPaul Mackerras /** 1709b6b563cSPaul Mackerras * This function can be used within scan_flattened_dt callback to get 1719b6b563cSPaul Mackerras * access to properties 1729b6b563cSPaul Mackerras */ 1733c726f8dSBenjamin Herrenschmidt void* __init of_get_flat_dt_prop(unsigned long node, const char *name, 1749b6b563cSPaul Mackerras unsigned long *size) 1759b6b563cSPaul Mackerras { 1769b6b563cSPaul Mackerras unsigned long p = node; 1779b6b563cSPaul Mackerras 1789b6b563cSPaul Mackerras do { 1799b6b563cSPaul Mackerras u32 tag = *((u32 *)p); 1809b6b563cSPaul Mackerras u32 sz, noff; 1819b6b563cSPaul Mackerras const char *nstr; 1829b6b563cSPaul Mackerras 1839b6b563cSPaul Mackerras p += 4; 1849b6b563cSPaul Mackerras if (tag == OF_DT_NOP) 1859b6b563cSPaul Mackerras continue; 1869b6b563cSPaul Mackerras if (tag != OF_DT_PROP) 1879b6b563cSPaul Mackerras return NULL; 1889b6b563cSPaul Mackerras 1899b6b563cSPaul Mackerras sz = *((u32 *)p); 1909b6b563cSPaul Mackerras noff = *((u32 *)(p + 4)); 1919b6b563cSPaul Mackerras p += 8; 1929b6b563cSPaul Mackerras if (initial_boot_params->version < 0x10) 1939b6b563cSPaul Mackerras p = _ALIGN(p, sz >= 8 ? 8 : 4); 1949b6b563cSPaul Mackerras 1959b6b563cSPaul Mackerras nstr = find_flat_dt_string(noff); 1969b6b563cSPaul Mackerras if (nstr == NULL) { 1979b6b563cSPaul Mackerras printk(KERN_WARNING "Can't find property index" 1989b6b563cSPaul Mackerras " name !\n"); 1999b6b563cSPaul Mackerras return NULL; 2009b6b563cSPaul Mackerras } 2019b6b563cSPaul Mackerras if (strcmp(name, nstr) == 0) { 2029b6b563cSPaul Mackerras if (size) 2039b6b563cSPaul Mackerras *size = sz; 2049b6b563cSPaul Mackerras return (void *)p; 2059b6b563cSPaul Mackerras } 2069b6b563cSPaul Mackerras p += sz; 2079b6b563cSPaul Mackerras p = _ALIGN(p, 4); 2089b6b563cSPaul Mackerras } while(1); 2099b6b563cSPaul Mackerras } 2109b6b563cSPaul Mackerras 211e8222502SBenjamin Herrenschmidt int __init of_flat_dt_is_compatible(unsigned long node, const char *compat) 212e8222502SBenjamin Herrenschmidt { 213e8222502SBenjamin Herrenschmidt const char* cp; 214e8222502SBenjamin Herrenschmidt unsigned long cplen, l; 215e8222502SBenjamin Herrenschmidt 216e8222502SBenjamin Herrenschmidt cp = of_get_flat_dt_prop(node, "compatible", &cplen); 217e8222502SBenjamin Herrenschmidt if (cp == NULL) 218e8222502SBenjamin Herrenschmidt return 0; 219e8222502SBenjamin Herrenschmidt while (cplen > 0) { 220e8222502SBenjamin Herrenschmidt if (strncasecmp(cp, compat, strlen(compat)) == 0) 221e8222502SBenjamin Herrenschmidt return 1; 222e8222502SBenjamin Herrenschmidt l = strlen(cp) + 1; 223e8222502SBenjamin Herrenschmidt cp += l; 224e8222502SBenjamin Herrenschmidt cplen -= l; 225e8222502SBenjamin Herrenschmidt } 226e8222502SBenjamin Herrenschmidt 227e8222502SBenjamin Herrenschmidt return 0; 228e8222502SBenjamin Herrenschmidt } 229e8222502SBenjamin Herrenschmidt 2309b6b563cSPaul Mackerras static void *__init unflatten_dt_alloc(unsigned long *mem, unsigned long size, 2319b6b563cSPaul Mackerras unsigned long align) 2329b6b563cSPaul Mackerras { 2339b6b563cSPaul Mackerras void *res; 2349b6b563cSPaul Mackerras 2359b6b563cSPaul Mackerras *mem = _ALIGN(*mem, align); 2369b6b563cSPaul Mackerras res = (void *)*mem; 2379b6b563cSPaul Mackerras *mem += size; 2389b6b563cSPaul Mackerras 2399b6b563cSPaul Mackerras return res; 2409b6b563cSPaul Mackerras } 2419b6b563cSPaul Mackerras 2429b6b563cSPaul Mackerras static unsigned long __init unflatten_dt_node(unsigned long mem, 2439b6b563cSPaul Mackerras unsigned long *p, 2449b6b563cSPaul Mackerras struct device_node *dad, 2459b6b563cSPaul Mackerras struct device_node ***allnextpp, 2469b6b563cSPaul Mackerras unsigned long fpsize) 2479b6b563cSPaul Mackerras { 2489b6b563cSPaul Mackerras struct device_node *np; 2499b6b563cSPaul Mackerras struct property *pp, **prev_pp = NULL; 2509b6b563cSPaul Mackerras char *pathp; 2519b6b563cSPaul Mackerras u32 tag; 2529b6b563cSPaul Mackerras unsigned int l, allocl; 2539b6b563cSPaul Mackerras int has_name = 0; 2549b6b563cSPaul Mackerras int new_format = 0; 2559b6b563cSPaul Mackerras 2569b6b563cSPaul Mackerras tag = *((u32 *)(*p)); 2579b6b563cSPaul Mackerras if (tag != OF_DT_BEGIN_NODE) { 2589b6b563cSPaul Mackerras printk("Weird tag at start of node: %x\n", tag); 2599b6b563cSPaul Mackerras return mem; 2609b6b563cSPaul Mackerras } 2619b6b563cSPaul Mackerras *p += 4; 2629b6b563cSPaul Mackerras pathp = (char *)*p; 2639b6b563cSPaul Mackerras l = allocl = strlen(pathp) + 1; 2649b6b563cSPaul Mackerras *p = _ALIGN(*p + l, 4); 2659b6b563cSPaul Mackerras 2669b6b563cSPaul Mackerras /* version 0x10 has a more compact unit name here instead of the full 2679b6b563cSPaul Mackerras * path. we accumulate the full path size using "fpsize", we'll rebuild 2689b6b563cSPaul Mackerras * it later. We detect this because the first character of the name is 2699b6b563cSPaul Mackerras * not '/'. 2709b6b563cSPaul Mackerras */ 2719b6b563cSPaul Mackerras if ((*pathp) != '/') { 2729b6b563cSPaul Mackerras new_format = 1; 2739b6b563cSPaul Mackerras if (fpsize == 0) { 2749b6b563cSPaul Mackerras /* root node: special case. fpsize accounts for path 2759b6b563cSPaul Mackerras * plus terminating zero. root node only has '/', so 2769b6b563cSPaul Mackerras * fpsize should be 2, but we want to avoid the first 2779b6b563cSPaul Mackerras * level nodes to have two '/' so we use fpsize 1 here 2789b6b563cSPaul Mackerras */ 2799b6b563cSPaul Mackerras fpsize = 1; 2809b6b563cSPaul Mackerras allocl = 2; 2819b6b563cSPaul Mackerras } else { 2829b6b563cSPaul Mackerras /* account for '/' and path size minus terminal 0 2839b6b563cSPaul Mackerras * already in 'l' 2849b6b563cSPaul Mackerras */ 2859b6b563cSPaul Mackerras fpsize += l; 2869b6b563cSPaul Mackerras allocl = fpsize; 2879b6b563cSPaul Mackerras } 2889b6b563cSPaul Mackerras } 2899b6b563cSPaul Mackerras 2909b6b563cSPaul Mackerras 2919b6b563cSPaul Mackerras np = unflatten_dt_alloc(&mem, sizeof(struct device_node) + allocl, 2929b6b563cSPaul Mackerras __alignof__(struct device_node)); 2939b6b563cSPaul Mackerras if (allnextpp) { 2949b6b563cSPaul Mackerras memset(np, 0, sizeof(*np)); 2959b6b563cSPaul Mackerras np->full_name = ((char*)np) + sizeof(struct device_node); 2969b6b563cSPaul Mackerras if (new_format) { 2979b6b563cSPaul Mackerras char *p = np->full_name; 2989b6b563cSPaul Mackerras /* rebuild full path for new format */ 2999b6b563cSPaul Mackerras if (dad && dad->parent) { 3009b6b563cSPaul Mackerras strcpy(p, dad->full_name); 3019b6b563cSPaul Mackerras #ifdef DEBUG 3029b6b563cSPaul Mackerras if ((strlen(p) + l + 1) != allocl) { 3039b6b563cSPaul Mackerras DBG("%s: p: %d, l: %d, a: %d\n", 304e8222502SBenjamin Herrenschmidt pathp, (int)strlen(p), l, allocl); 3059b6b563cSPaul Mackerras } 3069b6b563cSPaul Mackerras #endif 3079b6b563cSPaul Mackerras p += strlen(p); 3089b6b563cSPaul Mackerras } 3099b6b563cSPaul Mackerras *(p++) = '/'; 3109b6b563cSPaul Mackerras memcpy(p, pathp, l); 3119b6b563cSPaul Mackerras } else 3129b6b563cSPaul Mackerras memcpy(np->full_name, pathp, l); 3139b6b563cSPaul Mackerras prev_pp = &np->properties; 3149b6b563cSPaul Mackerras **allnextpp = np; 3159b6b563cSPaul Mackerras *allnextpp = &np->allnext; 3169b6b563cSPaul Mackerras if (dad != NULL) { 3179b6b563cSPaul Mackerras np->parent = dad; 3189b6b563cSPaul Mackerras /* we temporarily use the next field as `last_child'*/ 3199b6b563cSPaul Mackerras if (dad->next == 0) 3209b6b563cSPaul Mackerras dad->child = np; 3219b6b563cSPaul Mackerras else 3229b6b563cSPaul Mackerras dad->next->sibling = np; 3239b6b563cSPaul Mackerras dad->next = np; 3249b6b563cSPaul Mackerras } 3259b6b563cSPaul Mackerras kref_init(&np->kref); 3269b6b563cSPaul Mackerras } 3279b6b563cSPaul Mackerras while(1) { 3289b6b563cSPaul Mackerras u32 sz, noff; 3299b6b563cSPaul Mackerras char *pname; 3309b6b563cSPaul Mackerras 3319b6b563cSPaul Mackerras tag = *((u32 *)(*p)); 3329b6b563cSPaul Mackerras if (tag == OF_DT_NOP) { 3339b6b563cSPaul Mackerras *p += 4; 3349b6b563cSPaul Mackerras continue; 3359b6b563cSPaul Mackerras } 3369b6b563cSPaul Mackerras if (tag != OF_DT_PROP) 3379b6b563cSPaul Mackerras break; 3389b6b563cSPaul Mackerras *p += 4; 3399b6b563cSPaul Mackerras sz = *((u32 *)(*p)); 3409b6b563cSPaul Mackerras noff = *((u32 *)((*p) + 4)); 3419b6b563cSPaul Mackerras *p += 8; 3429b6b563cSPaul Mackerras if (initial_boot_params->version < 0x10) 3439b6b563cSPaul Mackerras *p = _ALIGN(*p, sz >= 8 ? 8 : 4); 3449b6b563cSPaul Mackerras 3459b6b563cSPaul Mackerras pname = find_flat_dt_string(noff); 3469b6b563cSPaul Mackerras if (pname == NULL) { 3479b6b563cSPaul Mackerras printk("Can't find property name in list !\n"); 3489b6b563cSPaul Mackerras break; 3499b6b563cSPaul Mackerras } 3509b6b563cSPaul Mackerras if (strcmp(pname, "name") == 0) 3519b6b563cSPaul Mackerras has_name = 1; 3529b6b563cSPaul Mackerras l = strlen(pname) + 1; 3539b6b563cSPaul Mackerras pp = unflatten_dt_alloc(&mem, sizeof(struct property), 3549b6b563cSPaul Mackerras __alignof__(struct property)); 3559b6b563cSPaul Mackerras if (allnextpp) { 3569b6b563cSPaul Mackerras if (strcmp(pname, "linux,phandle") == 0) { 3579b6b563cSPaul Mackerras np->node = *((u32 *)*p); 3589b6b563cSPaul Mackerras if (np->linux_phandle == 0) 3599b6b563cSPaul Mackerras np->linux_phandle = np->node; 3609b6b563cSPaul Mackerras } 3619b6b563cSPaul Mackerras if (strcmp(pname, "ibm,phandle") == 0) 3629b6b563cSPaul Mackerras np->linux_phandle = *((u32 *)*p); 3639b6b563cSPaul Mackerras pp->name = pname; 3649b6b563cSPaul Mackerras pp->length = sz; 3659b6b563cSPaul Mackerras pp->value = (void *)*p; 3669b6b563cSPaul Mackerras *prev_pp = pp; 3679b6b563cSPaul Mackerras prev_pp = &pp->next; 3689b6b563cSPaul Mackerras } 3699b6b563cSPaul Mackerras *p = _ALIGN((*p) + sz, 4); 3709b6b563cSPaul Mackerras } 3719b6b563cSPaul Mackerras /* with version 0x10 we may not have the name property, recreate 3729b6b563cSPaul Mackerras * it here from the unit name if absent 3739b6b563cSPaul Mackerras */ 3749b6b563cSPaul Mackerras if (!has_name) { 3759b6b563cSPaul Mackerras char *p = pathp, *ps = pathp, *pa = NULL; 3769b6b563cSPaul Mackerras int sz; 3779b6b563cSPaul Mackerras 3789b6b563cSPaul Mackerras while (*p) { 3799b6b563cSPaul Mackerras if ((*p) == '@') 3809b6b563cSPaul Mackerras pa = p; 3819b6b563cSPaul Mackerras if ((*p) == '/') 3829b6b563cSPaul Mackerras ps = p + 1; 3839b6b563cSPaul Mackerras p++; 3849b6b563cSPaul Mackerras } 3859b6b563cSPaul Mackerras if (pa < ps) 3869b6b563cSPaul Mackerras pa = p; 3879b6b563cSPaul Mackerras sz = (pa - ps) + 1; 3889b6b563cSPaul Mackerras pp = unflatten_dt_alloc(&mem, sizeof(struct property) + sz, 3899b6b563cSPaul Mackerras __alignof__(struct property)); 3909b6b563cSPaul Mackerras if (allnextpp) { 3919b6b563cSPaul Mackerras pp->name = "name"; 3929b6b563cSPaul Mackerras pp->length = sz; 3939b6b563cSPaul Mackerras pp->value = (unsigned char *)(pp + 1); 3949b6b563cSPaul Mackerras *prev_pp = pp; 3959b6b563cSPaul Mackerras prev_pp = &pp->next; 3969b6b563cSPaul Mackerras memcpy(pp->value, ps, sz - 1); 3979b6b563cSPaul Mackerras ((char *)pp->value)[sz - 1] = 0; 3989b6b563cSPaul Mackerras DBG("fixed up name for %s -> %s\n", pathp, pp->value); 3999b6b563cSPaul Mackerras } 4009b6b563cSPaul Mackerras } 4019b6b563cSPaul Mackerras if (allnextpp) { 4029b6b563cSPaul Mackerras *prev_pp = NULL; 4039b6b563cSPaul Mackerras np->name = get_property(np, "name", NULL); 4049b6b563cSPaul Mackerras np->type = get_property(np, "device_type", NULL); 4059b6b563cSPaul Mackerras 4069b6b563cSPaul Mackerras if (!np->name) 4079b6b563cSPaul Mackerras np->name = "<NULL>"; 4089b6b563cSPaul Mackerras if (!np->type) 4099b6b563cSPaul Mackerras np->type = "<NULL>"; 4109b6b563cSPaul Mackerras } 4119b6b563cSPaul Mackerras while (tag == OF_DT_BEGIN_NODE) { 4129b6b563cSPaul Mackerras mem = unflatten_dt_node(mem, p, np, allnextpp, fpsize); 4139b6b563cSPaul Mackerras tag = *((u32 *)(*p)); 4149b6b563cSPaul Mackerras } 4159b6b563cSPaul Mackerras if (tag != OF_DT_END_NODE) { 4169b6b563cSPaul Mackerras printk("Weird tag at end of node: %x\n", tag); 4179b6b563cSPaul Mackerras return mem; 4189b6b563cSPaul Mackerras } 4199b6b563cSPaul Mackerras *p += 4; 4209b6b563cSPaul Mackerras return mem; 4219b6b563cSPaul Mackerras } 4229b6b563cSPaul Mackerras 4232babf5c2SMichael Ellerman static int __init early_parse_mem(char *p) 4242babf5c2SMichael Ellerman { 4252babf5c2SMichael Ellerman if (!p) 4262babf5c2SMichael Ellerman return 1; 4272babf5c2SMichael Ellerman 4282babf5c2SMichael Ellerman memory_limit = PAGE_ALIGN(memparse(p, &p)); 4292babf5c2SMichael Ellerman DBG("memory limit = 0x%lx\n", memory_limit); 4302babf5c2SMichael Ellerman 4312babf5c2SMichael Ellerman return 0; 4322babf5c2SMichael Ellerman } 4332babf5c2SMichael Ellerman early_param("mem", early_parse_mem); 4342babf5c2SMichael Ellerman 4352babf5c2SMichael Ellerman /* 4362babf5c2SMichael Ellerman * The device tree may be allocated below our memory limit, or inside the 4372babf5c2SMichael Ellerman * crash kernel region for kdump. If so, move it out now. 4382babf5c2SMichael Ellerman */ 4392babf5c2SMichael Ellerman static void move_device_tree(void) 4402babf5c2SMichael Ellerman { 4412babf5c2SMichael Ellerman unsigned long start, size; 4422babf5c2SMichael Ellerman void *p; 4432babf5c2SMichael Ellerman 4442babf5c2SMichael Ellerman DBG("-> move_device_tree\n"); 4452babf5c2SMichael Ellerman 4462babf5c2SMichael Ellerman start = __pa(initial_boot_params); 4472babf5c2SMichael Ellerman size = initial_boot_params->totalsize; 4482babf5c2SMichael Ellerman 4492babf5c2SMichael Ellerman if ((memory_limit && (start + size) > memory_limit) || 4502babf5c2SMichael Ellerman overlaps_crashkernel(start, size)) { 4512babf5c2SMichael Ellerman p = __va(lmb_alloc_base(size, PAGE_SIZE, lmb.rmo_size)); 4522babf5c2SMichael Ellerman memcpy(p, initial_boot_params, size); 4532babf5c2SMichael Ellerman initial_boot_params = (struct boot_param_header *)p; 4542babf5c2SMichael Ellerman DBG("Moved device tree to 0x%p\n", p); 4552babf5c2SMichael Ellerman } 4562babf5c2SMichael Ellerman 4572babf5c2SMichael Ellerman DBG("<- move_device_tree\n"); 4582babf5c2SMichael Ellerman } 4599b6b563cSPaul Mackerras 4609b6b563cSPaul Mackerras /** 4619b6b563cSPaul Mackerras * unflattens the device-tree passed by the firmware, creating the 4629b6b563cSPaul Mackerras * tree of struct device_node. It also fills the "name" and "type" 4639b6b563cSPaul Mackerras * pointers of the nodes so the normal device-tree walking functions 4649b6b563cSPaul Mackerras * can be used (this used to be done by finish_device_tree) 4659b6b563cSPaul Mackerras */ 4669b6b563cSPaul Mackerras void __init unflatten_device_tree(void) 4679b6b563cSPaul Mackerras { 4689b6b563cSPaul Mackerras unsigned long start, mem, size; 4699b6b563cSPaul Mackerras struct device_node **allnextp = &allnodes; 4709b6b563cSPaul Mackerras 4719b6b563cSPaul Mackerras DBG(" -> unflatten_device_tree()\n"); 4729b6b563cSPaul Mackerras 4739b6b563cSPaul Mackerras /* First pass, scan for size */ 4749b6b563cSPaul Mackerras start = ((unsigned long)initial_boot_params) + 4759b6b563cSPaul Mackerras initial_boot_params->off_dt_struct; 4769b6b563cSPaul Mackerras size = unflatten_dt_node(0, &start, NULL, NULL, 0); 4779b6b563cSPaul Mackerras size = (size | 3) + 1; 4789b6b563cSPaul Mackerras 4799b6b563cSPaul Mackerras DBG(" size is %lx, allocating...\n", size); 4809b6b563cSPaul Mackerras 4819b6b563cSPaul Mackerras /* Allocate memory for the expanded device tree */ 4829b6b563cSPaul Mackerras mem = lmb_alloc(size + 4, __alignof__(struct device_node)); 4839b6b563cSPaul Mackerras mem = (unsigned long) __va(mem); 4849b6b563cSPaul Mackerras 4859b6b563cSPaul Mackerras ((u32 *)mem)[size / 4] = 0xdeadbeef; 4869b6b563cSPaul Mackerras 4879b6b563cSPaul Mackerras DBG(" unflattening %lx...\n", mem); 4889b6b563cSPaul Mackerras 4899b6b563cSPaul Mackerras /* Second pass, do actual unflattening */ 4909b6b563cSPaul Mackerras start = ((unsigned long)initial_boot_params) + 4919b6b563cSPaul Mackerras initial_boot_params->off_dt_struct; 4929b6b563cSPaul Mackerras unflatten_dt_node(mem, &start, NULL, &allnextp, 0); 4939b6b563cSPaul Mackerras if (*((u32 *)start) != OF_DT_END) 4949b6b563cSPaul Mackerras printk(KERN_WARNING "Weird tag at end of tree: %08x\n", *((u32 *)start)); 4959b6b563cSPaul Mackerras if (((u32 *)mem)[size / 4] != 0xdeadbeef) 4969b6b563cSPaul Mackerras printk(KERN_WARNING "End of tree marker overwritten: %08x\n", 4979b6b563cSPaul Mackerras ((u32 *)mem)[size / 4] ); 4989b6b563cSPaul Mackerras *allnextp = NULL; 4999b6b563cSPaul Mackerras 5009b6b563cSPaul Mackerras /* Get pointer to OF "/chosen" node for use everywhere */ 5019b6b563cSPaul Mackerras of_chosen = of_find_node_by_path("/chosen"); 502a575b807SPaul Mackerras if (of_chosen == NULL) 503a575b807SPaul Mackerras of_chosen = of_find_node_by_path("/chosen@0"); 5049b6b563cSPaul Mackerras 5059b6b563cSPaul Mackerras DBG(" <- unflatten_device_tree()\n"); 5069b6b563cSPaul Mackerras } 5079b6b563cSPaul Mackerras 508d205819eSPaul Mackerras /* 509d205819eSPaul Mackerras * ibm,pa-features is a per-cpu property that contains a string of 510d205819eSPaul Mackerras * attribute descriptors, each of which has a 2 byte header plus up 511d205819eSPaul Mackerras * to 254 bytes worth of processor attribute bits. First header 512d205819eSPaul Mackerras * byte specifies the number of bytes following the header. 513d205819eSPaul Mackerras * Second header byte is an "attribute-specifier" type, of which 514d205819eSPaul Mackerras * zero is the only currently-defined value. 515d205819eSPaul Mackerras * Implementation: Pass in the byte and bit offset for the feature 516d205819eSPaul Mackerras * that we are interested in. The function will return -1 if the 517d205819eSPaul Mackerras * pa-features property is missing, or a 1/0 to indicate if the feature 518d205819eSPaul Mackerras * is supported/not supported. Note that the bit numbers are 519d205819eSPaul Mackerras * big-endian to match the definition in PAPR. 520d205819eSPaul Mackerras */ 521d205819eSPaul Mackerras static struct ibm_pa_feature { 522d205819eSPaul Mackerras unsigned long cpu_features; /* CPU_FTR_xxx bit */ 523d205819eSPaul Mackerras unsigned int cpu_user_ftrs; /* PPC_FEATURE_xxx bit */ 524d205819eSPaul Mackerras unsigned char pabyte; /* byte number in ibm,pa-features */ 525d205819eSPaul Mackerras unsigned char pabit; /* bit number (big-endian) */ 526d205819eSPaul Mackerras unsigned char invert; /* if 1, pa bit set => clear feature */ 527d205819eSPaul Mackerras } ibm_pa_features[] __initdata = { 528d205819eSPaul Mackerras {0, PPC_FEATURE_HAS_MMU, 0, 0, 0}, 529d205819eSPaul Mackerras {0, PPC_FEATURE_HAS_FPU, 0, 1, 0}, 530d205819eSPaul Mackerras {CPU_FTR_SLB, 0, 0, 2, 0}, 531d205819eSPaul Mackerras {CPU_FTR_CTRL, 0, 0, 3, 0}, 532d205819eSPaul Mackerras {CPU_FTR_NOEXECUTE, 0, 0, 6, 0}, 533d205819eSPaul Mackerras {CPU_FTR_NODSISRALIGN, 0, 1, 1, 1}, 534bf72aebaSPaul Mackerras #if 0 535bf72aebaSPaul Mackerras /* put this back once we know how to test if firmware does 64k IO */ 536d205819eSPaul Mackerras {CPU_FTR_CI_LARGE_PAGE, 0, 1, 2, 0}, 537bf72aebaSPaul Mackerras #endif 538339d76c5SPaul Mackerras {CPU_FTR_REAL_LE, PPC_FEATURE_TRUE_LE, 5, 0, 0}, 539d205819eSPaul Mackerras }; 540d205819eSPaul Mackerras 541d205819eSPaul Mackerras static void __init check_cpu_pa_features(unsigned long node) 542d205819eSPaul Mackerras { 543d205819eSPaul Mackerras unsigned char *pa_ftrs; 544d205819eSPaul Mackerras unsigned long len, tablelen, i, bit; 545d205819eSPaul Mackerras 546d205819eSPaul Mackerras pa_ftrs = of_get_flat_dt_prop(node, "ibm,pa-features", &tablelen); 547d205819eSPaul Mackerras if (pa_ftrs == NULL) 548d205819eSPaul Mackerras return; 549d205819eSPaul Mackerras 550d205819eSPaul Mackerras /* find descriptor with type == 0 */ 551d205819eSPaul Mackerras for (;;) { 552d205819eSPaul Mackerras if (tablelen < 3) 553d205819eSPaul Mackerras return; 554d205819eSPaul Mackerras len = 2 + pa_ftrs[0]; 555d205819eSPaul Mackerras if (tablelen < len) 556d205819eSPaul Mackerras return; /* descriptor 0 not found */ 557d205819eSPaul Mackerras if (pa_ftrs[1] == 0) 558d205819eSPaul Mackerras break; 559d205819eSPaul Mackerras tablelen -= len; 560d205819eSPaul Mackerras pa_ftrs += len; 561d205819eSPaul Mackerras } 562d205819eSPaul Mackerras 563d205819eSPaul Mackerras /* loop over bits we know about */ 564d205819eSPaul Mackerras for (i = 0; i < ARRAY_SIZE(ibm_pa_features); ++i) { 565d205819eSPaul Mackerras struct ibm_pa_feature *fp = &ibm_pa_features[i]; 566d205819eSPaul Mackerras 567d205819eSPaul Mackerras if (fp->pabyte >= pa_ftrs[0]) 568d205819eSPaul Mackerras continue; 569d205819eSPaul Mackerras bit = (pa_ftrs[2 + fp->pabyte] >> (7 - fp->pabit)) & 1; 570d205819eSPaul Mackerras if (bit ^ fp->invert) { 571d205819eSPaul Mackerras cur_cpu_spec->cpu_features |= fp->cpu_features; 572d205819eSPaul Mackerras cur_cpu_spec->cpu_user_features |= fp->cpu_user_ftrs; 573d205819eSPaul Mackerras } else { 574d205819eSPaul Mackerras cur_cpu_spec->cpu_features &= ~fp->cpu_features; 575d205819eSPaul Mackerras cur_cpu_spec->cpu_user_features &= ~fp->cpu_user_ftrs; 576d205819eSPaul Mackerras } 577d205819eSPaul Mackerras } 578d205819eSPaul Mackerras } 579d205819eSPaul Mackerras 5809b6b563cSPaul Mackerras static int __init early_init_dt_scan_cpus(unsigned long node, 5814df20460SAnton Blanchard const char *uname, int depth, 5824df20460SAnton Blanchard void *data) 5839b6b563cSPaul Mackerras { 5844df20460SAnton Blanchard static int logical_cpuid = 0; 5854df20460SAnton Blanchard char *type = of_get_flat_dt_prop(node, "device_type", NULL); 5864d177fbfSStephen Rothwell #ifdef CONFIG_ALTIVEC 5874d177fbfSStephen Rothwell u32 *prop; 5884d177fbfSStephen Rothwell #endif 5894d177fbfSStephen Rothwell u32 *intserv; 5904df20460SAnton Blanchard int i, nthreads; 5914df20460SAnton Blanchard unsigned long len; 5924df20460SAnton Blanchard int found = 0; 5939b6b563cSPaul Mackerras 5949b6b563cSPaul Mackerras /* We are scanning "cpu" nodes only */ 5959b6b563cSPaul Mackerras if (type == NULL || strcmp(type, "cpu") != 0) 5969b6b563cSPaul Mackerras return 0; 5979b6b563cSPaul Mackerras 5984df20460SAnton Blanchard /* Get physical cpuid */ 5994df20460SAnton Blanchard intserv = of_get_flat_dt_prop(node, "ibm,ppc-interrupt-server#s", &len); 6004df20460SAnton Blanchard if (intserv) { 6014df20460SAnton Blanchard nthreads = len / sizeof(int); 6029b6b563cSPaul Mackerras } else { 6034df20460SAnton Blanchard intserv = of_get_flat_dt_prop(node, "reg", NULL); 6044df20460SAnton Blanchard nthreads = 1; 6054df20460SAnton Blanchard } 6064df20460SAnton Blanchard 6074df20460SAnton Blanchard /* 6084df20460SAnton Blanchard * Now see if any of these threads match our boot cpu. 6094df20460SAnton Blanchard * NOTE: This must match the parsing done in smp_setup_cpu_maps. 6104df20460SAnton Blanchard */ 6114df20460SAnton Blanchard for (i = 0; i < nthreads; i++) { 6124df20460SAnton Blanchard /* 6134df20460SAnton Blanchard * version 2 of the kexec param format adds the phys cpuid of 6144df20460SAnton Blanchard * booted proc. 6154df20460SAnton Blanchard */ 6164df20460SAnton Blanchard if (initial_boot_params && initial_boot_params->version >= 2) { 6174df20460SAnton Blanchard if (intserv[i] == 6184df20460SAnton Blanchard initial_boot_params->boot_cpuid_phys) { 6194df20460SAnton Blanchard found = 1; 6204df20460SAnton Blanchard break; 6214df20460SAnton Blanchard } 6224df20460SAnton Blanchard } else { 6234df20460SAnton Blanchard /* 6244df20460SAnton Blanchard * Check if it's the boot-cpu, set it's hw index now, 6254df20460SAnton Blanchard * unfortunately this format did not support booting 6264df20460SAnton Blanchard * off secondary threads. 6274df20460SAnton Blanchard */ 6283c726f8dSBenjamin Herrenschmidt if (of_get_flat_dt_prop(node, 6293c726f8dSBenjamin Herrenschmidt "linux,boot-cpu", NULL) != NULL) { 6304df20460SAnton Blanchard found = 1; 6314df20460SAnton Blanchard break; 6329b6b563cSPaul Mackerras } 6339b6b563cSPaul Mackerras } 6344df20460SAnton Blanchard 6354df20460SAnton Blanchard #ifdef CONFIG_SMP 6364df20460SAnton Blanchard /* logical cpu id is always 0 on UP kernels */ 6374df20460SAnton Blanchard logical_cpuid++; 6384df20460SAnton Blanchard #endif 6394df20460SAnton Blanchard } 6404df20460SAnton Blanchard 6414df20460SAnton Blanchard if (found) { 6424df20460SAnton Blanchard DBG("boot cpu: logical %d physical %d\n", logical_cpuid, 6434df20460SAnton Blanchard intserv[i]); 6444df20460SAnton Blanchard boot_cpuid = logical_cpuid; 6454df20460SAnton Blanchard set_hard_smp_processor_id(boot_cpuid, intserv[i]); 6464df20460SAnton Blanchard } 6479b6b563cSPaul Mackerras 6489b6b563cSPaul Mackerras #ifdef CONFIG_ALTIVEC 6499b6b563cSPaul Mackerras /* Check if we have a VMX and eventually update CPU features */ 650676e2497SStephen Rothwell prop = (u32 *)of_get_flat_dt_prop(node, "ibm,vmx", NULL); 6519b6b563cSPaul Mackerras if (prop && (*prop) > 0) { 6529b6b563cSPaul Mackerras cur_cpu_spec->cpu_features |= CPU_FTR_ALTIVEC; 6539b6b563cSPaul Mackerras cur_cpu_spec->cpu_user_features |= PPC_FEATURE_HAS_ALTIVEC; 6549b6b563cSPaul Mackerras } 6559b6b563cSPaul Mackerras 6569b6b563cSPaul Mackerras /* Same goes for Apple's "altivec" property */ 6573c726f8dSBenjamin Herrenschmidt prop = (u32 *)of_get_flat_dt_prop(node, "altivec", NULL); 6589b6b563cSPaul Mackerras if (prop) { 6599b6b563cSPaul Mackerras cur_cpu_spec->cpu_features |= CPU_FTR_ALTIVEC; 6609b6b563cSPaul Mackerras cur_cpu_spec->cpu_user_features |= PPC_FEATURE_HAS_ALTIVEC; 6619b6b563cSPaul Mackerras } 6629b6b563cSPaul Mackerras #endif /* CONFIG_ALTIVEC */ 6639b6b563cSPaul Mackerras 664d205819eSPaul Mackerras check_cpu_pa_features(node); 665d205819eSPaul Mackerras 6669b6b563cSPaul Mackerras #ifdef CONFIG_PPC_PSERIES 6674df20460SAnton Blanchard if (nthreads > 1) 6689b6b563cSPaul Mackerras cur_cpu_spec->cpu_features |= CPU_FTR_SMT; 6694df20460SAnton Blanchard else 6704df20460SAnton Blanchard cur_cpu_spec->cpu_features &= ~CPU_FTR_SMT; 6719b6b563cSPaul Mackerras #endif 6729b6b563cSPaul Mackerras 6739b6b563cSPaul Mackerras return 0; 6749b6b563cSPaul Mackerras } 6759b6b563cSPaul Mackerras 6769b6b563cSPaul Mackerras static int __init early_init_dt_scan_chosen(unsigned long node, 6779b6b563cSPaul Mackerras const char *uname, int depth, void *data) 6789b6b563cSPaul Mackerras { 6799b6b563cSPaul Mackerras unsigned long *lprop; 680329dda08SKumar Gala unsigned long l; 681329dda08SKumar Gala char *p; 6829b6b563cSPaul Mackerras 6839b6b563cSPaul Mackerras DBG("search \"chosen\", depth: %d, uname: %s\n", depth, uname); 6849b6b563cSPaul Mackerras 685a575b807SPaul Mackerras if (depth != 1 || 686a575b807SPaul Mackerras (strcmp(uname, "chosen") != 0 && strcmp(uname, "chosen@0") != 0)) 6879b6b563cSPaul Mackerras return 0; 6889b6b563cSPaul Mackerras 6899b6b563cSPaul Mackerras #ifdef CONFIG_PPC64 6909b6b563cSPaul Mackerras /* check if iommu is forced on or off */ 6913c726f8dSBenjamin Herrenschmidt if (of_get_flat_dt_prop(node, "linux,iommu-off", NULL) != NULL) 6929b6b563cSPaul Mackerras iommu_is_off = 1; 6933c726f8dSBenjamin Herrenschmidt if (of_get_flat_dt_prop(node, "linux,iommu-force-on", NULL) != NULL) 6949b6b563cSPaul Mackerras iommu_force_on = 1; 6959b6b563cSPaul Mackerras #endif 6969b6b563cSPaul Mackerras 6972babf5c2SMichael Ellerman /* mem=x on the command line is the preferred mechanism */ 6983c726f8dSBenjamin Herrenschmidt lprop = of_get_flat_dt_prop(node, "linux,memory-limit", NULL); 6999b6b563cSPaul Mackerras if (lprop) 7009b6b563cSPaul Mackerras memory_limit = *lprop; 7019b6b563cSPaul Mackerras 7029b6b563cSPaul Mackerras #ifdef CONFIG_PPC64 7033c726f8dSBenjamin Herrenschmidt lprop = of_get_flat_dt_prop(node, "linux,tce-alloc-start", NULL); 7049b6b563cSPaul Mackerras if (lprop) 7059b6b563cSPaul Mackerras tce_alloc_start = *lprop; 7063c726f8dSBenjamin Herrenschmidt lprop = of_get_flat_dt_prop(node, "linux,tce-alloc-end", NULL); 7079b6b563cSPaul Mackerras if (lprop) 7089b6b563cSPaul Mackerras tce_alloc_end = *lprop; 7099b6b563cSPaul Mackerras #endif 7109b6b563cSPaul Mackerras 711dcee3036SMichael Ellerman #ifdef CONFIG_KEXEC 712dcee3036SMichael Ellerman lprop = (u64*)of_get_flat_dt_prop(node, "linux,crashkernel-base", NULL); 713dcee3036SMichael Ellerman if (lprop) 714dcee3036SMichael Ellerman crashk_res.start = *lprop; 715dcee3036SMichael Ellerman 716dcee3036SMichael Ellerman lprop = (u64*)of_get_flat_dt_prop(node, "linux,crashkernel-size", NULL); 717dcee3036SMichael Ellerman if (lprop) 718dcee3036SMichael Ellerman crashk_res.end = crashk_res.start + *lprop - 1; 719dcee3036SMichael Ellerman #endif 720dcee3036SMichael Ellerman 721329dda08SKumar Gala /* Retreive command line */ 722329dda08SKumar Gala p = of_get_flat_dt_prop(node, "bootargs", &l); 723329dda08SKumar Gala if (p != NULL && l > 0) 724329dda08SKumar Gala strlcpy(cmd_line, p, min((int)l, COMMAND_LINE_SIZE)); 725329dda08SKumar Gala 726329dda08SKumar Gala #ifdef CONFIG_CMDLINE 727329dda08SKumar Gala if (l == 0 || (l == 1 && (*p) == 0)) 728329dda08SKumar Gala strlcpy(cmd_line, CONFIG_CMDLINE, COMMAND_LINE_SIZE); 729329dda08SKumar Gala #endif /* CONFIG_CMDLINE */ 730329dda08SKumar Gala 731329dda08SKumar Gala DBG("Command line is: %s\n", cmd_line); 732329dda08SKumar Gala 7339b6b563cSPaul Mackerras /* break now */ 7349b6b563cSPaul Mackerras return 1; 7359b6b563cSPaul Mackerras } 7369b6b563cSPaul Mackerras 7379b6b563cSPaul Mackerras static int __init early_init_dt_scan_root(unsigned long node, 7389b6b563cSPaul Mackerras const char *uname, int depth, void *data) 7399b6b563cSPaul Mackerras { 7409b6b563cSPaul Mackerras u32 *prop; 7419b6b563cSPaul Mackerras 7429b6b563cSPaul Mackerras if (depth != 0) 7439b6b563cSPaul Mackerras return 0; 7449b6b563cSPaul Mackerras 7453c726f8dSBenjamin Herrenschmidt prop = of_get_flat_dt_prop(node, "#size-cells", NULL); 7469b6b563cSPaul Mackerras dt_root_size_cells = (prop == NULL) ? 1 : *prop; 7479b6b563cSPaul Mackerras DBG("dt_root_size_cells = %x\n", dt_root_size_cells); 7489b6b563cSPaul Mackerras 7493c726f8dSBenjamin Herrenschmidt prop = of_get_flat_dt_prop(node, "#address-cells", NULL); 7509b6b563cSPaul Mackerras dt_root_addr_cells = (prop == NULL) ? 2 : *prop; 7519b6b563cSPaul Mackerras DBG("dt_root_addr_cells = %x\n", dt_root_addr_cells); 7529b6b563cSPaul Mackerras 7539b6b563cSPaul Mackerras /* break now */ 7549b6b563cSPaul Mackerras return 1; 7559b6b563cSPaul Mackerras } 7569b6b563cSPaul Mackerras 7579b6b563cSPaul Mackerras static unsigned long __init dt_mem_next_cell(int s, cell_t **cellp) 7589b6b563cSPaul Mackerras { 7599b6b563cSPaul Mackerras cell_t *p = *cellp; 7609b6b563cSPaul Mackerras unsigned long r; 7619b6b563cSPaul Mackerras 7629b6b563cSPaul Mackerras /* Ignore more than 2 cells */ 7639b6b563cSPaul Mackerras while (s > sizeof(unsigned long) / 4) { 7649b6b563cSPaul Mackerras p++; 7659b6b563cSPaul Mackerras s--; 7669b6b563cSPaul Mackerras } 7679b6b563cSPaul Mackerras r = *p++; 7689b6b563cSPaul Mackerras #ifdef CONFIG_PPC64 7699b6b563cSPaul Mackerras if (s > 1) { 7709b6b563cSPaul Mackerras r <<= 32; 7719b6b563cSPaul Mackerras r |= *(p++); 7729b6b563cSPaul Mackerras s--; 7739b6b563cSPaul Mackerras } 7749b6b563cSPaul Mackerras #endif 7759b6b563cSPaul Mackerras 7769b6b563cSPaul Mackerras *cellp = p; 7779b6b563cSPaul Mackerras return r; 7789b6b563cSPaul Mackerras } 7799b6b563cSPaul Mackerras 7809b6b563cSPaul Mackerras 7819b6b563cSPaul Mackerras static int __init early_init_dt_scan_memory(unsigned long node, 7829b6b563cSPaul Mackerras const char *uname, int depth, void *data) 7839b6b563cSPaul Mackerras { 7843c726f8dSBenjamin Herrenschmidt char *type = of_get_flat_dt_prop(node, "device_type", NULL); 7859b6b563cSPaul Mackerras cell_t *reg, *endp; 7869b6b563cSPaul Mackerras unsigned long l; 7879b6b563cSPaul Mackerras 7889b6b563cSPaul Mackerras /* We are scanning "memory" nodes only */ 789a23414beSPaul Mackerras if (type == NULL) { 790a23414beSPaul Mackerras /* 791a23414beSPaul Mackerras * The longtrail doesn't have a device_type on the 792a23414beSPaul Mackerras * /memory node, so look for the node called /memory@0. 793a23414beSPaul Mackerras */ 794a23414beSPaul Mackerras if (depth != 1 || strcmp(uname, "memory@0") != 0) 795a23414beSPaul Mackerras return 0; 796a23414beSPaul Mackerras } else if (strcmp(type, "memory") != 0) 7979b6b563cSPaul Mackerras return 0; 7989b6b563cSPaul Mackerras 799ba759485SMichael Ellerman reg = (cell_t *)of_get_flat_dt_prop(node, "linux,usable-memory", &l); 800ba759485SMichael Ellerman if (reg == NULL) 8013c726f8dSBenjamin Herrenschmidt reg = (cell_t *)of_get_flat_dt_prop(node, "reg", &l); 8029b6b563cSPaul Mackerras if (reg == NULL) 8039b6b563cSPaul Mackerras return 0; 8049b6b563cSPaul Mackerras 8059b6b563cSPaul Mackerras endp = reg + (l / sizeof(cell_t)); 8069b6b563cSPaul Mackerras 807358c86fdSMichael Ellerman DBG("memory scan node %s, reg size %ld, data: %x %x %x %x,\n", 8089b6b563cSPaul Mackerras uname, l, reg[0], reg[1], reg[2], reg[3]); 8099b6b563cSPaul Mackerras 8109b6b563cSPaul Mackerras while ((endp - reg) >= (dt_root_addr_cells + dt_root_size_cells)) { 8119b6b563cSPaul Mackerras unsigned long base, size; 8129b6b563cSPaul Mackerras 8139b6b563cSPaul Mackerras base = dt_mem_next_cell(dt_root_addr_cells, ®); 8149b6b563cSPaul Mackerras size = dt_mem_next_cell(dt_root_size_cells, ®); 8159b6b563cSPaul Mackerras 8169b6b563cSPaul Mackerras if (size == 0) 8179b6b563cSPaul Mackerras continue; 8189b6b563cSPaul Mackerras DBG(" - %lx , %lx\n", base, size); 8199b6b563cSPaul Mackerras #ifdef CONFIG_PPC64 8209b6b563cSPaul Mackerras if (iommu_is_off) { 8219b6b563cSPaul Mackerras if (base >= 0x80000000ul) 8229b6b563cSPaul Mackerras continue; 8239b6b563cSPaul Mackerras if ((base + size) > 0x80000000ul) 8249b6b563cSPaul Mackerras size = 0x80000000ul - base; 8259b6b563cSPaul Mackerras } 8269b6b563cSPaul Mackerras #endif 8279b6b563cSPaul Mackerras lmb_add(base, size); 8289b6b563cSPaul Mackerras } 8299b6b563cSPaul Mackerras return 0; 8309b6b563cSPaul Mackerras } 8319b6b563cSPaul Mackerras 8329b6b563cSPaul Mackerras static void __init early_reserve_mem(void) 8339b6b563cSPaul Mackerras { 834cbbcf340SKumar Gala u64 base, size; 835cbbcf340SKumar Gala u64 *reserve_map; 8368a300887SJon Loeliger unsigned long self_base; 8378a300887SJon Loeliger unsigned long self_size; 8389b6b563cSPaul Mackerras 839cbbcf340SKumar Gala reserve_map = (u64 *)(((unsigned long)initial_boot_params) + 8409b6b563cSPaul Mackerras initial_boot_params->off_mem_rsvmap); 8414d1f3f25SJimi Xenidis 8424d1f3f25SJimi Xenidis /* before we do anything, lets reserve the dt blob */ 8438a300887SJon Loeliger self_base = __pa((unsigned long)initial_boot_params); 8448a300887SJon Loeliger self_size = initial_boot_params->totalsize; 8458a300887SJon Loeliger lmb_reserve(self_base, self_size); 8464d1f3f25SJimi Xenidis 847cbbcf340SKumar Gala #ifdef CONFIG_PPC32 848cbbcf340SKumar Gala /* 849cbbcf340SKumar Gala * Handle the case where we might be booting from an old kexec 850cbbcf340SKumar Gala * image that setup the mem_rsvmap as pairs of 32-bit values 851cbbcf340SKumar Gala */ 852cbbcf340SKumar Gala if (*reserve_map > 0xffffffffull) { 853cbbcf340SKumar Gala u32 base_32, size_32; 854cbbcf340SKumar Gala u32 *reserve_map_32 = (u32 *)reserve_map; 855cbbcf340SKumar Gala 856cbbcf340SKumar Gala while (1) { 857cbbcf340SKumar Gala base_32 = *(reserve_map_32++); 858cbbcf340SKumar Gala size_32 = *(reserve_map_32++); 859cbbcf340SKumar Gala if (size_32 == 0) 860cbbcf340SKumar Gala break; 8618a300887SJon Loeliger /* skip if the reservation is for the blob */ 8628a300887SJon Loeliger if (base_32 == self_base && size_32 == self_size) 8638a300887SJon Loeliger continue; 864329dda08SKumar Gala DBG("reserving: %x -> %x\n", base_32, size_32); 865cbbcf340SKumar Gala lmb_reserve(base_32, size_32); 866cbbcf340SKumar Gala } 867cbbcf340SKumar Gala return; 868cbbcf340SKumar Gala } 869cbbcf340SKumar Gala #endif 8709b6b563cSPaul Mackerras while (1) { 8719b6b563cSPaul Mackerras base = *(reserve_map++); 8729b6b563cSPaul Mackerras size = *(reserve_map++); 8739b6b563cSPaul Mackerras if (size == 0) 8749b6b563cSPaul Mackerras break; 8758a300887SJon Loeliger /* skip if the reservation is for the blob */ 8768a300887SJon Loeliger if (base == self_base && size == self_size) 8778a300887SJon Loeliger continue; 878cbbcf340SKumar Gala DBG("reserving: %llx -> %llx\n", base, size); 8799b6b563cSPaul Mackerras lmb_reserve(base, size); 8809b6b563cSPaul Mackerras } 8819b6b563cSPaul Mackerras 8829b6b563cSPaul Mackerras #if 0 8839b6b563cSPaul Mackerras DBG("memory reserved, lmbs :\n"); 8849b6b563cSPaul Mackerras lmb_dump_all(); 8859b6b563cSPaul Mackerras #endif 8869b6b563cSPaul Mackerras } 8879b6b563cSPaul Mackerras 8889b6b563cSPaul Mackerras void __init early_init_devtree(void *params) 8899b6b563cSPaul Mackerras { 8909b6b563cSPaul Mackerras DBG(" -> early_init_devtree()\n"); 8919b6b563cSPaul Mackerras 8929b6b563cSPaul Mackerras /* Setup flat device-tree pointer */ 8939b6b563cSPaul Mackerras initial_boot_params = params; 8949b6b563cSPaul Mackerras 895458148c0SMichael Ellerman #ifdef CONFIG_PPC_RTAS 896458148c0SMichael Ellerman /* Some machines might need RTAS info for debugging, grab it now. */ 897458148c0SMichael Ellerman of_scan_flat_dt(early_init_dt_scan_rtas, NULL); 898458148c0SMichael Ellerman #endif 899458148c0SMichael Ellerman 9009b6b563cSPaul Mackerras /* Retrieve various informations from the /chosen node of the 9019b6b563cSPaul Mackerras * device-tree, including the platform type, initrd location and 9029b6b563cSPaul Mackerras * size, TCE reserve, and more ... 9039b6b563cSPaul Mackerras */ 9043c726f8dSBenjamin Herrenschmidt of_scan_flat_dt(early_init_dt_scan_chosen, NULL); 9059b6b563cSPaul Mackerras 9069b6b563cSPaul Mackerras /* Scan memory nodes and rebuild LMBs */ 9079b6b563cSPaul Mackerras lmb_init(); 9083c726f8dSBenjamin Herrenschmidt of_scan_flat_dt(early_init_dt_scan_root, NULL); 9093c726f8dSBenjamin Herrenschmidt of_scan_flat_dt(early_init_dt_scan_memory, NULL); 910846f77b0SMichael Ellerman 911846f77b0SMichael Ellerman /* Save command line for /proc/cmdline and then parse parameters */ 912846f77b0SMichael Ellerman strlcpy(saved_command_line, cmd_line, COMMAND_LINE_SIZE); 913846f77b0SMichael Ellerman parse_early_param(); 914846f77b0SMichael Ellerman 9159b6b563cSPaul Mackerras /* Reserve LMB regions used by kernel, initrd, dt, etc... */ 9160cc4746cSMichael Ellerman lmb_reserve(PHYSICAL_START, __pa(klimit) - PHYSICAL_START); 91747310413SMichael Ellerman reserve_kdump_trampoline(); 91835dd5432SMichael Ellerman reserve_crashkernel(); 9199b6b563cSPaul Mackerras early_reserve_mem(); 9209b6b563cSPaul Mackerras 9212babf5c2SMichael Ellerman lmb_enforce_memory_limit(memory_limit); 9222babf5c2SMichael Ellerman lmb_analyze(); 9232babf5c2SMichael Ellerman 9242babf5c2SMichael Ellerman DBG("Phys. mem: %lx\n", lmb_phys_mem_size()); 9252babf5c2SMichael Ellerman 9262babf5c2SMichael Ellerman /* We may need to relocate the flat tree, do it now. 9272babf5c2SMichael Ellerman * FIXME .. and the initrd too? */ 9282babf5c2SMichael Ellerman move_device_tree(); 9292babf5c2SMichael Ellerman 9309b6b563cSPaul Mackerras DBG("Scanning CPUs ...\n"); 9319b6b563cSPaul Mackerras 9323c726f8dSBenjamin Herrenschmidt /* Retreive CPU related informations from the flat tree 9333c726f8dSBenjamin Herrenschmidt * (altivec support, boot CPU ID, ...) 9349b6b563cSPaul Mackerras */ 9353c726f8dSBenjamin Herrenschmidt of_scan_flat_dt(early_init_dt_scan_cpus, NULL); 9369b6b563cSPaul Mackerras 9379b6b563cSPaul Mackerras DBG(" <- early_init_devtree()\n"); 9389b6b563cSPaul Mackerras } 9399b6b563cSPaul Mackerras 9409b6b563cSPaul Mackerras #undef printk 9419b6b563cSPaul Mackerras 9429b6b563cSPaul Mackerras int 9439b6b563cSPaul Mackerras prom_n_addr_cells(struct device_node* np) 9449b6b563cSPaul Mackerras { 9459b6b563cSPaul Mackerras int* ip; 9469b6b563cSPaul Mackerras do { 9479b6b563cSPaul Mackerras if (np->parent) 9489b6b563cSPaul Mackerras np = np->parent; 9499b6b563cSPaul Mackerras ip = (int *) get_property(np, "#address-cells", NULL); 9509b6b563cSPaul Mackerras if (ip != NULL) 9519b6b563cSPaul Mackerras return *ip; 9529b6b563cSPaul Mackerras } while (np->parent); 9539b6b563cSPaul Mackerras /* No #address-cells property for the root node, default to 1 */ 9549b6b563cSPaul Mackerras return 1; 9559b6b563cSPaul Mackerras } 9561dfc6772SPaul Mackerras EXPORT_SYMBOL(prom_n_addr_cells); 9579b6b563cSPaul Mackerras 9589b6b563cSPaul Mackerras int 9599b6b563cSPaul Mackerras prom_n_size_cells(struct device_node* np) 9609b6b563cSPaul Mackerras { 9619b6b563cSPaul Mackerras int* ip; 9629b6b563cSPaul Mackerras do { 9639b6b563cSPaul Mackerras if (np->parent) 9649b6b563cSPaul Mackerras np = np->parent; 9659b6b563cSPaul Mackerras ip = (int *) get_property(np, "#size-cells", NULL); 9669b6b563cSPaul Mackerras if (ip != NULL) 9679b6b563cSPaul Mackerras return *ip; 9689b6b563cSPaul Mackerras } while (np->parent); 9699b6b563cSPaul Mackerras /* No #size-cells property for the root node, default to 1 */ 9709b6b563cSPaul Mackerras return 1; 9719b6b563cSPaul Mackerras } 9721dfc6772SPaul Mackerras EXPORT_SYMBOL(prom_n_size_cells); 9739b6b563cSPaul Mackerras 9749b6b563cSPaul Mackerras /** 9759b6b563cSPaul Mackerras * Construct and return a list of the device_nodes with a given name. 9769b6b563cSPaul Mackerras */ 9779b6b563cSPaul Mackerras struct device_node *find_devices(const char *name) 9789b6b563cSPaul Mackerras { 9799b6b563cSPaul Mackerras struct device_node *head, **prevp, *np; 9809b6b563cSPaul Mackerras 9819b6b563cSPaul Mackerras prevp = &head; 9829b6b563cSPaul Mackerras for (np = allnodes; np != 0; np = np->allnext) { 9839b6b563cSPaul Mackerras if (np->name != 0 && strcasecmp(np->name, name) == 0) { 9849b6b563cSPaul Mackerras *prevp = np; 9859b6b563cSPaul Mackerras prevp = &np->next; 9869b6b563cSPaul Mackerras } 9879b6b563cSPaul Mackerras } 9889b6b563cSPaul Mackerras *prevp = NULL; 9899b6b563cSPaul Mackerras return head; 9909b6b563cSPaul Mackerras } 9919b6b563cSPaul Mackerras EXPORT_SYMBOL(find_devices); 9929b6b563cSPaul Mackerras 9939b6b563cSPaul Mackerras /** 9949b6b563cSPaul Mackerras * Construct and return a list of the device_nodes with a given type. 9959b6b563cSPaul Mackerras */ 9969b6b563cSPaul Mackerras struct device_node *find_type_devices(const char *type) 9979b6b563cSPaul Mackerras { 9989b6b563cSPaul Mackerras struct device_node *head, **prevp, *np; 9999b6b563cSPaul Mackerras 10009b6b563cSPaul Mackerras prevp = &head; 10019b6b563cSPaul Mackerras for (np = allnodes; np != 0; np = np->allnext) { 10029b6b563cSPaul Mackerras if (np->type != 0 && strcasecmp(np->type, type) == 0) { 10039b6b563cSPaul Mackerras *prevp = np; 10049b6b563cSPaul Mackerras prevp = &np->next; 10059b6b563cSPaul Mackerras } 10069b6b563cSPaul Mackerras } 10079b6b563cSPaul Mackerras *prevp = NULL; 10089b6b563cSPaul Mackerras return head; 10099b6b563cSPaul Mackerras } 10109b6b563cSPaul Mackerras EXPORT_SYMBOL(find_type_devices); 10119b6b563cSPaul Mackerras 10129b6b563cSPaul Mackerras /** 10139b6b563cSPaul Mackerras * Returns all nodes linked together 10149b6b563cSPaul Mackerras */ 10159b6b563cSPaul Mackerras struct device_node *find_all_nodes(void) 10169b6b563cSPaul Mackerras { 10179b6b563cSPaul Mackerras struct device_node *head, **prevp, *np; 10189b6b563cSPaul Mackerras 10199b6b563cSPaul Mackerras prevp = &head; 10209b6b563cSPaul Mackerras for (np = allnodes; np != 0; np = np->allnext) { 10219b6b563cSPaul Mackerras *prevp = np; 10229b6b563cSPaul Mackerras prevp = &np->next; 10239b6b563cSPaul Mackerras } 10249b6b563cSPaul Mackerras *prevp = NULL; 10259b6b563cSPaul Mackerras return head; 10269b6b563cSPaul Mackerras } 10279b6b563cSPaul Mackerras EXPORT_SYMBOL(find_all_nodes); 10289b6b563cSPaul Mackerras 10299b6b563cSPaul Mackerras /** Checks if the given "compat" string matches one of the strings in 10309b6b563cSPaul Mackerras * the device's "compatible" property 10319b6b563cSPaul Mackerras */ 10329b6b563cSPaul Mackerras int device_is_compatible(struct device_node *device, const char *compat) 10339b6b563cSPaul Mackerras { 10349b6b563cSPaul Mackerras const char* cp; 10359b6b563cSPaul Mackerras int cplen, l; 10369b6b563cSPaul Mackerras 10379b6b563cSPaul Mackerras cp = (char *) get_property(device, "compatible", &cplen); 10389b6b563cSPaul Mackerras if (cp == NULL) 10399b6b563cSPaul Mackerras return 0; 10409b6b563cSPaul Mackerras while (cplen > 0) { 10419b6b563cSPaul Mackerras if (strncasecmp(cp, compat, strlen(compat)) == 0) 10429b6b563cSPaul Mackerras return 1; 10439b6b563cSPaul Mackerras l = strlen(cp) + 1; 10449b6b563cSPaul Mackerras cp += l; 10459b6b563cSPaul Mackerras cplen -= l; 10469b6b563cSPaul Mackerras } 10479b6b563cSPaul Mackerras 10489b6b563cSPaul Mackerras return 0; 10499b6b563cSPaul Mackerras } 10509b6b563cSPaul Mackerras EXPORT_SYMBOL(device_is_compatible); 10519b6b563cSPaul Mackerras 10529b6b563cSPaul Mackerras 10539b6b563cSPaul Mackerras /** 10549b6b563cSPaul Mackerras * Indicates whether the root node has a given value in its 10559b6b563cSPaul Mackerras * compatible property. 10569b6b563cSPaul Mackerras */ 10579b6b563cSPaul Mackerras int machine_is_compatible(const char *compat) 10589b6b563cSPaul Mackerras { 10599b6b563cSPaul Mackerras struct device_node *root; 10609b6b563cSPaul Mackerras int rc = 0; 10619b6b563cSPaul Mackerras 10629b6b563cSPaul Mackerras root = of_find_node_by_path("/"); 10639b6b563cSPaul Mackerras if (root) { 10649b6b563cSPaul Mackerras rc = device_is_compatible(root, compat); 10659b6b563cSPaul Mackerras of_node_put(root); 10669b6b563cSPaul Mackerras } 10679b6b563cSPaul Mackerras return rc; 10689b6b563cSPaul Mackerras } 10699b6b563cSPaul Mackerras EXPORT_SYMBOL(machine_is_compatible); 10709b6b563cSPaul Mackerras 10719b6b563cSPaul Mackerras /** 10729b6b563cSPaul Mackerras * Construct and return a list of the device_nodes with a given type 10739b6b563cSPaul Mackerras * and compatible property. 10749b6b563cSPaul Mackerras */ 10759b6b563cSPaul Mackerras struct device_node *find_compatible_devices(const char *type, 10769b6b563cSPaul Mackerras const char *compat) 10779b6b563cSPaul Mackerras { 10789b6b563cSPaul Mackerras struct device_node *head, **prevp, *np; 10799b6b563cSPaul Mackerras 10809b6b563cSPaul Mackerras prevp = &head; 10819b6b563cSPaul Mackerras for (np = allnodes; np != 0; np = np->allnext) { 10829b6b563cSPaul Mackerras if (type != NULL 10839b6b563cSPaul Mackerras && !(np->type != 0 && strcasecmp(np->type, type) == 0)) 10849b6b563cSPaul Mackerras continue; 10859b6b563cSPaul Mackerras if (device_is_compatible(np, compat)) { 10869b6b563cSPaul Mackerras *prevp = np; 10879b6b563cSPaul Mackerras prevp = &np->next; 10889b6b563cSPaul Mackerras } 10899b6b563cSPaul Mackerras } 10909b6b563cSPaul Mackerras *prevp = NULL; 10919b6b563cSPaul Mackerras return head; 10929b6b563cSPaul Mackerras } 10939b6b563cSPaul Mackerras EXPORT_SYMBOL(find_compatible_devices); 10949b6b563cSPaul Mackerras 10959b6b563cSPaul Mackerras /** 10969b6b563cSPaul Mackerras * Find the device_node with a given full_name. 10979b6b563cSPaul Mackerras */ 10989b6b563cSPaul Mackerras struct device_node *find_path_device(const char *path) 10999b6b563cSPaul Mackerras { 11009b6b563cSPaul Mackerras struct device_node *np; 11019b6b563cSPaul Mackerras 11029b6b563cSPaul Mackerras for (np = allnodes; np != 0; np = np->allnext) 11039b6b563cSPaul Mackerras if (np->full_name != 0 && strcasecmp(np->full_name, path) == 0) 11049b6b563cSPaul Mackerras return np; 11059b6b563cSPaul Mackerras return NULL; 11069b6b563cSPaul Mackerras } 11079b6b563cSPaul Mackerras EXPORT_SYMBOL(find_path_device); 11089b6b563cSPaul Mackerras 11099b6b563cSPaul Mackerras /******* 11109b6b563cSPaul Mackerras * 11119b6b563cSPaul Mackerras * New implementation of the OF "find" APIs, return a refcounted 11129b6b563cSPaul Mackerras * object, call of_node_put() when done. The device tree and list 11139b6b563cSPaul Mackerras * are protected by a rw_lock. 11149b6b563cSPaul Mackerras * 11159b6b563cSPaul Mackerras * Note that property management will need some locking as well, 11169b6b563cSPaul Mackerras * this isn't dealt with yet. 11179b6b563cSPaul Mackerras * 11189b6b563cSPaul Mackerras *******/ 11199b6b563cSPaul Mackerras 11209b6b563cSPaul Mackerras /** 11219b6b563cSPaul Mackerras * of_find_node_by_name - Find a node by its "name" property 11229b6b563cSPaul Mackerras * @from: The node to start searching from or NULL, the node 11239b6b563cSPaul Mackerras * you pass will not be searched, only the next one 11249b6b563cSPaul Mackerras * will; typically, you pass what the previous call 11259b6b563cSPaul Mackerras * returned. of_node_put() will be called on it 11269b6b563cSPaul Mackerras * @name: The name string to match against 11279b6b563cSPaul Mackerras * 11289b6b563cSPaul Mackerras * Returns a node pointer with refcount incremented, use 11299b6b563cSPaul Mackerras * of_node_put() on it when done. 11309b6b563cSPaul Mackerras */ 11319b6b563cSPaul Mackerras struct device_node *of_find_node_by_name(struct device_node *from, 11329b6b563cSPaul Mackerras const char *name) 11339b6b563cSPaul Mackerras { 11349b6b563cSPaul Mackerras struct device_node *np; 11359b6b563cSPaul Mackerras 11369b6b563cSPaul Mackerras read_lock(&devtree_lock); 11379b6b563cSPaul Mackerras np = from ? from->allnext : allnodes; 1138090db7c8SOlaf Hering for (; np != NULL; np = np->allnext) 1139090db7c8SOlaf Hering if (np->name != NULL && strcasecmp(np->name, name) == 0 11409b6b563cSPaul Mackerras && of_node_get(np)) 11419b6b563cSPaul Mackerras break; 11429b6b563cSPaul Mackerras if (from) 11439b6b563cSPaul Mackerras of_node_put(from); 11449b6b563cSPaul Mackerras read_unlock(&devtree_lock); 11459b6b563cSPaul Mackerras return np; 11469b6b563cSPaul Mackerras } 11479b6b563cSPaul Mackerras EXPORT_SYMBOL(of_find_node_by_name); 11489b6b563cSPaul Mackerras 11499b6b563cSPaul Mackerras /** 11509b6b563cSPaul Mackerras * of_find_node_by_type - Find a node by its "device_type" property 11519b6b563cSPaul Mackerras * @from: The node to start searching from or NULL, the node 11529b6b563cSPaul Mackerras * you pass will not be searched, only the next one 11539b6b563cSPaul Mackerras * will; typically, you pass what the previous call 11549b6b563cSPaul Mackerras * returned. of_node_put() will be called on it 11559b6b563cSPaul Mackerras * @name: The type string to match against 11569b6b563cSPaul Mackerras * 11579b6b563cSPaul Mackerras * Returns a node pointer with refcount incremented, use 11589b6b563cSPaul Mackerras * of_node_put() on it when done. 11599b6b563cSPaul Mackerras */ 11609b6b563cSPaul Mackerras struct device_node *of_find_node_by_type(struct device_node *from, 11619b6b563cSPaul Mackerras const char *type) 11629b6b563cSPaul Mackerras { 11639b6b563cSPaul Mackerras struct device_node *np; 11649b6b563cSPaul Mackerras 11659b6b563cSPaul Mackerras read_lock(&devtree_lock); 11669b6b563cSPaul Mackerras np = from ? from->allnext : allnodes; 11679b6b563cSPaul Mackerras for (; np != 0; np = np->allnext) 11689b6b563cSPaul Mackerras if (np->type != 0 && strcasecmp(np->type, type) == 0 11699b6b563cSPaul Mackerras && of_node_get(np)) 11709b6b563cSPaul Mackerras break; 11719b6b563cSPaul Mackerras if (from) 11729b6b563cSPaul Mackerras of_node_put(from); 11739b6b563cSPaul Mackerras read_unlock(&devtree_lock); 11749b6b563cSPaul Mackerras return np; 11759b6b563cSPaul Mackerras } 11769b6b563cSPaul Mackerras EXPORT_SYMBOL(of_find_node_by_type); 11779b6b563cSPaul Mackerras 11789b6b563cSPaul Mackerras /** 11799b6b563cSPaul Mackerras * of_find_compatible_node - Find a node based on type and one of the 11809b6b563cSPaul Mackerras * tokens in its "compatible" property 11819b6b563cSPaul Mackerras * @from: The node to start searching from or NULL, the node 11829b6b563cSPaul Mackerras * you pass will not be searched, only the next one 11839b6b563cSPaul Mackerras * will; typically, you pass what the previous call 11849b6b563cSPaul Mackerras * returned. of_node_put() will be called on it 11859b6b563cSPaul Mackerras * @type: The type string to match "device_type" or NULL to ignore 11869b6b563cSPaul Mackerras * @compatible: The string to match to one of the tokens in the device 11879b6b563cSPaul Mackerras * "compatible" list. 11889b6b563cSPaul Mackerras * 11899b6b563cSPaul Mackerras * Returns a node pointer with refcount incremented, use 11909b6b563cSPaul Mackerras * of_node_put() on it when done. 11919b6b563cSPaul Mackerras */ 11929b6b563cSPaul Mackerras struct device_node *of_find_compatible_node(struct device_node *from, 11939b6b563cSPaul Mackerras const char *type, const char *compatible) 11949b6b563cSPaul Mackerras { 11959b6b563cSPaul Mackerras struct device_node *np; 11969b6b563cSPaul Mackerras 11979b6b563cSPaul Mackerras read_lock(&devtree_lock); 11989b6b563cSPaul Mackerras np = from ? from->allnext : allnodes; 11999b6b563cSPaul Mackerras for (; np != 0; np = np->allnext) { 12009b6b563cSPaul Mackerras if (type != NULL 12019b6b563cSPaul Mackerras && !(np->type != 0 && strcasecmp(np->type, type) == 0)) 12029b6b563cSPaul Mackerras continue; 12039b6b563cSPaul Mackerras if (device_is_compatible(np, compatible) && of_node_get(np)) 12049b6b563cSPaul Mackerras break; 12059b6b563cSPaul Mackerras } 12069b6b563cSPaul Mackerras if (from) 12079b6b563cSPaul Mackerras of_node_put(from); 12089b6b563cSPaul Mackerras read_unlock(&devtree_lock); 12099b6b563cSPaul Mackerras return np; 12109b6b563cSPaul Mackerras } 12119b6b563cSPaul Mackerras EXPORT_SYMBOL(of_find_compatible_node); 12129b6b563cSPaul Mackerras 12139b6b563cSPaul Mackerras /** 12149b6b563cSPaul Mackerras * of_find_node_by_path - Find a node matching a full OF path 12159b6b563cSPaul Mackerras * @path: The full path to match 12169b6b563cSPaul Mackerras * 12179b6b563cSPaul Mackerras * Returns a node pointer with refcount incremented, use 12189b6b563cSPaul Mackerras * of_node_put() on it when done. 12199b6b563cSPaul Mackerras */ 12209b6b563cSPaul Mackerras struct device_node *of_find_node_by_path(const char *path) 12219b6b563cSPaul Mackerras { 12229b6b563cSPaul Mackerras struct device_node *np = allnodes; 12239b6b563cSPaul Mackerras 12249b6b563cSPaul Mackerras read_lock(&devtree_lock); 12259b6b563cSPaul Mackerras for (; np != 0; np = np->allnext) { 12269b6b563cSPaul Mackerras if (np->full_name != 0 && strcasecmp(np->full_name, path) == 0 12279b6b563cSPaul Mackerras && of_node_get(np)) 12289b6b563cSPaul Mackerras break; 12299b6b563cSPaul Mackerras } 12309b6b563cSPaul Mackerras read_unlock(&devtree_lock); 12319b6b563cSPaul Mackerras return np; 12329b6b563cSPaul Mackerras } 12339b6b563cSPaul Mackerras EXPORT_SYMBOL(of_find_node_by_path); 12349b6b563cSPaul Mackerras 12359b6b563cSPaul Mackerras /** 12369b6b563cSPaul Mackerras * of_find_node_by_phandle - Find a node given a phandle 12379b6b563cSPaul Mackerras * @handle: phandle of the node to find 12389b6b563cSPaul Mackerras * 12399b6b563cSPaul Mackerras * Returns a node pointer with refcount incremented, use 12409b6b563cSPaul Mackerras * of_node_put() on it when done. 12419b6b563cSPaul Mackerras */ 12429b6b563cSPaul Mackerras struct device_node *of_find_node_by_phandle(phandle handle) 12439b6b563cSPaul Mackerras { 12449b6b563cSPaul Mackerras struct device_node *np; 12459b6b563cSPaul Mackerras 12469b6b563cSPaul Mackerras read_lock(&devtree_lock); 12479b6b563cSPaul Mackerras for (np = allnodes; np != 0; np = np->allnext) 12489b6b563cSPaul Mackerras if (np->linux_phandle == handle) 12499b6b563cSPaul Mackerras break; 12509b6b563cSPaul Mackerras if (np) 12519b6b563cSPaul Mackerras of_node_get(np); 12529b6b563cSPaul Mackerras read_unlock(&devtree_lock); 12539b6b563cSPaul Mackerras return np; 12549b6b563cSPaul Mackerras } 12559b6b563cSPaul Mackerras EXPORT_SYMBOL(of_find_node_by_phandle); 12569b6b563cSPaul Mackerras 12579b6b563cSPaul Mackerras /** 12589b6b563cSPaul Mackerras * of_find_all_nodes - Get next node in global list 12599b6b563cSPaul Mackerras * @prev: Previous node or NULL to start iteration 12609b6b563cSPaul Mackerras * of_node_put() will be called on it 12619b6b563cSPaul Mackerras * 12629b6b563cSPaul Mackerras * Returns a node pointer with refcount incremented, use 12639b6b563cSPaul Mackerras * of_node_put() on it when done. 12649b6b563cSPaul Mackerras */ 12659b6b563cSPaul Mackerras struct device_node *of_find_all_nodes(struct device_node *prev) 12669b6b563cSPaul Mackerras { 12679b6b563cSPaul Mackerras struct device_node *np; 12689b6b563cSPaul Mackerras 12699b6b563cSPaul Mackerras read_lock(&devtree_lock); 12709b6b563cSPaul Mackerras np = prev ? prev->allnext : allnodes; 12719b6b563cSPaul Mackerras for (; np != 0; np = np->allnext) 12729b6b563cSPaul Mackerras if (of_node_get(np)) 12739b6b563cSPaul Mackerras break; 12749b6b563cSPaul Mackerras if (prev) 12759b6b563cSPaul Mackerras of_node_put(prev); 12769b6b563cSPaul Mackerras read_unlock(&devtree_lock); 12779b6b563cSPaul Mackerras return np; 12789b6b563cSPaul Mackerras } 12799b6b563cSPaul Mackerras EXPORT_SYMBOL(of_find_all_nodes); 12809b6b563cSPaul Mackerras 12819b6b563cSPaul Mackerras /** 12829b6b563cSPaul Mackerras * of_get_parent - Get a node's parent if any 12839b6b563cSPaul Mackerras * @node: Node to get parent 12849b6b563cSPaul Mackerras * 12859b6b563cSPaul Mackerras * Returns a node pointer with refcount incremented, use 12869b6b563cSPaul Mackerras * of_node_put() on it when done. 12879b6b563cSPaul Mackerras */ 12889b6b563cSPaul Mackerras struct device_node *of_get_parent(const struct device_node *node) 12899b6b563cSPaul Mackerras { 12909b6b563cSPaul Mackerras struct device_node *np; 12919b6b563cSPaul Mackerras 12929b6b563cSPaul Mackerras if (!node) 12939b6b563cSPaul Mackerras return NULL; 12949b6b563cSPaul Mackerras 12959b6b563cSPaul Mackerras read_lock(&devtree_lock); 12969b6b563cSPaul Mackerras np = of_node_get(node->parent); 12979b6b563cSPaul Mackerras read_unlock(&devtree_lock); 12989b6b563cSPaul Mackerras return np; 12999b6b563cSPaul Mackerras } 13009b6b563cSPaul Mackerras EXPORT_SYMBOL(of_get_parent); 13019b6b563cSPaul Mackerras 13029b6b563cSPaul Mackerras /** 13039b6b563cSPaul Mackerras * of_get_next_child - Iterate a node childs 13049b6b563cSPaul Mackerras * @node: parent node 13059b6b563cSPaul Mackerras * @prev: previous child of the parent node, or NULL to get first 13069b6b563cSPaul Mackerras * 13079b6b563cSPaul Mackerras * Returns a node pointer with refcount incremented, use 13089b6b563cSPaul Mackerras * of_node_put() on it when done. 13099b6b563cSPaul Mackerras */ 13109b6b563cSPaul Mackerras struct device_node *of_get_next_child(const struct device_node *node, 13119b6b563cSPaul Mackerras struct device_node *prev) 13129b6b563cSPaul Mackerras { 13139b6b563cSPaul Mackerras struct device_node *next; 13149b6b563cSPaul Mackerras 13159b6b563cSPaul Mackerras read_lock(&devtree_lock); 13169b6b563cSPaul Mackerras next = prev ? prev->sibling : node->child; 13179b6b563cSPaul Mackerras for (; next != 0; next = next->sibling) 13189b6b563cSPaul Mackerras if (of_node_get(next)) 13199b6b563cSPaul Mackerras break; 13209b6b563cSPaul Mackerras if (prev) 13219b6b563cSPaul Mackerras of_node_put(prev); 13229b6b563cSPaul Mackerras read_unlock(&devtree_lock); 13239b6b563cSPaul Mackerras return next; 13249b6b563cSPaul Mackerras } 13259b6b563cSPaul Mackerras EXPORT_SYMBOL(of_get_next_child); 13269b6b563cSPaul Mackerras 13279b6b563cSPaul Mackerras /** 13289b6b563cSPaul Mackerras * of_node_get - Increment refcount of a node 13299b6b563cSPaul Mackerras * @node: Node to inc refcount, NULL is supported to 13309b6b563cSPaul Mackerras * simplify writing of callers 13319b6b563cSPaul Mackerras * 13329b6b563cSPaul Mackerras * Returns node. 13339b6b563cSPaul Mackerras */ 13349b6b563cSPaul Mackerras struct device_node *of_node_get(struct device_node *node) 13359b6b563cSPaul Mackerras { 13369b6b563cSPaul Mackerras if (node) 13379b6b563cSPaul Mackerras kref_get(&node->kref); 13389b6b563cSPaul Mackerras return node; 13399b6b563cSPaul Mackerras } 13409b6b563cSPaul Mackerras EXPORT_SYMBOL(of_node_get); 13419b6b563cSPaul Mackerras 13429b6b563cSPaul Mackerras static inline struct device_node * kref_to_device_node(struct kref *kref) 13439b6b563cSPaul Mackerras { 13449b6b563cSPaul Mackerras return container_of(kref, struct device_node, kref); 13459b6b563cSPaul Mackerras } 13469b6b563cSPaul Mackerras 13479b6b563cSPaul Mackerras /** 13489b6b563cSPaul Mackerras * of_node_release - release a dynamically allocated node 13499b6b563cSPaul Mackerras * @kref: kref element of the node to be released 13509b6b563cSPaul Mackerras * 13519b6b563cSPaul Mackerras * In of_node_put() this function is passed to kref_put() 13529b6b563cSPaul Mackerras * as the destructor. 13539b6b563cSPaul Mackerras */ 13549b6b563cSPaul Mackerras static void of_node_release(struct kref *kref) 13559b6b563cSPaul Mackerras { 13569b6b563cSPaul Mackerras struct device_node *node = kref_to_device_node(kref); 13579b6b563cSPaul Mackerras struct property *prop = node->properties; 13589b6b563cSPaul Mackerras 13599b6b563cSPaul Mackerras if (!OF_IS_DYNAMIC(node)) 13609b6b563cSPaul Mackerras return; 13619b6b563cSPaul Mackerras while (prop) { 13629b6b563cSPaul Mackerras struct property *next = prop->next; 13639b6b563cSPaul Mackerras kfree(prop->name); 13649b6b563cSPaul Mackerras kfree(prop->value); 13659b6b563cSPaul Mackerras kfree(prop); 13669b6b563cSPaul Mackerras prop = next; 1367088186deSDave C Boutcher 1368088186deSDave C Boutcher if (!prop) { 1369088186deSDave C Boutcher prop = node->deadprops; 1370088186deSDave C Boutcher node->deadprops = NULL; 1371088186deSDave C Boutcher } 13729b6b563cSPaul Mackerras } 13739b6b563cSPaul Mackerras kfree(node->full_name); 13749b6b563cSPaul Mackerras kfree(node->data); 13759b6b563cSPaul Mackerras kfree(node); 13769b6b563cSPaul Mackerras } 13779b6b563cSPaul Mackerras 13789b6b563cSPaul Mackerras /** 13799b6b563cSPaul Mackerras * of_node_put - Decrement refcount of a node 13809b6b563cSPaul Mackerras * @node: Node to dec refcount, NULL is supported to 13819b6b563cSPaul Mackerras * simplify writing of callers 13829b6b563cSPaul Mackerras * 13839b6b563cSPaul Mackerras */ 13849b6b563cSPaul Mackerras void of_node_put(struct device_node *node) 13859b6b563cSPaul Mackerras { 13869b6b563cSPaul Mackerras if (node) 13879b6b563cSPaul Mackerras kref_put(&node->kref, of_node_release); 13889b6b563cSPaul Mackerras } 13899b6b563cSPaul Mackerras EXPORT_SYMBOL(of_node_put); 13909b6b563cSPaul Mackerras 13919b6b563cSPaul Mackerras /* 13929b6b563cSPaul Mackerras * Plug a device node into the tree and global list. 13939b6b563cSPaul Mackerras */ 13949b6b563cSPaul Mackerras void of_attach_node(struct device_node *np) 13959b6b563cSPaul Mackerras { 13969b6b563cSPaul Mackerras write_lock(&devtree_lock); 13979b6b563cSPaul Mackerras np->sibling = np->parent->child; 13989b6b563cSPaul Mackerras np->allnext = allnodes; 13999b6b563cSPaul Mackerras np->parent->child = np; 14009b6b563cSPaul Mackerras allnodes = np; 14019b6b563cSPaul Mackerras write_unlock(&devtree_lock); 14029b6b563cSPaul Mackerras } 14039b6b563cSPaul Mackerras 14049b6b563cSPaul Mackerras /* 14059b6b563cSPaul Mackerras * "Unplug" a node from the device tree. The caller must hold 14069b6b563cSPaul Mackerras * a reference to the node. The memory associated with the node 14079b6b563cSPaul Mackerras * is not freed until its refcount goes to zero. 14089b6b563cSPaul Mackerras */ 14099b6b563cSPaul Mackerras void of_detach_node(const struct device_node *np) 14109b6b563cSPaul Mackerras { 14119b6b563cSPaul Mackerras struct device_node *parent; 14129b6b563cSPaul Mackerras 14139b6b563cSPaul Mackerras write_lock(&devtree_lock); 14149b6b563cSPaul Mackerras 14159b6b563cSPaul Mackerras parent = np->parent; 14169b6b563cSPaul Mackerras 14179b6b563cSPaul Mackerras if (allnodes == np) 14189b6b563cSPaul Mackerras allnodes = np->allnext; 14199b6b563cSPaul Mackerras else { 14209b6b563cSPaul Mackerras struct device_node *prev; 14219b6b563cSPaul Mackerras for (prev = allnodes; 14229b6b563cSPaul Mackerras prev->allnext != np; 14239b6b563cSPaul Mackerras prev = prev->allnext) 14249b6b563cSPaul Mackerras ; 14259b6b563cSPaul Mackerras prev->allnext = np->allnext; 14269b6b563cSPaul Mackerras } 14279b6b563cSPaul Mackerras 14289b6b563cSPaul Mackerras if (parent->child == np) 14299b6b563cSPaul Mackerras parent->child = np->sibling; 14309b6b563cSPaul Mackerras else { 14319b6b563cSPaul Mackerras struct device_node *prevsib; 14329b6b563cSPaul Mackerras for (prevsib = np->parent->child; 14339b6b563cSPaul Mackerras prevsib->sibling != np; 14349b6b563cSPaul Mackerras prevsib = prevsib->sibling) 14359b6b563cSPaul Mackerras ; 14369b6b563cSPaul Mackerras prevsib->sibling = np->sibling; 14379b6b563cSPaul Mackerras } 14389b6b563cSPaul Mackerras 14399b6b563cSPaul Mackerras write_unlock(&devtree_lock); 14409b6b563cSPaul Mackerras } 14419b6b563cSPaul Mackerras 14429b6b563cSPaul Mackerras #ifdef CONFIG_PPC_PSERIES 14439b6b563cSPaul Mackerras /* 14449b6b563cSPaul Mackerras * Fix up the uninitialized fields in a new device node: 1445*0ebfff14SBenjamin Herrenschmidt * name, type and pci-specific fields 14469b6b563cSPaul Mackerras */ 14479b6b563cSPaul Mackerras 1448cc5d0189SBenjamin Herrenschmidt static int of_finish_dynamic_node(struct device_node *node) 14499b6b563cSPaul Mackerras { 14509b6b563cSPaul Mackerras struct device_node *parent = of_get_parent(node); 14519b6b563cSPaul Mackerras int err = 0; 14529b6b563cSPaul Mackerras phandle *ibm_phandle; 14539b6b563cSPaul Mackerras 14549b6b563cSPaul Mackerras node->name = get_property(node, "name", NULL); 14559b6b563cSPaul Mackerras node->type = get_property(node, "device_type", NULL); 14569b6b563cSPaul Mackerras 14579b6b563cSPaul Mackerras if (!parent) { 14589b6b563cSPaul Mackerras err = -ENODEV; 14599b6b563cSPaul Mackerras goto out; 14609b6b563cSPaul Mackerras } 14619b6b563cSPaul Mackerras 14629b6b563cSPaul Mackerras /* We don't support that function on PowerMac, at least 14639b6b563cSPaul Mackerras * not yet 14649b6b563cSPaul Mackerras */ 1465e8222502SBenjamin Herrenschmidt if (machine_is(powermac)) 14669b6b563cSPaul Mackerras return -ENODEV; 14679b6b563cSPaul Mackerras 14689b6b563cSPaul Mackerras /* fix up new node's linux_phandle field */ 1469cc5d0189SBenjamin Herrenschmidt if ((ibm_phandle = (unsigned int *)get_property(node, 1470cc5d0189SBenjamin Herrenschmidt "ibm,phandle", NULL))) 14719b6b563cSPaul Mackerras node->linux_phandle = *ibm_phandle; 14729b6b563cSPaul Mackerras 14739b6b563cSPaul Mackerras out: 14749b6b563cSPaul Mackerras of_node_put(parent); 14759b6b563cSPaul Mackerras return err; 14769b6b563cSPaul Mackerras } 14779b6b563cSPaul Mackerras 14789b6b563cSPaul Mackerras static int prom_reconfig_notifier(struct notifier_block *nb, 14799b6b563cSPaul Mackerras unsigned long action, void *node) 14809b6b563cSPaul Mackerras { 14819b6b563cSPaul Mackerras int err; 14829b6b563cSPaul Mackerras 14839b6b563cSPaul Mackerras switch (action) { 14849b6b563cSPaul Mackerras case PSERIES_RECONFIG_ADD: 1485cc5d0189SBenjamin Herrenschmidt err = of_finish_dynamic_node(node); 14869b6b563cSPaul Mackerras if (err < 0) { 14879b6b563cSPaul Mackerras printk(KERN_ERR "finish_node returned %d\n", err); 14889b6b563cSPaul Mackerras err = NOTIFY_BAD; 14899b6b563cSPaul Mackerras } 14909b6b563cSPaul Mackerras break; 14919b6b563cSPaul Mackerras default: 14929b6b563cSPaul Mackerras err = NOTIFY_DONE; 14939b6b563cSPaul Mackerras break; 14949b6b563cSPaul Mackerras } 14959b6b563cSPaul Mackerras return err; 14969b6b563cSPaul Mackerras } 14979b6b563cSPaul Mackerras 14989b6b563cSPaul Mackerras static struct notifier_block prom_reconfig_nb = { 14999b6b563cSPaul Mackerras .notifier_call = prom_reconfig_notifier, 15009b6b563cSPaul Mackerras .priority = 10, /* This one needs to run first */ 15019b6b563cSPaul Mackerras }; 15029b6b563cSPaul Mackerras 15039b6b563cSPaul Mackerras static int __init prom_reconfig_setup(void) 15049b6b563cSPaul Mackerras { 15059b6b563cSPaul Mackerras return pSeries_reconfig_notifier_register(&prom_reconfig_nb); 15069b6b563cSPaul Mackerras } 15079b6b563cSPaul Mackerras __initcall(prom_reconfig_setup); 15089b6b563cSPaul Mackerras #endif 15099b6b563cSPaul Mackerras 1510ecaa8b0fSDave C Boutcher struct property *of_find_property(struct device_node *np, const char *name, 15119b6b563cSPaul Mackerras int *lenp) 15129b6b563cSPaul Mackerras { 15139b6b563cSPaul Mackerras struct property *pp; 15149b6b563cSPaul Mackerras 1515088186deSDave C Boutcher read_lock(&devtree_lock); 15169b6b563cSPaul Mackerras for (pp = np->properties; pp != 0; pp = pp->next) 15179b6b563cSPaul Mackerras if (strcmp(pp->name, name) == 0) { 15189b6b563cSPaul Mackerras if (lenp != 0) 15199b6b563cSPaul Mackerras *lenp = pp->length; 1520088186deSDave C Boutcher break; 15219b6b563cSPaul Mackerras } 1522088186deSDave C Boutcher read_unlock(&devtree_lock); 1523088186deSDave C Boutcher 1524ecaa8b0fSDave C Boutcher return pp; 1525ecaa8b0fSDave C Boutcher } 1526ecaa8b0fSDave C Boutcher 1527ecaa8b0fSDave C Boutcher /* 1528ecaa8b0fSDave C Boutcher * Find a property with a given name for a given node 1529ecaa8b0fSDave C Boutcher * and return the value. 1530ecaa8b0fSDave C Boutcher */ 1531a1af5b2fSJeremy Kerr void *get_property(struct device_node *np, const char *name, int *lenp) 1532ecaa8b0fSDave C Boutcher { 1533ecaa8b0fSDave C Boutcher struct property *pp = of_find_property(np,name,lenp); 1534088186deSDave C Boutcher return pp ? pp->value : NULL; 15359b6b563cSPaul Mackerras } 15369b6b563cSPaul Mackerras EXPORT_SYMBOL(get_property); 15379b6b563cSPaul Mackerras 15389b6b563cSPaul Mackerras /* 15399b6b563cSPaul Mackerras * Add a property to a node 15409b6b563cSPaul Mackerras */ 1541183d0202SBenjamin Herrenschmidt int prom_add_property(struct device_node* np, struct property* prop) 15429b6b563cSPaul Mackerras { 1543183d0202SBenjamin Herrenschmidt struct property **next; 15449b6b563cSPaul Mackerras 15459b6b563cSPaul Mackerras prop->next = NULL; 1546183d0202SBenjamin Herrenschmidt write_lock(&devtree_lock); 1547183d0202SBenjamin Herrenschmidt next = &np->properties; 1548183d0202SBenjamin Herrenschmidt while (*next) { 1549183d0202SBenjamin Herrenschmidt if (strcmp(prop->name, (*next)->name) == 0) { 1550183d0202SBenjamin Herrenschmidt /* duplicate ! don't insert it */ 1551183d0202SBenjamin Herrenschmidt write_unlock(&devtree_lock); 1552183d0202SBenjamin Herrenschmidt return -1; 1553183d0202SBenjamin Herrenschmidt } 15549b6b563cSPaul Mackerras next = &(*next)->next; 1555183d0202SBenjamin Herrenschmidt } 15569b6b563cSPaul Mackerras *next = prop; 1557183d0202SBenjamin Herrenschmidt write_unlock(&devtree_lock); 1558183d0202SBenjamin Herrenschmidt 1559799d6046SPaul Mackerras #ifdef CONFIG_PROC_DEVICETREE 1560183d0202SBenjamin Herrenschmidt /* try to add to proc as well if it was initialized */ 1561183d0202SBenjamin Herrenschmidt if (np->pde) 1562183d0202SBenjamin Herrenschmidt proc_device_tree_add_prop(np->pde, prop); 1563799d6046SPaul Mackerras #endif /* CONFIG_PROC_DEVICETREE */ 1564183d0202SBenjamin Herrenschmidt 1565183d0202SBenjamin Herrenschmidt return 0; 15669b6b563cSPaul Mackerras } 15679b6b563cSPaul Mackerras 1568088186deSDave C Boutcher /* 1569088186deSDave C Boutcher * Remove a property from a node. Note that we don't actually 1570088186deSDave C Boutcher * remove it, since we have given out who-knows-how-many pointers 1571088186deSDave C Boutcher * to the data using get-property. Instead we just move the property 1572088186deSDave C Boutcher * to the "dead properties" list, so it won't be found any more. 1573088186deSDave C Boutcher */ 1574088186deSDave C Boutcher int prom_remove_property(struct device_node *np, struct property *prop) 1575088186deSDave C Boutcher { 1576088186deSDave C Boutcher struct property **next; 1577088186deSDave C Boutcher int found = 0; 15789b6b563cSPaul Mackerras 1579088186deSDave C Boutcher write_lock(&devtree_lock); 1580088186deSDave C Boutcher next = &np->properties; 1581088186deSDave C Boutcher while (*next) { 1582088186deSDave C Boutcher if (*next == prop) { 1583088186deSDave C Boutcher /* found the node */ 1584088186deSDave C Boutcher *next = prop->next; 1585088186deSDave C Boutcher prop->next = np->deadprops; 1586088186deSDave C Boutcher np->deadprops = prop; 1587088186deSDave C Boutcher found = 1; 1588088186deSDave C Boutcher break; 1589088186deSDave C Boutcher } 1590088186deSDave C Boutcher next = &(*next)->next; 1591088186deSDave C Boutcher } 1592088186deSDave C Boutcher write_unlock(&devtree_lock); 1593088186deSDave C Boutcher 1594088186deSDave C Boutcher if (!found) 1595088186deSDave C Boutcher return -ENODEV; 1596088186deSDave C Boutcher 1597088186deSDave C Boutcher #ifdef CONFIG_PROC_DEVICETREE 1598088186deSDave C Boutcher /* try to remove the proc node as well */ 1599088186deSDave C Boutcher if (np->pde) 1600088186deSDave C Boutcher proc_device_tree_remove_prop(np->pde, prop); 1601088186deSDave C Boutcher #endif /* CONFIG_PROC_DEVICETREE */ 1602088186deSDave C Boutcher 1603088186deSDave C Boutcher return 0; 1604088186deSDave C Boutcher } 1605088186deSDave C Boutcher 1606088186deSDave C Boutcher /* 1607088186deSDave C Boutcher * Update a property in a node. Note that we don't actually 1608088186deSDave C Boutcher * remove it, since we have given out who-knows-how-many pointers 1609088186deSDave C Boutcher * to the data using get-property. Instead we just move the property 1610088186deSDave C Boutcher * to the "dead properties" list, and add the new property to the 1611088186deSDave C Boutcher * property list 1612088186deSDave C Boutcher */ 1613088186deSDave C Boutcher int prom_update_property(struct device_node *np, 1614088186deSDave C Boutcher struct property *newprop, 1615088186deSDave C Boutcher struct property *oldprop) 1616088186deSDave C Boutcher { 1617088186deSDave C Boutcher struct property **next; 1618088186deSDave C Boutcher int found = 0; 1619088186deSDave C Boutcher 1620088186deSDave C Boutcher write_lock(&devtree_lock); 1621088186deSDave C Boutcher next = &np->properties; 1622088186deSDave C Boutcher while (*next) { 1623088186deSDave C Boutcher if (*next == oldprop) { 1624088186deSDave C Boutcher /* found the node */ 1625088186deSDave C Boutcher newprop->next = oldprop->next; 1626088186deSDave C Boutcher *next = newprop; 1627088186deSDave C Boutcher oldprop->next = np->deadprops; 1628088186deSDave C Boutcher np->deadprops = oldprop; 1629088186deSDave C Boutcher found = 1; 1630088186deSDave C Boutcher break; 1631088186deSDave C Boutcher } 1632088186deSDave C Boutcher next = &(*next)->next; 1633088186deSDave C Boutcher } 1634088186deSDave C Boutcher write_unlock(&devtree_lock); 1635088186deSDave C Boutcher 1636088186deSDave C Boutcher if (!found) 1637088186deSDave C Boutcher return -ENODEV; 1638088186deSDave C Boutcher 1639088186deSDave C Boutcher #ifdef CONFIG_PROC_DEVICETREE 1640088186deSDave C Boutcher /* try to add to proc as well if it was initialized */ 1641088186deSDave C Boutcher if (np->pde) 1642088186deSDave C Boutcher proc_device_tree_update_prop(np->pde, newprop, oldprop); 1643088186deSDave C Boutcher #endif /* CONFIG_PROC_DEVICETREE */ 1644088186deSDave C Boutcher 1645088186deSDave C Boutcher return 0; 1646088186deSDave C Boutcher } 1647b68239eeSMichael Ellerman 1648acf7d768SBenjamin Herrenschmidt 1649acf7d768SBenjamin Herrenschmidt /* Find the device node for a given logical cpu number, also returns the cpu 1650acf7d768SBenjamin Herrenschmidt * local thread number (index in ibm,interrupt-server#s) if relevant and 1651acf7d768SBenjamin Herrenschmidt * asked for (non NULL) 1652acf7d768SBenjamin Herrenschmidt */ 1653acf7d768SBenjamin Herrenschmidt struct device_node *of_get_cpu_node(int cpu, unsigned int *thread) 1654acf7d768SBenjamin Herrenschmidt { 1655acf7d768SBenjamin Herrenschmidt int hardid; 1656acf7d768SBenjamin Herrenschmidt struct device_node *np; 1657acf7d768SBenjamin Herrenschmidt 1658acf7d768SBenjamin Herrenschmidt hardid = get_hard_smp_processor_id(cpu); 1659acf7d768SBenjamin Herrenschmidt 1660acf7d768SBenjamin Herrenschmidt for_each_node_by_type(np, "cpu") { 1661acf7d768SBenjamin Herrenschmidt u32 *intserv; 1662acf7d768SBenjamin Herrenschmidt unsigned int plen, t; 1663acf7d768SBenjamin Herrenschmidt 1664acf7d768SBenjamin Herrenschmidt /* Check for ibm,ppc-interrupt-server#s. If it doesn't exist 1665acf7d768SBenjamin Herrenschmidt * fallback to "reg" property and assume no threads 1666acf7d768SBenjamin Herrenschmidt */ 1667acf7d768SBenjamin Herrenschmidt intserv = (u32 *)get_property(np, "ibm,ppc-interrupt-server#s", 1668acf7d768SBenjamin Herrenschmidt &plen); 1669acf7d768SBenjamin Herrenschmidt if (intserv == NULL) { 1670acf7d768SBenjamin Herrenschmidt u32 *reg = (u32 *)get_property(np, "reg", NULL); 1671acf7d768SBenjamin Herrenschmidt if (reg == NULL) 1672acf7d768SBenjamin Herrenschmidt continue; 1673acf7d768SBenjamin Herrenschmidt if (*reg == hardid) { 1674acf7d768SBenjamin Herrenschmidt if (thread) 1675acf7d768SBenjamin Herrenschmidt *thread = 0; 1676acf7d768SBenjamin Herrenschmidt return np; 1677acf7d768SBenjamin Herrenschmidt } 1678acf7d768SBenjamin Herrenschmidt } else { 1679acf7d768SBenjamin Herrenschmidt plen /= sizeof(u32); 1680acf7d768SBenjamin Herrenschmidt for (t = 0; t < plen; t++) { 1681acf7d768SBenjamin Herrenschmidt if (hardid == intserv[t]) { 1682acf7d768SBenjamin Herrenschmidt if (thread) 1683acf7d768SBenjamin Herrenschmidt *thread = t; 1684acf7d768SBenjamin Herrenschmidt return np; 1685acf7d768SBenjamin Herrenschmidt } 1686acf7d768SBenjamin Herrenschmidt } 1687acf7d768SBenjamin Herrenschmidt } 1688acf7d768SBenjamin Herrenschmidt } 1689acf7d768SBenjamin Herrenschmidt return NULL; 1690acf7d768SBenjamin Herrenschmidt } 16917a4571aeSMichael Ellerman 16927a4571aeSMichael Ellerman #ifdef DEBUG 16937a4571aeSMichael Ellerman static struct debugfs_blob_wrapper flat_dt_blob; 16947a4571aeSMichael Ellerman 16957a4571aeSMichael Ellerman static int __init export_flat_device_tree(void) 16967a4571aeSMichael Ellerman { 16977a4571aeSMichael Ellerman struct dentry *d; 16987a4571aeSMichael Ellerman 16997a4571aeSMichael Ellerman d = debugfs_create_dir("powerpc", NULL); 17007a4571aeSMichael Ellerman if (!d) 17017a4571aeSMichael Ellerman return 1; 17027a4571aeSMichael Ellerman 17037a4571aeSMichael Ellerman flat_dt_blob.data = initial_boot_params; 17047a4571aeSMichael Ellerman flat_dt_blob.size = initial_boot_params->totalsize; 17057a4571aeSMichael Ellerman 17067a4571aeSMichael Ellerman d = debugfs_create_blob("flat-device-tree", S_IFREG | S_IRUSR, 17077a4571aeSMichael Ellerman d, &flat_dt_blob); 17087a4571aeSMichael Ellerman if (!d) 17097a4571aeSMichael Ellerman return 1; 17107a4571aeSMichael Ellerman 17117a4571aeSMichael Ellerman return 0; 17127a4571aeSMichael Ellerman } 17137a4571aeSMichael Ellerman __initcall(export_flat_device_tree); 17147a4571aeSMichael Ellerman #endif 1715