xref: /openbmc/u-boot/drivers/fastboot/fb_getvar.c (revision caa2a2e5ab44d87faf51fafc780ecc985e0c05d6)
1*f73a7df9SAlex Kiernan // SPDX-License-Identifier: BSD-2-Clause
2*f73a7df9SAlex Kiernan /*
3*f73a7df9SAlex Kiernan  * Copyright (C) 2016 The Android Open Source Project
4*f73a7df9SAlex Kiernan  */
5*f73a7df9SAlex Kiernan 
6*f73a7df9SAlex Kiernan #include <common.h>
7*f73a7df9SAlex Kiernan #include <fastboot.h>
8*f73a7df9SAlex Kiernan #include <fastboot-internal.h>
9*f73a7df9SAlex Kiernan #include <fb_mmc.h>
10*f73a7df9SAlex Kiernan #include <fb_nand.h>
11*f73a7df9SAlex Kiernan #include <fs.h>
12*f73a7df9SAlex Kiernan #include <version.h>
13*f73a7df9SAlex Kiernan 
14*f73a7df9SAlex Kiernan static void getvar_version(char *var_parameter, char *response);
15*f73a7df9SAlex Kiernan static void getvar_bootloader_version(char *var_parameter, char *response);
16*f73a7df9SAlex Kiernan static void getvar_downloadsize(char *var_parameter, char *response);
17*f73a7df9SAlex Kiernan static void getvar_serialno(char *var_parameter, char *response);
18*f73a7df9SAlex Kiernan static void getvar_version_baseband(char *var_parameter, char *response);
19*f73a7df9SAlex Kiernan static void getvar_product(char *var_parameter, char *response);
20*f73a7df9SAlex Kiernan static void getvar_current_slot(char *var_parameter, char *response);
21*f73a7df9SAlex Kiernan static void getvar_slot_suffixes(char *var_parameter, char *response);
22*f73a7df9SAlex Kiernan static void getvar_has_slot(char *var_parameter, char *response);
23*f73a7df9SAlex Kiernan #if CONFIG_IS_ENABLED(FASTBOOT_FLASH_MMC)
24*f73a7df9SAlex Kiernan static void getvar_partition_type(char *part_name, char *response);
25*f73a7df9SAlex Kiernan #endif
26*f73a7df9SAlex Kiernan #if CONFIG_IS_ENABLED(FASTBOOT_FLASH)
27*f73a7df9SAlex Kiernan static void getvar_partition_size(char *part_name, char *response);
28*f73a7df9SAlex Kiernan #endif
29*f73a7df9SAlex Kiernan 
30*f73a7df9SAlex Kiernan static const struct {
31*f73a7df9SAlex Kiernan 	const char *variable;
32*f73a7df9SAlex Kiernan 	void (*dispatch)(char *var_parameter, char *response);
33*f73a7df9SAlex Kiernan } getvar_dispatch[] = {
34*f73a7df9SAlex Kiernan 	{
35*f73a7df9SAlex Kiernan 		.variable = "version",
36*f73a7df9SAlex Kiernan 		.dispatch = getvar_version
37*f73a7df9SAlex Kiernan 	}, {
38*f73a7df9SAlex Kiernan 		.variable = "bootloader-version",
39*f73a7df9SAlex Kiernan 		.dispatch = getvar_bootloader_version
40*f73a7df9SAlex Kiernan 	}, {
41*f73a7df9SAlex Kiernan 		.variable = "version-bootloader",
42*f73a7df9SAlex Kiernan 		.dispatch = getvar_bootloader_version
43*f73a7df9SAlex Kiernan 	}, {
44*f73a7df9SAlex Kiernan 		.variable = "downloadsize",
45*f73a7df9SAlex Kiernan 		.dispatch = getvar_downloadsize
46*f73a7df9SAlex Kiernan 	}, {
47*f73a7df9SAlex Kiernan 		.variable = "max-download-size",
48*f73a7df9SAlex Kiernan 		.dispatch = getvar_downloadsize
49*f73a7df9SAlex Kiernan 	}, {
50*f73a7df9SAlex Kiernan 		.variable = "serialno",
51*f73a7df9SAlex Kiernan 		.dispatch = getvar_serialno
52*f73a7df9SAlex Kiernan 	}, {
53*f73a7df9SAlex Kiernan 		.variable = "version-baseband",
54*f73a7df9SAlex Kiernan 		.dispatch = getvar_version_baseband
55*f73a7df9SAlex Kiernan 	}, {
56*f73a7df9SAlex Kiernan 		.variable = "product",
57*f73a7df9SAlex Kiernan 		.dispatch = getvar_product
58*f73a7df9SAlex Kiernan 	}, {
59*f73a7df9SAlex Kiernan 		.variable = "current-slot",
60*f73a7df9SAlex Kiernan 		.dispatch = getvar_current_slot
61*f73a7df9SAlex Kiernan 	}, {
62*f73a7df9SAlex Kiernan 		.variable = "slot-suffixes",
63*f73a7df9SAlex Kiernan 		.dispatch = getvar_slot_suffixes
64*f73a7df9SAlex Kiernan 	}, {
65*f73a7df9SAlex Kiernan 		.variable = "has_slot",
66*f73a7df9SAlex Kiernan 		.dispatch = getvar_has_slot
67*f73a7df9SAlex Kiernan #if CONFIG_IS_ENABLED(FASTBOOT_FLASH_MMC)
68*f73a7df9SAlex Kiernan 	}, {
69*f73a7df9SAlex Kiernan 		.variable = "partition-type",
70*f73a7df9SAlex Kiernan 		.dispatch = getvar_partition_type
71*f73a7df9SAlex Kiernan #endif
72*f73a7df9SAlex Kiernan #if CONFIG_IS_ENABLED(FASTBOOT_FLASH)
73*f73a7df9SAlex Kiernan 	}, {
74*f73a7df9SAlex Kiernan 		.variable = "partition-size",
75*f73a7df9SAlex Kiernan 		.dispatch = getvar_partition_size
76*f73a7df9SAlex Kiernan #endif
77*f73a7df9SAlex Kiernan 	}
78*f73a7df9SAlex Kiernan };
79*f73a7df9SAlex Kiernan 
getvar_version(char * var_parameter,char * response)80*f73a7df9SAlex Kiernan static void getvar_version(char *var_parameter, char *response)
81*f73a7df9SAlex Kiernan {
82*f73a7df9SAlex Kiernan 	fastboot_okay(FASTBOOT_VERSION, response);
83*f73a7df9SAlex Kiernan }
84*f73a7df9SAlex Kiernan 
getvar_bootloader_version(char * var_parameter,char * response)85*f73a7df9SAlex Kiernan static void getvar_bootloader_version(char *var_parameter, char *response)
86*f73a7df9SAlex Kiernan {
87*f73a7df9SAlex Kiernan 	fastboot_okay(U_BOOT_VERSION, response);
88*f73a7df9SAlex Kiernan }
89*f73a7df9SAlex Kiernan 
getvar_downloadsize(char * var_parameter,char * response)90*f73a7df9SAlex Kiernan static void getvar_downloadsize(char *var_parameter, char *response)
91*f73a7df9SAlex Kiernan {
92*f73a7df9SAlex Kiernan 	fastboot_response("OKAY", response, "0x%08x", fastboot_buf_size);
93*f73a7df9SAlex Kiernan }
94*f73a7df9SAlex Kiernan 
getvar_serialno(char * var_parameter,char * response)95*f73a7df9SAlex Kiernan static void getvar_serialno(char *var_parameter, char *response)
96*f73a7df9SAlex Kiernan {
97*f73a7df9SAlex Kiernan 	const char *tmp = env_get("serial#");
98*f73a7df9SAlex Kiernan 
99*f73a7df9SAlex Kiernan 	if (tmp)
100*f73a7df9SAlex Kiernan 		fastboot_okay(tmp, response);
101*f73a7df9SAlex Kiernan 	else
102*f73a7df9SAlex Kiernan 		fastboot_fail("Value not set", response);
103*f73a7df9SAlex Kiernan }
104*f73a7df9SAlex Kiernan 
getvar_version_baseband(char * var_parameter,char * response)105*f73a7df9SAlex Kiernan static void getvar_version_baseband(char *var_parameter, char *response)
106*f73a7df9SAlex Kiernan {
107*f73a7df9SAlex Kiernan 	fastboot_okay("N/A", response);
108*f73a7df9SAlex Kiernan }
109*f73a7df9SAlex Kiernan 
getvar_product(char * var_parameter,char * response)110*f73a7df9SAlex Kiernan static void getvar_product(char *var_parameter, char *response)
111*f73a7df9SAlex Kiernan {
112*f73a7df9SAlex Kiernan 	const char *board = env_get("board");
113*f73a7df9SAlex Kiernan 
114*f73a7df9SAlex Kiernan 	if (board)
115*f73a7df9SAlex Kiernan 		fastboot_okay(board, response);
116*f73a7df9SAlex Kiernan 	else
117*f73a7df9SAlex Kiernan 		fastboot_fail("Board not set", response);
118*f73a7df9SAlex Kiernan }
119*f73a7df9SAlex Kiernan 
getvar_current_slot(char * var_parameter,char * response)120*f73a7df9SAlex Kiernan static void getvar_current_slot(char *var_parameter, char *response)
121*f73a7df9SAlex Kiernan {
122*f73a7df9SAlex Kiernan 	/* A/B not implemented, for now always return _a */
123*f73a7df9SAlex Kiernan 	fastboot_okay("_a", response);
124*f73a7df9SAlex Kiernan }
125*f73a7df9SAlex Kiernan 
getvar_slot_suffixes(char * var_parameter,char * response)126*f73a7df9SAlex Kiernan static void getvar_slot_suffixes(char *var_parameter, char *response)
127*f73a7df9SAlex Kiernan {
128*f73a7df9SAlex Kiernan 	fastboot_okay("_a,_b", response);
129*f73a7df9SAlex Kiernan }
130*f73a7df9SAlex Kiernan 
getvar_has_slot(char * part_name,char * response)131*f73a7df9SAlex Kiernan static void getvar_has_slot(char *part_name, char *response)
132*f73a7df9SAlex Kiernan {
133*f73a7df9SAlex Kiernan 	if (part_name && (!strcmp(part_name, "boot") ||
134*f73a7df9SAlex Kiernan 			  !strcmp(part_name, "system")))
135*f73a7df9SAlex Kiernan 		fastboot_okay("yes", response);
136*f73a7df9SAlex Kiernan 	else
137*f73a7df9SAlex Kiernan 		fastboot_okay("no", response);
138*f73a7df9SAlex Kiernan }
139*f73a7df9SAlex Kiernan 
140*f73a7df9SAlex Kiernan #if CONFIG_IS_ENABLED(FASTBOOT_FLASH_MMC)
getvar_partition_type(char * part_name,char * response)141*f73a7df9SAlex Kiernan static void getvar_partition_type(char *part_name, char *response)
142*f73a7df9SAlex Kiernan {
143*f73a7df9SAlex Kiernan 	int r;
144*f73a7df9SAlex Kiernan 	struct blk_desc *dev_desc;
145*f73a7df9SAlex Kiernan 	disk_partition_t part_info;
146*f73a7df9SAlex Kiernan 
147*f73a7df9SAlex Kiernan 	r = fastboot_mmc_get_part_info(part_name, &dev_desc, &part_info,
148*f73a7df9SAlex Kiernan 				       response);
149*f73a7df9SAlex Kiernan 	if (r >= 0) {
150*f73a7df9SAlex Kiernan 		r = fs_set_blk_dev_with_part(dev_desc, r);
151*f73a7df9SAlex Kiernan 		if (r < 0)
152*f73a7df9SAlex Kiernan 			fastboot_fail("failed to set partition", response);
153*f73a7df9SAlex Kiernan 		else
154*f73a7df9SAlex Kiernan 			fastboot_okay(fs_get_type_name(), response);
155*f73a7df9SAlex Kiernan 	}
156*f73a7df9SAlex Kiernan }
157*f73a7df9SAlex Kiernan #endif
158*f73a7df9SAlex Kiernan 
159*f73a7df9SAlex Kiernan #if CONFIG_IS_ENABLED(FASTBOOT_FLASH)
getvar_partition_size(char * part_name,char * response)160*f73a7df9SAlex Kiernan static void getvar_partition_size(char *part_name, char *response)
161*f73a7df9SAlex Kiernan {
162*f73a7df9SAlex Kiernan 	int r;
163*f73a7df9SAlex Kiernan 	size_t size;
164*f73a7df9SAlex Kiernan 
165*f73a7df9SAlex Kiernan #if CONFIG_IS_ENABLED(FASTBOOT_FLASH_MMC)
166*f73a7df9SAlex Kiernan 	struct blk_desc *dev_desc;
167*f73a7df9SAlex Kiernan 	disk_partition_t part_info;
168*f73a7df9SAlex Kiernan 
169*f73a7df9SAlex Kiernan 	r = fastboot_mmc_get_part_info(part_name, &dev_desc, &part_info,
170*f73a7df9SAlex Kiernan 				       response);
171*f73a7df9SAlex Kiernan 	if (r >= 0)
172*f73a7df9SAlex Kiernan 		size = part_info.size;
173*f73a7df9SAlex Kiernan #endif
174*f73a7df9SAlex Kiernan #if CONFIG_IS_ENABLED(FASTBOOT_FLASH_NAND)
175*f73a7df9SAlex Kiernan 	struct part_info *part_info;
176*f73a7df9SAlex Kiernan 
177*f73a7df9SAlex Kiernan 	r = fastboot_nand_get_part_info(part_name, &part_info, response);
178*f73a7df9SAlex Kiernan 	if (r >= 0)
179*f73a7df9SAlex Kiernan 		size = part_info->size;
180*f73a7df9SAlex Kiernan #endif
181*f73a7df9SAlex Kiernan 	if (r >= 0)
182*f73a7df9SAlex Kiernan 		fastboot_response("OKAY", response, "0x%016zx", size);
183*f73a7df9SAlex Kiernan }
184*f73a7df9SAlex Kiernan #endif
185*f73a7df9SAlex Kiernan 
186*f73a7df9SAlex Kiernan /**
187*f73a7df9SAlex Kiernan  * fastboot_getvar() - Writes variable indicated by cmd_parameter to response.
188*f73a7df9SAlex Kiernan  *
189*f73a7df9SAlex Kiernan  * @cmd_parameter: Pointer to command parameter
190*f73a7df9SAlex Kiernan  * @response: Pointer to fastboot response buffer
191*f73a7df9SAlex Kiernan  *
192*f73a7df9SAlex Kiernan  * Look up cmd_parameter first as an environment variable of the form
193*f73a7df9SAlex Kiernan  * fastboot.<cmd_parameter>, if that exists return use its value to set
194*f73a7df9SAlex Kiernan  * response.
195*f73a7df9SAlex Kiernan  *
196*f73a7df9SAlex Kiernan  * Otherwise lookup the name of variable and execute the appropriate
197*f73a7df9SAlex Kiernan  * function to return the requested value.
198*f73a7df9SAlex Kiernan  */
fastboot_getvar(char * cmd_parameter,char * response)199*f73a7df9SAlex Kiernan void fastboot_getvar(char *cmd_parameter, char *response)
200*f73a7df9SAlex Kiernan {
201*f73a7df9SAlex Kiernan 	if (!cmd_parameter) {
202*f73a7df9SAlex Kiernan 		fastboot_fail("missing var", response);
203*f73a7df9SAlex Kiernan 	} else {
204*f73a7df9SAlex Kiernan #define FASTBOOT_ENV_PREFIX	"fastboot."
205*f73a7df9SAlex Kiernan 		int i;
206*f73a7df9SAlex Kiernan 		char *var_parameter = cmd_parameter;
207*f73a7df9SAlex Kiernan 		char envstr[FASTBOOT_RESPONSE_LEN];
208*f73a7df9SAlex Kiernan 		const char *s;
209*f73a7df9SAlex Kiernan 
210*f73a7df9SAlex Kiernan 		snprintf(envstr, sizeof(envstr) - 1,
211*f73a7df9SAlex Kiernan 			 FASTBOOT_ENV_PREFIX "%s", cmd_parameter);
212*f73a7df9SAlex Kiernan 		s = env_get(envstr);
213*f73a7df9SAlex Kiernan 		if (s) {
214*f73a7df9SAlex Kiernan 			fastboot_response("OKAY", response, "%s", s);
215*f73a7df9SAlex Kiernan 			return;
216*f73a7df9SAlex Kiernan 		}
217*f73a7df9SAlex Kiernan 
218*f73a7df9SAlex Kiernan 		strsep(&var_parameter, ":");
219*f73a7df9SAlex Kiernan 		for (i = 0; i < ARRAY_SIZE(getvar_dispatch); ++i) {
220*f73a7df9SAlex Kiernan 			if (!strcmp(getvar_dispatch[i].variable,
221*f73a7df9SAlex Kiernan 				    cmd_parameter)) {
222*f73a7df9SAlex Kiernan 				getvar_dispatch[i].dispatch(var_parameter,
223*f73a7df9SAlex Kiernan 							    response);
224*f73a7df9SAlex Kiernan 				return;
225*f73a7df9SAlex Kiernan 			}
226*f73a7df9SAlex Kiernan 		}
227*f73a7df9SAlex Kiernan 		pr_warn("WARNING: unknown variable: %s\n", cmd_parameter);
228*f73a7df9SAlex Kiernan 		fastboot_fail("Variable not implemented", response);
229*f73a7df9SAlex Kiernan 	}
230*f73a7df9SAlex Kiernan }
231