]>
Commit | Line | Data |
---|---|---|
83d290c5 | 1 | // SPDX-License-Identifier: GPL-2.0+ |
fd1000b9 SL |
2 | /* |
3 | * (c) Copyright 2016 by VRT Technology | |
4 | * | |
5 | * Author: | |
6 | * Stuart Longland <[email protected]> | |
7 | * | |
8 | * Based on FAT environment driver | |
9 | * (c) Copyright 2011 by Tigris Elektronik GmbH | |
10 | * | |
11 | * Author: | |
12 | * Maximilian Schwerin <[email protected]> | |
13 | * | |
14 | * and EXT4 filesystem implementation | |
15 | * (C) Copyright 2011 - 2012 Samsung Electronics | |
16 | * EXT4 filesystem implementation in Uboot by | |
17 | * Uma Shankar <[email protected]> | |
18 | * Manjunatha C Achar <[email protected]> | |
fd1000b9 SL |
19 | */ |
20 | ||
e6f6f9e6 | 21 | #include <part.h> |
fd1000b9 SL |
22 | |
23 | #include <command.h> | |
0ac7d722 | 24 | #include <env.h> |
f3998fdc | 25 | #include <env_internal.h> |
fd1000b9 SL |
26 | #include <linux/stddef.h> |
27 | #include <malloc.h> | |
bd62e241 | 28 | #include <memalign.h> |
fd1000b9 SL |
29 | #include <search.h> |
30 | #include <errno.h> | |
31 | #include <ext4fs.h> | |
32 | #include <mmc.h> | |
54ee5ae8 | 33 | #include <scsi.h> |
3be9f399 | 34 | #include <virtio.h> |
401d1c4f | 35 | #include <asm/global_data.h> |
fd1000b9 | 36 | |
286fee50 PD |
37 | DECLARE_GLOBAL_DATA_PTR; |
38 | ||
d319edca PC |
39 | __weak const char *env_ext4_get_intf(void) |
40 | { | |
41 | return (const char *)CONFIG_ENV_EXT4_INTERFACE; | |
42 | } | |
43 | ||
44 | __weak const char *env_ext4_get_dev_part(void) | |
45 | { | |
b0493bb7 DW |
46 | #ifdef CONFIG_MMC |
47 | static char *part_str; | |
48 | ||
49 | if (!part_str) { | |
50 | part_str = CONFIG_ENV_EXT4_DEVICE_AND_PART; | |
51 | if (!strcmp(CONFIG_ENV_EXT4_INTERFACE, "mmc") && part_str[0] == ':') { | |
52 | part_str = "0" CONFIG_ENV_EXT4_DEVICE_AND_PART; | |
53 | part_str[0] += mmc_get_env_dev(); | |
54 | } | |
55 | } | |
56 | ||
57 | return part_str; | |
58 | #else | |
d319edca | 59 | return (const char *)CONFIG_ENV_EXT4_DEVICE_AND_PART; |
b0493bb7 | 60 | #endif |
d319edca PC |
61 | } |
62 | ||
f6de047e | 63 | static int env_ext4_save_buffer(env_t *env_new) |
fd1000b9 | 64 | { |
bd62e241 | 65 | struct blk_desc *dev_desc = NULL; |
0528979f | 66 | struct disk_partition info; |
fd1000b9 SL |
67 | int dev, part; |
68 | int err; | |
d319edca PC |
69 | const char *ifname = env_ext4_get_intf(); |
70 | const char *dev_and_part = env_ext4_get_dev_part(); | |
fd1000b9 | 71 | |
d319edca | 72 | part = blk_get_device_part_str(ifname, dev_and_part, |
1087a794 | 73 | &dev_desc, &info, 1); |
fd1000b9 SL |
74 | if (part < 0) |
75 | return 1; | |
76 | ||
bd62e241 | 77 | dev = dev_desc->devnum; |
fd1000b9 SL |
78 | ext4fs_set_blk_dev(dev_desc, &info); |
79 | ||
7667bdeb | 80 | if (!ext4fs_mount()) { |
fd1000b9 | 81 | printf("\n** Unable to use %s %s for saveenv **\n", |
d319edca | 82 | ifname, dev_and_part); |
fd1000b9 SL |
83 | return 1; |
84 | } | |
85 | ||
f6de047e | 86 | err = ext4fs_write(CONFIG_ENV_EXT4_FILE, (void *)env_new, |
5efc0686 | 87 | sizeof(env_t), FILETYPE_REG); |
fd1000b9 SL |
88 | ext4fs_close(); |
89 | ||
90 | if (err == -1) { | |
91 | printf("\n** Unable to write \"%s\" from %s%d:%d **\n", | |
d319edca | 92 | CONFIG_ENV_EXT4_FILE, ifname, dev, part); |
fd1000b9 SL |
93 | return 1; |
94 | } | |
95 | ||
f6de047e PD |
96 | return 0; |
97 | } | |
98 | ||
99 | static int env_ext4_save(void) | |
100 | { | |
101 | env_t env_new; | |
102 | int err; | |
103 | ||
104 | err = env_export(&env_new); | |
105 | if (err) | |
106 | return err; | |
107 | ||
108 | err = env_ext4_save_buffer(&env_new); | |
109 | if (err) | |
110 | return err; | |
111 | ||
112 | gd->env_valid = ENV_VALID; | |
fd1000b9 | 113 | puts("done\n"); |
f6de047e | 114 | |
fd1000b9 SL |
115 | return 0; |
116 | } | |
fd1000b9 | 117 | |
0718f743 PD |
118 | static int env_ext4_erase(void) |
119 | { | |
120 | env_t env_new; | |
121 | int err; | |
122 | ||
123 | memset(&env_new, 0, sizeof(env_t)); | |
124 | ||
125 | err = env_ext4_save_buffer(&env_new); | |
126 | if (err) | |
127 | return err; | |
128 | ||
129 | gd->env_valid = ENV_INVALID; | |
130 | puts("done\n"); | |
131 | ||
132 | return 0; | |
133 | } | |
134 | ||
c5951991 | 135 | static int env_ext4_load(void) |
fd1000b9 SL |
136 | { |
137 | ALLOC_CACHE_ALIGN_BUFFER(char, buf, CONFIG_ENV_SIZE); | |
bd62e241 | 138 | struct blk_desc *dev_desc = NULL; |
0528979f | 139 | struct disk_partition info; |
fd1000b9 SL |
140 | int dev, part; |
141 | int err; | |
bd62e241 | 142 | loff_t off; |
d319edca PC |
143 | const char *ifname = env_ext4_get_intf(); |
144 | const char *dev_and_part = env_ext4_get_dev_part(); | |
fd1000b9 | 145 | |
95058fbb | 146 | #ifdef CONFIG_MMC |
d319edca | 147 | if (!strcmp(ifname, "mmc")) |
26862b4a | 148 | mmc_initialize(NULL); |
95058fbb | 149 | #endif |
54ee5ae8 RS |
150 | #if defined(CONFIG_AHCI) || defined(CONFIG_SCSI) |
151 | if (!strcmp(ifname, "scsi")) | |
152 | scsi_scan(true); | |
153 | #endif | |
3be9f399 FK |
154 | #if defined(CONFIG_VIRTIO) |
155 | if (!strcmp(ifname, "virtio")) | |
156 | virtio_init(); | |
157 | #endif | |
26862b4a | 158 | |
d319edca | 159 | part = blk_get_device_part_str(ifname, dev_and_part, |
1087a794 | 160 | &dev_desc, &info, 1); |
fd1000b9 SL |
161 | if (part < 0) |
162 | goto err_env_relocate; | |
163 | ||
bd62e241 | 164 | dev = dev_desc->devnum; |
fd1000b9 SL |
165 | ext4fs_set_blk_dev(dev_desc, &info); |
166 | ||
7667bdeb | 167 | if (!ext4fs_mount()) { |
fd1000b9 | 168 | printf("\n** Unable to use %s %s for loading the env **\n", |
d319edca | 169 | ifname, dev_and_part); |
fd1000b9 SL |
170 | goto err_env_relocate; |
171 | } | |
172 | ||
1087a794 JRO |
173 | err = ext4_read_file(CONFIG_ENV_EXT4_FILE, buf, 0, CONFIG_ENV_SIZE, |
174 | &off); | |
fd1000b9 SL |
175 | ext4fs_close(); |
176 | ||
177 | if (err == -1) { | |
178 | printf("\n** Unable to read \"%s\" from %s%d:%d **\n", | |
d319edca | 179 | CONFIG_ENV_EXT4_FILE, ifname, dev, part); |
fd1000b9 SL |
180 | goto err_env_relocate; |
181 | } | |
182 | ||
890feeca | 183 | err = env_import(buf, 1, H_EXTERNAL); |
286fee50 PD |
184 | if (!err) |
185 | gd->env_valid = ENV_VALID; | |
186 | ||
187 | return err; | |
fd1000b9 SL |
188 | |
189 | err_env_relocate: | |
0ac7d722 | 190 | env_set_default(NULL, 0); |
c5951991 SG |
191 | |
192 | return -EIO; | |
fd1000b9 | 193 | } |
4415f1d1 SG |
194 | |
195 | U_BOOT_ENV_LOCATION(ext4) = { | |
196 | .location = ENVL_EXT4, | |
ac358beb | 197 | ENV_NAME("EXT4") |
e5bce247 | 198 | .load = env_ext4_load, |
1df96a7e | 199 | .save = ENV_SAVE_PTR(env_ext4_save), |
1af031ac | 200 | .erase = ENV_ERASE_PTR(env_ext4_erase), |
4415f1d1 | 201 | }; |