mirror of
				https://github.com/smaeul/u-boot.git
				synced 2025-10-26 09:38:14 +00:00 
			
		
		
		
	In case we have restarted u-boot there is a chance that environment buffer contains old environment (from the previous boot). If UBI volume is zero size, ubi_volume_read() doesn't modify the buffer and exits successfully. We need to clear buffer manually before reading it from UBI, so the invalid CRC will cause setting default environment in case that the UBI volume is zero size. Signed-off-by: Marcin Niestroj <m.niestroj@grinn-global.com>
		
			
				
	
	
		
			215 lines
		
	
	
		
			5.2 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			215 lines
		
	
	
		
			5.2 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /*
 | |
|  * (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 */
 |