1 // SPDX-License-Identifier: GPL-2.0+
2 /*
3 * (c) Copyright 2011 by Tigris Elektronik GmbH
4 *
5 * Author:
6 * Maximilian Schwerin <mvs@tigris.de>
7 */
8
9 #include <common.h>
10
11 #include <command.h>
12 #include <environment.h>
13 #include <linux/stddef.h>
14 #include <malloc.h>
15 #include <memalign.h>
16 #include <search.h>
17 #include <errno.h>
18 #include <fat.h>
19 #include <mmc.h>
20
21 #ifdef CONFIG_SPL_BUILD
22 /* TODO(sjg@chromium.org): Figure out why this is needed */
23 # if !defined(CONFIG_TARGET_AM335X_EVM) || defined(CONFIG_SPL_OS_BOOT)
24 # define LOADENV
25 # endif
26 #else
27 # define LOADENV
28 # if defined(CONFIG_CMD_SAVEENV)
29 # define CMD_SAVEENV
30 # endif
31 #endif
32
33 #ifdef CMD_SAVEENV
env_fat_save(void)34 static int env_fat_save(void)
35 {
36 env_t __aligned(ARCH_DMA_MINALIGN) env_new;
37 struct blk_desc *dev_desc = NULL;
38 disk_partition_t info;
39 int dev, part;
40 int err;
41 loff_t size;
42
43 err = env_export(&env_new);
44 if (err)
45 return err;
46
47 part = blk_get_device_part_str(CONFIG_ENV_FAT_INTERFACE,
48 CONFIG_ENV_FAT_DEVICE_AND_PART,
49 &dev_desc, &info, 1);
50 if (part < 0)
51 return 1;
52
53 dev = dev_desc->devnum;
54 if (fat_set_blk_dev(dev_desc, &info) != 0) {
55 /*
56 * This printf is embedded in the messages from env_save that
57 * will calling it. The missing \n is intentional.
58 */
59 printf("Unable to use %s %d:%d... ",
60 CONFIG_ENV_FAT_INTERFACE, dev, part);
61 return 1;
62 }
63
64 err = file_fat_write(CONFIG_ENV_FAT_FILE, (void *)&env_new, 0, sizeof(env_t),
65 &size);
66 if (err == -1) {
67 /*
68 * This printf is embedded in the messages from env_save that
69 * will calling it. The missing \n is intentional.
70 */
71 printf("Unable to write \"%s\" from %s%d:%d... ",
72 CONFIG_ENV_FAT_FILE, CONFIG_ENV_FAT_INTERFACE, dev, part);
73 return 1;
74 }
75
76 return 0;
77 }
78 #endif /* CMD_SAVEENV */
79
80 #ifdef LOADENV
env_fat_load(void)81 static int env_fat_load(void)
82 {
83 ALLOC_CACHE_ALIGN_BUFFER(char, buf, CONFIG_ENV_SIZE);
84 struct blk_desc *dev_desc = NULL;
85 disk_partition_t info;
86 int dev, part;
87 int err;
88
89 #ifdef CONFIG_MMC
90 if (!strcmp(CONFIG_ENV_FAT_INTERFACE, "mmc"))
91 mmc_initialize(NULL);
92 #endif
93
94 part = blk_get_device_part_str(CONFIG_ENV_FAT_INTERFACE,
95 CONFIG_ENV_FAT_DEVICE_AND_PART,
96 &dev_desc, &info, 1);
97 if (part < 0)
98 goto err_env_relocate;
99
100 dev = dev_desc->devnum;
101 if (fat_set_blk_dev(dev_desc, &info) != 0) {
102 /*
103 * This printf is embedded in the messages from env_save that
104 * will calling it. The missing \n is intentional.
105 */
106 printf("Unable to use %s %d:%d... ",
107 CONFIG_ENV_FAT_INTERFACE, dev, part);
108 goto err_env_relocate;
109 }
110
111 err = file_fat_read(CONFIG_ENV_FAT_FILE, buf, CONFIG_ENV_SIZE);
112 if (err == -1) {
113 /*
114 * This printf is embedded in the messages from env_save that
115 * will calling it. The missing \n is intentional.
116 */
117 printf("Unable to read \"%s\" from %s%d:%d... ",
118 CONFIG_ENV_FAT_FILE, CONFIG_ENV_FAT_INTERFACE, dev, part);
119 goto err_env_relocate;
120 }
121
122 return env_import(buf, 1);
123
124 err_env_relocate:
125 set_default_env(NULL, 0);
126
127 return -EIO;
128 }
129 #endif /* LOADENV */
130
131 U_BOOT_ENV_LOCATION(fat) = {
132 .location = ENVL_FAT,
133 ENV_NAME("FAT")
134 #ifdef LOADENV
135 .load = env_fat_load,
136 #endif
137 #ifdef CMD_SAVEENV
138 .save = env_save_ptr(env_fat_save),
139 #endif
140 };
141