flash.c 9.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371
  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. * SPDX-License-Identifier: GPL-2.0+
  8. */
  9. /* #define DEBUG */
  10. #include <common.h>
  11. #include <command.h>
  12. #include <environment.h>
  13. #include <linux/stddef.h>
  14. #include <malloc.h>
  15. #include <search.h>
  16. #include <errno.h>
  17. DECLARE_GLOBAL_DATA_PTR;
  18. #ifndef CONFIG_SPL_BUILD
  19. # if defined(CONFIG_CMD_SAVEENV) && defined(CONFIG_CMD_FLASH)
  20. # define CMD_SAVEENV
  21. # elif defined(CONFIG_ENV_ADDR_REDUND)
  22. # error CONFIG_ENV_ADDR_REDUND must have CONFIG_CMD_SAVEENV & CONFIG_CMD_FLASH
  23. # endif
  24. #endif
  25. #if defined(CONFIG_ENV_SIZE_REDUND) && \
  26. (CONFIG_ENV_SIZE_REDUND < CONFIG_ENV_SIZE)
  27. #error CONFIG_ENV_SIZE_REDUND should not be less then CONFIG_ENV_SIZE
  28. #endif
  29. /* TODO(sjg@chromium.org): Figure out all these special cases */
  30. #if (!defined(CONFIG_MICROBLAZE) && !defined(CONFIG_ARCH_ZYNQ) && \
  31. !defined(CONFIG_TARGET_MCCMON6) && !defined(CONFIG_TARGET_X600) && \
  32. !defined(CONFIG_TARGET_EDMINIV2)) || \
  33. !defined(CONFIG_SPL_BUILD)
  34. #define LOADENV
  35. #endif
  36. #if !defined(CONFIG_TARGET_X600) || !defined(CONFIG_SPL_BUILD)
  37. #define INITENV
  38. #endif
  39. char *env_name_spec = "Flash";
  40. #ifdef ENV_IS_EMBEDDED
  41. env_t *env_ptr = &environment;
  42. static __maybe_unused env_t *flash_addr = (env_t *)CONFIG_ENV_ADDR;
  43. #else /* ! ENV_IS_EMBEDDED */
  44. env_t *env_ptr = (env_t *)CONFIG_ENV_ADDR;
  45. static __maybe_unused env_t *flash_addr = (env_t *)CONFIG_ENV_ADDR;
  46. #endif /* ENV_IS_EMBEDDED */
  47. /* CONFIG_ENV_ADDR is supposed to be on sector boundary */
  48. static ulong __maybe_unused end_addr =
  49. CONFIG_ENV_ADDR + CONFIG_ENV_SECT_SIZE - 1;
  50. #ifdef CONFIG_ENV_ADDR_REDUND
  51. static env_t __maybe_unused *flash_addr_new = (env_t *)CONFIG_ENV_ADDR_REDUND;
  52. /* CONFIG_ENV_ADDR_REDUND is supposed to be on sector boundary */
  53. static ulong __maybe_unused end_addr_new =
  54. CONFIG_ENV_ADDR_REDUND + CONFIG_ENV_SECT_SIZE - 1;
  55. #endif /* CONFIG_ENV_ADDR_REDUND */
  56. #ifdef CONFIG_ENV_ADDR_REDUND
  57. #ifdef INITENV
  58. static int env_flash_init(void)
  59. {
  60. int crc1_ok = 0, crc2_ok = 0;
  61. uchar flag1 = flash_addr->flags;
  62. uchar flag2 = flash_addr_new->flags;
  63. ulong addr_default = (ulong)&default_environment[0];
  64. ulong addr1 = (ulong)&(flash_addr->data);
  65. ulong addr2 = (ulong)&(flash_addr_new->data);
  66. crc1_ok = crc32(0, flash_addr->data, ENV_SIZE) == flash_addr->crc;
  67. crc2_ok =
  68. crc32(0, flash_addr_new->data, ENV_SIZE) == flash_addr_new->crc;
  69. if (crc1_ok && !crc2_ok) {
  70. gd->env_addr = addr1;
  71. gd->env_valid = ENV_VALID;
  72. } else if (!crc1_ok && crc2_ok) {
  73. gd->env_addr = addr2;
  74. gd->env_valid = ENV_VALID;
  75. } else if (!crc1_ok && !crc2_ok) {
  76. gd->env_addr = addr_default;
  77. gd->env_valid = 0;
  78. } else if (flag1 == ACTIVE_FLAG && flag2 == OBSOLETE_FLAG) {
  79. gd->env_addr = addr1;
  80. gd->env_valid = ENV_VALID;
  81. } else if (flag1 == OBSOLETE_FLAG && flag2 == ACTIVE_FLAG) {
  82. gd->env_addr = addr2;
  83. gd->env_valid = ENV_VALID;
  84. } else if (flag1 == flag2) {
  85. gd->env_addr = addr1;
  86. gd->env_valid = ENV_REDUND;
  87. } else if (flag1 == 0xFF) {
  88. gd->env_addr = addr1;
  89. gd->env_valid = ENV_REDUND;
  90. } else if (flag2 == 0xFF) {
  91. gd->env_addr = addr2;
  92. gd->env_valid = ENV_REDUND;
  93. }
  94. return 0;
  95. }
  96. #endif
  97. #ifdef CMD_SAVEENV
  98. static int env_flash_save(void)
  99. {
  100. env_t env_new;
  101. char *saved_data = NULL;
  102. char flag = OBSOLETE_FLAG, new_flag = ACTIVE_FLAG;
  103. int rc = 1;
  104. #if CONFIG_ENV_SECT_SIZE > CONFIG_ENV_SIZE
  105. ulong up_data = 0;
  106. #endif
  107. debug("Protect off %08lX ... %08lX\n", (ulong)flash_addr, end_addr);
  108. if (flash_sect_protect(0, (ulong)flash_addr, end_addr))
  109. goto done;
  110. debug("Protect off %08lX ... %08lX\n",
  111. (ulong)flash_addr_new, end_addr_new);
  112. if (flash_sect_protect(0, (ulong)flash_addr_new, end_addr_new))
  113. goto done;
  114. rc = env_export(&env_new);
  115. if (rc)
  116. return rc;
  117. env_new.flags = new_flag;
  118. #if CONFIG_ENV_SECT_SIZE > CONFIG_ENV_SIZE
  119. up_data = end_addr_new + 1 - ((long)flash_addr_new + CONFIG_ENV_SIZE);
  120. debug("Data to save 0x%lX\n", up_data);
  121. if (up_data) {
  122. saved_data = malloc(up_data);
  123. if (saved_data == NULL) {
  124. printf("Unable to save the rest of sector (%ld)\n",
  125. up_data);
  126. goto done;
  127. }
  128. memcpy(saved_data,
  129. (void *)((long)flash_addr_new + CONFIG_ENV_SIZE),
  130. up_data);
  131. debug("Data (start 0x%lX, len 0x%lX) saved at 0x%p\n",
  132. (long)flash_addr_new + CONFIG_ENV_SIZE,
  133. up_data, saved_data);
  134. }
  135. #endif
  136. puts("Erasing Flash...");
  137. debug(" %08lX ... %08lX ...", (ulong)flash_addr_new, end_addr_new);
  138. if (flash_sect_erase((ulong)flash_addr_new, end_addr_new))
  139. goto done;
  140. puts("Writing to Flash... ");
  141. debug(" %08lX ... %08lX ...",
  142. (ulong)&(flash_addr_new->data),
  143. sizeof(env_ptr->data) + (ulong)&(flash_addr_new->data));
  144. rc = flash_write((char *)&env_new, (ulong)flash_addr_new,
  145. sizeof(env_new));
  146. if (rc)
  147. goto perror;
  148. rc = flash_write(&flag, (ulong)&(flash_addr->flags),
  149. sizeof(flash_addr->flags));
  150. if (rc)
  151. goto perror;
  152. #if CONFIG_ENV_SECT_SIZE > CONFIG_ENV_SIZE
  153. if (up_data) { /* restore the rest of sector */
  154. debug("Restoring the rest of data to 0x%lX len 0x%lX\n",
  155. (long)flash_addr_new + CONFIG_ENV_SIZE, up_data);
  156. if (flash_write(saved_data,
  157. (long)flash_addr_new + CONFIG_ENV_SIZE,
  158. up_data))
  159. goto perror;
  160. }
  161. #endif
  162. puts("done\n");
  163. {
  164. env_t *etmp = flash_addr;
  165. ulong ltmp = end_addr;
  166. flash_addr = flash_addr_new;
  167. flash_addr_new = etmp;
  168. end_addr = end_addr_new;
  169. end_addr_new = ltmp;
  170. }
  171. rc = 0;
  172. goto done;
  173. perror:
  174. flash_perror(rc);
  175. done:
  176. if (saved_data)
  177. free(saved_data);
  178. /* try to re-protect */
  179. flash_sect_protect(1, (ulong)flash_addr, end_addr);
  180. flash_sect_protect(1, (ulong)flash_addr_new, end_addr_new);
  181. return rc;
  182. }
  183. #endif /* CMD_SAVEENV */
  184. #else /* ! CONFIG_ENV_ADDR_REDUND */
  185. #ifdef INITENV
  186. static int env_flash_init(void)
  187. {
  188. if (crc32(0, env_ptr->data, ENV_SIZE) == env_ptr->crc) {
  189. gd->env_addr = (ulong)&(env_ptr->data);
  190. gd->env_valid = ENV_VALID;
  191. return 0;
  192. }
  193. gd->env_addr = (ulong)&default_environment[0];
  194. gd->env_valid = 0;
  195. return 0;
  196. }
  197. #endif
  198. #ifdef CMD_SAVEENV
  199. static int env_flash_save(void)
  200. {
  201. env_t env_new;
  202. int rc = 1;
  203. char *saved_data = NULL;
  204. #if CONFIG_ENV_SECT_SIZE > CONFIG_ENV_SIZE
  205. ulong up_data = 0;
  206. up_data = end_addr + 1 - ((long)flash_addr + CONFIG_ENV_SIZE);
  207. debug("Data to save 0x%lx\n", up_data);
  208. if (up_data) {
  209. saved_data = malloc(up_data);
  210. if (saved_data == NULL) {
  211. printf("Unable to save the rest of sector (%ld)\n",
  212. up_data);
  213. goto done;
  214. }
  215. memcpy(saved_data,
  216. (void *)((long)flash_addr + CONFIG_ENV_SIZE), up_data);
  217. debug("Data (start 0x%lx, len 0x%lx) saved at 0x%lx\n",
  218. (ulong)flash_addr + CONFIG_ENV_SIZE,
  219. up_data,
  220. (ulong)saved_data);
  221. }
  222. #endif /* CONFIG_ENV_SECT_SIZE */
  223. debug("Protect off %08lX ... %08lX\n", (ulong)flash_addr, end_addr);
  224. if (flash_sect_protect(0, (long)flash_addr, end_addr))
  225. goto done;
  226. rc = env_export(&env_new);
  227. if (rc)
  228. goto done;
  229. puts("Erasing Flash...");
  230. if (flash_sect_erase((long)flash_addr, end_addr))
  231. goto done;
  232. puts("Writing to Flash... ");
  233. rc = flash_write((char *)&env_new, (long)flash_addr, CONFIG_ENV_SIZE);
  234. if (rc != 0)
  235. goto perror;
  236. #if CONFIG_ENV_SECT_SIZE > CONFIG_ENV_SIZE
  237. if (up_data) { /* restore the rest of sector */
  238. debug("Restoring the rest of data to 0x%lx len 0x%lx\n",
  239. (ulong)flash_addr + CONFIG_ENV_SIZE, up_data);
  240. if (flash_write(saved_data,
  241. (long)flash_addr + CONFIG_ENV_SIZE,
  242. up_data))
  243. goto perror;
  244. }
  245. #endif
  246. puts("done\n");
  247. rc = 0;
  248. goto done;
  249. perror:
  250. flash_perror(rc);
  251. done:
  252. if (saved_data)
  253. free(saved_data);
  254. /* try to re-protect */
  255. flash_sect_protect(1, (long)flash_addr, end_addr);
  256. return rc;
  257. }
  258. #endif /* CMD_SAVEENV */
  259. #endif /* CONFIG_ENV_ADDR_REDUND */
  260. #ifdef LOADENV
  261. static void env_flash_load(void)
  262. {
  263. #ifdef CONFIG_ENV_ADDR_REDUND
  264. if (gd->env_addr != (ulong)&(flash_addr->data)) {
  265. env_t *etmp = flash_addr;
  266. ulong ltmp = end_addr;
  267. flash_addr = flash_addr_new;
  268. flash_addr_new = etmp;
  269. end_addr = end_addr_new;
  270. end_addr_new = ltmp;
  271. }
  272. if (flash_addr_new->flags != OBSOLETE_FLAG &&
  273. crc32(0, flash_addr_new->data, ENV_SIZE) == flash_addr_new->crc) {
  274. char flag = OBSOLETE_FLAG;
  275. gd->env_valid = ENV_REDUND;
  276. flash_sect_protect(0, (ulong)flash_addr_new, end_addr_new);
  277. flash_write(&flag,
  278. (ulong)&(flash_addr_new->flags),
  279. sizeof(flash_addr_new->flags));
  280. flash_sect_protect(1, (ulong)flash_addr_new, end_addr_new);
  281. }
  282. if (flash_addr->flags != ACTIVE_FLAG &&
  283. (flash_addr->flags & ACTIVE_FLAG) == ACTIVE_FLAG) {
  284. char flag = ACTIVE_FLAG;
  285. gd->env_valid = ENV_REDUND;
  286. flash_sect_protect(0, (ulong)flash_addr, end_addr);
  287. flash_write(&flag,
  288. (ulong)&(flash_addr->flags),
  289. sizeof(flash_addr->flags));
  290. flash_sect_protect(1, (ulong)flash_addr, end_addr);
  291. }
  292. if (gd->env_valid == ENV_REDUND)
  293. puts("*** Warning - some problems detected "
  294. "reading environment; recovered successfully\n\n");
  295. #endif /* CONFIG_ENV_ADDR_REDUND */
  296. env_import((char *)flash_addr, 1);
  297. }
  298. #endif /* LOADENV */
  299. U_BOOT_ENV_LOCATION(flash) = {
  300. .location = ENVL_FLASH,
  301. #ifdef LOADENV
  302. .load = env_flash_load,
  303. #endif
  304. #ifdef CMD_SAVEENV
  305. .save = env_save_ptr(env_flash_save),
  306. #endif
  307. #ifdef INITENV
  308. .init = env_flash_init,
  309. #endif
  310. };