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