ext4.c 3.8 KB
Newer Older
1
// SPDX-License-Identifier: GPL-2.0+
2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21
/*
 * (c) Copyright 2016 by VRT Technology
 *
 * Author:
 *  Stuart Longland <stuartl@vrt.com.au>
 *
 * Based on FAT environment driver
 * (c) Copyright 2011 by Tigris Elektronik GmbH
 *
 * Author:
 *  Maximilian Schwerin <mvs@tigris.de>
 *
 * and EXT4 filesystem implementation
 * (C) Copyright 2011 - 2012 Samsung Electronics
 * EXT4 filesystem implementation in Uboot by
 * Uma Shankar <uma.shankar@samsung.com>
 * Manjunatha C Achar <a.manjunatha@samsung.com>
 */

#include <common.h>
22
#include <part.h>
23 24

#include <command.h>
25
#include <env.h>
26
#include <env_internal.h>
27 28
#include <linux/stddef.h>
#include <malloc.h>
29
#include <memalign.h>
30 31 32 33 34
#include <search.h>
#include <errno.h>
#include <ext4fs.h>
#include <mmc.h>

P
Patrick Delaunay 已提交
35 36
DECLARE_GLOBAL_DATA_PTR;

37 38 39 40 41 42 43
__weak const char *env_ext4_get_intf(void)
{
	return (const char *)CONFIG_ENV_EXT4_INTERFACE;
}

__weak const char *env_ext4_get_dev_part(void)
{
44 45 46 47 48 49 50 51 52 53 54 55 56
#ifdef CONFIG_MMC
	static char *part_str;

	if (!part_str) {
		part_str = CONFIG_ENV_EXT4_DEVICE_AND_PART;
		if (!strcmp(CONFIG_ENV_EXT4_INTERFACE, "mmc") && part_str[0] == ':') {
			part_str = "0" CONFIG_ENV_EXT4_DEVICE_AND_PART;
			part_str[0] += mmc_get_env_dev();
		}
	}

	return part_str;
#else
57
	return (const char *)CONFIG_ENV_EXT4_DEVICE_AND_PART;
58
#endif
59 60
}

61
static int env_ext4_save_buffer(env_t *env_new)
62
{
63
	struct blk_desc *dev_desc = NULL;
64
	struct disk_partition info;
65 66
	int dev, part;
	int err;
67 68
	const char *ifname = env_ext4_get_intf();
	const char *dev_and_part = env_ext4_get_dev_part();
69

70
	part = blk_get_device_part_str(ifname, dev_and_part,
71
				       &dev_desc, &info, 1);
72 73 74
	if (part < 0)
		return 1;

75
	dev = dev_desc->devnum;
76 77 78 79
	ext4fs_set_blk_dev(dev_desc, &info);

	if (!ext4fs_mount(info.size)) {
		printf("\n** Unable to use %s %s for saveenv **\n",
80
		       ifname, dev_and_part);
81 82 83
		return 1;
	}

84
	err = ext4fs_write(CONFIG_ENV_EXT4_FILE, (void *)env_new,
85
			   sizeof(env_t), FILETYPE_REG);
86 87 88 89
	ext4fs_close();

	if (err == -1) {
		printf("\n** Unable to write \"%s\" from %s%d:%d **\n",
90
			CONFIG_ENV_EXT4_FILE, ifname, dev, part);
91 92 93
		return 1;
	}

94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110
	return 0;
}

static int env_ext4_save(void)
{
	env_t env_new;
	int err;

	err = env_export(&env_new);
	if (err)
		return err;

	err = env_ext4_save_buffer(&env_new);
	if (err)
		return err;

	gd->env_valid = ENV_VALID;
111
	puts("done\n");
112

113 114 115
	return 0;
}

116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132
static int env_ext4_erase(void)
{
	env_t env_new;
	int err;

	memset(&env_new, 0, sizeof(env_t));

	err = env_ext4_save_buffer(&env_new);
	if (err)
		return err;

	gd->env_valid = ENV_INVALID;
	puts("done\n");

	return 0;
}

133
static int env_ext4_load(void)
134 135
{
	ALLOC_CACHE_ALIGN_BUFFER(char, buf, CONFIG_ENV_SIZE);
136
	struct blk_desc *dev_desc = NULL;
137
	struct disk_partition info;
138 139
	int dev, part;
	int err;
140
	loff_t off;
141 142
	const char *ifname = env_ext4_get_intf();
	const char *dev_and_part = env_ext4_get_dev_part();
143

144
#ifdef CONFIG_MMC
145
	if (!strcmp(ifname, "mmc"))
146
		mmc_initialize(NULL);
147
#endif
148

149
	part = blk_get_device_part_str(ifname, dev_and_part,
150
				       &dev_desc, &info, 1);
151 152 153
	if (part < 0)
		goto err_env_relocate;

154
	dev = dev_desc->devnum;
155 156 157 158
	ext4fs_set_blk_dev(dev_desc, &info);

	if (!ext4fs_mount(info.size)) {
		printf("\n** Unable to use %s %s for loading the env **\n",
159
		       ifname, dev_and_part);
160 161 162
		goto err_env_relocate;
	}

163 164
	err = ext4_read_file(CONFIG_ENV_EXT4_FILE, buf, 0, CONFIG_ENV_SIZE,
			     &off);
165 166 167 168
	ext4fs_close();

	if (err == -1) {
		printf("\n** Unable to read \"%s\" from %s%d:%d **\n",
169
			CONFIG_ENV_EXT4_FILE, ifname, dev, part);
170 171 172
		goto err_env_relocate;
	}

173
	err = env_import(buf, 1, H_EXTERNAL);
P
Patrick Delaunay 已提交
174 175 176 177
	if (!err)
		gd->env_valid = ENV_VALID;

	return err;
178 179

err_env_relocate:
180
	env_set_default(NULL, 0);
181 182

	return -EIO;
183
}
184 185 186

U_BOOT_ENV_LOCATION(ext4) = {
	.location	= ENVL_EXT4,
S
Simon Glass 已提交
187
	ENV_NAME("EXT4")
188
	.load		= env_ext4_load,
189
	.save		= ENV_SAVE_PTR(env_ext4_save),
190 191
	.erase		= CONFIG_IS_ENABLED(CMD_ERASEENV) ? env_ext4_erase :
							    NULL,
192
};