blob: f21939186f015cd8e38c713343b75664d8effd93 [file] [log] [blame]
Tom Rini10e47792018-05-06 17:58:06 -04001// SPDX-License-Identifier: GPL-2.0+
Stuart Longlandd1f40632016-02-23 15:51:26 +10002/*
3 * (c) Copyright 2016 by VRT Technology
4 *
5 * Author:
6 * Stuart Longland <stuartl@vrt.com.au>
7 *
8 * Based on FAT environment driver
9 * (c) Copyright 2011 by Tigris Elektronik GmbH
10 *
11 * Author:
12 * Maximilian Schwerin <mvs@tigris.de>
13 *
14 * and EXT4 filesystem implementation
15 * (C) Copyright 2011 - 2012 Samsung Electronics
16 * EXT4 filesystem implementation in Uboot by
17 * Uma Shankar <uma.shankar@samsung.com>
18 * Manjunatha C Achar <a.manjunatha@samsung.com>
Stuart Longlandd1f40632016-02-23 15:51:26 +100019 */
20
Tom Riniabb9a042024-05-18 20:20:43 -060021#include <common.h>
Simon Glass655306c2020-05-10 11:39:58 -060022#include <part.h>
Stuart Longlandd1f40632016-02-23 15:51:26 +100023
24#include <command.h>
Simon Glass97385862019-08-01 09:47:00 -060025#include <env.h>
Simon Glass9d1f6192019-08-02 09:44:25 -060026#include <env_internal.h>
Stuart Longlandd1f40632016-02-23 15:51:26 +100027#include <linux/stddef.h>
28#include <malloc.h>
Andrej Rosanob7763472016-06-21 17:54:25 +020029#include <memalign.h>
Stuart Longlandd1f40632016-02-23 15:51:26 +100030#include <search.h>
31#include <errno.h>
32#include <ext4fs.h>
33#include <mmc.h>
Rogier Stam454aa192022-05-11 23:20:28 +020034#include <scsi.h>
Fiona Kluted4b57ec2024-05-01 10:54:09 +020035#include <virtio.h>
Simon Glass3ba929a2020-10-30 21:38:53 -060036#include <asm/global_data.h>
Stuart Longlandd1f40632016-02-23 15:51:26 +100037
Patrick Delaunay87c8a502020-07-28 11:51:15 +020038DECLARE_GLOBAL_DATA_PTR;
39
Patrice Chotard15dc5c02019-01-17 14:45:26 +010040__weak const char *env_ext4_get_intf(void)
41{
42 return (const char *)CONFIG_ENV_EXT4_INTERFACE;
43}
44
45__weak const char *env_ext4_get_dev_part(void)
46{
David Woodhouse6815e2c2020-08-04 10:05:47 +010047#ifdef CONFIG_MMC
48 static char *part_str;
49
50 if (!part_str) {
51 part_str = CONFIG_ENV_EXT4_DEVICE_AND_PART;
52 if (!strcmp(CONFIG_ENV_EXT4_INTERFACE, "mmc") && part_str[0] == ':') {
53 part_str = "0" CONFIG_ENV_EXT4_DEVICE_AND_PART;
54 part_str[0] += mmc_get_env_dev();
55 }
56 }
57
58 return part_str;
59#else
Patrice Chotard15dc5c02019-01-17 14:45:26 +010060 return (const char *)CONFIG_ENV_EXT4_DEVICE_AND_PART;
David Woodhouse6815e2c2020-08-04 10:05:47 +010061#endif
Patrice Chotard15dc5c02019-01-17 14:45:26 +010062}
63
Patrick Delaunay112a1952020-07-28 11:51:25 +020064static int env_ext4_save_buffer(env_t *env_new)
Stuart Longlandd1f40632016-02-23 15:51:26 +100065{
Andrej Rosanob7763472016-06-21 17:54:25 +020066 struct blk_desc *dev_desc = NULL;
Simon Glassc1c4a8f2020-05-10 11:39:57 -060067 struct disk_partition info;
Stuart Longlandd1f40632016-02-23 15:51:26 +100068 int dev, part;
69 int err;
Patrice Chotard15dc5c02019-01-17 14:45:26 +010070 const char *ifname = env_ext4_get_intf();
71 const char *dev_and_part = env_ext4_get_dev_part();
Stuart Longlandd1f40632016-02-23 15:51:26 +100072
Patrice Chotard15dc5c02019-01-17 14:45:26 +010073 part = blk_get_device_part_str(ifname, dev_and_part,
Jorge Ramirez-Ortiz4a4c26d2018-01-10 11:33:48 +010074 &dev_desc, &info, 1);
Stuart Longlandd1f40632016-02-23 15:51:26 +100075 if (part < 0)
76 return 1;
77
Andrej Rosanob7763472016-06-21 17:54:25 +020078 dev = dev_desc->devnum;
Stuart Longlandd1f40632016-02-23 15:51:26 +100079 ext4fs_set_blk_dev(dev_desc, &info);
80
Sean Andersonab125f52023-11-08 12:51:09 -050081 if (!ext4fs_mount()) {
Stuart Longlandd1f40632016-02-23 15:51:26 +100082 printf("\n** Unable to use %s %s for saveenv **\n",
Patrice Chotard15dc5c02019-01-17 14:45:26 +010083 ifname, dev_and_part);
Stuart Longlandd1f40632016-02-23 15:51:26 +100084 return 1;
85 }
86
Patrick Delaunay112a1952020-07-28 11:51:25 +020087 err = ext4fs_write(CONFIG_ENV_EXT4_FILE, (void *)env_new,
Jean-Jacques Hiblot448dd272019-02-13 12:15:25 +010088 sizeof(env_t), FILETYPE_REG);
Stuart Longlandd1f40632016-02-23 15:51:26 +100089 ext4fs_close();
90
91 if (err == -1) {
92 printf("\n** Unable to write \"%s\" from %s%d:%d **\n",
Patrice Chotard15dc5c02019-01-17 14:45:26 +010093 CONFIG_ENV_EXT4_FILE, ifname, dev, part);
Stuart Longlandd1f40632016-02-23 15:51:26 +100094 return 1;
95 }
Patrick Delaunay112a1952020-07-28 11:51:25 +020096
97 return 0;
98}
99
100static int env_ext4_save(void)
101{
102 env_t env_new;
103 int err;
104
105 err = env_export(&env_new);
106 if (err)
107 return err;
Stuart Longlandd1f40632016-02-23 15:51:26 +1000108
Patrick Delaunay112a1952020-07-28 11:51:25 +0200109 err = env_ext4_save_buffer(&env_new);
110 if (err)
111 return err;
112
113 gd->env_valid = ENV_VALID;
Stuart Longlandd1f40632016-02-23 15:51:26 +1000114 puts("done\n");
Patrick Delaunay112a1952020-07-28 11:51:25 +0200115
Stuart Longlandd1f40632016-02-23 15:51:26 +1000116 return 0;
117}
Stuart Longlandd1f40632016-02-23 15:51:26 +1000118
Patrick Delaunay53606502020-07-28 11:51:26 +0200119static int env_ext4_erase(void)
120{
121 env_t env_new;
122 int err;
123
124 memset(&env_new, 0, sizeof(env_t));
125
126 err = env_ext4_save_buffer(&env_new);
127 if (err)
128 return err;
129
130 gd->env_valid = ENV_INVALID;
131 puts("done\n");
132
133 return 0;
134}
135
Simon Glass99778492017-08-03 12:22:17 -0600136static int env_ext4_load(void)
Stuart Longlandd1f40632016-02-23 15:51:26 +1000137{
138 ALLOC_CACHE_ALIGN_BUFFER(char, buf, CONFIG_ENV_SIZE);
Andrej Rosanob7763472016-06-21 17:54:25 +0200139 struct blk_desc *dev_desc = NULL;
Simon Glassc1c4a8f2020-05-10 11:39:57 -0600140 struct disk_partition info;
Stuart Longlandd1f40632016-02-23 15:51:26 +1000141 int dev, part;
142 int err;
Andrej Rosanob7763472016-06-21 17:54:25 +0200143 loff_t off;
Patrice Chotard15dc5c02019-01-17 14:45:26 +0100144 const char *ifname = env_ext4_get_intf();
145 const char *dev_and_part = env_ext4_get_dev_part();
Stuart Longlandd1f40632016-02-23 15:51:26 +1000146
Heinrich Schuchardt9991fe82018-04-14 15:41:00 +0200147#ifdef CONFIG_MMC
Patrice Chotard15dc5c02019-01-17 14:45:26 +0100148 if (!strcmp(ifname, "mmc"))
Faiz Abbasaae518f2018-02-12 19:24:31 +0530149 mmc_initialize(NULL);
Heinrich Schuchardt9991fe82018-04-14 15:41:00 +0200150#endif
Rogier Stam454aa192022-05-11 23:20:28 +0200151#if defined(CONFIG_AHCI) || defined(CONFIG_SCSI)
152 if (!strcmp(ifname, "scsi"))
153 scsi_scan(true);
154#endif
Fiona Kluted4b57ec2024-05-01 10:54:09 +0200155#if defined(CONFIG_VIRTIO)
156 if (!strcmp(ifname, "virtio"))
157 virtio_init();
158#endif
Faiz Abbasaae518f2018-02-12 19:24:31 +0530159
Patrice Chotard15dc5c02019-01-17 14:45:26 +0100160 part = blk_get_device_part_str(ifname, dev_and_part,
Jorge Ramirez-Ortiz4a4c26d2018-01-10 11:33:48 +0100161 &dev_desc, &info, 1);
Stuart Longlandd1f40632016-02-23 15:51:26 +1000162 if (part < 0)
163 goto err_env_relocate;
164
Andrej Rosanob7763472016-06-21 17:54:25 +0200165 dev = dev_desc->devnum;
Stuart Longlandd1f40632016-02-23 15:51:26 +1000166 ext4fs_set_blk_dev(dev_desc, &info);
167
Sean Andersonab125f52023-11-08 12:51:09 -0500168 if (!ext4fs_mount()) {
Stuart Longlandd1f40632016-02-23 15:51:26 +1000169 printf("\n** Unable to use %s %s for loading the env **\n",
Patrice Chotard15dc5c02019-01-17 14:45:26 +0100170 ifname, dev_and_part);
Stuart Longlandd1f40632016-02-23 15:51:26 +1000171 goto err_env_relocate;
172 }
173
Jorge Ramirez-Ortiz4a4c26d2018-01-10 11:33:48 +0100174 err = ext4_read_file(CONFIG_ENV_EXT4_FILE, buf, 0, CONFIG_ENV_SIZE,
175 &off);
Stuart Longlandd1f40632016-02-23 15:51:26 +1000176 ext4fs_close();
177
178 if (err == -1) {
179 printf("\n** Unable to read \"%s\" from %s%d:%d **\n",
Patrice Chotard15dc5c02019-01-17 14:45:26 +0100180 CONFIG_ENV_EXT4_FILE, ifname, dev, part);
Stuart Longlandd1f40632016-02-23 15:51:26 +1000181 goto err_env_relocate;
182 }
183
Marek Vasutdfe4b7e2020-07-07 20:51:35 +0200184 err = env_import(buf, 1, H_EXTERNAL);
Patrick Delaunay87c8a502020-07-28 11:51:15 +0200185 if (!err)
186 gd->env_valid = ENV_VALID;
187
188 return err;
Stuart Longlandd1f40632016-02-23 15:51:26 +1000189
190err_env_relocate:
Simon Glass97385862019-08-01 09:47:00 -0600191 env_set_default(NULL, 0);
Simon Glass99778492017-08-03 12:22:17 -0600192
193 return -EIO;
Stuart Longlandd1f40632016-02-23 15:51:26 +1000194}
Simon Glassc10a88e2017-08-03 12:21:58 -0600195
196U_BOOT_ENV_LOCATION(ext4) = {
197 .location = ENVL_EXT4,
Simon Glassd8273ed2017-08-03 12:22:03 -0600198 ENV_NAME("EXT4")
Simon Glass082af922017-08-03 12:22:01 -0600199 .load = env_ext4_load,
Rasmus Villemoes89809e82020-02-19 09:47:42 +0000200 .save = ENV_SAVE_PTR(env_ext4_save),
Patrick Delaunay50f6f112021-02-09 11:48:50 +0100201 .erase = ENV_ERASE_PTR(env_ext4_erase),
Simon Glassc10a88e2017-08-03 12:21:58 -0600202};