1 /*
2 * Copyright (c) 2020-2021 Huawei Device Co., Ltd. All rights reserved.
3 *
4 * Redistribution and use in source and binary forms, with or without modification,
5 * are permitted provided that the following conditions are met:
6 *
7 * 1. Redistributions of source code must retain the above copyright notice, this list of
8 * conditions and the following disclaimer.
9 *
10 * 2. Redistributions in binary form must reproduce the above copyright notice, this list
11 * of conditions and the following disclaimer in the documentation and/or other materials
12 * provided with the distribution.
13 *
14 * 3. Neither the name of the copyright holder nor the names of its contributors may be used
15 * to endorse or promote products derived from this software without specific prior written
16 * permission.
17 *
18 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
19 * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO,
20 * THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
21 * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR
22 * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
23 * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
24 * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS;
25 * OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY,
26 * WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR
27 * OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF
28 * ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
29 */
30
31 #include "los_bootargs.h"
32 #include "los_base.h"
33 #include "string.h"
34
35 #if defined(LOSCFG_STORAGE_SPINOR) || defined(LOSCFG_STORAGE_SPINAND)
36 #include "mtd_list.h"
37 #endif
38
39 #ifdef LOSCFG_STORAGE_EMMC
40 #include "disk.h"
41 #endif
42
43 STATIC CHAR *g_cmdLine = NULL;
44 STATIC UINT64 g_alignSize = 0;
45 STATIC struct BootArgs g_bootArgs[MAX_ARGS_NUM] = {0};
46
LOS_GetCmdLine(VOID)47 INT32 LOS_GetCmdLine(VOID)
48 {
49 int ret;
50
51 g_cmdLine = (CHAR *)malloc(COMMAND_LINE_SIZE);
52 if (g_cmdLine == NULL) {
53 PRINT_ERR("Malloc g_cmdLine space error!\n");
54 return LOS_NOK;
55 }
56
57 #ifdef LOSCFG_STORAGE_EMMC
58 los_disk *emmcDisk = los_get_mmcdisk_bytype(EMMC);
59 if (emmcDisk == NULL) {
60 PRINT_ERR("Get EMMC disk failed!\n");
61 goto ERROUT;
62 }
63 g_alignSize = EMMC_SEC_SIZE;
64 ret = los_disk_read(emmcDisk->disk_id, g_cmdLine, COMMAND_LINE_ADDR / EMMC_SEC_SIZE,
65 COMMAND_LINE_SIZE / EMMC_SEC_SIZE, TRUE);
66 if (ret == 0) {
67 return LOS_OK;
68 }
69 #endif
70
71 #ifdef LOSCFG_STORAGE_SPINOR
72 struct MtdDev *mtd = GetMtd("spinor");
73 if (mtd == NULL) {
74 PRINT_ERR("Get spinor mtd failed!\n");
75 goto ERROUT;
76 }
77 g_alignSize = mtd->eraseSize;
78 ret = mtd->read(mtd, COMMAND_LINE_ADDR, COMMAND_LINE_SIZE, g_cmdLine);
79 if (ret == COMMAND_LINE_SIZE) {
80 return LOS_OK;
81 }
82 #endif
83
84 #ifdef LOSCFG_STORAGE_SPINAND
85 struct MtdDev *mtd = GetMtd("nand");
86 if (mtd == NULL) {
87 PRINT_ERR("Get nand mtd failed!\n");
88 goto ERROUT;
89 }
90 g_alignSize = mtd->eraseSize;
91 ret = mtd->read(mtd, COMMAND_LINE_ADDR, COMMAND_LINE_SIZE, g_cmdLine);
92 if (ret == COMMAND_LINE_SIZE) {
93 return LOS_OK;
94 }
95 #endif
96
97 PRINT_ERR("Read cmdline error!\n");
98 ERROUT:
99 free(g_cmdLine);
100 g_cmdLine = NULL;
101 return LOS_NOK;
102 }
103
LOS_FreeCmdLine(VOID)104 VOID LOS_FreeCmdLine(VOID)
105 {
106 if (g_cmdLine != NULL) {
107 free(g_cmdLine);
108 g_cmdLine = NULL;
109 }
110 }
111
GetBootargs(CHAR ** args)112 STATIC INT32 GetBootargs(CHAR **args)
113 {
114 #ifdef LOSCFG_BOOTENV_RAM
115 *args = OsGetArgsAddr();
116 return LOS_OK;
117 #else
118 INT32 i;
119 INT32 len = 0;
120 CHAR *tmp = NULL;
121 const CHAR *bootargsName = "bootargs=";
122
123 if (g_cmdLine == NULL) {
124 PRINT_ERR("Should call LOS_GetCmdLine() first!\n");
125 return LOS_NOK;
126 }
127
128 for (i = 0; i < COMMAND_LINE_SIZE; i += len + 1) {
129 len = strlen(g_cmdLine + i);
130 tmp = strstr(g_cmdLine + i, bootargsName);
131 if (tmp != NULL) {
132 *args = tmp + strlen(bootargsName);
133 return LOS_OK;
134 }
135 }
136 PRINT_ERR("Cannot find bootargs!\n");
137 return LOS_NOK;
138 #endif
139 }
140
LOS_ParseBootargs(VOID)141 INT32 LOS_ParseBootargs(VOID)
142 {
143 INT32 idx = 0;
144 INT32 ret;
145 CHAR *args = NULL;
146 CHAR *argName = NULL;
147 CHAR *argValue = NULL;
148
149 ret = GetBootargs(&args);
150 if (ret != LOS_OK) {
151 return LOS_NOK;
152 }
153
154 while ((argValue = strsep(&args, " ")) != NULL) {
155 argName = strsep(&argValue, "=");
156 if (argValue == NULL) {
157 /* If the argument is not compliance with the format 'foo=bar' */
158 g_bootArgs[idx].argName = argName;
159 g_bootArgs[idx].argValue = argName;
160 } else {
161 g_bootArgs[idx].argName = argName;
162 g_bootArgs[idx].argValue = argValue;
163 }
164 if (++idx >= MAX_ARGS_NUM) {
165 /* Discard the rest arguments */
166 break;
167 }
168 }
169 return LOS_OK;
170 }
171
LOS_GetArgValue(CHAR * argName,CHAR ** argValue)172 INT32 LOS_GetArgValue(CHAR *argName, CHAR **argValue)
173 {
174 INT32 idx = 0;
175
176 while (idx < MAX_ARGS_NUM) {
177 if (g_bootArgs[idx].argName == NULL) {
178 break;
179 }
180 if (strcmp(argName, g_bootArgs[idx].argName) == 0) {
181 *argValue = g_bootArgs[idx].argValue;
182 return LOS_OK;
183 }
184 idx++;
185 }
186
187 return LOS_NOK;
188 }
189
LOS_GetAlignsize(VOID)190 UINT64 LOS_GetAlignsize(VOID)
191 {
192 return g_alignSize;
193 }
194
LOS_SizeStrToNum(CHAR * value)195 UINT64 LOS_SizeStrToNum(CHAR *value)
196 {
197 UINT64 num = 0;
198
199 /* If the string is a hexadecimal value */
200 if (sscanf_s(value, "0x%llx", &num) > 0) {
201 value += strlen("0x");
202 if (strspn(value, "0123456789abcdefABCDEF") < strlen(value)) {
203 goto ERROUT;
204 }
205 return num;
206 }
207
208 /* If the string is a decimal value in unit *Bytes */
209 INT32 ret = sscanf_s(value, "%d", &num);
210 INT32 decOffset = strspn(value, "0123456789");
211 CHAR *endPos = value + decOffset;
212 if ((ret <= 0) || (decOffset < (strlen(value) - 1))) {
213 goto ERROUT;
214 }
215
216 if (strlen(endPos) == 0) {
217 return num;
218 } else if (strcasecmp(endPos, "k") == 0) {
219 num = num * BYTES_PER_KBYTE;
220 } else if (strcasecmp(endPos, "m") == 0) {
221 num = num * BYTES_PER_MBYTE;
222 } else if (strcasecmp(endPos, "g") == 0) {
223 num = num * BYTES_PER_GBYTE;
224 } else {
225 goto ERROUT;
226 }
227
228 return num;
229
230 ERROUT:
231 PRINT_ERR("Invalid value string \"%s\"!\n", value);
232 return num;
233 }
234