2011-10-24 19:15:32 +00:00
/*
* Copyright ( c ) 2011 The Chromium OS Authors .
2013-07-08 07:37:19 +00:00
* SPDX - License - Identifier : GPL - 2.0 +
2011-10-24 19:15:32 +00:00
*/
2014-03-03 11:19:30 +00:00
# ifndef USE_HOSTCC
2011-10-24 19:15:32 +00:00
# include <common.h>
2014-07-23 12:55:09 +00:00
# include <errno.h>
2011-10-24 19:15:32 +00:00
# include <serial.h>
# include <libfdt.h>
# include <fdtdec.h>
2015-02-28 05:06:34 +00:00
# include <asm/sections.h>
2014-07-23 12:55:09 +00:00
# include <linux/ctype.h>
2011-10-24 19:15:32 +00:00
DECLARE_GLOBAL_DATA_PTR ;
/*
* Here are the type we know about . One day we might allow drivers to
* register . For now we just put them here . The COMPAT macro allows us to
* turn this into a sparse list later , and keeps the ID with the name .
*/
# define COMPAT(id, name) name
static const char * const compat_names [ COMPAT_COUNT ] = {
2012-02-27 10:52:34 +00:00
COMPAT ( UNKNOWN , " <none> " ) ,
2012-04-02 13:18:52 +00:00
COMPAT ( NVIDIA_TEGRA20_EMC , " nvidia,tegra20-emc " ) ,
COMPAT ( NVIDIA_TEGRA20_EMC_TABLE , " nvidia,tegra20-emc-table " ) ,
2012-04-17 09:01:35 +00:00
COMPAT ( NVIDIA_TEGRA20_KBC , " nvidia,tegra20-kbc " ) ,
2012-07-29 20:53:29 +00:00
COMPAT ( NVIDIA_TEGRA20_NAND , " nvidia,tegra20-nand " ) ,
2012-10-17 13:24:49 +00:00
COMPAT ( NVIDIA_TEGRA20_PWM , " nvidia,tegra20-pwm " ) ,
2015-04-15 03:03:40 +00:00
COMPAT ( NVIDIA_TEGRA124_DC , " nvidia,tegra124-dc " ) ,
COMPAT ( NVIDIA_TEGRA124_SOR , " nvidia,tegra124-sor " ) ,
COMPAT ( NVIDIA_TEGRA124_PMC , " nvidia,tegra124-pmc " ) ,
2012-10-17 13:24:50 +00:00
COMPAT ( NVIDIA_TEGRA20_DC , " nvidia,tegra20-dc " ) ,
2015-03-04 23:36:00 +00:00
COMPAT ( NVIDIA_TEGRA210_SDMMC , " nvidia,tegra210-sdhci " ) ,
2014-01-24 19:46:06 +00:00
COMPAT ( NVIDIA_TEGRA124_SDMMC , " nvidia,tegra124-sdhci " ) ,
2013-03-04 21:07:18 +00:00
COMPAT ( NVIDIA_TEGRA30_SDMMC , " nvidia,tegra30-sdhci " ) ,
2013-02-21 12:31:30 +00:00
COMPAT ( NVIDIA_TEGRA20_SDMMC , " nvidia,tegra20-sdhci " ) ,
2014-12-10 05:25:12 +00:00
COMPAT ( NVIDIA_TEGRA124_PCIE , " nvidia,tegra124-pcie " ) ,
COMPAT ( NVIDIA_TEGRA30_PCIE , " nvidia,tegra30-pcie " ) ,
COMPAT ( NVIDIA_TEGRA20_PCIE , " nvidia,tegra20-pcie " ) ,
2014-12-10 05:25:09 +00:00
COMPAT ( NVIDIA_TEGRA124_XUSB_PADCTL , " nvidia,tegra124-xusb-padctl " ) ,
2015-03-04 23:36:00 +00:00
COMPAT ( NVIDIA_TEGRA210_XUSB_PADCTL , " nvidia,tegra210-xusb-padctl " ) ,
2012-12-11 00:52:46 +00:00
COMPAT ( SMSC_LAN9215 , " smsc,lan9215 " ) ,
COMPAT ( SAMSUNG_EXYNOS5_SROMC , " samsung,exynos-sromc " ) ,
2012-12-26 20:03:10 +00:00
COMPAT ( SAMSUNG_S3C2440_I2C , " samsung,s3c2440-i2c " ) ,
2012-12-26 20:03:16 +00:00
COMPAT ( SAMSUNG_EXYNOS5_SOUND , " samsung,exynos-sound " ) ,
COMPAT ( WOLFSON_WM8994_CODEC , " wolfson,wm8994-codec " ) ,
2013-05-15 10:27:32 +00:00
COMPAT ( GOOGLE_CROS_EC_KEYB , " google,cros-ec-keyb " ) ,
2013-01-07 23:35:05 +00:00
COMPAT ( SAMSUNG_EXYNOS_USB_PHY , " samsung,exynos-usb-phy " ) ,
2013-09-14 08:32:48 +00:00
COMPAT ( SAMSUNG_EXYNOS5_USB3_PHY , " samsung,exynos5250-usb3-phy " ) ,
2013-02-25 01:13:01 +00:00
COMPAT ( SAMSUNG_EXYNOS_TMU , " samsung,exynos-tmu " ) ,
2013-02-21 23:53:00 +00:00
COMPAT ( SAMSUNG_EXYNOS_FIMD , " samsung,exynos-fimd " ) ,
2014-03-07 13:59:39 +00:00
COMPAT ( SAMSUNG_EXYNOS_MIPI_DSI , " samsung,exynos-mipi-dsi " ) ,
2013-02-21 23:53:05 +00:00
COMPAT ( SAMSUNG_EXYNOS5_DP , " samsung,exynos5-dp " ) ,
2014-05-16 04:59:51 +00:00
COMPAT ( SAMSUNG_EXYNOS_DWMMC , " samsung,exynos-dwmmc " ) ,
2014-03-07 13:59:41 +00:00
COMPAT ( SAMSUNG_EXYNOS_MMC , " samsung,exynos-mmc " ) ,
2013-06-24 11:17:20 +00:00
COMPAT ( SAMSUNG_EXYNOS_SERIAL , " samsung,exynos4210-uart " ) ,
2015-04-20 18:07:45 +00:00
COMPAT ( MAXIM_MAX77686_PMIC , " maxim,max77686 " ) ,
2013-03-11 06:08:08 +00:00
COMPAT ( GENERIC_SPI_FLASH , " spi-flash " ) ,
2013-02-14 19:46:15 +00:00
COMPAT ( MAXIM_98095_CODEC , " maxim,max98095-codec " ) ,
2013-04-12 10:44:57 +00:00
COMPAT ( INFINEON_SLB9635_TPM , " infineon,slb9635-tpm " ) ,
2015-05-04 17:31:00 +00:00
COMPAT ( INFINEON_SLB9645_TPM , " infineon,slb9645tt " ) ,
2013-04-29 22:58:52 +00:00
COMPAT ( SAMSUNG_EXYNOS5_I2C , " samsung,exynos5-hsi2c " ) ,
2014-02-27 20:26:19 +00:00
COMPAT ( SANDBOX_LCD_SDL , " sandbox,lcd-sdl " ) ,
2014-09-05 11:23:33 +00:00
COMPAT ( SAMSUNG_EXYNOS_SYSMMU , " samsung,sysmmu-v3.3 " ) ,
2014-11-13 05:42:21 +00:00
COMPAT ( INTEL_MICROCODE , " intel,microcode " ) ,
2014-11-13 05:42:28 +00:00
COMPAT ( MEMORY_SPD , " memory-spd " ) ,
2014-11-15 01:18:38 +00:00
COMPAT ( INTEL_PANTHERPOINT_AHCI , " intel,pantherpoint-ahci " ) ,
2014-11-25 04:18:16 +00:00
COMPAT ( INTEL_MODEL_206AX , " intel,model-206ax " ) ,
2014-11-15 03:56:36 +00:00
COMPAT ( INTEL_GMA , " intel,gma " ) ,
2014-12-10 05:25:05 +00:00
COMPAT ( AMS_AS3722 , " ams,as3722 " ) ,
2015-01-20 05:16:06 +00:00
COMPAT ( INTEL_ICH_SPI , " intel,ich-spi " ) ,
2015-02-05 15:42:26 +00:00
COMPAT ( INTEL_QRK_MRC , " intel,quark-mrc " ) ,
2015-05-26 05:27:37 +00:00
COMPAT ( INTEL_X86_PINCTRL , " intel,x86-pinctrl " ) ,
2015-03-11 06:54:46 +00:00
COMPAT ( SOCIONEXT_XHCI , " socionext,uniphier-xhci " ) ,
2015-03-26 15:29:29 +00:00
COMPAT ( COMPAT_INTEL_PCH , " intel,bd82x6x " ) ,
2015-05-25 14:35:04 +00:00
COMPAT ( COMPAT_INTEL_IRQ_ROUTER , " intel,irq-router " ) ,
2015-07-25 17:33:56 +00:00
COMPAT ( ALTERA_SOCFPGA_DWMAC , " altr,socfpga-stmmac " ) ,
2015-08-07 12:36:35 +00:00
COMPAT ( COMPAT_INTEL_BAYTRAIL_FSP , " intel,baytrail-fsp " ) ,
COMPAT ( COMPAT_INTEL_BAYTRAIL_FSP_MDP , " intel,baytrail-fsp-mdp " ) ,
2011-10-24 19:15:32 +00:00
} ;
2012-01-17 08:20:50 +00:00
const char * fdtdec_get_compatible ( enum fdt_compat_id id )
{
/* We allow reading of the 'unknown' ID for testing purposes */
assert ( id > = 0 & & id < COMPAT_COUNT ) ;
return compat_names [ id ] ;
}
2013-03-19 04:58:51 +00:00
fdt_addr_t fdtdec_get_addr_size ( const void * blob , int node ,
const char * prop_name , fdt_size_t * sizep )
2011-10-24 19:15:32 +00:00
{
2015-08-03 00:13:50 +00:00
const fdt_addr_t * cell ;
int len ;
2011-10-24 19:15:32 +00:00
2012-07-12 05:25:01 +00:00
debug ( " %s: %s: " , __func__ , prop_name ) ;
2015-08-03 00:13:50 +00:00
cell = fdt_getprop ( blob , node , prop_name , & len ) ;
if ( cell & & ( ( ! sizep & & len = = sizeof ( fdt_addr_t ) ) | |
len = = sizeof ( fdt_addr_t ) * 2 ) ) {
fdt_addr_t addr = fdt_addr_to_cpu ( * cell ) ;
if ( sizep ) {
const fdt_size_t * size ;
size = ( fdt_size_t * ) ( ( char * ) cell +
sizeof ( fdt_addr_t ) ) ;
* sizep = fdt_size_to_cpu ( * size ) ;
debug ( " addr=%08lx, size=%llx \n " ,
( ulong ) addr , ( u64 ) * sizep ) ;
} else {
debug ( " %08lx \n " , ( ulong ) addr ) ;
}
return addr ;
2015-07-23 16:51:30 +00:00
}
2015-08-03 00:13:50 +00:00
debug ( " (not found) \n " ) ;
return FDT_ADDR_T_NONE ;
2011-10-24 19:15:32 +00:00
}
2013-03-19 04:58:51 +00:00
fdt_addr_t fdtdec_get_addr ( const void * blob , int node ,
const char * prop_name )
{
return fdtdec_get_addr_size ( blob , node , prop_name , NULL ) ;
}
2014-12-31 08:05:11 +00:00
# ifdef CONFIG_PCI
int fdtdec_get_pci_addr ( const void * blob , int node , enum fdt_pci_space type ,
const char * prop_name , struct fdt_pci_addr * addr )
{
const u32 * cell ;
int len ;
int ret = - ENOENT ;
debug ( " %s: %s: " , __func__ , prop_name ) ;
/*
* If we follow the pci bus bindings strictly , we should check
* the value of the node ' s parent node ' s # address - cells and
* # size - cells . They need to be 3 and 2 accordingly . However ,
* for simplicity we skip the check here .
*/
cell = fdt_getprop ( blob , node , prop_name , & len ) ;
if ( ! cell )
goto fail ;
if ( ( len % FDT_PCI_REG_SIZE ) = = 0 ) {
int num = len / FDT_PCI_REG_SIZE ;
int i ;
for ( i = 0 ; i < num ; i + + ) {
debug ( " pci address #%d: %08lx %08lx %08lx \n " , i ,
( ulong ) fdt_addr_to_cpu ( cell [ 0 ] ) ,
( ulong ) fdt_addr_to_cpu ( cell [ 1 ] ) ,
( ulong ) fdt_addr_to_cpu ( cell [ 2 ] ) ) ;
if ( ( fdt_addr_to_cpu ( * cell ) & type ) = = type ) {
addr - > phys_hi = fdt_addr_to_cpu ( cell [ 0 ] ) ;
addr - > phys_mid = fdt_addr_to_cpu ( cell [ 1 ] ) ;
addr - > phys_lo = fdt_addr_to_cpu ( cell [ 2 ] ) ;
break ;
} else {
cell + = ( FDT_PCI_ADDR_CELLS +
FDT_PCI_SIZE_CELLS ) ;
}
}
2015-03-05 19:25:19 +00:00
if ( i = = num ) {
ret = - ENXIO ;
2014-12-31 08:05:11 +00:00
goto fail ;
2015-03-05 19:25:19 +00:00
}
2014-12-31 08:05:11 +00:00
return 0 ;
} else {
ret = - EINVAL ;
}
fail :
debug ( " (not found) \n " ) ;
return ret ;
}
int fdtdec_get_pci_vendev ( const void * blob , int node , u16 * vendor , u16 * device )
{
const char * list , * end ;
int len ;
list = fdt_getprop ( blob , node , " compatible " , & len ) ;
if ( ! list )
return - ENOENT ;
end = list + len ;
while ( list < end ) {
char * s ;
len = strlen ( list ) ;
if ( len > = strlen ( " pciVVVV,DDDD " ) ) {
s = strstr ( list , " pci " ) ;
/*
* check if the string is something like pciVVVV , DDDD . RR
* or just pciVVVV , DDDD
*/
if ( s & & s [ 7 ] = = ' , ' & &
( s [ 12 ] = = ' . ' | | s [ 12 ] = = 0 ) ) {
s + = 3 ;
* vendor = simple_strtol ( s , NULL , 16 ) ;
s + = 5 ;
* device = simple_strtol ( s , NULL , 16 ) ;
return 0 ;
}
}
2015-08-20 13:40:25 +00:00
list + = ( len + 1 ) ;
2014-12-31 08:05:11 +00:00
}
return - ENOENT ;
}
int fdtdec_get_pci_bdf ( const void * blob , int node ,
struct fdt_pci_addr * addr , pci_dev_t * bdf )
{
u16 dt_vendor , dt_device , vendor , device ;
int ret ;
/* get vendor id & device id from the compatible string */
ret = fdtdec_get_pci_vendev ( blob , node , & dt_vendor , & dt_device ) ;
if ( ret )
return ret ;
/* extract the bdf from fdt_pci_addr */
* bdf = addr - > phys_hi & 0xffff00 ;
/* read vendor id & device id based on bdf */
pci_read_config_word ( * bdf , PCI_VENDOR_ID , & vendor ) ;
pci_read_config_word ( * bdf , PCI_DEVICE_ID , & device ) ;
/*
* Note there are two places in the device tree to fully describe
* a pci device : one is via compatible string with a format of
* " pciVVVV,DDDD " and the other one is the bdf numbers encoded in
* the device node ' s reg address property . We read the vendor id
* and device id based on bdf and compare the values with the
* " VVVV,DDDD " . If they are the same , then we are good to use bdf
* to read device ' s bar . But if they are different , we have to rely
* on the vendor id and device id extracted from the compatible
* string and locate the real bdf by pci_find_device ( ) . This is
* because normally we may only know device ' s device number and
* function number when writing device tree . The bus number is
* dynamically assigned during the pci enumeration process .
*/
if ( ( dt_vendor ! = vendor ) | | ( dt_device ! = device ) ) {
* bdf = pci_find_device ( dt_vendor , dt_device , 0 ) ;
if ( * bdf = = - 1 )
return - ENODEV ;
}
return 0 ;
}
int fdtdec_get_pci_bar32 ( const void * blob , int node ,
struct fdt_pci_addr * addr , u32 * bar )
{
pci_dev_t bdf ;
int barnum ;
int ret ;
/* get pci devices's bdf */
ret = fdtdec_get_pci_bdf ( blob , node , addr , & bdf ) ;
if ( ret )
return ret ;
/* extract the bar number from fdt_pci_addr */
barnum = addr - > phys_hi & 0xff ;
if ( ( barnum < PCI_BASE_ADDRESS_0 ) | | ( barnum > PCI_CARDBUS_CIS ) )
return - EINVAL ;
barnum = ( barnum - PCI_BASE_ADDRESS_0 ) / 4 ;
* bar = pci_read_bar32 ( pci_bus_to_hose ( PCI_BUS ( bdf ) ) , bdf , barnum ) ;
return 0 ;
}
# endif
2012-10-25 16:31:05 +00:00
uint64_t fdtdec_get_uint64 ( const void * blob , int node , const char * prop_name ,
uint64_t default_val )
{
const uint64_t * cell64 ;
int length ;
cell64 = fdt_getprop ( blob , node , prop_name , & length ) ;
if ( ! cell64 | | length < sizeof ( * cell64 ) )
return default_val ;
return fdt64_to_cpu ( * cell64 ) ;
}
2012-02-27 10:52:34 +00:00
int fdtdec_get_is_enabled ( const void * blob , int node )
2011-10-24 19:15:32 +00:00
{
const char * cell ;
2012-02-27 10:52:34 +00:00
/*
* It should say " okay " , so only allow that . Some fdts use " ok " but
* this is a bug . Please fix your device tree source file . See here
* for discussion :
*
* http : //www.mail-archive.com/u-boot@lists.denx.de/msg71598.html
*/
2011-10-24 19:15:32 +00:00
cell = fdt_getprop ( blob , node , " status " , NULL ) ;
if ( cell )
2012-02-27 10:52:34 +00:00
return 0 = = strcmp ( cell , " okay " ) ;
return 1 ;
2011-10-24 19:15:32 +00:00
}
2012-11-13 04:13:54 +00:00
enum fdt_compat_id fdtdec_lookup ( const void * blob , int node )
2011-10-24 19:15:32 +00:00
{
enum fdt_compat_id id ;
/* Search our drivers */
for ( id = COMPAT_UNKNOWN ; id < COMPAT_COUNT ; id + + )
if ( 0 = = fdt_node_check_compatible ( blob , node ,
compat_names [ id ] ) )
return id ;
return COMPAT_UNKNOWN ;
}
int fdtdec_next_compatible ( const void * blob , int node ,
enum fdt_compat_id id )
{
return fdt_node_offset_by_compatible ( blob , node , compat_names [ id ] ) ;
}
2012-04-02 13:18:42 +00:00
int fdtdec_next_compatible_subnode ( const void * blob , int node ,
enum fdt_compat_id id , int * depthp )
{
do {
node = fdt_next_node ( blob , node , depthp ) ;
} while ( * depthp > 1 ) ;
/* If this is a direct subnode, and compatible, return it */
if ( * depthp = = 1 & & 0 = = fdt_node_check_compatible (
blob , node , compat_names [ id ] ) )
return node ;
return - FDT_ERR_NOTFOUND ;
}
2011-10-24 19:15:32 +00:00
int fdtdec_next_alias ( const void * blob , const char * name ,
enum fdt_compat_id id , int * upto )
{
# define MAX_STR_LEN 20
char str [ MAX_STR_LEN + 20 ] ;
int node , err ;
/* snprintf() is not available */
assert ( strlen ( name ) < MAX_STR_LEN ) ;
sprintf ( str , " %.*s%d " , MAX_STR_LEN , name , * upto ) ;
2012-10-31 14:02:42 +00:00
node = fdt_path_offset ( blob , str ) ;
2011-10-24 19:15:32 +00:00
if ( node < 0 )
return node ;
err = fdt_node_check_compatible ( blob , node , compat_names [ id ] ) ;
if ( err < 0 )
return err ;
2012-02-27 10:52:34 +00:00
if ( err )
return - FDT_ERR_NOTFOUND ;
( * upto ) + + ;
return node ;
2011-10-24 19:15:32 +00:00
}
2012-01-17 08:20:50 +00:00
int fdtdec_find_aliases_for_id ( const void * blob , const char * name ,
enum fdt_compat_id id , int * node_list , int maxcount )
2012-02-03 15:13:53 +00:00
{
memset ( node_list , ' \0 ' , sizeof ( * node_list ) * maxcount ) ;
return fdtdec_add_aliases_for_id ( blob , name , id , node_list , maxcount ) ;
}
/* TODO: Can we tighten this code up a little? */
int fdtdec_add_aliases_for_id ( const void * blob , const char * name ,
enum fdt_compat_id id , int * node_list , int maxcount )
2012-01-17 08:20:50 +00:00
{
int name_len = strlen ( name ) ;
int nodes [ maxcount ] ;
int num_found = 0 ;
int offset , node ;
int alias_node ;
int count ;
int i , j ;
/* find the alias node if present */
alias_node = fdt_path_offset ( blob , " /aliases " ) ;
/*
* start with nothing , and we can assume that the root node can ' t
* match
*/
memset ( nodes , ' \0 ' , sizeof ( nodes ) ) ;
/* First find all the compatible nodes */
for ( node = count = 0 ; node > = 0 & & count < maxcount ; ) {
node = fdtdec_next_compatible ( blob , node , id ) ;
if ( node > = 0 )
nodes [ count + + ] = node ;
}
if ( node > = 0 )
debug ( " %s: warning: maxcount exceeded with alias '%s' \n " ,
__func__ , name ) ;
/* Now find all the aliases */
for ( offset = fdt_first_property_offset ( blob , alias_node ) ;
offset > 0 ;
offset = fdt_next_property_offset ( blob , offset ) ) {
const struct fdt_property * prop ;
const char * path ;
int number ;
int found ;
node = 0 ;
prop = fdt_get_property_by_offset ( blob , offset , NULL ) ;
path = fdt_string ( blob , fdt32_to_cpu ( prop - > nameoff ) ) ;
if ( prop - > len & & 0 = = strncmp ( path , name , name_len ) )
node = fdt_path_offset ( blob , prop - > data ) ;
if ( node < = 0 )
continue ;
/* Get the alias number */
number = simple_strtoul ( path + name_len , NULL , 10 ) ;
if ( number < 0 | | number > = maxcount ) {
debug ( " %s: warning: alias '%s' is out of range \n " ,
__func__ , path ) ;
continue ;
}
/* Make sure the node we found is actually in our list! */
found = - 1 ;
for ( j = 0 ; j < count ; j + + )
if ( nodes [ j ] = = node ) {
found = j ;
break ;
}
if ( found = = - 1 ) {
debug ( " %s: warning: alias '%s' points to a node "
" '%s' that is missing or is not compatible "
" with '%s' \n " , __func__ , path ,
fdt_get_name ( blob , node , NULL ) ,
compat_names [ id ] ) ;
continue ;
}
/*
* Add this node to our list in the right place , and mark
* it as done .
*/
if ( fdtdec_get_is_enabled ( blob , node ) ) {
2012-02-03 15:13:53 +00:00
if ( node_list [ number ] ) {
debug ( " %s: warning: alias '%s' requires that "
" a node be placed in the list in a "
" position which is already filled by "
" node '%s' \n " , __func__ , path ,
fdt_get_name ( blob , node , NULL ) ) ;
continue ;
}
2012-01-17 08:20:50 +00:00
node_list [ number ] = node ;
if ( number > = num_found )
num_found = number + 1 ;
}
2012-02-03 15:13:53 +00:00
nodes [ found ] = 0 ;
2012-01-17 08:20:50 +00:00
}
/* Add any nodes not mentioned by an alias */
for ( i = j = 0 ; i < maxcount ; i + + ) {
if ( ! node_list [ i ] ) {
for ( ; j < maxcount ; j + + )
if ( nodes [ j ] & &
fdtdec_get_is_enabled ( blob , nodes [ j ] ) )
break ;
/* Have we run out of nodes to add? */
if ( j = = maxcount )
break ;
assert ( ! node_list [ i ] ) ;
node_list [ i ] = nodes [ j + + ] ;
if ( i > = num_found )
num_found = i + 1 ;
}
}
return num_found ;
}
2014-07-23 12:55:09 +00:00
int fdtdec_get_alias_seq ( const void * blob , const char * base , int offset ,
int * seqp )
{
int base_len = strlen ( base ) ;
const char * find_name ;
int find_namelen ;
int prop_offset ;
int aliases ;
find_name = fdt_get_name ( blob , offset , & find_namelen ) ;
debug ( " Looking for '%s' at %d, name %s \n " , base , offset , find_name ) ;
aliases = fdt_path_offset ( blob , " /aliases " ) ;
for ( prop_offset = fdt_first_property_offset ( blob , aliases ) ;
prop_offset > 0 ;
prop_offset = fdt_next_property_offset ( blob , prop_offset ) ) {
const char * prop ;
const char * name ;
const char * slash ;
2015-06-23 21:39:08 +00:00
int len , val ;
2014-07-23 12:55:09 +00:00
prop = fdt_getprop_by_offset ( blob , prop_offset , & name , & len ) ;
debug ( " - %s, %s \n " , name , prop ) ;
if ( len < find_namelen | | * prop ! = ' / ' | | prop [ len - 1 ] | |
strncmp ( name , base , base_len ) )
continue ;
slash = strrchr ( prop , ' / ' ) ;
if ( strcmp ( slash + 1 , find_name ) )
continue ;
2015-06-23 21:39:08 +00:00
val = trailing_strtol ( name ) ;
if ( val ! = - 1 ) {
* seqp = val ;
debug ( " Found seq %d \n " , * seqp ) ;
return 0 ;
2014-07-23 12:55:09 +00:00
}
}
debug ( " Not found \n " ) ;
return - ENOENT ;
}
2014-09-04 22:27:24 +00:00
int fdtdec_get_chosen_node ( const void * blob , const char * name )
{
const char * prop ;
int chosen_node ;
int len ;
if ( ! blob )
return - FDT_ERR_NOTFOUND ;
chosen_node = fdt_path_offset ( blob , " /chosen " ) ;
prop = fdt_getprop ( blob , chosen_node , name , & len ) ;
if ( ! prop )
return - FDT_ERR_NOTFOUND ;
return fdt_path_offset ( blob , prop ) ;
}
2012-03-28 10:08:24 +00:00
int fdtdec_check_fdt ( void )
{
/*
* We must have an FDT , but we cannot panic ( ) yet since the console
* is not ready . So for now , just assert ( ) . Boards which need an early
* FDT ( prior to console ready ) will need to make their own
* arrangements and do their own checks .
*/
assert ( ! fdtdec_prepare_fdt ( ) ) ;
return 0 ;
}
2011-10-24 19:15:32 +00:00
/*
* This function is a little odd in that it accesses global data . At some
* point if the architecture board . c files merge this will make more sense .
* Even now , it is common code .
*/
2012-03-28 10:08:24 +00:00
int fdtdec_prepare_fdt ( void )
2011-10-24 19:15:32 +00:00
{
2013-04-20 08:42:46 +00:00
if ( ! gd - > fdt_blob | | ( ( uintptr_t ) gd - > fdt_blob & 3 ) | |
fdt_check_header ( gd - > fdt_blob ) ) {
2015-02-28 05:06:32 +00:00
# ifdef CONFIG_SPL_BUILD
puts ( " Missing DTB \n " ) ;
# else
puts ( " No valid device tree binary found - please append one to U-Boot binary, use u-boot-dtb.bin or define CONFIG_OF_EMBED. For sandbox, use -d <file.dtb> \n " ) ;
2015-06-23 21:39:09 +00:00
# ifdef DEBUG
if ( gd - > fdt_blob ) {
printf ( " fdt_blob=%p \n " , gd - > fdt_blob ) ;
print_buffer ( ( ulong ) gd - > fdt_blob , gd - > fdt_blob , 4 ,
32 , 0 ) ;
}
# endif
2015-02-28 05:06:32 +00:00
# endif
2012-03-28 10:08:24 +00:00
return - 1 ;
}
2011-10-24 19:15:32 +00:00
return 0 ;
}
2012-02-27 10:52:35 +00:00
int fdtdec_lookup_phandle ( const void * blob , int node , const char * prop_name )
{
const u32 * phandle ;
int lookup ;
2012-07-12 05:25:01 +00:00
debug ( " %s: %s \n " , __func__ , prop_name ) ;
2012-02-27 10:52:35 +00:00
phandle = fdt_getprop ( blob , node , prop_name , NULL ) ;
if ( ! phandle )
return - FDT_ERR_NOTFOUND ;
lookup = fdt_node_offset_by_phandle ( blob , fdt32_to_cpu ( * phandle ) ) ;
return lookup ;
}
/**
* Look up a property in a node and check that it has a minimum length .
*
* @ param blob FDT blob
* @ param node node to examine
* @ param prop_name name of property to find
* @ param min_len minimum property length in bytes
* @ param err 0 if ok , or - FDT_ERR_NOTFOUND if the property is not
found , or - FDT_ERR_BADLAYOUT if not enough data
* @ return pointer to cell , which is only valid if err = = 0
*/
static const void * get_prop_check_min_len ( const void * blob , int node ,
const char * prop_name , int min_len , int * err )
{
const void * cell ;
int len ;
debug ( " %s: %s \n " , __func__ , prop_name ) ;
cell = fdt_getprop ( blob , node , prop_name , & len ) ;
if ( ! cell )
* err = - FDT_ERR_NOTFOUND ;
else if ( len < min_len )
* err = - FDT_ERR_BADLAYOUT ;
else
* err = 0 ;
return cell ;
}
int fdtdec_get_int_array ( const void * blob , int node , const char * prop_name ,
u32 * array , int count )
{
const u32 * cell ;
int i , err = 0 ;
debug ( " %s: %s \n " , __func__ , prop_name ) ;
cell = get_prop_check_min_len ( blob , node , prop_name ,
sizeof ( u32 ) * count , & err ) ;
if ( ! err ) {
for ( i = 0 ; i < count ; i + + )
array [ i ] = fdt32_to_cpu ( cell [ i ] ) ;
}
return err ;
}
2014-11-11 01:00:19 +00:00
int fdtdec_get_int_array_count ( const void * blob , int node ,
const char * prop_name , u32 * array , int count )
{
const u32 * cell ;
int len , elems ;
int i ;
debug ( " %s: %s \n " , __func__ , prop_name ) ;
cell = fdt_getprop ( blob , node , prop_name , & len ) ;
if ( ! cell )
return - FDT_ERR_NOTFOUND ;
elems = len / sizeof ( u32 ) ;
if ( count > elems )
count = elems ;
for ( i = 0 ; i < count ; i + + )
array [ i ] = fdt32_to_cpu ( cell [ i ] ) ;
return count ;
}
2012-04-02 13:18:41 +00:00
const u32 * fdtdec_locate_array ( const void * blob , int node ,
const char * prop_name , int count )
{
const u32 * cell ;
int err ;
cell = get_prop_check_min_len ( blob , node , prop_name ,
sizeof ( u32 ) * count , & err ) ;
return err ? NULL : cell ;
}
2012-02-27 10:52:35 +00:00
int fdtdec_get_bool ( const void * blob , int node , const char * prop_name )
{
const s32 * cell ;
int len ;
debug ( " %s: %s \n " , __func__ , prop_name ) ;
cell = fdt_getprop ( blob , node , prop_name , & len ) ;
return cell ! = NULL ;
}
2012-02-27 10:52:36 +00:00
2015-01-06 03:05:26 +00:00
int fdtdec_parse_phandle_with_args ( const void * blob , int src_node ,
const char * list_name ,
const char * cells_name ,
int cell_count , int index ,
struct fdtdec_phandle_args * out_args )
{
const __be32 * list , * list_end ;
int rc = 0 , size , cur_index = 0 ;
uint32_t count = 0 ;
int node = - 1 ;
int phandle ;
/* Retrieve the phandle list property */
list = fdt_getprop ( blob , src_node , list_name , & size ) ;
if ( ! list )
return - ENOENT ;
list_end = list + size / sizeof ( * list ) ;
/* Loop over the phandles until all the requested entry is found */
while ( list < list_end ) {
rc = - EINVAL ;
count = 0 ;
/*
* If phandle is 0 , then it is an empty entry with no
* arguments . Skip forward to the next entry .
*/
phandle = be32_to_cpup ( list + + ) ;
if ( phandle ) {
/*
* Find the provider node and parse the # * - cells
* property to determine the argument length .
*
* This is not needed if the cell count is hard - coded
* ( i . e . cells_name not set , but cell_count is set ) ,
* except when we ' re going to return the found node
* below .
*/
if ( cells_name | | cur_index = = index ) {
node = fdt_node_offset_by_phandle ( blob ,
phandle ) ;
if ( ! node ) {
debug ( " %s: could not find phandle \n " ,
fdt_get_name ( blob , src_node ,
NULL ) ) ;
goto err ;
}
}
if ( cells_name ) {
count = fdtdec_get_int ( blob , node , cells_name ,
- 1 ) ;
if ( count = = - 1 ) {
debug ( " %s: could not get %s for %s \n " ,
fdt_get_name ( blob , src_node ,
NULL ) ,
cells_name ,
fdt_get_name ( blob , node ,
NULL ) ) ;
goto err ;
}
} else {
count = cell_count ;
}
/*
* Make sure that the arguments actually fit in the
* remaining property data length
*/
if ( list + count > list_end ) {
debug ( " %s: arguments longer than property \n " ,
fdt_get_name ( blob , src_node , NULL ) ) ;
goto err ;
}
}
/*
* All of the error cases above bail out of the loop , so at
* this point , the parsing is successful . If the requested
* index matches , then fill the out_args structure and return ,
* or return - ENOENT for an empty entry .
*/
rc = - ENOENT ;
if ( cur_index = = index ) {
if ( ! phandle )
goto err ;
if ( out_args ) {
int i ;
if ( count > MAX_PHANDLE_ARGS ) {
debug ( " %s: too many arguments %d \n " ,
fdt_get_name ( blob , src_node ,
NULL ) , count ) ;
count = MAX_PHANDLE_ARGS ;
}
out_args - > node = node ;
out_args - > args_count = count ;
for ( i = 0 ; i < count ; i + + ) {
out_args - > args [ i ] =
be32_to_cpup ( list + + ) ;
}
}
/* Found it! return success */
return 0 ;
}
node = - 1 ;
list + = count ;
cur_index + + ;
}
/*
* Result will be one of :
* - ENOENT : index is for empty phandle
* - EINVAL : parsing error on data
* [ 1. . n ] : Number of phandle ( count mode ; when index = - 1 )
*/
rc = index < 0 ? cur_index : - ENOENT ;
err :
return rc ;
}
2012-04-17 09:01:28 +00:00
int fdtdec_get_byte_array ( const void * blob , int node , const char * prop_name ,
u8 * array , int count )
{
const u8 * cell ;
int err ;
cell = get_prop_check_min_len ( blob , node , prop_name , count , & err ) ;
if ( ! err )
memcpy ( array , cell , count ) ;
return err ;
}
const u8 * fdtdec_locate_byte_array ( const void * blob , int node ,
const char * prop_name , int count )
{
const u8 * cell ;
int err ;
cell = get_prop_check_min_len ( blob , node , prop_name , count , & err ) ;
if ( err )
return NULL ;
return cell ;
}
2012-10-25 16:30:58 +00:00
int fdtdec_get_config_int ( const void * blob , const char * prop_name ,
int default_val )
{
int config_node ;
debug ( " %s: %s \n " , __func__ , prop_name ) ;
config_node = fdt_path_offset ( blob , " /config " ) ;
if ( config_node < 0 )
return default_val ;
return fdtdec_get_int ( blob , config_node , prop_name , default_val ) ;
}
2012-10-25 16:30:59 +00:00
2012-10-25 16:31:04 +00:00
int fdtdec_get_config_bool ( const void * blob , const char * prop_name )
{
int config_node ;
const void * prop ;
debug ( " %s: %s \n " , __func__ , prop_name ) ;
config_node = fdt_path_offset ( blob , " /config " ) ;
if ( config_node < 0 )
return 0 ;
prop = fdt_get_property ( blob , config_node , prop_name , NULL ) ;
return prop ! = NULL ;
}
2012-10-25 16:30:59 +00:00
char * fdtdec_get_config_string ( const void * blob , const char * prop_name )
{
const char * nodep ;
int nodeoffset ;
int len ;
debug ( " %s: %s \n " , __func__ , prop_name ) ;
nodeoffset = fdt_path_offset ( blob , " /config " ) ;
if ( nodeoffset < 0 )
return NULL ;
nodep = fdt_getprop ( blob , nodeoffset , prop_name , & len ) ;
if ( ! nodep )
return NULL ;
return ( char * ) nodep ;
}
2012-10-25 16:31:00 +00:00
2014-10-24 00:58:51 +00:00
int fdtdec_decode_region ( const void * blob , int node , const char * prop_name ,
fdt_addr_t * basep , fdt_size_t * sizep )
2012-10-25 16:31:00 +00:00
{
const fdt_addr_t * cell ;
int len ;
2014-10-24 00:58:51 +00:00
debug ( " %s: %s: %s \n " , __func__ , fdt_get_name ( blob , node , NULL ) ,
prop_name ) ;
2012-10-25 16:31:00 +00:00
cell = fdt_getprop ( blob , node , prop_name , & len ) ;
2014-10-24 00:58:51 +00:00
if ( ! cell | | ( len < sizeof ( fdt_addr_t ) * 2 ) ) {
debug ( " cell=%p, len=%d \n " , cell , len ) ;
2012-10-25 16:31:00 +00:00
return - 1 ;
2014-10-24 00:58:51 +00:00
}
* basep = fdt_addr_to_cpu ( * cell ) ;
* sizep = fdt_size_to_cpu ( cell [ 1 ] ) ;
debug ( " %s: base=%08lx, size=%lx \n " , __func__ , ( ulong ) * basep ,
( ulong ) * sizep ) ;
2012-10-25 16:31:00 +00:00
return 0 ;
}
2014-02-27 20:26:01 +00:00
/**
* Read a flash entry from the fdt
*
* @ param blob FDT blob
* @ param node Offset of node to read
* @ param name Name of node being read
* @ param entry Place to put offset and size of this node
* @ return 0 if ok , - ve on error
*/
int fdtdec_read_fmap_entry ( const void * blob , int node , const char * name ,
struct fmap_entry * entry )
{
2014-10-24 00:58:52 +00:00
const char * prop ;
2014-02-27 20:26:01 +00:00
u32 reg [ 2 ] ;
if ( fdtdec_get_int_array ( blob , node , " reg " , reg , 2 ) ) {
debug ( " Node '%s' has bad/missing 'reg' property \n " , name ) ;
return - FDT_ERR_NOTFOUND ;
}
entry - > offset = reg [ 0 ] ;
entry - > length = reg [ 1 ] ;
2014-10-24 00:58:52 +00:00
entry - > used = fdtdec_get_int ( blob , node , " used " , entry - > length ) ;
prop = fdt_getprop ( blob , node , " compress " , NULL ) ;
entry - > compress_algo = prop & & ! strcmp ( prop , " lzo " ) ?
FMAP_COMPRESS_LZO : FMAP_COMPRESS_NONE ;
prop = fdt_getprop ( blob , node , " hash " , & entry - > hash_size ) ;
entry - > hash_algo = prop ? FMAP_HASH_SHA256 : FMAP_HASH_NONE ;
entry - > hash = ( uint8_t * ) prop ;
2014-02-27 20:26:01 +00:00
return 0 ;
}
2014-08-26 15:33:53 +00:00
2015-03-05 19:25:14 +00:00
u64 fdtdec_get_number ( const fdt32_t * ptr , unsigned int cells )
2014-08-26 15:33:53 +00:00
{
u64 number = 0 ;
while ( cells - - )
number = ( number < < 32 ) | fdt32_to_cpu ( * ptr + + ) ;
return number ;
}
int fdt_get_resource ( const void * fdt , int node , const char * property ,
unsigned int index , struct fdt_resource * res )
{
const fdt32_t * ptr , * end ;
int na , ns , len , parent ;
unsigned int i = 0 ;
parent = fdt_parent_offset ( fdt , node ) ;
if ( parent < 0 )
return parent ;
na = fdt_address_cells ( fdt , parent ) ;
ns = fdt_size_cells ( fdt , parent ) ;
ptr = fdt_getprop ( fdt , node , property , & len ) ;
if ( ! ptr )
return len ;
end = ptr + len / sizeof ( * ptr ) ;
while ( ptr + na + ns < = end ) {
if ( i = = index ) {
res - > start = res - > end = fdtdec_get_number ( ptr , na ) ;
res - > end + = fdtdec_get_number ( & ptr [ na ] , ns ) - 1 ;
return 0 ;
}
ptr + = na + ns ;
i + + ;
}
return - FDT_ERR_NOTFOUND ;
}
int fdt_get_named_resource ( const void * fdt , int node , const char * property ,
const char * prop_names , const char * name ,
struct fdt_resource * res )
{
int index ;
index = fdt_find_string ( fdt , node , prop_names , name ) ;
if ( index < 0 )
return index ;
return fdt_get_resource ( fdt , node , property , index , res ) ;
}
2014-08-26 15:33:54 +00:00
2014-10-24 00:58:56 +00:00
int fdtdec_decode_memory_region ( const void * blob , int config_node ,
const char * mem_type , const char * suffix ,
fdt_addr_t * basep , fdt_size_t * sizep )
{
char prop_name [ 50 ] ;
const char * mem ;
fdt_size_t size , offset_size ;
fdt_addr_t base , offset ;
int node ;
if ( config_node = = - 1 ) {
config_node = fdt_path_offset ( blob , " /config " ) ;
if ( config_node < 0 ) {
debug ( " %s: Cannot find /config node \n " , __func__ ) ;
return - ENOENT ;
}
}
if ( ! suffix )
suffix = " " ;
snprintf ( prop_name , sizeof ( prop_name ) , " %s-memory%s " , mem_type ,
suffix ) ;
mem = fdt_getprop ( blob , config_node , prop_name , NULL ) ;
if ( ! mem ) {
debug ( " %s: No memory type for '%s', using /memory \n " , __func__ ,
prop_name ) ;
mem = " /memory " ;
}
node = fdt_path_offset ( blob , mem ) ;
if ( node < 0 ) {
debug ( " %s: Failed to find node '%s': %s \n " , __func__ , mem ,
fdt_strerror ( node ) ) ;
return - ENOENT ;
}
/*
* Not strictly correct - the memory may have multiple banks . We just
* use the first
*/
if ( fdtdec_decode_region ( blob , node , " reg " , & base , & size ) ) {
debug ( " %s: Failed to decode memory region %s \n " , __func__ ,
mem ) ;
return - EINVAL ;
}
snprintf ( prop_name , sizeof ( prop_name ) , " %s-offset%s " , mem_type ,
suffix ) ;
if ( fdtdec_decode_region ( blob , config_node , prop_name , & offset ,
& offset_size ) ) {
debug ( " %s: Failed to decode memory region '%s' \n " , __func__ ,
prop_name ) ;
return - EINVAL ;
}
* basep = base + offset ;
* sizep = offset_size ;
return 0 ;
}
2015-02-28 05:06:34 +00:00
2015-04-15 03:03:21 +00:00
static int decode_timing_property ( const void * blob , int node , const char * name ,
struct timing_entry * result )
{
int length , ret = 0 ;
const u32 * prop ;
prop = fdt_getprop ( blob , node , name , & length ) ;
if ( ! prop ) {
debug ( " %s: could not find property %s \n " ,
fdt_get_name ( blob , node , NULL ) , name ) ;
return length ;
}
if ( length = = sizeof ( u32 ) ) {
result - > typ = fdtdec_get_int ( blob , node , name , 0 ) ;
result - > min = result - > typ ;
result - > max = result - > typ ;
} else {
ret = fdtdec_get_int_array ( blob , node , name , & result - > min , 3 ) ;
}
return ret ;
}
int fdtdec_decode_display_timing ( const void * blob , int parent , int index ,
struct display_timing * dt )
{
int i , node , timings_node ;
u32 val = 0 ;
int ret = 0 ;
timings_node = fdt_subnode_offset ( blob , parent , " display-timings " ) ;
if ( timings_node < 0 )
return timings_node ;
for ( i = 0 , node = fdt_first_subnode ( blob , timings_node ) ;
node > 0 & & i ! = index ;
node = fdt_next_subnode ( blob , node ) )
i + + ;
if ( node < 0 )
return node ;
memset ( dt , 0 , sizeof ( * dt ) ) ;
ret | = decode_timing_property ( blob , node , " hback-porch " ,
& dt - > hback_porch ) ;
ret | = decode_timing_property ( blob , node , " hfront-porch " ,
& dt - > hfront_porch ) ;
ret | = decode_timing_property ( blob , node , " hactive " , & dt - > hactive ) ;
ret | = decode_timing_property ( blob , node , " hsync-len " , & dt - > hsync_len ) ;
ret | = decode_timing_property ( blob , node , " vback-porch " ,
& dt - > vback_porch ) ;
ret | = decode_timing_property ( blob , node , " vfront-porch " ,
& dt - > vfront_porch ) ;
ret | = decode_timing_property ( blob , node , " vactive " , & dt - > vactive ) ;
ret | = decode_timing_property ( blob , node , " vsync-len " , & dt - > vsync_len ) ;
ret | = decode_timing_property ( blob , node , " clock-frequency " ,
& dt - > pixelclock ) ;
dt - > flags = 0 ;
val = fdtdec_get_int ( blob , node , " vsync-active " , - 1 ) ;
if ( val ! = - 1 ) {
dt - > flags | = val ? DISPLAY_FLAGS_VSYNC_HIGH :
DISPLAY_FLAGS_VSYNC_LOW ;
}
val = fdtdec_get_int ( blob , node , " hsync-active " , - 1 ) ;
if ( val ! = - 1 ) {
dt - > flags | = val ? DISPLAY_FLAGS_HSYNC_HIGH :
DISPLAY_FLAGS_HSYNC_LOW ;
}
val = fdtdec_get_int ( blob , node , " de-active " , - 1 ) ;
if ( val ! = - 1 ) {
dt - > flags | = val ? DISPLAY_FLAGS_DE_HIGH :
DISPLAY_FLAGS_DE_LOW ;
}
val = fdtdec_get_int ( blob , node , " pixelclk-active " , - 1 ) ;
if ( val ! = - 1 ) {
dt - > flags | = val ? DISPLAY_FLAGS_PIXDATA_POSEDGE :
DISPLAY_FLAGS_PIXDATA_NEGEDGE ;
}
if ( fdtdec_get_bool ( blob , node , " interlaced " ) )
dt - > flags | = DISPLAY_FLAGS_INTERLACED ;
if ( fdtdec_get_bool ( blob , node , " doublescan " ) )
dt - > flags | = DISPLAY_FLAGS_DOUBLESCAN ;
if ( fdtdec_get_bool ( blob , node , " doubleclk " ) )
dt - > flags | = DISPLAY_FLAGS_DOUBLECLK ;
return 0 ;
}
2015-02-28 05:06:35 +00:00
int fdtdec_setup ( void )
2015-02-28 05:06:34 +00:00
{
2015-08-11 22:31:55 +00:00
# if CONFIG_IS_ENABLED(OF_CONTROL)
2015-02-28 05:06:34 +00:00
# ifdef CONFIG_OF_EMBED
/* Get a pointer to the FDT */
gd - > fdt_blob = __dtb_dt_begin ;
# elif defined CONFIG_OF_SEPARATE
# ifdef CONFIG_SPL_BUILD
/* FDT is at end of BSS */
gd - > fdt_blob = ( ulong * ) & __bss_end ;
# else
/* FDT is at end of image */
gd - > fdt_blob = ( ulong * ) & _end ;
2015-08-01 07:03:25 +00:00
# endif
2015-02-28 05:06:34 +00:00
# elif defined(CONFIG_OF_HOSTFILE)
if ( sandbox_read_fdt_from_file ( ) ) {
puts ( " Failed to read control FDT \n " ) ;
return - 1 ;
}
# endif
# ifndef CONFIG_SPL_BUILD
/* Allow the early environment to override the fdt address */
gd - > fdt_blob = ( void * ) getenv_ulong ( " fdtcontroladdr " , 16 ,
( uintptr_t ) gd - > fdt_blob ) ;
# endif
2014-03-03 11:19:30 +00:00
# endif
2015-02-28 05:06:35 +00:00
return fdtdec_prepare_fdt ( ) ;
2015-02-28 05:06:34 +00:00
}
# endif /* !USE_HOSTCC */