123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214 |
- /*
- * (c) Copyright 2012 by National Instruments,
- * Joe Hershberger <joe.hershberger@ni.com>
- *
- * SPDX-License-Identifier: GPL-2.0+
- */
- #include <common.h>
- #include <command.h>
- #include <environment.h>
- #include <errno.h>
- #include <malloc.h>
- #include <memalign.h>
- #include <search.h>
- #include <ubi_uboot.h>
- #undef crc32
- char *env_name_spec = "UBI";
- env_t *env_ptr;
- DECLARE_GLOBAL_DATA_PTR;
- int env_init(void)
- {
- /* use default */
- gd->env_addr = (ulong)&default_environment[0];
- gd->env_valid = 1;
- return 0;
- }
- #ifdef CONFIG_CMD_SAVEENV
- #ifdef CONFIG_SYS_REDUNDAND_ENVIRONMENT
- static unsigned char env_flags;
- int saveenv(void)
- {
- ALLOC_CACHE_ALIGN_BUFFER(env_t, env_new, 1);
- int ret;
- ret = env_export(env_new);
- if (ret)
- return ret;
- if (ubi_part(CONFIG_ENV_UBI_PART, NULL)) {
- printf("\n** Cannot find mtd partition \"%s\"\n",
- CONFIG_ENV_UBI_PART);
- return 1;
- }
- env_new->flags = ++env_flags; /* increase the serial */
- if (gd->env_valid == 1) {
- puts("Writing to redundant UBI... ");
- if (ubi_volume_write(CONFIG_ENV_UBI_VOLUME_REDUND,
- (void *)env_new, CONFIG_ENV_SIZE)) {
- printf("\n** Unable to write env to %s:%s **\n",
- CONFIG_ENV_UBI_PART,
- CONFIG_ENV_UBI_VOLUME_REDUND);
- return 1;
- }
- } else {
- puts("Writing to UBI... ");
- if (ubi_volume_write(CONFIG_ENV_UBI_VOLUME,
- (void *)env_new, CONFIG_ENV_SIZE)) {
- printf("\n** Unable to write env to %s:%s **\n",
- CONFIG_ENV_UBI_PART,
- CONFIG_ENV_UBI_VOLUME);
- return 1;
- }
- }
- puts("done\n");
- gd->env_valid = gd->env_valid == 2 ? 1 : 2;
- return 0;
- }
- #else /* ! CONFIG_SYS_REDUNDAND_ENVIRONMENT */
- int saveenv(void)
- {
- ALLOC_CACHE_ALIGN_BUFFER(env_t, env_new, 1);
- int ret;
- ret = env_export(env_new);
- if (ret)
- return ret;
- if (ubi_part(CONFIG_ENV_UBI_PART, NULL)) {
- printf("\n** Cannot find mtd partition \"%s\"\n",
- CONFIG_ENV_UBI_PART);
- return 1;
- }
- if (ubi_volume_write(CONFIG_ENV_UBI_VOLUME, (void *)env_new,
- CONFIG_ENV_SIZE)) {
- printf("\n** Unable to write env to %s:%s **\n",
- CONFIG_ENV_UBI_PART, CONFIG_ENV_UBI_VOLUME);
- return 1;
- }
- puts("done\n");
- return 0;
- }
- #endif /* CONFIG_SYS_REDUNDAND_ENVIRONMENT */
- #endif /* CONFIG_CMD_SAVEENV */
- #ifdef CONFIG_SYS_REDUNDAND_ENVIRONMENT
- void env_relocate_spec(void)
- {
- ALLOC_CACHE_ALIGN_BUFFER(char, env1_buf, CONFIG_ENV_SIZE);
- ALLOC_CACHE_ALIGN_BUFFER(char, env2_buf, CONFIG_ENV_SIZE);
- int crc1_ok = 0, crc2_ok = 0;
- env_t *ep, *tmp_env1, *tmp_env2;
- /*
- * In case we have restarted u-boot there is a chance that buffer
- * contains old environment (from the previous boot).
- * If UBI volume is zero size, ubi_volume_read() doesn't modify the
- * buffer.
- * We need to clear buffer manually here, so the invalid CRC will
- * cause setting default environment as expected.
- */
- memset(env1_buf, 0x0, CONFIG_ENV_SIZE);
- memset(env2_buf, 0x0, CONFIG_ENV_SIZE);
- tmp_env1 = (env_t *)env1_buf;
- tmp_env2 = (env_t *)env2_buf;
- if (ubi_part(CONFIG_ENV_UBI_PART, NULL)) {
- printf("\n** Cannot find mtd partition \"%s\"\n",
- CONFIG_ENV_UBI_PART);
- set_default_env(NULL);
- return;
- }
- if (ubi_volume_read(CONFIG_ENV_UBI_VOLUME, (void *)tmp_env1,
- CONFIG_ENV_SIZE)) {
- printf("\n** Unable to read env from %s:%s **\n",
- CONFIG_ENV_UBI_PART, CONFIG_ENV_UBI_VOLUME);
- }
- if (ubi_volume_read(CONFIG_ENV_UBI_VOLUME_REDUND, (void *)tmp_env2,
- CONFIG_ENV_SIZE)) {
- printf("\n** Unable to read redundant env from %s:%s **\n",
- CONFIG_ENV_UBI_PART, CONFIG_ENV_UBI_VOLUME_REDUND);
- }
- crc1_ok = crc32(0, tmp_env1->data, ENV_SIZE) == tmp_env1->crc;
- crc2_ok = crc32(0, tmp_env2->data, ENV_SIZE) == tmp_env2->crc;
- if (!crc1_ok && !crc2_ok) {
- set_default_env("!bad CRC");
- return;
- } else if (crc1_ok && !crc2_ok) {
- gd->env_valid = 1;
- } else if (!crc1_ok && crc2_ok) {
- gd->env_valid = 2;
- } else {
- /* both ok - check serial */
- if (tmp_env1->flags == 255 && tmp_env2->flags == 0)
- gd->env_valid = 2;
- else if (tmp_env2->flags == 255 && tmp_env1->flags == 0)
- gd->env_valid = 1;
- else if (tmp_env1->flags > tmp_env2->flags)
- gd->env_valid = 1;
- else if (tmp_env2->flags > tmp_env1->flags)
- gd->env_valid = 2;
- else /* flags are equal - almost impossible */
- gd->env_valid = 1;
- }
- if (gd->env_valid == 1)
- ep = tmp_env1;
- else
- ep = tmp_env2;
- env_flags = ep->flags;
- env_import((char *)ep, 0);
- }
- #else /* ! CONFIG_SYS_REDUNDAND_ENVIRONMENT */
- void env_relocate_spec(void)
- {
- ALLOC_CACHE_ALIGN_BUFFER(char, buf, CONFIG_ENV_SIZE);
- /*
- * In case we have restarted u-boot there is a chance that buffer
- * contains old environment (from the previous boot).
- * If UBI volume is zero size, ubi_volume_read() doesn't modify the
- * buffer.
- * We need to clear buffer manually here, so the invalid CRC will
- * cause setting default environment as expected.
- */
- memset(buf, 0x0, CONFIG_ENV_SIZE);
- if (ubi_part(CONFIG_ENV_UBI_PART, NULL)) {
- printf("\n** Cannot find mtd partition \"%s\"\n",
- CONFIG_ENV_UBI_PART);
- set_default_env(NULL);
- return;
- }
- if (ubi_volume_read(CONFIG_ENV_UBI_VOLUME, buf, CONFIG_ENV_SIZE)) {
- printf("\n** Unable to read env from %s:%s **\n",
- CONFIG_ENV_UBI_PART, CONFIG_ENV_UBI_VOLUME);
- set_default_env(NULL);
- return;
- }
- env_import(buf, 1);
- }
- #endif /* CONFIG_SYS_REDUNDAND_ENVIRONMENT */
|