16aff3115Swdenk /* 2*bc11756dSGrant Erickson * (C) Copyright 2000-2008 36aff3115Swdenk * Wolfgang Denk, DENX Software Engineering, wd@denx.de. 46aff3115Swdenk * 56aff3115Swdenk * See file CREDITS for list of people who contributed to this 66aff3115Swdenk * project. 76aff3115Swdenk * 86aff3115Swdenk * This program is free software; you can redistribute it and/or 96aff3115Swdenk * modify it under the terms of the GNU General Public License as 106aff3115Swdenk * published by the Free Software Foundation; either version 2 of 116aff3115Swdenk * the License, or (at your option) any later version. 126aff3115Swdenk * 136aff3115Swdenk * This program is distributed in the hope that it will be useful, 146aff3115Swdenk * but WITHOUT ANY WARRANTY; without even the implied warranty of 156aff3115Swdenk * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the 166aff3115Swdenk * GNU General Public License for more details. 176aff3115Swdenk * 186aff3115Swdenk * You should have received a copy of the GNU General Public License 196aff3115Swdenk * along with this program; if not, write to the Free Software 206aff3115Swdenk * Foundation, Inc., 59 Temple Place, Suite 330, Boston, 216aff3115Swdenk * MA 02111-1307 USA 226aff3115Swdenk */ 236aff3115Swdenk 246aff3115Swdenk #include <errno.h> 256aff3115Swdenk #include <fcntl.h> 266aff3115Swdenk #include <stdio.h> 276aff3115Swdenk #include <stdlib.h> 286aff3115Swdenk #include <stddef.h> 296aff3115Swdenk #include <string.h> 306aff3115Swdenk #include <sys/types.h> 316aff3115Swdenk #include <sys/ioctl.h> 326aff3115Swdenk #include <sys/stat.h> 336aff3115Swdenk #include <unistd.h> 346aff3115Swdenk 356de66b35SMarkus Klotzbücher #ifdef MTD_OLD 366de66b35SMarkus Klotzbücher # include <linux/mtd/mtd.h> 376de66b35SMarkus Klotzbücher #else 38c83d7ca4SWolfgang Denk # define __user /* nothing */ 396de66b35SMarkus Klotzbücher # include <mtd/mtd-user.h> 406de66b35SMarkus Klotzbücher #endif 416de66b35SMarkus Klotzbücher 426de66b35SMarkus Klotzbücher #include "fw_env.h" 436aff3115Swdenk 446aff3115Swdenk #define CMD_GETENV "fw_printenv" 456aff3115Swdenk #define CMD_SETENV "fw_setenv" 466aff3115Swdenk 476aff3115Swdenk typedef struct envdev_s { 486de66b35SMarkus Klotzbücher char devname[16]; /* Device name */ 49d0fb80c3Swdenk ulong devoff; /* Device offset */ 506aff3115Swdenk ulong env_size; /* environment size */ 516aff3115Swdenk ulong erase_size; /* device erase size */ 526aff3115Swdenk } envdev_t; 536aff3115Swdenk 546aff3115Swdenk static envdev_t envdevices[2]; 556aff3115Swdenk static int curdev; 566aff3115Swdenk 576aff3115Swdenk #define DEVNAME(i) envdevices[(i)].devname 58d0fb80c3Swdenk #define DEVOFFSET(i) envdevices[(i)].devoff 596aff3115Swdenk #define ENVSIZE(i) envdevices[(i)].env_size 606aff3115Swdenk #define DEVESIZE(i) envdevices[(i)].erase_size 616aff3115Swdenk 626aff3115Swdenk #define CFG_ENV_SIZE ENVSIZE(curdev) 636aff3115Swdenk 64d0fb80c3Swdenk #define ENV_SIZE getenvsize() 656aff3115Swdenk 666aff3115Swdenk typedef struct environment_s { 676aff3115Swdenk ulong crc; /* CRC32 over data bytes */ 686de66b35SMarkus Klotzbücher unsigned char flags; /* active or obsolete */ 696de66b35SMarkus Klotzbücher char *data; 706aff3115Swdenk } env_t; 716aff3115Swdenk 726aff3115Swdenk static env_t environment; 736aff3115Swdenk 74d0fb80c3Swdenk static int HaveRedundEnv = 0; 75d0fb80c3Swdenk 766de66b35SMarkus Klotzbücher static unsigned char active_flag = 1; 776de66b35SMarkus Klotzbücher static unsigned char obsolete_flag = 0; 78d0fb80c3Swdenk 796aff3115Swdenk 806aff3115Swdenk #define XMK_STR(x) #x 816aff3115Swdenk #define MK_STR(x) XMK_STR(x) 826aff3115Swdenk 836de66b35SMarkus Klotzbücher static char default_environment[] = { 84d0fb80c3Swdenk #if defined(CONFIG_BOOTARGS) 856aff3115Swdenk "bootargs=" CONFIG_BOOTARGS "\0" 866aff3115Swdenk #endif 87d0fb80c3Swdenk #if defined(CONFIG_BOOTCOMMAND) 886aff3115Swdenk "bootcmd=" CONFIG_BOOTCOMMAND "\0" 896aff3115Swdenk #endif 90d0fb80c3Swdenk #if defined(CONFIG_RAMBOOTCOMMAND) 91d0fb80c3Swdenk "ramboot=" CONFIG_RAMBOOTCOMMAND "\0" 92d0fb80c3Swdenk #endif 93d0fb80c3Swdenk #if defined(CONFIG_NFSBOOTCOMMAND) 94d0fb80c3Swdenk "nfsboot=" CONFIG_NFSBOOTCOMMAND "\0" 95d0fb80c3Swdenk #endif 96d0fb80c3Swdenk #if defined(CONFIG_BOOTDELAY) && (CONFIG_BOOTDELAY >= 0) 976aff3115Swdenk "bootdelay=" MK_STR (CONFIG_BOOTDELAY) "\0" 986aff3115Swdenk #endif 99d0fb80c3Swdenk #if defined(CONFIG_BAUDRATE) && (CONFIG_BAUDRATE >= 0) 1006aff3115Swdenk "baudrate=" MK_STR (CONFIG_BAUDRATE) "\0" 1016aff3115Swdenk #endif 102d0fb80c3Swdenk #ifdef CONFIG_LOADS_ECHO 103d0fb80c3Swdenk "loads_echo=" MK_STR (CONFIG_LOADS_ECHO) "\0" 104d0fb80c3Swdenk #endif 1056aff3115Swdenk #ifdef CONFIG_ETHADDR 1066aff3115Swdenk "ethaddr=" MK_STR (CONFIG_ETHADDR) "\0" 1076aff3115Swdenk #endif 108d0fb80c3Swdenk #ifdef CONFIG_ETH1ADDR 109d0fb80c3Swdenk "eth1addr=" MK_STR (CONFIG_ETH1ADDR) "\0" 110d0fb80c3Swdenk #endif 111d0fb80c3Swdenk #ifdef CONFIG_ETH2ADDR 112d0fb80c3Swdenk "eth2addr=" MK_STR (CONFIG_ETH2ADDR) "\0" 113d0fb80c3Swdenk #endif 114e2ffd59bSwdenk #ifdef CONFIG_ETH3ADDR 115e2ffd59bSwdenk "eth3addr=" MK_STR (CONFIG_ETH3ADDR) "\0" 116e2ffd59bSwdenk #endif 117d0fb80c3Swdenk #ifdef CONFIG_ETHPRIME 118d0fb80c3Swdenk "ethprime=" CONFIG_ETHPRIME "\0" 119d0fb80c3Swdenk #endif 1206aff3115Swdenk #ifdef CONFIG_IPADDR 1216aff3115Swdenk "ipaddr=" MK_STR (CONFIG_IPADDR) "\0" 1226aff3115Swdenk #endif 1236aff3115Swdenk #ifdef CONFIG_SERVERIP 1246aff3115Swdenk "serverip=" MK_STR (CONFIG_SERVERIP) "\0" 1256aff3115Swdenk #endif 126d0fb80c3Swdenk #ifdef CFG_AUTOLOAD 127d0fb80c3Swdenk "autoload=" CFG_AUTOLOAD "\0" 128d0fb80c3Swdenk #endif 129d0fb80c3Swdenk #ifdef CONFIG_ROOTPATH 130d0fb80c3Swdenk "rootpath=" MK_STR (CONFIG_ROOTPATH) "\0" 131d0fb80c3Swdenk #endif 132d0fb80c3Swdenk #ifdef CONFIG_GATEWAYIP 133d0fb80c3Swdenk "gatewayip=" MK_STR (CONFIG_GATEWAYIP) "\0" 134d0fb80c3Swdenk #endif 135d0fb80c3Swdenk #ifdef CONFIG_NETMASK 136d0fb80c3Swdenk "netmask=" MK_STR (CONFIG_NETMASK) "\0" 137d0fb80c3Swdenk #endif 138d0fb80c3Swdenk #ifdef CONFIG_HOSTNAME 139d0fb80c3Swdenk "hostname=" MK_STR (CONFIG_HOSTNAME) "\0" 140d0fb80c3Swdenk #endif 141d0fb80c3Swdenk #ifdef CONFIG_BOOTFILE 142d0fb80c3Swdenk "bootfile=" MK_STR (CONFIG_BOOTFILE) "\0" 143d0fb80c3Swdenk #endif 144d0fb80c3Swdenk #ifdef CONFIG_LOADADDR 145d0fb80c3Swdenk "loadaddr=" MK_STR (CONFIG_LOADADDR) "\0" 146d0fb80c3Swdenk #endif 147d0fb80c3Swdenk #ifdef CONFIG_PREBOOT 148d0fb80c3Swdenk "preboot=" CONFIG_PREBOOT "\0" 149d0fb80c3Swdenk #endif 150d0fb80c3Swdenk #ifdef CONFIG_CLOCKS_IN_MHZ 151d0fb80c3Swdenk "clocks_in_mhz=" "1" "\0" 152d0fb80c3Swdenk #endif 153ad10dd9aSstroese #if defined(CONFIG_PCI_BOOTDELAY) && (CONFIG_PCI_BOOTDELAY > 0) 154ad10dd9aSstroese "pcidelay=" MK_STR (CONFIG_PCI_BOOTDELAY) "\0" 155ad10dd9aSstroese #endif 156d0fb80c3Swdenk #ifdef CONFIG_EXTRA_ENV_SETTINGS 157d0fb80c3Swdenk CONFIG_EXTRA_ENV_SETTINGS 158d0fb80c3Swdenk #endif 159d0fb80c3Swdenk "\0" /* Termimate env_t data with 2 NULs */ 1606aff3115Swdenk }; 1616aff3115Swdenk 1626aff3115Swdenk static int flash_io (int mode); 1636de66b35SMarkus Klotzbücher static char *envmatch (char * s1, char * s2); 1646aff3115Swdenk static int env_init (void); 1656aff3115Swdenk static int parse_config (void); 1664a6fd34bSwdenk 167d0fb80c3Swdenk #if defined(CONFIG_FILE) 168d0fb80c3Swdenk static int get_config (char *); 169d0fb80c3Swdenk #endif 170d0fb80c3Swdenk static inline ulong getenvsize (void) 171d0fb80c3Swdenk { 172d0fb80c3Swdenk ulong rc = CFG_ENV_SIZE - sizeof (long); 1734a6fd34bSwdenk 174d0fb80c3Swdenk if (HaveRedundEnv) 175d0fb80c3Swdenk rc -= sizeof (char); 176d0fb80c3Swdenk return rc; 177d0fb80c3Swdenk } 1786aff3115Swdenk 1796aff3115Swdenk /* 1806aff3115Swdenk * Search the environment for a variable. 1816aff3115Swdenk * Return the value, if found, or NULL, if not found. 1826aff3115Swdenk */ 1836de66b35SMarkus Klotzbücher char *fw_getenv (char *name) 1846aff3115Swdenk { 1856de66b35SMarkus Klotzbücher char *env, *nxt; 1866aff3115Swdenk 1876aff3115Swdenk if (env_init ()) 1886aff3115Swdenk return (NULL); 1896aff3115Swdenk 1906aff3115Swdenk for (env = environment.data; *env; env = nxt + 1) { 1916de66b35SMarkus Klotzbücher char *val; 1926aff3115Swdenk 1936aff3115Swdenk for (nxt = env; *nxt; ++nxt) { 1946aff3115Swdenk if (nxt >= &environment.data[ENV_SIZE]) { 1956aff3115Swdenk fprintf (stderr, "## Error: " 1966aff3115Swdenk "environment not terminated\n"); 1976aff3115Swdenk return (NULL); 1986aff3115Swdenk } 1996aff3115Swdenk } 2006aff3115Swdenk val = envmatch (name, env); 2016aff3115Swdenk if (!val) 2026aff3115Swdenk continue; 2036aff3115Swdenk return (val); 2046aff3115Swdenk } 2056aff3115Swdenk return (NULL); 2066aff3115Swdenk } 2076aff3115Swdenk 2086aff3115Swdenk /* 2096aff3115Swdenk * Print the current definition of one, or more, or all 2106aff3115Swdenk * environment variables 2116aff3115Swdenk */ 212*bc11756dSGrant Erickson int fw_printenv (int argc, char *argv[]) 2136aff3115Swdenk { 2146de66b35SMarkus Klotzbücher char *env, *nxt; 2156aff3115Swdenk int i, n_flag; 216*bc11756dSGrant Erickson int rc = 0; 2176aff3115Swdenk 2186aff3115Swdenk if (env_init ()) 219*bc11756dSGrant Erickson return (-1); 2206aff3115Swdenk 2216aff3115Swdenk if (argc == 1) { /* Print all env variables */ 2226aff3115Swdenk for (env = environment.data; *env; env = nxt + 1) { 2236aff3115Swdenk for (nxt = env; *nxt; ++nxt) { 2246aff3115Swdenk if (nxt >= &environment.data[ENV_SIZE]) { 2256aff3115Swdenk fprintf (stderr, "## Error: " 2266aff3115Swdenk "environment not terminated\n"); 227*bc11756dSGrant Erickson return (-1); 2286aff3115Swdenk } 2296aff3115Swdenk } 2306aff3115Swdenk 2316aff3115Swdenk printf ("%s\n", env); 2326aff3115Swdenk } 233*bc11756dSGrant Erickson return (0); 2346aff3115Swdenk } 2356aff3115Swdenk 2366aff3115Swdenk if (strcmp (argv[1], "-n") == 0) { 2376aff3115Swdenk n_flag = 1; 2386aff3115Swdenk ++argv; 2396aff3115Swdenk --argc; 2406aff3115Swdenk if (argc != 2) { 2416aff3115Swdenk fprintf (stderr, "## Error: " 2426aff3115Swdenk "`-n' option requires exactly one argument\n"); 243*bc11756dSGrant Erickson return (-1); 2446aff3115Swdenk } 2456aff3115Swdenk } else { 2466aff3115Swdenk n_flag = 0; 2476aff3115Swdenk } 2486aff3115Swdenk 2496aff3115Swdenk for (i = 1; i < argc; ++i) { /* print single env variables */ 2506de66b35SMarkus Klotzbücher char *name = argv[i]; 2516de66b35SMarkus Klotzbücher char *val = NULL; 2526aff3115Swdenk 2536aff3115Swdenk for (env = environment.data; *env; env = nxt + 1) { 2546aff3115Swdenk 2556aff3115Swdenk for (nxt = env; *nxt; ++nxt) { 2566aff3115Swdenk if (nxt >= &environment.data[ENV_SIZE]) { 2576aff3115Swdenk fprintf (stderr, "## Error: " 2586aff3115Swdenk "environment not terminated\n"); 259*bc11756dSGrant Erickson return (-1); 2606aff3115Swdenk } 2616aff3115Swdenk } 2626aff3115Swdenk val = envmatch (name, env); 2636aff3115Swdenk if (val) { 2646aff3115Swdenk if (!n_flag) { 2656aff3115Swdenk fputs (name, stdout); 2666aff3115Swdenk putc ('=', stdout); 2676aff3115Swdenk } 2686aff3115Swdenk puts (val); 2696aff3115Swdenk break; 2706aff3115Swdenk } 2716aff3115Swdenk } 272*bc11756dSGrant Erickson if (!val) { 2734a6fd34bSwdenk fprintf (stderr, "## Error: \"%s\" not defined\n", name); 274*bc11756dSGrant Erickson rc = -1; 2756aff3115Swdenk } 2766aff3115Swdenk } 2776aff3115Swdenk 278*bc11756dSGrant Erickson return (rc); 279*bc11756dSGrant Erickson } 280*bc11756dSGrant Erickson 2816aff3115Swdenk /* 2826aff3115Swdenk * Deletes or sets environment variables. Returns errno style error codes: 2836aff3115Swdenk * 0 - OK 2846aff3115Swdenk * EINVAL - need at least 1 argument 2856aff3115Swdenk * EROFS - certain variables ("ethaddr", "serial#") cannot be 2866aff3115Swdenk * modified or deleted 2876aff3115Swdenk * 2886aff3115Swdenk */ 2896aff3115Swdenk int fw_setenv (int argc, char *argv[]) 2906aff3115Swdenk { 2916aff3115Swdenk int i, len; 2926de66b35SMarkus Klotzbücher char *env, *nxt; 2936de66b35SMarkus Klotzbücher char *oldval = NULL; 2946de66b35SMarkus Klotzbücher char *name; 2956aff3115Swdenk 2966aff3115Swdenk if (argc < 2) { 2976aff3115Swdenk return (EINVAL); 2986aff3115Swdenk } 2996aff3115Swdenk 3006aff3115Swdenk if (env_init ()) 3016aff3115Swdenk return (errno); 3026aff3115Swdenk 3036aff3115Swdenk name = argv[1]; 3046aff3115Swdenk 3056aff3115Swdenk /* 3066aff3115Swdenk * search if variable with this name already exists 3076aff3115Swdenk */ 3084a6fd34bSwdenk for (nxt = env = environment.data; *env; env = nxt + 1) { 3096aff3115Swdenk for (nxt = env; *nxt; ++nxt) { 3106aff3115Swdenk if (nxt >= &environment.data[ENV_SIZE]) { 3116aff3115Swdenk fprintf (stderr, "## Error: " 3126aff3115Swdenk "environment not terminated\n"); 3136aff3115Swdenk return (EINVAL); 3146aff3115Swdenk } 3156aff3115Swdenk } 3166aff3115Swdenk if ((oldval = envmatch (name, env)) != NULL) 3176aff3115Swdenk break; 3186aff3115Swdenk } 3196aff3115Swdenk 3206aff3115Swdenk /* 3216aff3115Swdenk * Delete any existing definition 3226aff3115Swdenk */ 3236aff3115Swdenk if (oldval) { 3246aff3115Swdenk /* 3256aff3115Swdenk * Ethernet Address and serial# can be set only once 3266aff3115Swdenk */ 3276aff3115Swdenk if ((strcmp (name, "ethaddr") == 0) || 3286aff3115Swdenk (strcmp (name, "serial#") == 0)) { 3296aff3115Swdenk fprintf (stderr, "Can't overwrite \"%s\"\n", name); 3306aff3115Swdenk return (EROFS); 3316aff3115Swdenk } 3326aff3115Swdenk 3336aff3115Swdenk if (*++nxt == '\0') { 3346aff3115Swdenk *env = '\0'; 3356aff3115Swdenk } else { 3366aff3115Swdenk for (;;) { 3376aff3115Swdenk *env = *nxt++; 3386aff3115Swdenk if ((*env == '\0') && (*nxt == '\0')) 3396aff3115Swdenk break; 3406aff3115Swdenk ++env; 3416aff3115Swdenk } 3426aff3115Swdenk } 3436aff3115Swdenk *++env = '\0'; 3446aff3115Swdenk } 3456aff3115Swdenk 3466aff3115Swdenk /* Delete only ? */ 3476aff3115Swdenk if (argc < 3) 3486aff3115Swdenk goto WRITE_FLASH; 3496aff3115Swdenk 3506aff3115Swdenk /* 3516aff3115Swdenk * Append new definition at the end 3526aff3115Swdenk */ 3534a6fd34bSwdenk for (env = environment.data; *env || *(env + 1); ++env); 3546aff3115Swdenk if (env > environment.data) 3556aff3115Swdenk ++env; 3566aff3115Swdenk /* 3576aff3115Swdenk * Overflow when: 3586aff3115Swdenk * "name" + "=" + "val" +"\0\0" > CFG_ENV_SIZE - (env-environment) 3596aff3115Swdenk */ 3606aff3115Swdenk len = strlen (name) + 2; 3616aff3115Swdenk /* add '=' for first arg, ' ' for all others */ 3626aff3115Swdenk for (i = 2; i < argc; ++i) { 3636aff3115Swdenk len += strlen (argv[i]) + 1; 3646aff3115Swdenk } 3656aff3115Swdenk if (len > (&environment.data[ENV_SIZE] - env)) { 3666aff3115Swdenk fprintf (stderr, 3676aff3115Swdenk "Error: environment overflow, \"%s\" deleted\n", 3686aff3115Swdenk name); 3696aff3115Swdenk return (-1); 3706aff3115Swdenk } 3716aff3115Swdenk while ((*env = *name++) != '\0') 3726aff3115Swdenk env++; 3736aff3115Swdenk for (i = 2; i < argc; ++i) { 3746de66b35SMarkus Klotzbücher char *val = argv[i]; 3756aff3115Swdenk 3766aff3115Swdenk *env = (i == 2) ? '=' : ' '; 3774a6fd34bSwdenk while ((*++env = *val++) != '\0'); 3786aff3115Swdenk } 3796aff3115Swdenk 3806aff3115Swdenk /* end is marked with double '\0' */ 3816aff3115Swdenk *++env = '\0'; 3826aff3115Swdenk 3836aff3115Swdenk WRITE_FLASH: 3846aff3115Swdenk 3856aff3115Swdenk /* Update CRC */ 3866de66b35SMarkus Klotzbücher environment.crc = crc32 (0, (uint8_t*) environment.data, ENV_SIZE); 3876aff3115Swdenk 3886aff3115Swdenk /* write environment back to flash */ 3896aff3115Swdenk if (flash_io (O_RDWR)) { 3904a6fd34bSwdenk fprintf (stderr, "Error: can't write fw_env to flash\n"); 3916aff3115Swdenk return (-1); 3926aff3115Swdenk } 3936aff3115Swdenk 3946aff3115Swdenk return (0); 3956aff3115Swdenk } 3966aff3115Swdenk 3976aff3115Swdenk static int flash_io (int mode) 3986aff3115Swdenk { 3996aff3115Swdenk int fd, fdr, rc, otherdev, len, resid; 4006aff3115Swdenk erase_info_t erase; 401592c5cabSwdenk char *data = NULL; 4026aff3115Swdenk 4036aff3115Swdenk if ((fd = open (DEVNAME (curdev), mode)) < 0) { 4046aff3115Swdenk fprintf (stderr, 4056aff3115Swdenk "Can't open %s: %s\n", 4066aff3115Swdenk DEVNAME (curdev), strerror (errno)); 4076aff3115Swdenk return (-1); 4086aff3115Swdenk } 4096aff3115Swdenk 410d0fb80c3Swdenk len = sizeof (environment.crc); 411d0fb80c3Swdenk if (HaveRedundEnv) { 412d0fb80c3Swdenk len += sizeof (environment.flags); 413d0fb80c3Swdenk } 4146aff3115Swdenk 4156aff3115Swdenk if (mode == O_RDWR) { 416d0fb80c3Swdenk if (HaveRedundEnv) { 4176aff3115Swdenk /* switch to next partition for writing */ 4186aff3115Swdenk otherdev = !curdev; 4196aff3115Swdenk if ((fdr = open (DEVNAME (otherdev), mode)) < 0) { 4206aff3115Swdenk fprintf (stderr, 4216aff3115Swdenk "Can't open %s: %s\n", 4224a6fd34bSwdenk DEVNAME (otherdev), 4234a6fd34bSwdenk strerror (errno)); 4246aff3115Swdenk return (-1); 4256aff3115Swdenk } 426d0fb80c3Swdenk } else { 4276aff3115Swdenk otherdev = curdev; 4286aff3115Swdenk fdr = fd; 429d0fb80c3Swdenk } 4306aff3115Swdenk printf ("Unlocking flash...\n"); 4316aff3115Swdenk erase.length = DEVESIZE (otherdev); 432d0fb80c3Swdenk erase.start = DEVOFFSET (otherdev); 4336aff3115Swdenk ioctl (fdr, MEMUNLOCK, &erase); 4346aff3115Swdenk 435d0fb80c3Swdenk if (HaveRedundEnv) { 4366aff3115Swdenk erase.length = DEVESIZE (curdev); 437d0fb80c3Swdenk erase.start = DEVOFFSET (curdev); 4386aff3115Swdenk ioctl (fd, MEMUNLOCK, &erase); 4396aff3115Swdenk environment.flags = active_flag; 440d0fb80c3Swdenk } 441d0fb80c3Swdenk 4426aff3115Swdenk printf ("Done\n"); 4436aff3115Swdenk resid = DEVESIZE (otherdev) - CFG_ENV_SIZE; 4446aff3115Swdenk if (resid) { 4456aff3115Swdenk if ((data = malloc (resid)) == NULL) { 4466aff3115Swdenk fprintf (stderr, 4476aff3115Swdenk "Cannot malloc %d bytes: %s\n", 4484a6fd34bSwdenk resid, 4494a6fd34bSwdenk strerror (errno)); 4506aff3115Swdenk return (-1); 4516aff3115Swdenk } 4524a6fd34bSwdenk if (lseek (fdr, DEVOFFSET (otherdev) + CFG_ENV_SIZE, SEEK_SET) 4534a6fd34bSwdenk == -1) { 4544a6fd34bSwdenk fprintf (stderr, "seek error on %s: %s\n", 4554a6fd34bSwdenk DEVNAME (otherdev), 4564a6fd34bSwdenk strerror (errno)); 4576aff3115Swdenk return (-1); 4586aff3115Swdenk } 4596aff3115Swdenk if ((rc = read (fdr, data, resid)) != resid) { 4606aff3115Swdenk fprintf (stderr, 4616aff3115Swdenk "read error on %s: %s\n", 4624a6fd34bSwdenk DEVNAME (otherdev), 4634a6fd34bSwdenk strerror (errno)); 4646aff3115Swdenk return (-1); 4656aff3115Swdenk } 4666aff3115Swdenk } 4676aff3115Swdenk 4686aff3115Swdenk printf ("Erasing old environment...\n"); 4696aff3115Swdenk 4706aff3115Swdenk erase.length = DEVESIZE (otherdev); 471d0fb80c3Swdenk erase.start = DEVOFFSET (otherdev); 4726aff3115Swdenk if (ioctl (fdr, MEMERASE, &erase) != 0) { 4736aff3115Swdenk fprintf (stderr, "MTD erase error on %s: %s\n", 4744a6fd34bSwdenk DEVNAME (otherdev), 4754a6fd34bSwdenk strerror (errno)); 4766aff3115Swdenk return (-1); 4776aff3115Swdenk } 4786aff3115Swdenk 4796aff3115Swdenk printf ("Done\n"); 4806aff3115Swdenk 4816aff3115Swdenk printf ("Writing environment to %s...\n", DEVNAME (otherdev)); 482d0fb80c3Swdenk if (lseek (fdr, DEVOFFSET (otherdev), SEEK_SET) == -1) { 483d0fb80c3Swdenk fprintf (stderr, 484d0fb80c3Swdenk "seek error on %s: %s\n", 485d0fb80c3Swdenk DEVNAME (otherdev), strerror (errno)); 486d0fb80c3Swdenk return (-1); 487d0fb80c3Swdenk } 4886aff3115Swdenk if (write (fdr, &environment, len) != len) { 4896aff3115Swdenk fprintf (stderr, 4906aff3115Swdenk "CRC write error on %s: %s\n", 4916aff3115Swdenk DEVNAME (otherdev), strerror (errno)); 4926aff3115Swdenk return (-1); 4936aff3115Swdenk } 4946aff3115Swdenk if (write (fdr, environment.data, ENV_SIZE) != ENV_SIZE) { 4956aff3115Swdenk fprintf (stderr, 4966aff3115Swdenk "Write error on %s: %s\n", 4976aff3115Swdenk DEVNAME (otherdev), strerror (errno)); 4986aff3115Swdenk return (-1); 4996aff3115Swdenk } 5006aff3115Swdenk if (resid) { 5016aff3115Swdenk if (write (fdr, data, resid) != resid) { 5026aff3115Swdenk fprintf (stderr, 5036aff3115Swdenk "write error on %s: %s\n", 5046aff3115Swdenk DEVNAME (curdev), strerror (errno)); 5056aff3115Swdenk return (-1); 5066aff3115Swdenk } 5076aff3115Swdenk free (data); 5086aff3115Swdenk } 509d0fb80c3Swdenk if (HaveRedundEnv) { 5106aff3115Swdenk /* change flag on current active env partition */ 5114a6fd34bSwdenk if (lseek (fd, DEVOFFSET (curdev) + sizeof (ulong), SEEK_SET) 5124a6fd34bSwdenk == -1) { 5134a6fd34bSwdenk fprintf (stderr, "seek error on %s: %s\n", 5146aff3115Swdenk DEVNAME (curdev), strerror (errno)); 5156aff3115Swdenk return (-1); 5166aff3115Swdenk } 5176aff3115Swdenk if (write (fd, &obsolete_flag, sizeof (obsolete_flag)) != 5186aff3115Swdenk sizeof (obsolete_flag)) { 5196aff3115Swdenk fprintf (stderr, 5206aff3115Swdenk "Write error on %s: %s\n", 5216aff3115Swdenk DEVNAME (curdev), strerror (errno)); 5226aff3115Swdenk return (-1); 5236aff3115Swdenk } 524d0fb80c3Swdenk } 5256aff3115Swdenk printf ("Done\n"); 5266aff3115Swdenk printf ("Locking ...\n"); 5276aff3115Swdenk erase.length = DEVESIZE (otherdev); 528d0fb80c3Swdenk erase.start = DEVOFFSET (otherdev); 5296aff3115Swdenk ioctl (fdr, MEMLOCK, &erase); 530d0fb80c3Swdenk if (HaveRedundEnv) { 5316aff3115Swdenk erase.length = DEVESIZE (curdev); 532d0fb80c3Swdenk erase.start = DEVOFFSET (curdev); 5336aff3115Swdenk ioctl (fd, MEMLOCK, &erase); 5346aff3115Swdenk if (close (fdr)) { 5356aff3115Swdenk fprintf (stderr, 5366aff3115Swdenk "I/O error on %s: %s\n", 5374a6fd34bSwdenk DEVNAME (otherdev), 5384a6fd34bSwdenk strerror (errno)); 5396aff3115Swdenk return (-1); 5406aff3115Swdenk } 541d0fb80c3Swdenk } 5426aff3115Swdenk printf ("Done\n"); 5436aff3115Swdenk } else { 544d0fb80c3Swdenk 545d0fb80c3Swdenk if (lseek (fd, DEVOFFSET (curdev), SEEK_SET) == -1) { 546d0fb80c3Swdenk fprintf (stderr, 547d0fb80c3Swdenk "seek error on %s: %s\n", 548d0fb80c3Swdenk DEVNAME (curdev), strerror (errno)); 549d0fb80c3Swdenk return (-1); 550d0fb80c3Swdenk } 5516aff3115Swdenk if (read (fd, &environment, len) != len) { 5526aff3115Swdenk fprintf (stderr, 5536aff3115Swdenk "CRC read error on %s: %s\n", 5546aff3115Swdenk DEVNAME (curdev), strerror (errno)); 5556aff3115Swdenk return (-1); 5566aff3115Swdenk } 5576aff3115Swdenk if ((rc = read (fd, environment.data, ENV_SIZE)) != ENV_SIZE) { 5586aff3115Swdenk fprintf (stderr, 5596aff3115Swdenk "Read error on %s: %s\n", 5606aff3115Swdenk DEVNAME (curdev), strerror (errno)); 5616aff3115Swdenk return (-1); 5626aff3115Swdenk } 5636aff3115Swdenk } 5646aff3115Swdenk 5656aff3115Swdenk if (close (fd)) { 5666aff3115Swdenk fprintf (stderr, 5676aff3115Swdenk "I/O error on %s: %s\n", 5686aff3115Swdenk DEVNAME (curdev), strerror (errno)); 5696aff3115Swdenk return (-1); 5706aff3115Swdenk } 5716aff3115Swdenk 5726aff3115Swdenk /* everything ok */ 5736aff3115Swdenk return (0); 5746aff3115Swdenk } 5756aff3115Swdenk 5766aff3115Swdenk /* 5776aff3115Swdenk * s1 is either a simple 'name', or a 'name=value' pair. 5786aff3115Swdenk * s2 is a 'name=value' pair. 5796aff3115Swdenk * If the names match, return the value of s2, else NULL. 5806aff3115Swdenk */ 5816aff3115Swdenk 5826de66b35SMarkus Klotzbücher static char *envmatch (char * s1, char * s2) 5836aff3115Swdenk { 5846aff3115Swdenk 5856aff3115Swdenk while (*s1 == *s2++) 5866aff3115Swdenk if (*s1++ == '=') 5876aff3115Swdenk return (s2); 5886aff3115Swdenk if (*s1 == '\0' && *(s2 - 1) == '=') 5896aff3115Swdenk return (s2); 5906aff3115Swdenk return (NULL); 5916aff3115Swdenk } 5926aff3115Swdenk 5936aff3115Swdenk /* 5946aff3115Swdenk * Prevent confusion if running from erased flash memory 5956aff3115Swdenk */ 5966aff3115Swdenk static int env_init (void) 5976aff3115Swdenk { 5986aff3115Swdenk int crc1, crc1_ok; 5996de66b35SMarkus Klotzbücher char *addr1; 600d0fb80c3Swdenk 6016aff3115Swdenk int crc2, crc2_ok; 6026de66b35SMarkus Klotzbücher char flag1, flag2, *addr2; 6036aff3115Swdenk 6046aff3115Swdenk if (parse_config ()) /* should fill envdevices */ 6056aff3115Swdenk return 1; 6066aff3115Swdenk 6076aff3115Swdenk if ((addr1 = calloc (1, ENV_SIZE)) == NULL) { 6086aff3115Swdenk fprintf (stderr, 6096aff3115Swdenk "Not enough memory for environment (%ld bytes)\n", 6106aff3115Swdenk ENV_SIZE); 6116aff3115Swdenk return (errno); 6126aff3115Swdenk } 6136aff3115Swdenk 6146aff3115Swdenk /* read environment from FLASH to local buffer */ 6156aff3115Swdenk environment.data = addr1; 6166aff3115Swdenk curdev = 0; 6176aff3115Swdenk if (flash_io (O_RDONLY)) { 6186aff3115Swdenk return (errno); 6196aff3115Swdenk } 6206aff3115Swdenk 6216de66b35SMarkus Klotzbücher crc1_ok = ((crc1 = crc32 (0, (uint8_t *) environment.data, ENV_SIZE)) 6226aff3115Swdenk == environment.crc); 623d0fb80c3Swdenk if (!HaveRedundEnv) { 6246aff3115Swdenk if (!crc1_ok) { 6256aff3115Swdenk fprintf (stderr, 6266aff3115Swdenk "Warning: Bad CRC, using default environment\n"); 627f07217c9SWolfgang Denk memcpy(environment.data, default_environment, sizeof default_environment); 6286aff3115Swdenk } 629d0fb80c3Swdenk } else { 6306aff3115Swdenk flag1 = environment.flags; 6316aff3115Swdenk 6326aff3115Swdenk curdev = 1; 6336aff3115Swdenk if ((addr2 = calloc (1, ENV_SIZE)) == NULL) { 6346aff3115Swdenk fprintf (stderr, 6356aff3115Swdenk "Not enough memory for environment (%ld bytes)\n", 6366aff3115Swdenk ENV_SIZE); 6376aff3115Swdenk return (errno); 6386aff3115Swdenk } 6396aff3115Swdenk environment.data = addr2; 6406aff3115Swdenk 6416aff3115Swdenk if (flash_io (O_RDONLY)) { 6426aff3115Swdenk return (errno); 6436aff3115Swdenk } 6446aff3115Swdenk 6456de66b35SMarkus Klotzbücher crc2_ok = ((crc2 = crc32 (0, (uint8_t *) environment.data, ENV_SIZE)) 6466aff3115Swdenk == environment.crc); 6476aff3115Swdenk flag2 = environment.flags; 6486aff3115Swdenk 6496aff3115Swdenk if (crc1_ok && !crc2_ok) { 6506aff3115Swdenk environment.data = addr1; 6516aff3115Swdenk environment.flags = flag1; 6526aff3115Swdenk environment.crc = crc1; 6536aff3115Swdenk curdev = 0; 6546aff3115Swdenk free (addr2); 6554a6fd34bSwdenk } else if (!crc1_ok && crc2_ok) { 6566aff3115Swdenk environment.data = addr2; 6576aff3115Swdenk environment.flags = flag2; 6586aff3115Swdenk environment.crc = crc2; 6596aff3115Swdenk curdev = 1; 6606aff3115Swdenk free (addr1); 6614a6fd34bSwdenk } else if (!crc1_ok && !crc2_ok) { 6626aff3115Swdenk fprintf (stderr, 6636aff3115Swdenk "Warning: Bad CRC, using default environment\n"); 664f07217c9SWolfgang Denk memcpy(environment.data, default_environment, sizeof default_environment); 6656aff3115Swdenk curdev = 0; 6666aff3115Swdenk free (addr1); 6674a6fd34bSwdenk } else if (flag1 == active_flag && flag2 == obsolete_flag) { 6686aff3115Swdenk environment.data = addr1; 6696aff3115Swdenk environment.flags = flag1; 6706aff3115Swdenk environment.crc = crc1; 6716aff3115Swdenk curdev = 0; 6726aff3115Swdenk free (addr2); 6734a6fd34bSwdenk } else if (flag1 == obsolete_flag && flag2 == active_flag) { 6746aff3115Swdenk environment.data = addr2; 6756aff3115Swdenk environment.flags = flag2; 6766aff3115Swdenk environment.crc = crc2; 6776aff3115Swdenk curdev = 1; 6786aff3115Swdenk free (addr1); 6794a6fd34bSwdenk } else if (flag1 == flag2) { 6806aff3115Swdenk environment.data = addr1; 6816aff3115Swdenk environment.flags = flag1; 6826aff3115Swdenk environment.crc = crc1; 6836aff3115Swdenk curdev = 0; 6846aff3115Swdenk free (addr2); 6854a6fd34bSwdenk } else if (flag1 == 0xFF) { 6866aff3115Swdenk environment.data = addr1; 6876aff3115Swdenk environment.flags = flag1; 6886aff3115Swdenk environment.crc = crc1; 6896aff3115Swdenk curdev = 0; 6906aff3115Swdenk free (addr2); 6914a6fd34bSwdenk } else if (flag2 == 0xFF) { 6926aff3115Swdenk environment.data = addr2; 6936aff3115Swdenk environment.flags = flag2; 6946aff3115Swdenk environment.crc = crc2; 6956aff3115Swdenk curdev = 1; 6966aff3115Swdenk free (addr1); 6976aff3115Swdenk } 6986aff3115Swdenk } 6996aff3115Swdenk return (0); 7006aff3115Swdenk } 7016aff3115Swdenk 7026aff3115Swdenk 7036aff3115Swdenk static int parse_config () 7046aff3115Swdenk { 7056aff3115Swdenk struct stat st; 7066aff3115Swdenk 707d0fb80c3Swdenk #if defined(CONFIG_FILE) 708d0fb80c3Swdenk /* Fills in DEVNAME(), ENVSIZE(), DEVESIZE(). Or don't. */ 709d0fb80c3Swdenk if (get_config (CONFIG_FILE)) { 7106aff3115Swdenk fprintf (stderr, 7114a6fd34bSwdenk "Cannot parse config file: %s\n", strerror (errno)); 7126aff3115Swdenk return 1; 7136aff3115Swdenk } 714d0fb80c3Swdenk #else 7156aff3115Swdenk strcpy (DEVNAME (0), DEVICE1_NAME); 716d0fb80c3Swdenk DEVOFFSET (0) = DEVICE1_OFFSET; 7176aff3115Swdenk ENVSIZE (0) = ENV1_SIZE; 7186aff3115Swdenk DEVESIZE (0) = DEVICE1_ESIZE; 7196aff3115Swdenk #ifdef HAVE_REDUND 7206aff3115Swdenk strcpy (DEVNAME (1), DEVICE2_NAME); 721d0fb80c3Swdenk DEVOFFSET (1) = DEVICE2_OFFSET; 7226aff3115Swdenk ENVSIZE (1) = ENV2_SIZE; 7236aff3115Swdenk DEVESIZE (1) = DEVICE2_ESIZE; 724d0fb80c3Swdenk HaveRedundEnv = 1; 7256aff3115Swdenk #endif 726d0fb80c3Swdenk #endif 727d0fb80c3Swdenk if (stat (DEVNAME (0), &st)) { 728d0fb80c3Swdenk fprintf (stderr, 729d0fb80c3Swdenk "Cannot access MTD device %s: %s\n", 730d0fb80c3Swdenk DEVNAME (0), strerror (errno)); 731d0fb80c3Swdenk return 1; 732d0fb80c3Swdenk } 733d0fb80c3Swdenk 734d0fb80c3Swdenk if (HaveRedundEnv && stat (DEVNAME (1), &st)) { 735d0fb80c3Swdenk fprintf (stderr, 736d0fb80c3Swdenk "Cannot access MTD device %s: %s\n", 737e2146b6aSWolfgang Denk DEVNAME (1), strerror (errno)); 738d0fb80c3Swdenk return 1; 739d0fb80c3Swdenk } 7406aff3115Swdenk return 0; 7416aff3115Swdenk } 742d0fb80c3Swdenk 743d0fb80c3Swdenk #if defined(CONFIG_FILE) 744d0fb80c3Swdenk static int get_config (char *fname) 745d0fb80c3Swdenk { 746d0fb80c3Swdenk FILE *fp; 747d0fb80c3Swdenk int i = 0; 748d0fb80c3Swdenk int rc; 749d0fb80c3Swdenk char dump[128]; 750d0fb80c3Swdenk 751d0fb80c3Swdenk if ((fp = fopen (fname, "r")) == NULL) { 752d0fb80c3Swdenk return 1; 753d0fb80c3Swdenk } 754d0fb80c3Swdenk 7554a6fd34bSwdenk while ((i < 2) && ((rc = fscanf (fp, "%s %lx %lx %lx", 7564a6fd34bSwdenk DEVNAME (i), 7574a6fd34bSwdenk &DEVOFFSET (i), 7584a6fd34bSwdenk &ENVSIZE (i), 7594a6fd34bSwdenk &DEVESIZE (i) )) != EOF)) { 760d0fb80c3Swdenk 761d0fb80c3Swdenk /* Skip incomplete conversions and comment strings */ 762d0fb80c3Swdenk if ((rc < 3) || (*DEVNAME (i) == '#')) { 763d0fb80c3Swdenk fgets (dump, sizeof (dump), fp); /* Consume till end */ 764d0fb80c3Swdenk continue; 765d0fb80c3Swdenk } 766d0fb80c3Swdenk 767d0fb80c3Swdenk i++; 768d0fb80c3Swdenk } 769d0fb80c3Swdenk fclose (fp); 770d0fb80c3Swdenk 771d0fb80c3Swdenk HaveRedundEnv = i - 1; 772d0fb80c3Swdenk if (!i) { /* No valid entries found */ 773d0fb80c3Swdenk errno = EINVAL; 774d0fb80c3Swdenk return 1; 775d0fb80c3Swdenk } else 776d0fb80c3Swdenk return 0; 777d0fb80c3Swdenk } 778d0fb80c3Swdenk #endif 779