1 /* 2 * (C) Copyright 2000-2010 3 * Wolfgang Denk, DENX Software Engineering, wd@denx.de. 4 * 5 * (C) Copyright 2001 Sysgo Real-Time Solutions, GmbH <www.elinos.com> 6 * Andreas Heppel <aheppel@sysgo.de> 7 * 8 * SPDX-License-Identifier: GPL-2.0+ 9 */ 10 11 #include <common.h> 12 #include <command.h> 13 #include <environment.h> 14 #include <linux/stddef.h> 15 #if defined(CONFIG_I2C_ENV_EEPROM_BUS) 16 #include <i2c.h> 17 #endif 18 #include <search.h> 19 #include <errno.h> 20 #include <linux/compiler.h> /* for BUG_ON */ 21 22 DECLARE_GLOBAL_DATA_PTR; 23 24 static int eeprom_bus_read(unsigned dev_addr, unsigned offset, 25 uchar *buffer, unsigned cnt) 26 { 27 int rcode; 28 #if defined(CONFIG_I2C_ENV_EEPROM_BUS) 29 int old_bus = i2c_get_bus_num(); 30 31 if (old_bus != CONFIG_I2C_ENV_EEPROM_BUS) 32 i2c_set_bus_num(CONFIG_I2C_ENV_EEPROM_BUS); 33 #endif 34 35 rcode = eeprom_read(dev_addr, offset, buffer, cnt); 36 37 #if defined(CONFIG_I2C_ENV_EEPROM_BUS) 38 i2c_set_bus_num(old_bus); 39 #endif 40 41 return rcode; 42 } 43 44 static int eeprom_bus_write(unsigned dev_addr, unsigned offset, 45 uchar *buffer, unsigned cnt) 46 { 47 int rcode; 48 #if defined(CONFIG_I2C_ENV_EEPROM_BUS) 49 int old_bus = i2c_get_bus_num(); 50 51 if (old_bus != CONFIG_I2C_ENV_EEPROM_BUS) 52 i2c_set_bus_num(CONFIG_I2C_ENV_EEPROM_BUS); 53 #endif 54 55 rcode = eeprom_write(dev_addr, offset, buffer, cnt); 56 57 #if defined(CONFIG_I2C_ENV_EEPROM_BUS) 58 i2c_set_bus_num(old_bus); 59 #endif 60 61 return rcode; 62 } 63 64 static uchar env_eeprom_get_char(int index) 65 { 66 uchar c; 67 unsigned int off = CONFIG_ENV_OFFSET; 68 69 #ifdef CONFIG_ENV_OFFSET_REDUND 70 if (gd->env_valid == ENV_REDUND) 71 off = CONFIG_ENV_OFFSET_REDUND; 72 #endif 73 eeprom_bus_read(CONFIG_SYS_DEF_EEPROM_ADDR, 74 off + index + offsetof(env_t, data), &c, 1); 75 76 return c; 77 } 78 79 static void env_eeprom_load(void) 80 { 81 char buf_env[CONFIG_ENV_SIZE]; 82 unsigned int off = CONFIG_ENV_OFFSET; 83 84 #ifdef CONFIG_ENV_OFFSET_REDUND 85 ulong len, crc[2], crc_tmp; 86 unsigned int off_env[2]; 87 uchar rdbuf[64], flags[2]; 88 int i, crc_ok[2] = {0, 0}; 89 90 eeprom_init(-1); /* prepare for EEPROM read/write */ 91 92 off_env[0] = CONFIG_ENV_OFFSET; 93 off_env[1] = CONFIG_ENV_OFFSET_REDUND; 94 95 for (i = 0; i < 2; i++) { 96 /* read CRC */ 97 eeprom_bus_read(CONFIG_SYS_DEF_EEPROM_ADDR, 98 off_env[i] + offsetof(env_t, crc), 99 (uchar *)&crc[i], sizeof(ulong)); 100 /* read FLAGS */ 101 eeprom_bus_read(CONFIG_SYS_DEF_EEPROM_ADDR, 102 off_env[i] + offsetof(env_t, flags), 103 (uchar *)&flags[i], sizeof(uchar)); 104 105 crc_tmp = 0; 106 len = ENV_SIZE; 107 off = off_env[i] + offsetof(env_t, data); 108 while (len > 0) { 109 int n = (len > sizeof(rdbuf)) ? sizeof(rdbuf) : len; 110 111 eeprom_bus_read(CONFIG_SYS_DEF_EEPROM_ADDR, off, 112 rdbuf, n); 113 114 crc_tmp = crc32(crc_tmp, rdbuf, n); 115 len -= n; 116 off += n; 117 } 118 119 if (crc_tmp == crc[i]) 120 crc_ok[i] = 1; 121 } 122 123 if (!crc_ok[0] && !crc_ok[1]) { 124 gd->env_addr = 0; 125 gd->env_valid = 0; 126 } else if (crc_ok[0] && !crc_ok[1]) { 127 gd->env_valid = ENV_VALID; 128 } else if (!crc_ok[0] && crc_ok[1]) { 129 gd->env_valid = ENV_REDUND; 130 } else { 131 /* both ok - check serial */ 132 if (flags[0] == ACTIVE_FLAG && flags[1] == OBSOLETE_FLAG) 133 gd->env_valid = ENV_VALID; 134 else if (flags[0] == OBSOLETE_FLAG && flags[1] == ACTIVE_FLAG) 135 gd->env_valid = ENV_REDUND; 136 else if (flags[0] == 0xFF && flags[1] == 0) 137 gd->env_valid = ENV_REDUND; 138 else if (flags[1] == 0xFF && flags[0] == 0) 139 gd->env_valid = ENV_VALID; 140 else /* flags are equal - almost impossible */ 141 gd->env_valid = ENV_VALID; 142 } 143 144 #else /* CONFIG_ENV_OFFSET_REDUND */ 145 ulong crc, len, new; 146 uchar rdbuf[64]; 147 148 eeprom_init(-1); /* prepare for EEPROM read/write */ 149 150 /* read old CRC */ 151 eeprom_bus_read(CONFIG_SYS_DEF_EEPROM_ADDR, 152 CONFIG_ENV_OFFSET + offsetof(env_t, crc), 153 (uchar *)&crc, sizeof(ulong)); 154 155 new = 0; 156 len = ENV_SIZE; 157 off = offsetof(env_t, data); 158 while (len > 0) { 159 int n = (len > sizeof(rdbuf)) ? sizeof(rdbuf) : len; 160 161 eeprom_bus_read(CONFIG_SYS_DEF_EEPROM_ADDR, 162 CONFIG_ENV_OFFSET + off, rdbuf, n); 163 new = crc32(new, rdbuf, n); 164 len -= n; 165 off += n; 166 } 167 168 if (crc == new) { 169 gd->env_valid = ENV_VALID; 170 } else { 171 gd->env_valid = 0; 172 } 173 #endif /* CONFIG_ENV_OFFSET_REDUND */ 174 175 off = CONFIG_ENV_OFFSET; 176 #ifdef CONFIG_ENV_OFFSET_REDUND 177 if (gd->env_valid == ENV_REDUND) 178 off = CONFIG_ENV_OFFSET_REDUND; 179 #endif 180 181 eeprom_bus_read(CONFIG_SYS_DEF_EEPROM_ADDR, 182 off, (uchar *)buf_env, CONFIG_ENV_SIZE); 183 184 env_import(buf_env, 1); 185 } 186 187 static int env_eeprom_save(void) 188 { 189 env_t env_new; 190 int rc; 191 unsigned int off = CONFIG_ENV_OFFSET; 192 #ifdef CONFIG_ENV_OFFSET_REDUND 193 unsigned int off_red = CONFIG_ENV_OFFSET_REDUND; 194 char flag_obsolete = OBSOLETE_FLAG; 195 #endif 196 197 rc = env_export(&env_new); 198 if (rc) 199 return rc; 200 201 #ifdef CONFIG_ENV_OFFSET_REDUND 202 if (gd->env_valid == ENV_VALID) { 203 off = CONFIG_ENV_OFFSET_REDUND; 204 off_red = CONFIG_ENV_OFFSET; 205 } 206 207 env_new.flags = ACTIVE_FLAG; 208 #endif 209 210 rc = eeprom_bus_write(CONFIG_SYS_DEF_EEPROM_ADDR, 211 off, (uchar *)&env_new, CONFIG_ENV_SIZE); 212 213 #ifdef CONFIG_ENV_OFFSET_REDUND 214 if (rc == 0) { 215 eeprom_bus_write(CONFIG_SYS_DEF_EEPROM_ADDR, 216 off_red + offsetof(env_t, flags), 217 (uchar *)&flag_obsolete, 1); 218 219 if (gd->env_valid == ENV_VALID) 220 gd->env_valid = ENV_REDUND; 221 else 222 gd->env_valid = ENV_VALID; 223 } 224 #endif 225 return rc; 226 } 227 228 U_BOOT_ENV_LOCATION(eeprom) = { 229 .location = ENVL_EEPROM, 230 ENV_NAME("EEPROM") 231 .get_char = env_eeprom_get_char, 232 .load = env_eeprom_load, 233 .save = env_save_ptr(env_eeprom_save), 234 }; 235