gpt.c 9.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465
  1. /*
  2. * cmd_gpt.c -- GPT (GUID Partition Table) handling command
  3. *
  4. * Copyright (C) 2015
  5. * Lukasz Majewski <l.majewski@majess.pl>
  6. *
  7. * Copyright (C) 2012 Samsung Electronics
  8. * author: Lukasz Majewski <l.majewski@samsung.com>
  9. * author: Piotr Wilczek <p.wilczek@samsung.com>
  10. *
  11. * SPDX-License-Identifier: GPL-2.0+
  12. */
  13. #include <common.h>
  14. #include <malloc.h>
  15. #include <command.h>
  16. #include <part_efi.h>
  17. #include <exports.h>
  18. #include <linux/ctype.h>
  19. #include <div64.h>
  20. #include <memalign.h>
  21. #ifndef CONFIG_PARTITION_UUIDS
  22. #error CONFIG_PARTITION_UUIDS must be enabled for CONFIG_CMD_GPT to be enabled
  23. #endif
  24. /**
  25. * extract_env(): Expand env name from string format '&{env_name}'
  26. * and return pointer to the env (if the env is set)
  27. *
  28. * @param str - pointer to string
  29. * @param env - pointer to pointer to extracted env
  30. *
  31. * @return - zero on successful expand and env is set
  32. */
  33. static int extract_env(const char *str, char **env)
  34. {
  35. int ret = -1;
  36. char *e, *s;
  37. #ifdef CONFIG_RANDOM_UUID
  38. char uuid_str[UUID_STR_LEN + 1];
  39. #endif
  40. if (!str || strlen(str) < 4)
  41. return -1;
  42. if (!((strncmp(str, "${", 2) == 0) && (str[strlen(str) - 1] == '}')))
  43. return -1;
  44. s = strdup(str);
  45. if (s == NULL)
  46. return -1;
  47. memset(s + strlen(s) - 1, '\0', 1);
  48. memmove(s, s + 2, strlen(s) - 1);
  49. e = getenv(s);
  50. if (e == NULL) {
  51. #ifdef CONFIG_RANDOM_UUID
  52. debug("%s unset. ", str);
  53. gen_rand_uuid_str(uuid_str, UUID_STR_FORMAT_STD);
  54. setenv(s, uuid_str);
  55. e = getenv(s);
  56. if (e) {
  57. debug("Set to random.\n");
  58. ret = 0;
  59. } else {
  60. debug("Can't get random UUID.\n");
  61. }
  62. #else
  63. debug("%s unset.\n", str);
  64. #endif
  65. } else {
  66. debug("%s get from environment.\n", str);
  67. ret = 0;
  68. }
  69. *env = e;
  70. free(s);
  71. return ret;
  72. }
  73. /**
  74. * extract_val(): Extract value from a key=value pair list (comma separated).
  75. * Only value for the given key is returend.
  76. * Function allocates memory for the value, remember to free!
  77. *
  78. * @param str - pointer to string with key=values pairs
  79. * @param key - pointer to the key to search for
  80. *
  81. * @return - pointer to allocated string with the value
  82. */
  83. static char *extract_val(const char *str, const char *key)
  84. {
  85. char *v, *k;
  86. char *s, *strcopy;
  87. char *new = NULL;
  88. strcopy = strdup(str);
  89. if (strcopy == NULL)
  90. return NULL;
  91. s = strcopy;
  92. while (s) {
  93. v = strsep(&s, ",");
  94. if (!v)
  95. break;
  96. k = strsep(&v, "=");
  97. if (!k)
  98. break;
  99. if (strcmp(k, key) == 0) {
  100. new = strdup(v);
  101. break;
  102. }
  103. }
  104. free(strcopy);
  105. return new;
  106. }
  107. /**
  108. * found_key(): Found key without value in parameter list (comma separated).
  109. *
  110. * @param str - pointer to string with key
  111. * @param key - pointer to the key to search for
  112. *
  113. * @return - true on found key
  114. */
  115. static bool found_key(const char *str, const char *key)
  116. {
  117. char *k;
  118. char *s, *strcopy;
  119. bool result = false;
  120. strcopy = strdup(str);
  121. if (!strcopy)
  122. return NULL;
  123. s = strcopy;
  124. while (s) {
  125. k = strsep(&s, ",");
  126. if (!k)
  127. break;
  128. if (strcmp(k, key) == 0) {
  129. result = true;
  130. break;
  131. }
  132. }
  133. free(strcopy);
  134. return result;
  135. }
  136. /**
  137. * set_gpt_info(): Fill partition information from string
  138. * function allocates memory, remember to free!
  139. *
  140. * @param dev_desc - pointer block device descriptor
  141. * @param str_part - pointer to string with partition information
  142. * @param str_disk_guid - pointer to pointer to allocated string with disk guid
  143. * @param partitions - pointer to pointer to allocated partitions array
  144. * @param parts_count - number of partitions
  145. *
  146. * @return - zero on success, otherwise error
  147. *
  148. */
  149. static int set_gpt_info(struct blk_desc *dev_desc,
  150. const char *str_part,
  151. char **str_disk_guid,
  152. disk_partition_t **partitions,
  153. u8 *parts_count)
  154. {
  155. char *tok, *str, *s;
  156. int i;
  157. char *val, *p;
  158. int p_count;
  159. disk_partition_t *parts;
  160. int errno = 0;
  161. uint64_t size_ll, start_ll;
  162. lbaint_t offset = 0;
  163. debug("%s: lba num: 0x%x %d\n", __func__,
  164. (unsigned int)dev_desc->lba, (unsigned int)dev_desc->lba);
  165. if (str_part == NULL)
  166. return -1;
  167. str = strdup(str_part);
  168. /* extract disk guid */
  169. s = str;
  170. val = extract_val(str, "uuid_disk");
  171. if (!val) {
  172. #ifdef CONFIG_RANDOM_UUID
  173. *str_disk_guid = malloc(UUID_STR_LEN + 1);
  174. gen_rand_uuid_str(*str_disk_guid, UUID_STR_FORMAT_STD);
  175. #else
  176. free(str);
  177. return -2;
  178. #endif
  179. } else {
  180. val = strsep(&val, ";");
  181. if (extract_env(val, &p))
  182. p = val;
  183. *str_disk_guid = strdup(p);
  184. free(val);
  185. /* Move s to first partition */
  186. strsep(&s, ";");
  187. }
  188. if (strlen(s) == 0)
  189. return -3;
  190. i = strlen(s) - 1;
  191. if (s[i] == ';')
  192. s[i] = '\0';
  193. /* calculate expected number of partitions */
  194. p_count = 1;
  195. p = s;
  196. while (*p) {
  197. if (*p++ == ';')
  198. p_count++;
  199. }
  200. /* allocate memory for partitions */
  201. parts = calloc(sizeof(disk_partition_t), p_count);
  202. /* retrieve partitions data from string */
  203. for (i = 0; i < p_count; i++) {
  204. tok = strsep(&s, ";");
  205. if (tok == NULL)
  206. break;
  207. /* uuid */
  208. val = extract_val(tok, "uuid");
  209. if (!val) {
  210. /* 'uuid' is optional if random uuid's are enabled */
  211. #ifdef CONFIG_RANDOM_UUID
  212. gen_rand_uuid_str(parts[i].uuid, UUID_STR_FORMAT_STD);
  213. #else
  214. errno = -4;
  215. goto err;
  216. #endif
  217. } else {
  218. if (extract_env(val, &p))
  219. p = val;
  220. if (strlen(p) >= sizeof(parts[i].uuid)) {
  221. printf("Wrong uuid format for partition %d\n", i);
  222. errno = -4;
  223. goto err;
  224. }
  225. strcpy((char *)parts[i].uuid, p);
  226. free(val);
  227. }
  228. #ifdef CONFIG_PARTITION_TYPE_GUID
  229. /* guid */
  230. val = extract_val(tok, "type");
  231. if (val) {
  232. /* 'type' is optional */
  233. if (extract_env(val, &p))
  234. p = val;
  235. if (strlen(p) >= sizeof(parts[i].type_guid)) {
  236. printf("Wrong type guid format for partition %d\n",
  237. i);
  238. errno = -4;
  239. goto err;
  240. }
  241. strcpy((char *)parts[i].type_guid, p);
  242. free(val);
  243. }
  244. #endif
  245. /* name */
  246. val = extract_val(tok, "name");
  247. if (!val) { /* name is mandatory */
  248. errno = -4;
  249. goto err;
  250. }
  251. if (extract_env(val, &p))
  252. p = val;
  253. if (strlen(p) >= sizeof(parts[i].name)) {
  254. errno = -4;
  255. goto err;
  256. }
  257. strcpy((char *)parts[i].name, p);
  258. free(val);
  259. /* size */
  260. val = extract_val(tok, "size");
  261. if (!val) { /* 'size' is mandatory */
  262. errno = -4;
  263. goto err;
  264. }
  265. if (extract_env(val, &p))
  266. p = val;
  267. if ((strcmp(p, "-") == 0)) {
  268. /* Let part efi module to auto extend the size */
  269. parts[i].size = 0;
  270. } else {
  271. size_ll = ustrtoull(p, &p, 0);
  272. parts[i].size = lldiv(size_ll, dev_desc->blksz);
  273. }
  274. free(val);
  275. /* start address */
  276. val = extract_val(tok, "start");
  277. if (val) { /* start address is optional */
  278. if (extract_env(val, &p))
  279. p = val;
  280. start_ll = ustrtoull(p, &p, 0);
  281. parts[i].start = lldiv(start_ll, dev_desc->blksz);
  282. free(val);
  283. }
  284. offset += parts[i].size + parts[i].start;
  285. /* bootable */
  286. if (found_key(tok, "bootable"))
  287. parts[i].bootable = 1;
  288. }
  289. *parts_count = p_count;
  290. *partitions = parts;
  291. free(str);
  292. return 0;
  293. err:
  294. free(str);
  295. free(*str_disk_guid);
  296. free(parts);
  297. return errno;
  298. }
  299. static int gpt_default(struct blk_desc *blk_dev_desc, const char *str_part)
  300. {
  301. int ret;
  302. char *str_disk_guid;
  303. u8 part_count = 0;
  304. disk_partition_t *partitions = NULL;
  305. /* fill partitions */
  306. ret = set_gpt_info(blk_dev_desc, str_part,
  307. &str_disk_guid, &partitions, &part_count);
  308. if (ret) {
  309. if (ret == -1)
  310. printf("No partition list provided\n");
  311. if (ret == -2)
  312. printf("Missing disk guid\n");
  313. if ((ret == -3) || (ret == -4))
  314. printf("Partition list incomplete\n");
  315. return -1;
  316. }
  317. /* save partitions layout to disk */
  318. ret = gpt_restore(blk_dev_desc, str_disk_guid, partitions, part_count);
  319. free(str_disk_guid);
  320. free(partitions);
  321. return ret;
  322. }
  323. static int gpt_verify(struct blk_desc *blk_dev_desc, const char *str_part)
  324. {
  325. ALLOC_CACHE_ALIGN_BUFFER_PAD(gpt_header, gpt_head, 1,
  326. blk_dev_desc->blksz);
  327. disk_partition_t *partitions = NULL;
  328. gpt_entry *gpt_pte = NULL;
  329. char *str_disk_guid;
  330. u8 part_count = 0;
  331. int ret = 0;
  332. /* fill partitions */
  333. ret = set_gpt_info(blk_dev_desc, str_part,
  334. &str_disk_guid, &partitions, &part_count);
  335. if (ret) {
  336. if (ret == -1) {
  337. printf("No partition list provided - only basic check\n");
  338. ret = gpt_verify_headers(blk_dev_desc, gpt_head,
  339. &gpt_pte);
  340. goto out;
  341. }
  342. if (ret == -2)
  343. printf("Missing disk guid\n");
  344. if ((ret == -3) || (ret == -4))
  345. printf("Partition list incomplete\n");
  346. return -1;
  347. }
  348. /* Check partition layout with provided pattern */
  349. ret = gpt_verify_partitions(blk_dev_desc, partitions, part_count,
  350. gpt_head, &gpt_pte);
  351. free(str_disk_guid);
  352. free(partitions);
  353. out:
  354. free(gpt_pte);
  355. return ret;
  356. }
  357. /**
  358. * do_gpt(): Perform GPT operations
  359. *
  360. * @param cmdtp - command name
  361. * @param flag
  362. * @param argc
  363. * @param argv
  364. *
  365. * @return zero on success; otherwise error
  366. */
  367. static int do_gpt(cmd_tbl_t *cmdtp, int flag, int argc, char * const argv[])
  368. {
  369. int ret = CMD_RET_SUCCESS;
  370. int dev = 0;
  371. char *ep;
  372. struct blk_desc *blk_dev_desc = NULL;
  373. if (argc < 4 || argc > 5)
  374. return CMD_RET_USAGE;
  375. dev = (int)simple_strtoul(argv[3], &ep, 10);
  376. if (!ep || ep[0] != '\0') {
  377. printf("'%s' is not a number\n", argv[3]);
  378. return CMD_RET_USAGE;
  379. }
  380. blk_dev_desc = blk_get_dev(argv[2], dev);
  381. if (!blk_dev_desc) {
  382. printf("%s: %s dev %d NOT available\n",
  383. __func__, argv[2], dev);
  384. return CMD_RET_FAILURE;
  385. }
  386. if ((strcmp(argv[1], "write") == 0) && (argc == 5)) {
  387. printf("Writing GPT: ");
  388. ret = gpt_default(blk_dev_desc, argv[4]);
  389. } else if ((strcmp(argv[1], "verify") == 0)) {
  390. ret = gpt_verify(blk_dev_desc, argv[4]);
  391. printf("Verify GPT: ");
  392. } else {
  393. return CMD_RET_USAGE;
  394. }
  395. if (ret) {
  396. printf("error!\n");
  397. return CMD_RET_FAILURE;
  398. }
  399. printf("success!\n");
  400. return CMD_RET_SUCCESS;
  401. }
  402. U_BOOT_CMD(gpt, CONFIG_SYS_MAXARGS, 1, do_gpt,
  403. "GUID Partition Table",
  404. "<command> <interface> <dev> <partitions_list>\n"
  405. " - GUID partition table restoration and validity check\n"
  406. " Restore or verify GPT information on a device connected\n"
  407. " to interface\n"
  408. " Example usage:\n"
  409. " gpt write mmc 0 $partitions\n"
  410. " gpt verify mmc 0 $partitions\n"
  411. );