2010-06-08 19:06:01 +02:00
/*
* BCM947xx nvram variable access
*
* Copyright ( C ) 2005 Broadcom Corporation
* Copyright ( C ) 2006 Felix Fietkau < nbd @ openwrt . org >
2012-12-26 19:51:13 +00:00
* Copyright ( C ) 2010 - 2012 Hauke Mehrtens < hauke @ hauke - m . de >
2010-06-08 19:06:01 +02:00
*
2013-01-22 12:59:30 +01:00
* This program is free software ; you can redistribute it and / or modify it
* under the terms of the GNU General Public License as published by the
2010-06-08 19:06:01 +02:00
* Free Software Foundation ; either version 2 of the License , or ( at your
* option ) any later version .
*/
# include <linux/types.h>
# include <linux/module.h>
# include <linux/kernel.h>
# include <linux/string.h>
2014-10-29 10:05:06 +01:00
# include <linux/mtd/mtd.h>
2014-12-01 07:58:18 +01:00
# include <linux/bcm47xx_nvram.h>
2010-06-08 19:06:01 +02:00
2014-12-10 17:38:26 +01:00
# define NVRAM_MAGIC 0x48534C46 /* 'FLSH' */
# define NVRAM_SPACE 0x8000
# define NVRAM_MAX_GPIO_ENTRIES 32
# define NVRAM_MAX_GPIO_VALUE_LEN 30
2014-10-30 12:50:03 +01:00
# define FLASH_MIN 0x00020000 /* Minimum flash size */
struct nvram_header {
u32 magic ;
u32 len ;
u32 crc_ver_init ; /* 0:7 crc, 8:15 ver, 16:31 sdram_init */
u32 config_refresh ; /* 0:15 sdram_config, 16:31 sdram_refresh */
u32 config_ncdl ; /* ncdl values for memc */
} ;
2010-06-08 19:06:01 +02:00
static char nvram_buf [ NVRAM_SPACE ] ;
2013-10-13 22:56:50 +02:00
static const u32 nvram_sizes [ ] = { 0x8000 , 0xF000 , 0x10000 } ;
2010-06-08 19:06:01 +02:00
2014-09-03 22:51:06 +02:00
static u32 find_nvram_size ( void __iomem * end )
2012-12-26 19:51:13 +00:00
{
2014-09-03 22:51:06 +02:00
struct nvram_header __iomem * header ;
2012-12-26 19:51:13 +00:00
int i ;
for ( i = 0 ; i < ARRAY_SIZE ( nvram_sizes ) ; i + + ) {
2014-09-03 22:51:06 +02:00
header = ( struct nvram_header * ) ( end - nvram_sizes [ i ] ) ;
2014-10-30 12:50:03 +01:00
if ( header - > magic = = NVRAM_MAGIC )
2012-12-26 19:51:13 +00:00
return nvram_sizes [ i ] ;
}
return 0 ;
}
2010-06-08 19:06:01 +02:00
/* Probe for NVRAM header */
2014-09-03 22:51:06 +02:00
static int nvram_find_and_copy ( void __iomem * iobase , u32 lim )
2010-06-08 19:06:01 +02:00
{
2014-09-03 22:51:06 +02:00
struct nvram_header __iomem * header ;
2010-06-08 19:06:01 +02:00
int i ;
2011-07-23 01:20:12 +02:00
u32 off ;
2010-06-08 19:06:01 +02:00
u32 * src , * dst ;
2012-12-26 19:51:13 +00:00
u32 size ;
2010-06-08 19:06:01 +02:00
2014-09-03 22:51:06 +02:00
if ( nvram_buf [ 0 ] ) {
pr_warn ( " nvram already initialized \n " ) ;
return - EEXIST ;
}
2012-12-26 19:51:11 +00:00
/* TODO: when nvram is on nand flash check for bad blocks first. */
2010-06-08 19:06:01 +02:00
off = FLASH_MIN ;
while ( off < = lim ) {
/* Windowed flash access */
2014-09-03 22:51:06 +02:00
size = find_nvram_size ( iobase + off ) ;
2012-12-26 19:51:13 +00:00
if ( size ) {
2014-09-03 22:51:06 +02:00
header = ( struct nvram_header * ) ( iobase + off - size ) ;
2010-06-08 19:06:01 +02:00
goto found ;
2012-12-26 19:51:13 +00:00
}
2010-06-08 19:06:01 +02:00
off < < = 1 ;
}
/* Try embedded NVRAM at 4 KB and 1 KB as last resorts */
2014-09-03 22:51:06 +02:00
header = ( struct nvram_header * ) ( iobase + 4096 ) ;
2014-10-30 12:50:03 +01:00
if ( header - > magic = = NVRAM_MAGIC ) {
2012-12-26 19:51:13 +00:00
size = NVRAM_SPACE ;
2010-06-08 19:06:01 +02:00
goto found ;
2012-12-26 19:51:13 +00:00
}
2010-06-08 19:06:01 +02:00
2014-09-03 22:51:06 +02:00
header = ( struct nvram_header * ) ( iobase + 1024 ) ;
2014-10-30 12:50:03 +01:00
if ( header - > magic = = NVRAM_MAGIC ) {
2012-12-26 19:51:13 +00:00
size = NVRAM_SPACE ;
2010-06-08 19:06:01 +02:00
goto found ;
2012-12-26 19:51:13 +00:00
}
2010-06-08 19:06:01 +02:00
2012-12-26 19:51:13 +00:00
pr_err ( " no nvram found \n " ) ;
2012-12-26 19:51:10 +00:00
return - ENXIO ;
2010-06-08 19:06:01 +02:00
found :
2012-12-26 19:51:13 +00:00
if ( header - > len > size )
pr_err ( " The nvram size accoridng to the header seems to be bigger than the partition on flash \n " ) ;
if ( header - > len > NVRAM_SPACE )
pr_err ( " nvram on flash (%i bytes) is bigger than the reserved space in memory, will just copy the first %i bytes \n " ,
header - > len , NVRAM_SPACE ) ;
2014-12-10 17:38:26 +01:00
src = ( u32 * ) header ;
dst = ( u32 * ) nvram_buf ;
2010-06-08 19:06:01 +02:00
for ( i = 0 ; i < sizeof ( struct nvram_header ) ; i + = 4 )
2014-12-10 11:49:53 +01:00
* dst + + = __raw_readl ( src + + ) ;
2012-12-26 19:51:13 +00:00
for ( ; i < header - > len & & i < NVRAM_SPACE & & i < size ; i + = 4 )
2014-12-10 11:49:53 +01:00
* dst + + = readl ( src + + ) ;
2012-12-26 19:51:10 +00:00
return 0 ;
2010-06-08 19:06:01 +02:00
}
2014-09-03 22:59:45 +02:00
/*
* On bcm47xx we need access to the NVRAM very early , so we can ' t use mtd
* subsystem to access flash . We can ' t even use platform device / driver to
* store memory offset .
* To handle this we provide following symbol . It ' s supposed to be called as
* soon as we get info about flash device , before any NVRAM entry is needed .
*/
int bcm47xx_nvram_init_from_mem ( u32 base , u32 lim )
2014-09-03 22:51:06 +02:00
{
void __iomem * iobase ;
int err ;
iobase = ioremap_nocache ( base , lim ) ;
if ( ! iobase )
return - ENOMEM ;
err = nvram_find_and_copy ( iobase , lim ) ;
iounmap ( iobase ) ;
return err ;
}
2012-12-26 19:51:12 +00:00
static int nvram_init ( void )
2012-12-26 08:29:17 +00:00
{
2014-10-29 10:05:06 +01:00
# ifdef CONFIG_MTD
struct mtd_info * mtd ;
struct nvram_header header ;
size_t bytes_read ;
int err , i ;
mtd = get_mtd_device_nm ( " nvram " ) ;
if ( IS_ERR ( mtd ) )
return - ENODEV ;
for ( i = 0 ; i < ARRAY_SIZE ( nvram_sizes ) ; i + + ) {
loff_t from = mtd - > size - nvram_sizes [ i ] ;
if ( from < 0 )
continue ;
err = mtd_read ( mtd , from , sizeof ( header ) , & bytes_read ,
( uint8_t * ) & header ) ;
2014-10-30 12:50:03 +01:00
if ( ! err & & header . magic = = NVRAM_MAGIC ) {
2014-10-29 10:05:06 +01:00
u8 * dst = ( uint8_t * ) nvram_buf ;
size_t len = header . len ;
if ( header . len > NVRAM_SPACE ) {
pr_err ( " nvram on flash (%i bytes) is bigger than the reserved space in memory, will just copy the first %i bytes \n " ,
header . len , NVRAM_SPACE ) ;
len = NVRAM_SPACE ;
}
err = mtd_read ( mtd , from , len , & bytes_read , dst ) ;
if ( err )
return err ;
return 0 ;
}
}
# endif
2014-10-28 13:30:23 +01:00
2012-12-26 19:51:10 +00:00
return - ENXIO ;
2012-12-26 08:29:17 +00:00
}
2014-10-30 12:50:03 +01:00
int bcm47xx_nvram_getenv ( const char * name , char * val , size_t val_len )
2010-06-08 19:06:01 +02:00
{
char * var , * value , * end , * eq ;
2014-12-10 11:49:54 +01:00
int data_left , err ;
2010-06-08 19:06:01 +02:00
if ( ! name )
2012-12-26 19:51:09 +00:00
return - EINVAL ;
2010-06-08 19:06:01 +02:00
2012-12-26 19:51:10 +00:00
if ( ! nvram_buf [ 0 ] ) {
2012-12-26 19:51:12 +00:00
err = nvram_init ( ) ;
2012-12-26 19:51:10 +00:00
if ( err )
return err ;
}
2010-06-08 19:06:01 +02:00
/* Look for name=value and return value */
var = & nvram_buf [ sizeof ( struct nvram_header ) ] ;
end = nvram_buf + sizeof ( nvram_buf ) - 2 ;
2014-12-10 17:38:26 +01:00
end [ 0 ] = ' \0 ' ;
end [ 1 ] = ' \0 ' ;
2010-06-08 19:06:01 +02:00
for ( ; * var ; var = value + strlen ( value ) + 1 ) {
2014-12-10 11:49:54 +01:00
data_left = end - var ;
eq = strnchr ( var , data_left , ' = ' ) ;
2010-06-08 19:06:01 +02:00
if ( ! eq )
break ;
value = eq + 1 ;
2014-12-10 17:38:26 +01:00
if ( eq - var = = strlen ( name ) & &
strncmp ( var , name , eq - var ) = = 0 )
2012-02-28 00:56:11 +01:00
return snprintf ( val , val_len , " %s " , value ) ;
2010-06-08 19:06:01 +02:00
}
2012-12-26 19:51:09 +00:00
return - ENOENT ;
2010-06-08 19:06:01 +02:00
}
2012-12-26 19:51:14 +00:00
EXPORT_SYMBOL ( bcm47xx_nvram_getenv ) ;
2013-09-18 13:31:15 +02:00
int bcm47xx_nvram_gpio_pin ( const char * name )
{
int i , err ;
2014-12-10 17:38:26 +01:00
char nvram_var [ ] = " gpioXX " ;
char buf [ NVRAM_MAX_GPIO_VALUE_LEN ] ;
2013-09-18 13:31:15 +02:00
2014-12-10 17:38:26 +01:00
/* TODO: Optimize it to don't call getenv so many times */
for ( i = 0 ; i < NVRAM_MAX_GPIO_ENTRIES ; i + + ) {
2013-09-18 13:31:15 +02:00
err = snprintf ( nvram_var , sizeof ( nvram_var ) , " gpio%i " , i ) ;
if ( err < = 0 )
continue ;
err = bcm47xx_nvram_getenv ( nvram_var , buf , sizeof ( buf ) ) ;
if ( err < = 0 )
continue ;
if ( ! strcmp ( name , buf ) )
return i ;
}
return - ENOENT ;
}
EXPORT_SYMBOL ( bcm47xx_nvram_gpio_pin ) ;