| /* |
| * Copyright (c) 2013 Qualcomm Atheros, Inc. |
| * |
| * (C) Copyright 2000-2002 |
| * Wolfgang Denk, DENX Software Engineering, wd@denx.de. |
| * |
| * (C) Copyright 2001 Sysgo Real-Time Solutions, GmbH <www.elinos.com> |
| * Andreas Heppel <aheppel@sysgo.de> |
| * |
| * See file CREDITS for list of people who contributed to this |
| * project. |
| * |
| * This program is free software; you can redistribute it and/or |
| * modify it under the terms of the GNU General Public License as |
| * published by the Free Software Foundation; either version 2 of |
| * the License, or (at your option) any later version. |
| * |
| * This program is distributed in the hope that it will be useful, |
| * but WITHOUT ANY WARRANTY; without even the implied warranty of |
| * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the |
| * GNU General Public License for more details. |
| * |
| * You should have received a copy of the GNU General Public License |
| * along with this program; if not, write to the Free Software |
| * Foundation, Inc., 59 Temple Place, Suite 330, Boston, |
| * MA 02111-1307 USA |
| */ |
| |
| #include <common.h> |
| #include <command.h> |
| #include <environment.h> |
| #include <linux/stddef.h> |
| #include <malloc.h> |
| |
| #ifdef CONFIG_SHOW_BOOT_PROGRESS |
| # include <status_led.h> |
| # define SHOW_BOOT_PROGRESS(arg) show_boot_progress(arg) |
| #else |
| # define SHOW_BOOT_PROGRESS(arg) |
| #endif |
| |
| DECLARE_GLOBAL_DATA_PTR; |
| |
| #ifdef CONFIG_AMIGAONEG3SE |
| extern void enable_nvram(void); |
| extern void disable_nvram(void); |
| #endif |
| |
| #undef DEBUG_ENV |
| #ifdef DEBUG_ENV |
| #define DEBUGF(fmt,args...) printf(fmt ,##args) |
| #else |
| #define DEBUGF(fmt,args...) |
| #endif |
| |
| extern env_t *env_ptr; |
| |
| extern void env_relocate_spec (void); |
| extern uchar env_get_char_spec(int); |
| |
| static uchar env_get_char_init (int index); |
| uchar (*env_get_char)(int) = env_get_char_init; |
| |
| /************************************************************************ |
| * Default settings to be used when no valid environment is found |
| */ |
| #define XMK_STR(x) #x |
| #define MK_STR(x) XMK_STR(x) |
| |
| uchar default_environment[] = { |
| #ifdef CONFIG_BOOTARGS |
| "bootargs=" CONFIG_BOOTARGS "\0" |
| #endif |
| #ifdef CONFIG_BOOTCOMMAND |
| "bootcmd=" CONFIG_BOOTCOMMAND "\0" |
| #endif |
| #ifdef CONFIG_RAMBOOTCOMMAND |
| "ramboot=" CONFIG_RAMBOOTCOMMAND "\0" |
| #endif |
| #ifdef CONFIG_NFSBOOTCOMMAND |
| "nfsboot=" CONFIG_NFSBOOTCOMMAND "\0" |
| #endif |
| #if defined(CONFIG_BOOTDELAY) && (CONFIG_BOOTDELAY >= 0) |
| "bootdelay=" MK_STR(CONFIG_BOOTDELAY) "\0" |
| #endif |
| #if defined(CONFIG_BAUDRATE) && (CONFIG_BAUDRATE >= 0) |
| "baudrate=" MK_STR(CONFIG_BAUDRATE) "\0" |
| #endif |
| #ifdef CONFIG_LOADS_ECHO |
| "loads_echo=" MK_STR(CONFIG_LOADS_ECHO) "\0" |
| #endif |
| #ifdef CONFIG_ETHADDR |
| "ethaddr=" MK_STR(CONFIG_ETHADDR) "\0" |
| #endif |
| #ifdef CONFIG_ETH1ADDR |
| "eth1addr=" MK_STR(CONFIG_ETH1ADDR) "\0" |
| #endif |
| #ifdef CONFIG_ETH2ADDR |
| "eth2addr=" MK_STR(CONFIG_ETH2ADDR) "\0" |
| #endif |
| #ifdef CONFIG_ETH3ADDR |
| "eth3addr=" MK_STR(CONFIG_ETH3ADDR) "\0" |
| #endif |
| #ifdef CONFIG_IPADDR |
| "ipaddr=" MK_STR(CONFIG_IPADDR) "\0" |
| #endif |
| #ifdef CONFIG_SERVERIP |
| "serverip=" MK_STR(CONFIG_SERVERIP) "\0" |
| #endif |
| #ifdef CFG_AUTOLOAD |
| "autoload=" CFG_AUTOLOAD "\0" |
| #endif |
| #ifdef CONFIG_PREBOOT |
| "preboot=" CONFIG_PREBOOT "\0" |
| #endif |
| #ifdef CONFIG_ROOTPATH |
| "rootpath=" MK_STR(CONFIG_ROOTPATH) "\0" |
| #endif |
| #ifdef CONFIG_GATEWAYIP |
| "gatewayip=" MK_STR(CONFIG_GATEWAYIP) "\0" |
| #endif |
| #ifdef CONFIG_NETMASK |
| "netmask=" MK_STR(CONFIG_NETMASK) "\0" |
| #endif |
| #ifdef CONFIG_HOSTNAME |
| "hostname=" MK_STR(CONFIG_HOSTNAME) "\0" |
| #endif |
| #ifdef CONFIG_BOOTFILE |
| "bootfile=" MK_STR(CONFIG_BOOTFILE) "\0" |
| #endif |
| #ifdef CONFIG_LOADADDR |
| "loadaddr=" MK_STR(CONFIG_LOADADDR) "\0" |
| #endif |
| #ifdef CONFIG_CLOCKS_IN_MHZ |
| "clocks_in_mhz=1\0" |
| #endif |
| #if defined(CONFIG_PCI_BOOTDELAY) && (CONFIG_PCI_BOOTDELAY > 0) |
| "pcidelay=" MK_STR(CONFIG_PCI_BOOTDELAY) "\0" |
| #endif |
| #ifdef CONFIG_EXTRA_ENV_SETTINGS |
| CONFIG_EXTRA_ENV_SETTINGS |
| #endif |
| "\0" |
| }; |
| |
| #if defined(CFG_ENV_IS_IN_NAND) /* Environment is in Nand Flash */ |
| int default_environment_size = sizeof(default_environment); |
| #endif |
| |
| void env_crc_update (void) |
| { |
| env_ptr->crc = crc32(0, env_ptr->data, ENV_SIZE); |
| } |
| |
| static uchar env_get_char_init (int index) |
| { |
| uchar c; |
| |
| /* if crc was bad, use the default environment */ |
| if (gd->env_valid) |
| { |
| c = env_get_char_spec(index); |
| } else { |
| c = default_environment[index]; |
| } |
| |
| return (c); |
| } |
| |
| #ifdef CONFIG_AMIGAONEG3SE |
| uchar env_get_char_memory (int index) |
| { |
| uchar retval; |
| enable_nvram(); |
| if (gd->env_valid) { |
| retval = ( *((uchar *)(gd->env_addr + index)) ); |
| } else { |
| retval = ( default_environment[index] ); |
| } |
| disable_nvram(); |
| return retval; |
| } |
| #else |
| uchar env_get_char_memory (int index) |
| { |
| if (gd->env_valid) { |
| return ( *((uchar *)(gd->env_addr + index)) ); |
| } else { |
| return ( default_environment[index] ); |
| } |
| } |
| #endif |
| |
| uchar *env_get_addr (int index) |
| { |
| if (gd->env_valid) { |
| return ( ((uchar *)(gd->env_addr + index)) ); |
| } else { |
| return (&default_environment[index]); |
| } |
| } |
| |
| void env_relocate (void) |
| { |
| DEBUGF ("%s[%d] offset = 0x%lx\n", __FUNCTION__,__LINE__, |
| gd->reloc_off); |
| |
| #ifdef CONFIG_AMIGAONEG3SE |
| enable_nvram(); |
| #endif |
| |
| #if defined(ENV_IS_EMBEDDED) |
| /* |
| * The environment buffer is embedded with the text segment, |
| * just relocate the environment pointer |
| */ |
| env_ptr = (env_t *)((ulong)env_ptr + gd->reloc_off); |
| DEBUGF ("%s[%d] embedded ENV at %p\n", __FUNCTION__,__LINE__,env_ptr); |
| #else |
| /* |
| * We must allocate a buffer for the environment |
| */ |
| env_ptr = (env_t *)malloc (CFG_ENV_SIZE); |
| DEBUGF ("%s[%d] malloced ENV at %p\n", __FUNCTION__,__LINE__,env_ptr); |
| #endif |
| |
| /* |
| * After relocation to RAM, we can always use the "memory" functions |
| */ |
| env_get_char = env_get_char_memory; |
| |
| if (gd->env_valid == 0) { |
| #if defined(CONFIG_GTH) || defined(CFG_ENV_IS_NOWHERE) /* Environment not changable */ |
| puts ("Using default environment\n\n"); |
| #else |
| puts ("*** Warning - bad CRC, using default environment\n\n"); |
| SHOW_BOOT_PROGRESS (-1); |
| #endif |
| |
| if (sizeof(default_environment) > ENV_SIZE) |
| { |
| puts ("*** Error - default environment is too large\n\n"); |
| return; |
| } |
| |
| memset (env_ptr, 0, sizeof(env_t)); |
| memcpy (env_ptr->data, |
| default_environment, |
| sizeof(default_environment)); |
| #ifdef CFG_REDUNDAND_ENVIRONMENT |
| env_ptr->flags = 0xFF; |
| #endif |
| env_crc_update (); |
| gd->env_valid = 1; |
| } |
| else { |
| env_relocate_spec (); |
| } |
| gd->env_addr = (ulong)&(env_ptr->data); |
| |
| #ifdef CONFIG_AMIGAONEG3SE |
| disable_nvram(); |
| #endif |
| } |
| |
| #ifdef CONFIG_AUTO_COMPLETE |
| int env_complete(char *var, int maxv, char *cmdv[], int bufsz, char *buf) |
| { |
| int i, nxt, len, vallen, found; |
| const char *lval, *rval; |
| |
| found = 0; |
| cmdv[0] = NULL; |
| |
| len = strlen(var); |
| /* now iterate over the variables and select those that match */ |
| for (i=0; env_get_char(i) != '\0'; i=nxt+1) { |
| |
| for (nxt=i; env_get_char(nxt) != '\0'; ++nxt) |
| ; |
| |
| lval = (char *)env_get_addr(i); |
| rval = strchr(lval, '='); |
| if (rval != NULL) { |
| vallen = rval - lval; |
| rval++; |
| } else |
| vallen = strlen(lval); |
| |
| if (len > 0 && (vallen < len || memcmp(lval, var, len) != 0)) |
| continue; |
| |
| if (found >= maxv - 2 || bufsz < vallen + 1) { |
| cmdv[found++] = "..."; |
| break; |
| } |
| cmdv[found++] = buf; |
| memcpy(buf, lval, vallen); buf += vallen; bufsz -= vallen; |
| *buf++ = '\0'; bufsz--; |
| } |
| |
| cmdv[found] = NULL; |
| return found; |
| } |
| #endif |
| |
| int env_restore_defaults(void) |
| { |
| int i; |
| const uchar delim = '='; |
| const int default_env_size = sizeof(default_environment) - 1; |
| char *s; |
| uchar *key = NULL; |
| uchar *val = NULL; |
| uchar *tmp_null = NULL; |
| for (i = 0; i < default_env_size; ++i) { |
| if (key == NULL) { |
| key = default_environment + i; |
| } |
| /* If delim is encountered, we know to check for key/val. */ |
| if (default_environment[i] == delim) { |
| /* Ignore empty keys. */ |
| if (key == (default_environment + i)) { |
| return -1; |
| } |
| /* |
| * Does in-place changing of null terminator to avoid |
| * extra buffers. |
| */ |
| tmp_null = default_environment + i; |
| val = tmp_null + 1; |
| *tmp_null = '\0'; |
| s = getenv(key); |
| if (!s) { |
| setenv(key, val); |
| } |
| *tmp_null = delim; |
| /* |
| * Advances pointer until we encounter the end of the |
| * 'val' string. |
| */ |
| for ( ; i < default_env_size; ++i) { |
| if (default_environment[i] == '\0') { |
| break; |
| } |
| } |
| key = NULL; |
| } |
| } |
| if (key != NULL && *key != '\0') { |
| printf("%s: incomplete parsing while restoring env.\n", __FILE__); |
| printf("\tPlease check default_environment var\n"); |
| return -1; |
| } |
| return 0; |
| } |