310 lines
7.1 KiB
C
310 lines
7.1 KiB
C
|
/*
|
||
|
* (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
|
||
|
|
||
|
#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)
|
||
|
{
|
||
|
DECLARE_GLOBAL_DATA_PTR;
|
||
|
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)
|
||
|
{
|
||
|
DECLARE_GLOBAL_DATA_PTR;
|
||
|
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)
|
||
|
{
|
||
|
DECLARE_GLOBAL_DATA_PTR;
|
||
|
|
||
|
if (gd->env_valid) {
|
||
|
return ( *((uchar *)(gd->env_addr + index)) );
|
||
|
} else {
|
||
|
return ( default_environment[index] );
|
||
|
}
|
||
|
}
|
||
|
#endif
|
||
|
|
||
|
uchar *env_get_addr (int index)
|
||
|
{
|
||
|
DECLARE_GLOBAL_DATA_PTR;
|
||
|
|
||
|
if (gd->env_valid) {
|
||
|
return ( ((uchar *)(gd->env_addr + index)) );
|
||
|
} else {
|
||
|
return (&default_environment[index]);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
void env_relocate (void)
|
||
|
{
|
||
|
DECLARE_GLOBAL_DATA_PTR;
|
||
|
|
||
|
DEBUGF ("%s[%d] offset = 0x%lx\n", __FUNCTION__,__LINE__,
|
||
|
gd->reloc_off);
|
||
|
|
||
|
#ifdef CONFIG_AMIGAONEG3SE
|
||
|
enable_nvram();
|
||
|
#endif
|
||
|
|
||
|
#ifdef 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
|