16aff3115Swdenk /* 26aff3115Swdenk * (C) Copyright 2000 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 #include <linux/mtd/mtd.h> 356aff3115Swdenk #include "fw_env.h" 366aff3115Swdenk 376aff3115Swdenk typedef unsigned char uchar; 386aff3115Swdenk 396aff3115Swdenk #define CMD_GETENV "fw_printenv" 406aff3115Swdenk #define CMD_SETENV "fw_setenv" 416aff3115Swdenk 426aff3115Swdenk typedef struct envdev_s { 436aff3115Swdenk uchar devname[16]; /* Device name */ 44*d0fb80c3Swdenk ulong devoff; /* Device offset */ 456aff3115Swdenk ulong env_size; /* environment size */ 466aff3115Swdenk ulong erase_size; /* device erase size */ 476aff3115Swdenk } envdev_t; 486aff3115Swdenk 496aff3115Swdenk static envdev_t envdevices[2]; 506aff3115Swdenk static int curdev; 516aff3115Swdenk 526aff3115Swdenk #define DEVNAME(i) envdevices[(i)].devname 53*d0fb80c3Swdenk #define DEVOFFSET(i) envdevices[(i)].devoff 546aff3115Swdenk #define ENVSIZE(i) envdevices[(i)].env_size 556aff3115Swdenk #define DEVESIZE(i) envdevices[(i)].erase_size 566aff3115Swdenk 576aff3115Swdenk #define CFG_ENV_SIZE ENVSIZE(curdev) 586aff3115Swdenk 59*d0fb80c3Swdenk #define ENV_SIZE getenvsize() 606aff3115Swdenk 616aff3115Swdenk typedef struct environment_s { 626aff3115Swdenk ulong crc; /* CRC32 over data bytes */ 636aff3115Swdenk uchar flags; /* active or obsolete */ 646aff3115Swdenk uchar *data; 656aff3115Swdenk } env_t; 666aff3115Swdenk 676aff3115Swdenk static env_t environment; 686aff3115Swdenk 69*d0fb80c3Swdenk static int HaveRedundEnv = 0; 70*d0fb80c3Swdenk 716aff3115Swdenk static uchar active_flag = 1; 726aff3115Swdenk static uchar obsolete_flag = 0; 73*d0fb80c3Swdenk 746aff3115Swdenk 756aff3115Swdenk #define XMK_STR(x) #x 766aff3115Swdenk #define MK_STR(x) XMK_STR(x) 776aff3115Swdenk 786aff3115Swdenk static uchar default_environment[] = { 79*d0fb80c3Swdenk #if defined(CONFIG_BOOTARGS) 806aff3115Swdenk "bootargs=" CONFIG_BOOTARGS "\0" 816aff3115Swdenk #endif 82*d0fb80c3Swdenk #if defined(CONFIG_BOOTCOMMAND) 836aff3115Swdenk "bootcmd=" CONFIG_BOOTCOMMAND "\0" 846aff3115Swdenk #endif 85*d0fb80c3Swdenk #if defined(CONFIG_RAMBOOTCOMMAND) 86*d0fb80c3Swdenk "ramboot=" CONFIG_RAMBOOTCOMMAND "\0" 87*d0fb80c3Swdenk #endif 88*d0fb80c3Swdenk #if defined(CONFIG_NFSBOOTCOMMAND) 89*d0fb80c3Swdenk "nfsboot=" CONFIG_NFSBOOTCOMMAND "\0" 90*d0fb80c3Swdenk #endif 91*d0fb80c3Swdenk #if defined(CONFIG_BOOTDELAY) && (CONFIG_BOOTDELAY >= 0) 926aff3115Swdenk "bootdelay=" MK_STR(CONFIG_BOOTDELAY) "\0" 936aff3115Swdenk #endif 94*d0fb80c3Swdenk #if defined(CONFIG_BAUDRATE) && (CONFIG_BAUDRATE >= 0) 956aff3115Swdenk "baudrate=" MK_STR(CONFIG_BAUDRATE) "\0" 966aff3115Swdenk #endif 97*d0fb80c3Swdenk #ifdef CONFIG_LOADS_ECHO 98*d0fb80c3Swdenk "loads_echo=" MK_STR(CONFIG_LOADS_ECHO) "\0" 99*d0fb80c3Swdenk #endif 1006aff3115Swdenk #ifdef CONFIG_ETHADDR 1016aff3115Swdenk "ethaddr=" MK_STR(CONFIG_ETHADDR) "\0" 1026aff3115Swdenk #endif 103*d0fb80c3Swdenk #ifdef CONFIG_ETH1ADDR 104*d0fb80c3Swdenk "eth1addr=" MK_STR(CONFIG_ETH1ADDR) "\0" 105*d0fb80c3Swdenk #endif 106*d0fb80c3Swdenk #ifdef CONFIG_ETH2ADDR 107*d0fb80c3Swdenk "eth2addr=" MK_STR(CONFIG_ETH2ADDR) "\0" 108*d0fb80c3Swdenk #endif 109*d0fb80c3Swdenk #ifdef CONFIG_ETHPRIME 110*d0fb80c3Swdenk "ethprime=" CONFIG_ETHPRIME "\0" 111*d0fb80c3Swdenk #endif 1126aff3115Swdenk #ifdef CONFIG_IPADDR 1136aff3115Swdenk "ipaddr=" MK_STR(CONFIG_IPADDR) "\0" 1146aff3115Swdenk #endif 1156aff3115Swdenk #ifdef CONFIG_SERVERIP 1166aff3115Swdenk "serverip=" MK_STR(CONFIG_SERVERIP) "\0" 1176aff3115Swdenk #endif 118*d0fb80c3Swdenk #ifdef CFG_AUTOLOAD 119*d0fb80c3Swdenk "autoload=" CFG_AUTOLOAD "\0" 120*d0fb80c3Swdenk #endif 121*d0fb80c3Swdenk #ifdef CONFIG_ROOTPATH 122*d0fb80c3Swdenk "rootpath=" MK_STR(CONFIG_ROOTPATH) "\0" 123*d0fb80c3Swdenk #endif 124*d0fb80c3Swdenk #ifdef CONFIG_GATEWAYIP 125*d0fb80c3Swdenk "gatewayip=" MK_STR(CONFIG_GATEWAYIP) "\0" 126*d0fb80c3Swdenk #endif 127*d0fb80c3Swdenk #ifdef CONFIG_NETMASK 128*d0fb80c3Swdenk "netmask=" MK_STR(CONFIG_NETMASK) "\0" 129*d0fb80c3Swdenk #endif 130*d0fb80c3Swdenk #ifdef CONFIG_HOSTNAME 131*d0fb80c3Swdenk "hostname=" MK_STR(CONFIG_HOSTNAME) "\0" 132*d0fb80c3Swdenk #endif 133*d0fb80c3Swdenk #ifdef CONFIG_BOOTFILE 134*d0fb80c3Swdenk "bootfile=" MK_STR(CONFIG_BOOTFILE) "\0" 135*d0fb80c3Swdenk #endif 136*d0fb80c3Swdenk #ifdef CONFIG_LOADADDR 137*d0fb80c3Swdenk "loadaddr=" MK_STR(CONFIG_LOADADDR) "\0" 138*d0fb80c3Swdenk #endif 139*d0fb80c3Swdenk #ifdef CONFIG_PREBOOT 140*d0fb80c3Swdenk "preboot=" CONFIG_PREBOOT "\0" 141*d0fb80c3Swdenk #endif 142*d0fb80c3Swdenk #ifdef CONFIG_CLOCKS_IN_MHZ 143*d0fb80c3Swdenk "clocks_in_mhz=" "1" "\0" 144*d0fb80c3Swdenk #endif 145*d0fb80c3Swdenk #ifdef CONFIG_EXTRA_ENV_SETTINGS 146*d0fb80c3Swdenk CONFIG_EXTRA_ENV_SETTINGS 147*d0fb80c3Swdenk #endif 148*d0fb80c3Swdenk "\0" /* Termimate env_t data with 2 NULs */ 1496aff3115Swdenk }; 1506aff3115Swdenk 1516aff3115Swdenk static int flash_io (int mode); 1526aff3115Swdenk static uchar *envmatch(uchar *s1, uchar *s2); 1536aff3115Swdenk static int env_init(void); 1546aff3115Swdenk static int parse_config(void); 155*d0fb80c3Swdenk #if defined(CONFIG_FILE) 156*d0fb80c3Swdenk static int get_config(char *); 157*d0fb80c3Swdenk #endif 158*d0fb80c3Swdenk static inline ulong getenvsize(void) 159*d0fb80c3Swdenk { 160*d0fb80c3Swdenk ulong rc = CFG_ENV_SIZE - sizeof(long); 161*d0fb80c3Swdenk if (HaveRedundEnv) 162*d0fb80c3Swdenk rc -= sizeof(char); 163*d0fb80c3Swdenk return rc; 164*d0fb80c3Swdenk } 1656aff3115Swdenk 1666aff3115Swdenk /* 1676aff3115Swdenk * Search the environment for a variable. 1686aff3115Swdenk * Return the value, if found, or NULL, if not found. 1696aff3115Swdenk */ 1706aff3115Swdenk unsigned char *fw_getenv (unsigned char *name) 1716aff3115Swdenk { 1726aff3115Swdenk uchar *env, *nxt; 1736aff3115Swdenk 1746aff3115Swdenk if (env_init()) 1756aff3115Swdenk return (NULL); 1766aff3115Swdenk 1776aff3115Swdenk for (env=environment.data; *env; env=nxt+1) { 1786aff3115Swdenk uchar *val; 1796aff3115Swdenk 1806aff3115Swdenk for (nxt=env; *nxt; ++nxt) { 1816aff3115Swdenk if (nxt >= &environment.data[ENV_SIZE]) { 1826aff3115Swdenk fprintf (stderr, "## Error: " 1836aff3115Swdenk "environment not terminated\n"); 1846aff3115Swdenk return (NULL); 1856aff3115Swdenk } 1866aff3115Swdenk } 1876aff3115Swdenk val=envmatch(name, env); 1886aff3115Swdenk if (!val) 1896aff3115Swdenk continue; 1906aff3115Swdenk return (val); 1916aff3115Swdenk } 1926aff3115Swdenk return (NULL); 1936aff3115Swdenk } 1946aff3115Swdenk 1956aff3115Swdenk /* 1966aff3115Swdenk * Print the current definition of one, or more, or all 1976aff3115Swdenk * environment variables 1986aff3115Swdenk */ 1996aff3115Swdenk void fw_printenv(int argc, char *argv[]) 2006aff3115Swdenk { 2016aff3115Swdenk uchar *env, *nxt; 2026aff3115Swdenk int i, n_flag; 2036aff3115Swdenk 2046aff3115Swdenk if (env_init()) 2056aff3115Swdenk return; 2066aff3115Swdenk 2076aff3115Swdenk if (argc == 1) { /* Print all env variables */ 2086aff3115Swdenk for (env=environment.data; *env; env=nxt+1) { 2096aff3115Swdenk for (nxt=env; *nxt; ++nxt) { 2106aff3115Swdenk if (nxt >= &environment.data[ENV_SIZE]) { 2116aff3115Swdenk fprintf (stderr, "## Error: " 2126aff3115Swdenk "environment not terminated\n"); 2136aff3115Swdenk return; 2146aff3115Swdenk } 2156aff3115Swdenk } 2166aff3115Swdenk 2176aff3115Swdenk printf("%s\n", env); 2186aff3115Swdenk } 2196aff3115Swdenk return; 2206aff3115Swdenk } 2216aff3115Swdenk 2226aff3115Swdenk if (strcmp(argv[1], "-n") == 0) { 2236aff3115Swdenk n_flag = 1; 2246aff3115Swdenk ++argv; 2256aff3115Swdenk --argc; 2266aff3115Swdenk if (argc != 2) { 2276aff3115Swdenk fprintf (stderr, "## Error: " 2286aff3115Swdenk "`-n' option requires exactly one argument\n"); 2296aff3115Swdenk return; 2306aff3115Swdenk } 2316aff3115Swdenk } else { 2326aff3115Swdenk n_flag = 0; 2336aff3115Swdenk } 2346aff3115Swdenk 2356aff3115Swdenk for (i=1; i<argc; ++i) { /* print single env variables */ 2366aff3115Swdenk uchar *name = argv[i]; 2376aff3115Swdenk uchar *val = NULL; 2386aff3115Swdenk 2396aff3115Swdenk for (env=environment.data; *env; env=nxt+1) { 2406aff3115Swdenk 2416aff3115Swdenk for (nxt=env; *nxt; ++nxt) { 2426aff3115Swdenk if (nxt >= &environment.data[ENV_SIZE]) { 2436aff3115Swdenk fprintf (stderr, "## Error: " 2446aff3115Swdenk "environment not terminated\n"); 2456aff3115Swdenk return; 2466aff3115Swdenk } 2476aff3115Swdenk } 2486aff3115Swdenk val=envmatch(name, env); 2496aff3115Swdenk if (val) { 2506aff3115Swdenk if (!n_flag) { 2516aff3115Swdenk fputs (name, stdout); 2526aff3115Swdenk putc ('=', stdout); 2536aff3115Swdenk } 2546aff3115Swdenk puts (val); 2556aff3115Swdenk break; 2566aff3115Swdenk } 2576aff3115Swdenk } 2586aff3115Swdenk if (!val) 2596aff3115Swdenk fprintf (stderr, "## Error: \"%s\" not defined\n", 2606aff3115Swdenk name); 2616aff3115Swdenk } 2626aff3115Swdenk } 2636aff3115Swdenk 2646aff3115Swdenk /* 2656aff3115Swdenk * Deletes or sets environment variables. Returns errno style error codes: 2666aff3115Swdenk * 0 - OK 2676aff3115Swdenk * EINVAL - need at least 1 argument 2686aff3115Swdenk * EROFS - certain variables ("ethaddr", "serial#") cannot be 2696aff3115Swdenk * modified or deleted 2706aff3115Swdenk * 2716aff3115Swdenk */ 2726aff3115Swdenk int fw_setenv (int argc, char *argv[]) 2736aff3115Swdenk { 2746aff3115Swdenk int i, len; 2756aff3115Swdenk uchar *env, *nxt; 2766aff3115Swdenk uchar *oldval = NULL; 2776aff3115Swdenk uchar *name; 2786aff3115Swdenk 2796aff3115Swdenk if (argc < 2) { 2806aff3115Swdenk return (EINVAL); 2816aff3115Swdenk } 2826aff3115Swdenk 2836aff3115Swdenk if (env_init()) 2846aff3115Swdenk return (errno); 2856aff3115Swdenk 2866aff3115Swdenk name = argv[1]; 2876aff3115Swdenk 2886aff3115Swdenk /* 2896aff3115Swdenk * search if variable with this name already exists 2906aff3115Swdenk */ 2916aff3115Swdenk for (env=environment.data; *env; env=nxt+1) { 2926aff3115Swdenk for (nxt=env; *nxt; ++nxt) { 2936aff3115Swdenk if (nxt >= &environment.data[ENV_SIZE]) { 2946aff3115Swdenk fprintf (stderr, "## Error: " 2956aff3115Swdenk "environment not terminated\n"); 2966aff3115Swdenk return (EINVAL); 2976aff3115Swdenk } 2986aff3115Swdenk } 2996aff3115Swdenk if ((oldval=envmatch(name, env)) != NULL) 3006aff3115Swdenk break; 3016aff3115Swdenk } 3026aff3115Swdenk 3036aff3115Swdenk /* 3046aff3115Swdenk * Delete any existing definition 3056aff3115Swdenk */ 3066aff3115Swdenk if (oldval) { 3076aff3115Swdenk /* 3086aff3115Swdenk * Ethernet Address and serial# can be set only once 3096aff3115Swdenk */ 3106aff3115Swdenk if ((strcmp (name, "ethaddr") == 0) || 3116aff3115Swdenk (strcmp (name, "serial#") == 0) ) { 3126aff3115Swdenk fprintf (stderr, "Can't overwrite \"%s\"\n", name); 3136aff3115Swdenk return (EROFS); 3146aff3115Swdenk } 3156aff3115Swdenk 3166aff3115Swdenk if (*++nxt == '\0') { 3176aff3115Swdenk *env = '\0'; 3186aff3115Swdenk } else { 3196aff3115Swdenk for (;;) { 3206aff3115Swdenk *env = *nxt++; 3216aff3115Swdenk if ((*env == '\0') && (*nxt == '\0')) 3226aff3115Swdenk break; 3236aff3115Swdenk ++env; 3246aff3115Swdenk } 3256aff3115Swdenk } 3266aff3115Swdenk *++env = '\0'; 3276aff3115Swdenk } 3286aff3115Swdenk 3296aff3115Swdenk /* Delete only ? */ 3306aff3115Swdenk if (argc < 3) 3316aff3115Swdenk goto WRITE_FLASH; 3326aff3115Swdenk 3336aff3115Swdenk /* 3346aff3115Swdenk * Append new definition at the end 3356aff3115Swdenk */ 3366aff3115Swdenk for (env=environment.data; *env || *(env+1); ++env) 3376aff3115Swdenk ; 3386aff3115Swdenk if (env > environment.data) 3396aff3115Swdenk ++env; 3406aff3115Swdenk /* 3416aff3115Swdenk * Overflow when: 3426aff3115Swdenk * "name" + "=" + "val" +"\0\0" > CFG_ENV_SIZE - (env-environment) 3436aff3115Swdenk */ 3446aff3115Swdenk len = strlen(name) + 2; 3456aff3115Swdenk /* add '=' for first arg, ' ' for all others */ 3466aff3115Swdenk for (i=2; i<argc; ++i) { 3476aff3115Swdenk len += strlen(argv[i]) + 1; 3486aff3115Swdenk } 3496aff3115Swdenk if (len > (&environment.data[ENV_SIZE]-env)) { 3506aff3115Swdenk fprintf (stderr, 3516aff3115Swdenk "Error: environment overflow, \"%s\" deleted\n", 3526aff3115Swdenk name); 3536aff3115Swdenk return (-1); 3546aff3115Swdenk } 3556aff3115Swdenk while ((*env = *name++) != '\0') 3566aff3115Swdenk env++; 3576aff3115Swdenk for (i=2; i<argc; ++i) { 3586aff3115Swdenk uchar *val = argv[i]; 3596aff3115Swdenk 3606aff3115Swdenk *env = (i==2) ? '=' : ' '; 3616aff3115Swdenk while ((*++env = *val++) != '\0') 3626aff3115Swdenk ; 3636aff3115Swdenk } 3646aff3115Swdenk 3656aff3115Swdenk /* end is marked with double '\0' */ 3666aff3115Swdenk *++env = '\0'; 3676aff3115Swdenk 3686aff3115Swdenk WRITE_FLASH: 3696aff3115Swdenk 3706aff3115Swdenk /* Update CRC */ 3716aff3115Swdenk environment.crc = crc32(0, environment.data, ENV_SIZE); 3726aff3115Swdenk 3736aff3115Swdenk /* write environment back to flash */ 3746aff3115Swdenk if (flash_io (O_RDWR)) { 3756aff3115Swdenk fprintf (stderr, 3766aff3115Swdenk "Error: can't write fw_env to flash\n"); 3776aff3115Swdenk return (-1); 3786aff3115Swdenk } 3796aff3115Swdenk 3806aff3115Swdenk return (0); 3816aff3115Swdenk } 3826aff3115Swdenk 3836aff3115Swdenk static int flash_io (int mode) 3846aff3115Swdenk { 3856aff3115Swdenk int fd, fdr, rc, otherdev, len, resid; 3866aff3115Swdenk erase_info_t erase; 3876aff3115Swdenk char *data; 3886aff3115Swdenk 3896aff3115Swdenk if ((fd = open(DEVNAME(curdev), mode)) < 0) { 3906aff3115Swdenk fprintf (stderr, 3916aff3115Swdenk "Can't open %s: %s\n", 3926aff3115Swdenk DEVNAME(curdev), strerror(errno)); 3936aff3115Swdenk return (-1); 3946aff3115Swdenk } 3956aff3115Swdenk 396*d0fb80c3Swdenk len = sizeof(environment.crc); 397*d0fb80c3Swdenk if (HaveRedundEnv) { 398*d0fb80c3Swdenk len += sizeof(environment.flags); 399*d0fb80c3Swdenk } 4006aff3115Swdenk 4016aff3115Swdenk if (mode == O_RDWR) { 402*d0fb80c3Swdenk if (HaveRedundEnv) { 4036aff3115Swdenk /* switch to next partition for writing */ 4046aff3115Swdenk otherdev = !curdev; 4056aff3115Swdenk if ((fdr = open(DEVNAME(otherdev), mode)) < 0) { 4066aff3115Swdenk fprintf (stderr, 4076aff3115Swdenk "Can't open %s: %s\n", 4086aff3115Swdenk DEVNAME(otherdev), strerror(errno)); 4096aff3115Swdenk return (-1); 4106aff3115Swdenk } 411*d0fb80c3Swdenk } else { 4126aff3115Swdenk otherdev = curdev; 4136aff3115Swdenk fdr = fd; 414*d0fb80c3Swdenk } 4156aff3115Swdenk printf("Unlocking flash...\n"); 4166aff3115Swdenk erase.length = DEVESIZE(otherdev); 417*d0fb80c3Swdenk erase.start = DEVOFFSET(otherdev); 4186aff3115Swdenk ioctl (fdr, MEMUNLOCK, &erase); 4196aff3115Swdenk 420*d0fb80c3Swdenk if (HaveRedundEnv) { 4216aff3115Swdenk erase.length = DEVESIZE(curdev); 422*d0fb80c3Swdenk erase.start = DEVOFFSET(curdev); 4236aff3115Swdenk ioctl (fd, MEMUNLOCK, &erase); 4246aff3115Swdenk environment.flags = active_flag; 425*d0fb80c3Swdenk } 426*d0fb80c3Swdenk 4276aff3115Swdenk printf("Done\n"); 4286aff3115Swdenk resid = DEVESIZE(otherdev) - CFG_ENV_SIZE; 4296aff3115Swdenk if (resid) { 4306aff3115Swdenk if ((data = malloc(resid)) == NULL) { 4316aff3115Swdenk fprintf(stderr, 4326aff3115Swdenk "Cannot malloc %d bytes: %s\n", 4336aff3115Swdenk resid, strerror(errno)); 4346aff3115Swdenk return (-1); 4356aff3115Swdenk } 436*d0fb80c3Swdenk if (lseek (fdr, DEVOFFSET(otherdev) + CFG_ENV_SIZE, SEEK_SET) == -1) { 4376aff3115Swdenk fprintf (stderr, 4386aff3115Swdenk "seek error on %s: %s\n", 439*d0fb80c3Swdenk DEVNAME(otherdev), strerror(errno)); 4406aff3115Swdenk return (-1); 4416aff3115Swdenk } 4426aff3115Swdenk if ((rc = read (fdr, data, resid)) != resid) { 4436aff3115Swdenk fprintf (stderr, 4446aff3115Swdenk "read error on %s: %s\n", 445*d0fb80c3Swdenk DEVNAME(otherdev), strerror(errno)); 4466aff3115Swdenk return (-1); 4476aff3115Swdenk } 4486aff3115Swdenk } 4496aff3115Swdenk 4506aff3115Swdenk printf("Erasing old environment...\n"); 4516aff3115Swdenk 4526aff3115Swdenk erase.length = DEVESIZE(otherdev); 453*d0fb80c3Swdenk erase.start = DEVOFFSET(otherdev); 4546aff3115Swdenk if (ioctl (fdr, MEMERASE, &erase) != 0) { 4556aff3115Swdenk fprintf (stderr, "MTD erase error on %s: %s\n", 4566aff3115Swdenk DEVNAME(otherdev), strerror(errno)); 4576aff3115Swdenk return (-1); 4586aff3115Swdenk } 4596aff3115Swdenk 4606aff3115Swdenk printf("Done\n"); 4616aff3115Swdenk 4626aff3115Swdenk printf("Writing environment to %s...\n",DEVNAME(otherdev)); 463*d0fb80c3Swdenk if (lseek (fdr, DEVOFFSET(otherdev), SEEK_SET) == -1) { 464*d0fb80c3Swdenk fprintf (stderr, 465*d0fb80c3Swdenk "seek error on %s: %s\n", 466*d0fb80c3Swdenk DEVNAME(otherdev), strerror(errno)); 467*d0fb80c3Swdenk return (-1); 468*d0fb80c3Swdenk } 4696aff3115Swdenk if (write(fdr, &environment, len) != len) { 4706aff3115Swdenk fprintf (stderr, 4716aff3115Swdenk "CRC write error on %s: %s\n", 4726aff3115Swdenk DEVNAME(otherdev), strerror(errno)); 4736aff3115Swdenk return (-1); 4746aff3115Swdenk } 4756aff3115Swdenk if (write(fdr, environment.data, ENV_SIZE) != ENV_SIZE) { 4766aff3115Swdenk fprintf (stderr, 4776aff3115Swdenk "Write error on %s: %s\n", 4786aff3115Swdenk DEVNAME(otherdev), strerror(errno)); 4796aff3115Swdenk return (-1); 4806aff3115Swdenk } 4816aff3115Swdenk if (resid) { 4826aff3115Swdenk if (write (fdr, data, resid) != resid) { 4836aff3115Swdenk fprintf (stderr, 4846aff3115Swdenk "write error on %s: %s\n", 4856aff3115Swdenk DEVNAME(curdev), strerror(errno)); 4866aff3115Swdenk return (-1); 4876aff3115Swdenk } 4886aff3115Swdenk free(data); 4896aff3115Swdenk } 490*d0fb80c3Swdenk if (HaveRedundEnv) { 4916aff3115Swdenk /* change flag on current active env partition */ 492*d0fb80c3Swdenk if (lseek (fd, DEVOFFSET(curdev) + sizeof(ulong), SEEK_SET) == -1) { 4936aff3115Swdenk fprintf (stderr, 4946aff3115Swdenk "seek error on %s: %s\n", 4956aff3115Swdenk DEVNAME(curdev), strerror(errno)); 4966aff3115Swdenk return (-1); 4976aff3115Swdenk } 4986aff3115Swdenk if (write (fd, &obsolete_flag, sizeof(obsolete_flag)) != 4996aff3115Swdenk sizeof(obsolete_flag)) { 5006aff3115Swdenk fprintf (stderr, 5016aff3115Swdenk "Write error on %s: %s\n", 5026aff3115Swdenk DEVNAME(curdev), strerror(errno)); 5036aff3115Swdenk return (-1); 5046aff3115Swdenk } 505*d0fb80c3Swdenk } 5066aff3115Swdenk printf("Done\n"); 5076aff3115Swdenk printf("Locking ...\n"); 5086aff3115Swdenk erase.length = DEVESIZE(otherdev); 509*d0fb80c3Swdenk erase.start = DEVOFFSET(otherdev); 5106aff3115Swdenk ioctl (fdr, MEMLOCK, &erase); 511*d0fb80c3Swdenk if (HaveRedundEnv) { 5126aff3115Swdenk erase.length = DEVESIZE(curdev); 513*d0fb80c3Swdenk erase.start = DEVOFFSET(curdev); 5146aff3115Swdenk ioctl (fd, MEMLOCK, &erase); 5156aff3115Swdenk if (close(fdr)) { 5166aff3115Swdenk fprintf (stderr, 5176aff3115Swdenk "I/O error on %s: %s\n", 5186aff3115Swdenk DEVNAME(otherdev), strerror(errno)); 5196aff3115Swdenk return (-1); 5206aff3115Swdenk } 521*d0fb80c3Swdenk } 5226aff3115Swdenk printf("Done\n"); 5236aff3115Swdenk } else { 524*d0fb80c3Swdenk 525*d0fb80c3Swdenk if (lseek (fd, DEVOFFSET(curdev), SEEK_SET) == -1) { 526*d0fb80c3Swdenk fprintf (stderr, 527*d0fb80c3Swdenk "seek error on %s: %s\n", 528*d0fb80c3Swdenk DEVNAME(curdev), strerror(errno)); 529*d0fb80c3Swdenk return (-1); 530*d0fb80c3Swdenk } 5316aff3115Swdenk if (read (fd, &environment, len) != len) { 5326aff3115Swdenk fprintf (stderr, 5336aff3115Swdenk "CRC read error on %s: %s\n", 5346aff3115Swdenk DEVNAME(curdev), strerror(errno)); 5356aff3115Swdenk return (-1); 5366aff3115Swdenk } 5376aff3115Swdenk if ((rc = read (fd, environment.data, ENV_SIZE)) != ENV_SIZE) { 5386aff3115Swdenk fprintf (stderr, 5396aff3115Swdenk "Read error on %s: %s\n", 5406aff3115Swdenk DEVNAME(curdev), strerror(errno)); 5416aff3115Swdenk return (-1); 5426aff3115Swdenk } 5436aff3115Swdenk } 5446aff3115Swdenk 5456aff3115Swdenk if (close(fd)) { 5466aff3115Swdenk fprintf (stderr, 5476aff3115Swdenk "I/O error on %s: %s\n", 5486aff3115Swdenk DEVNAME(curdev), strerror(errno)); 5496aff3115Swdenk return (-1); 5506aff3115Swdenk } 5516aff3115Swdenk 5526aff3115Swdenk /* everything ok */ 5536aff3115Swdenk return (0); 5546aff3115Swdenk } 5556aff3115Swdenk 5566aff3115Swdenk /* 5576aff3115Swdenk * s1 is either a simple 'name', or a 'name=value' pair. 5586aff3115Swdenk * s2 is a 'name=value' pair. 5596aff3115Swdenk * If the names match, return the value of s2, else NULL. 5606aff3115Swdenk */ 5616aff3115Swdenk 5626aff3115Swdenk static uchar * 5636aff3115Swdenk envmatch (uchar *s1, uchar *s2) 5646aff3115Swdenk { 5656aff3115Swdenk 5666aff3115Swdenk while (*s1 == *s2++) 5676aff3115Swdenk if (*s1++ == '=') 5686aff3115Swdenk return(s2); 5696aff3115Swdenk if (*s1 == '\0' && *(s2-1) == '=') 5706aff3115Swdenk return(s2); 5716aff3115Swdenk return(NULL); 5726aff3115Swdenk } 5736aff3115Swdenk 5746aff3115Swdenk /* 5756aff3115Swdenk * Prevent confusion if running from erased flash memory 5766aff3115Swdenk */ 5776aff3115Swdenk static int env_init(void) 5786aff3115Swdenk { 5796aff3115Swdenk int crc1, crc1_ok; 5806aff3115Swdenk uchar *addr1; 581*d0fb80c3Swdenk 5826aff3115Swdenk int crc2, crc2_ok; 5836aff3115Swdenk uchar flag1, flag2, *addr2; 5846aff3115Swdenk 5856aff3115Swdenk if (parse_config()) /* should fill envdevices */ 5866aff3115Swdenk return 1; 5876aff3115Swdenk 5886aff3115Swdenk if ((addr1 = calloc (1, ENV_SIZE)) == NULL) { 5896aff3115Swdenk fprintf (stderr, 5906aff3115Swdenk "Not enough memory for environment (%ld bytes)\n", 5916aff3115Swdenk ENV_SIZE); 5926aff3115Swdenk return (errno); 5936aff3115Swdenk } 5946aff3115Swdenk 5956aff3115Swdenk /* read environment from FLASH to local buffer */ 5966aff3115Swdenk environment.data = addr1; 5976aff3115Swdenk curdev = 0; 5986aff3115Swdenk if (flash_io (O_RDONLY)) { 5996aff3115Swdenk return (errno); 6006aff3115Swdenk } 6016aff3115Swdenk 6026aff3115Swdenk crc1_ok = ((crc1 = crc32(0, environment.data, ENV_SIZE)) 6036aff3115Swdenk == environment.crc); 604*d0fb80c3Swdenk if (!HaveRedundEnv) { 6056aff3115Swdenk if (!crc1_ok) { 6066aff3115Swdenk fprintf (stderr, 6076aff3115Swdenk "Warning: Bad CRC, using default environment\n"); 6086aff3115Swdenk environment.data = default_environment; 6096aff3115Swdenk free(addr1); 6106aff3115Swdenk } 611*d0fb80c3Swdenk } else { 6126aff3115Swdenk flag1 = environment.flags; 6136aff3115Swdenk 6146aff3115Swdenk curdev = 1; 6156aff3115Swdenk if ((addr2 = calloc (1, ENV_SIZE)) == NULL) { 6166aff3115Swdenk fprintf (stderr, 6176aff3115Swdenk "Not enough memory for environment (%ld bytes)\n", 6186aff3115Swdenk ENV_SIZE); 6196aff3115Swdenk return (errno); 6206aff3115Swdenk } 6216aff3115Swdenk environment.data = addr2; 6226aff3115Swdenk 6236aff3115Swdenk if (flash_io (O_RDONLY)) { 6246aff3115Swdenk return (errno); 6256aff3115Swdenk } 6266aff3115Swdenk 6276aff3115Swdenk crc2_ok = ((crc2 = crc32(0, environment.data, ENV_SIZE)) 6286aff3115Swdenk == environment.crc); 6296aff3115Swdenk flag2 = environment.flags; 6306aff3115Swdenk 6316aff3115Swdenk if (crc1_ok && ! crc2_ok) { 6326aff3115Swdenk environment.data = addr1; 6336aff3115Swdenk environment.flags = flag1; 6346aff3115Swdenk environment.crc = crc1; 6356aff3115Swdenk curdev = 0; 6366aff3115Swdenk free(addr2); 6376aff3115Swdenk } 6386aff3115Swdenk else if (! crc1_ok && crc2_ok) { 6396aff3115Swdenk environment.data = addr2; 6406aff3115Swdenk environment.flags = flag2; 6416aff3115Swdenk environment.crc = crc2; 6426aff3115Swdenk curdev = 1; 6436aff3115Swdenk free(addr1); 6446aff3115Swdenk } 6456aff3115Swdenk else if (! crc1_ok && ! crc2_ok) { 6466aff3115Swdenk fprintf (stderr, 6476aff3115Swdenk "Warning: Bad CRC, using default environment\n"); 6486aff3115Swdenk environment.data = default_environment; 6496aff3115Swdenk curdev = 0; 6506aff3115Swdenk free(addr2); 6516aff3115Swdenk free(addr1); 6526aff3115Swdenk } 6536aff3115Swdenk else if (flag1 == active_flag && flag2 == obsolete_flag) { 6546aff3115Swdenk environment.data = addr1; 6556aff3115Swdenk environment.flags = flag1; 6566aff3115Swdenk environment.crc = crc1; 6576aff3115Swdenk curdev = 0; 6586aff3115Swdenk free(addr2); 6596aff3115Swdenk } 6606aff3115Swdenk else if (flag1 == obsolete_flag && flag2 == active_flag) { 6616aff3115Swdenk environment.data = addr2; 6626aff3115Swdenk environment.flags = flag2; 6636aff3115Swdenk environment.crc = crc2; 6646aff3115Swdenk curdev = 1; 6656aff3115Swdenk free(addr1); 6666aff3115Swdenk } 6676aff3115Swdenk else if (flag1 == flag2) { 6686aff3115Swdenk environment.data = addr1; 6696aff3115Swdenk environment.flags = flag1; 6706aff3115Swdenk environment.crc = crc1; 6716aff3115Swdenk curdev = 0; 6726aff3115Swdenk free(addr2); 6736aff3115Swdenk } 6746aff3115Swdenk else if (flag1 == 0xFF) { 6756aff3115Swdenk environment.data = addr1; 6766aff3115Swdenk environment.flags = flag1; 6776aff3115Swdenk environment.crc = crc1; 6786aff3115Swdenk curdev = 0; 6796aff3115Swdenk free(addr2); 6806aff3115Swdenk } 6816aff3115Swdenk else if (flag2 == 0xFF) { 6826aff3115Swdenk environment.data = addr2; 6836aff3115Swdenk environment.flags = flag2; 6846aff3115Swdenk environment.crc = crc2; 6856aff3115Swdenk curdev = 1; 6866aff3115Swdenk free(addr1); 6876aff3115Swdenk } 6886aff3115Swdenk } 6896aff3115Swdenk return (0); 6906aff3115Swdenk } 6916aff3115Swdenk 6926aff3115Swdenk 6936aff3115Swdenk static int parse_config() 6946aff3115Swdenk { 6956aff3115Swdenk struct stat st; 6966aff3115Swdenk 697*d0fb80c3Swdenk #if defined(CONFIG_FILE) 698*d0fb80c3Swdenk /* Fills in DEVNAME(), ENVSIZE(), DEVESIZE(). Or don't. */ 699*d0fb80c3Swdenk if (get_config(CONFIG_FILE)) { 7006aff3115Swdenk fprintf (stderr, 701*d0fb80c3Swdenk "Cannot parse config file: %s\n", 702*d0fb80c3Swdenk strerror(errno)); 7036aff3115Swdenk return 1; 7046aff3115Swdenk } 7056aff3115Swdenk 706*d0fb80c3Swdenk #else 7076aff3115Swdenk strcpy(DEVNAME(0), DEVICE1_NAME); 708*d0fb80c3Swdenk DEVOFFSET(0) = DEVICE1_OFFSET; 7096aff3115Swdenk ENVSIZE(0) = ENV1_SIZE; 7106aff3115Swdenk DEVESIZE(0) = DEVICE1_ESIZE; 7116aff3115Swdenk #ifdef HAVE_REDUND 7126aff3115Swdenk strcpy(DEVNAME(1), DEVICE2_NAME); 713*d0fb80c3Swdenk DEVOFFSET(1) = DEVICE2_OFFSET; 7146aff3115Swdenk ENVSIZE(1) = ENV2_SIZE; 7156aff3115Swdenk DEVESIZE(1) = DEVICE2_ESIZE; 716*d0fb80c3Swdenk HaveRedundEnv = 1; 7176aff3115Swdenk #endif 718*d0fb80c3Swdenk #endif 719*d0fb80c3Swdenk if (stat (DEVNAME(0), &st)) { 720*d0fb80c3Swdenk fprintf (stderr, 721*d0fb80c3Swdenk "Cannot access MTD device %s: %s\n", 722*d0fb80c3Swdenk DEVNAME(0), strerror(errno)); 723*d0fb80c3Swdenk return 1; 724*d0fb80c3Swdenk } 725*d0fb80c3Swdenk 726*d0fb80c3Swdenk if (HaveRedundEnv && stat (DEVNAME(1), &st)) { 727*d0fb80c3Swdenk fprintf (stderr, 728*d0fb80c3Swdenk "Cannot access MTD device %s: %s\n", 729*d0fb80c3Swdenk DEVNAME(2), strerror(errno)); 730*d0fb80c3Swdenk return 1; 731*d0fb80c3Swdenk } 7326aff3115Swdenk return 0; 7336aff3115Swdenk } 734*d0fb80c3Swdenk 735*d0fb80c3Swdenk #if defined(CONFIG_FILE) 736*d0fb80c3Swdenk static int get_config (char *fname) 737*d0fb80c3Swdenk { 738*d0fb80c3Swdenk FILE *fp; 739*d0fb80c3Swdenk int i = 0; 740*d0fb80c3Swdenk int rc; 741*d0fb80c3Swdenk char dump[128]; 742*d0fb80c3Swdenk 743*d0fb80c3Swdenk if ((fp = fopen(fname, "r")) == NULL) { 744*d0fb80c3Swdenk return 1; 745*d0fb80c3Swdenk } 746*d0fb80c3Swdenk 747*d0fb80c3Swdenk while ((i < 2) && 748*d0fb80c3Swdenk ((rc = fscanf (fp, "%s %lx %lx %lx", 749*d0fb80c3Swdenk DEVNAME(i), &DEVOFFSET(i), &ENVSIZE(i), &DEVESIZE(i))) != EOF)) { 750*d0fb80c3Swdenk 751*d0fb80c3Swdenk /* Skip incomplete conversions and comment strings */ 752*d0fb80c3Swdenk if ((rc < 3) || (*DEVNAME(i) == '#')) { 753*d0fb80c3Swdenk fgets (dump, sizeof(dump), fp); /* Consume till end */ 754*d0fb80c3Swdenk continue; 755*d0fb80c3Swdenk } 756*d0fb80c3Swdenk 757*d0fb80c3Swdenk i++; 758*d0fb80c3Swdenk } 759*d0fb80c3Swdenk fclose(fp); 760*d0fb80c3Swdenk 761*d0fb80c3Swdenk HaveRedundEnv = i - 1; 762*d0fb80c3Swdenk if (!i) { /* No valid entries found */ 763*d0fb80c3Swdenk errno = EINVAL; 764*d0fb80c3Swdenk return 1; 765*d0fb80c3Swdenk } else 766*d0fb80c3Swdenk return 0; 767*d0fb80c3Swdenk } 768*d0fb80c3Swdenk #endif 769