avb_cmdline.c 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424
  1. // SPDX-License-Identifier: MIT
  2. /*
  3. * Copyright (C) 2016 The Android Open Source Project
  4. */
  5. #include "avb_cmdline.h"
  6. #include "avb_sha.h"
  7. #include "avb_util.h"
  8. #include "avb_version.h"
  9. #define NUM_GUIDS 3
  10. /* Substitutes all variables (e.g. $(ANDROID_SYSTEM_PARTUUID)) with
  11. * values. Returns NULL on OOM, otherwise the cmdline with values
  12. * replaced.
  13. */
  14. char* avb_sub_cmdline(AvbOps* ops,
  15. const char* cmdline,
  16. const char* ab_suffix,
  17. bool using_boot_for_vbmeta,
  18. const AvbCmdlineSubstList* additional_substitutions) {
  19. const char* part_name_str[NUM_GUIDS] = {"system", "boot", "vbmeta"};
  20. const char* replace_str[NUM_GUIDS] = {"$(ANDROID_SYSTEM_PARTUUID)",
  21. "$(ANDROID_BOOT_PARTUUID)",
  22. "$(ANDROID_VBMETA_PARTUUID)"};
  23. char* ret = NULL;
  24. AvbIOResult io_ret;
  25. size_t n;
  26. /* Special-case for when the top-level vbmeta struct is in the boot
  27. * partition.
  28. */
  29. if (using_boot_for_vbmeta) {
  30. part_name_str[2] = "boot";
  31. }
  32. /* Replace unique partition GUIDs */
  33. for (n = 0; n < NUM_GUIDS; n++) {
  34. char part_name[AVB_PART_NAME_MAX_SIZE];
  35. char guid_buf[37];
  36. if (!avb_str_concat(part_name,
  37. sizeof part_name,
  38. part_name_str[n],
  39. avb_strlen(part_name_str[n]),
  40. ab_suffix,
  41. avb_strlen(ab_suffix))) {
  42. avb_error("Partition name and suffix does not fit.\n");
  43. goto fail;
  44. }
  45. io_ret = ops->get_unique_guid_for_partition(
  46. ops, part_name, guid_buf, sizeof guid_buf);
  47. if (io_ret == AVB_IO_RESULT_ERROR_OOM) {
  48. goto fail;
  49. } else if (io_ret != AVB_IO_RESULT_OK) {
  50. avb_error("Error getting unique GUID for partition.\n");
  51. goto fail;
  52. }
  53. if (ret == NULL) {
  54. ret = avb_replace(cmdline, replace_str[n], guid_buf);
  55. } else {
  56. char* new_ret = avb_replace(ret, replace_str[n], guid_buf);
  57. avb_free(ret);
  58. ret = new_ret;
  59. }
  60. if (ret == NULL) {
  61. goto fail;
  62. }
  63. }
  64. avb_assert(ret != NULL);
  65. /* Replace any additional substitutions. */
  66. if (additional_substitutions != NULL) {
  67. for (n = 0; n < additional_substitutions->size; ++n) {
  68. char* new_ret = avb_replace(ret,
  69. additional_substitutions->tokens[n],
  70. additional_substitutions->values[n]);
  71. avb_free(ret);
  72. ret = new_ret;
  73. if (ret == NULL) {
  74. goto fail;
  75. }
  76. }
  77. }
  78. return ret;
  79. fail:
  80. if (ret != NULL) {
  81. avb_free(ret);
  82. }
  83. return NULL;
  84. }
  85. static int cmdline_append_option(AvbSlotVerifyData* slot_data,
  86. const char* key,
  87. const char* value) {
  88. size_t offset, key_len, value_len;
  89. char* new_cmdline;
  90. key_len = avb_strlen(key);
  91. value_len = avb_strlen(value);
  92. offset = 0;
  93. if (slot_data->cmdline != NULL) {
  94. offset = avb_strlen(slot_data->cmdline);
  95. if (offset > 0) {
  96. offset += 1;
  97. }
  98. }
  99. new_cmdline = avb_calloc(offset + key_len + value_len + 2);
  100. if (new_cmdline == NULL) {
  101. return 0;
  102. }
  103. if (offset > 0) {
  104. avb_memcpy(new_cmdline, slot_data->cmdline, offset - 1);
  105. new_cmdline[offset - 1] = ' ';
  106. }
  107. avb_memcpy(new_cmdline + offset, key, key_len);
  108. new_cmdline[offset + key_len] = '=';
  109. avb_memcpy(new_cmdline + offset + key_len + 1, value, value_len);
  110. if (slot_data->cmdline != NULL) {
  111. avb_free(slot_data->cmdline);
  112. }
  113. slot_data->cmdline = new_cmdline;
  114. return 1;
  115. }
  116. #define AVB_MAX_DIGITS_UINT64 32
  117. /* Writes |value| to |digits| in base 10 followed by a NUL byte.
  118. * Returns number of characters written excluding the NUL byte.
  119. */
  120. static size_t uint64_to_base10(uint64_t value,
  121. char digits[AVB_MAX_DIGITS_UINT64]) {
  122. char rev_digits[AVB_MAX_DIGITS_UINT64];
  123. size_t n, num_digits;
  124. for (num_digits = 0; num_digits < AVB_MAX_DIGITS_UINT64 - 1;) {
  125. rev_digits[num_digits++] = avb_div_by_10(&value) + '0';
  126. if (value == 0) {
  127. break;
  128. }
  129. }
  130. for (n = 0; n < num_digits; n++) {
  131. digits[n] = rev_digits[num_digits - 1 - n];
  132. }
  133. digits[n] = '\0';
  134. return n;
  135. }
  136. static int cmdline_append_version(AvbSlotVerifyData* slot_data,
  137. const char* key,
  138. uint64_t major_version,
  139. uint64_t minor_version) {
  140. char major_digits[AVB_MAX_DIGITS_UINT64];
  141. char minor_digits[AVB_MAX_DIGITS_UINT64];
  142. char combined[AVB_MAX_DIGITS_UINT64 * 2 + 1];
  143. size_t num_major_digits, num_minor_digits;
  144. num_major_digits = uint64_to_base10(major_version, major_digits);
  145. num_minor_digits = uint64_to_base10(minor_version, minor_digits);
  146. avb_memcpy(combined, major_digits, num_major_digits);
  147. combined[num_major_digits] = '.';
  148. avb_memcpy(combined + num_major_digits + 1, minor_digits, num_minor_digits);
  149. combined[num_major_digits + 1 + num_minor_digits] = '\0';
  150. return cmdline_append_option(slot_data, key, combined);
  151. }
  152. static int cmdline_append_uint64_base10(AvbSlotVerifyData* slot_data,
  153. const char* key,
  154. uint64_t value) {
  155. char digits[AVB_MAX_DIGITS_UINT64];
  156. uint64_to_base10(value, digits);
  157. return cmdline_append_option(slot_data, key, digits);
  158. }
  159. static int cmdline_append_hex(AvbSlotVerifyData* slot_data,
  160. const char* key,
  161. const uint8_t* data,
  162. size_t data_len) {
  163. int ret;
  164. char* hex_data = avb_bin2hex(data, data_len);
  165. if (hex_data == NULL) {
  166. return 0;
  167. }
  168. ret = cmdline_append_option(slot_data, key, hex_data);
  169. avb_free(hex_data);
  170. return ret;
  171. }
  172. AvbSlotVerifyResult avb_append_options(
  173. AvbOps* ops,
  174. AvbSlotVerifyData* slot_data,
  175. AvbVBMetaImageHeader* toplevel_vbmeta,
  176. AvbAlgorithmType algorithm_type,
  177. AvbHashtreeErrorMode hashtree_error_mode) {
  178. AvbSlotVerifyResult ret;
  179. const char* verity_mode;
  180. bool is_device_unlocked;
  181. AvbIOResult io_ret;
  182. /* Add androidboot.vbmeta.device option. */
  183. if (!cmdline_append_option(slot_data,
  184. "androidboot.vbmeta.device",
  185. "PARTUUID=$(ANDROID_VBMETA_PARTUUID)")) {
  186. ret = AVB_SLOT_VERIFY_RESULT_ERROR_OOM;
  187. goto out;
  188. }
  189. /* Add androidboot.vbmeta.avb_version option. */
  190. if (!cmdline_append_version(slot_data,
  191. "androidboot.vbmeta.avb_version",
  192. AVB_VERSION_MAJOR,
  193. AVB_VERSION_MINOR)) {
  194. ret = AVB_SLOT_VERIFY_RESULT_ERROR_OOM;
  195. goto out;
  196. }
  197. /* Set androidboot.avb.device_state to "locked" or "unlocked". */
  198. io_ret = ops->read_is_device_unlocked(ops, &is_device_unlocked);
  199. if (io_ret == AVB_IO_RESULT_ERROR_OOM) {
  200. ret = AVB_SLOT_VERIFY_RESULT_ERROR_OOM;
  201. goto out;
  202. } else if (io_ret != AVB_IO_RESULT_OK) {
  203. avb_error("Error getting device state.\n");
  204. ret = AVB_SLOT_VERIFY_RESULT_ERROR_IO;
  205. goto out;
  206. }
  207. if (!cmdline_append_option(slot_data,
  208. "androidboot.vbmeta.device_state",
  209. is_device_unlocked ? "unlocked" : "locked")) {
  210. ret = AVB_SLOT_VERIFY_RESULT_ERROR_OOM;
  211. goto out;
  212. }
  213. /* Set androidboot.vbmeta.{hash_alg, size, digest} - use same hash
  214. * function as is used to sign vbmeta.
  215. */
  216. switch (algorithm_type) {
  217. /* Explicit fallthrough. */
  218. case AVB_ALGORITHM_TYPE_NONE:
  219. case AVB_ALGORITHM_TYPE_SHA256_RSA2048:
  220. case AVB_ALGORITHM_TYPE_SHA256_RSA4096:
  221. case AVB_ALGORITHM_TYPE_SHA256_RSA8192: {
  222. size_t n, total_size = 0;
  223. uint8_t vbmeta_digest[AVB_SHA256_DIGEST_SIZE];
  224. avb_slot_verify_data_calculate_vbmeta_digest(
  225. slot_data, AVB_DIGEST_TYPE_SHA256, vbmeta_digest);
  226. for (n = 0; n < slot_data->num_vbmeta_images; n++) {
  227. total_size += slot_data->vbmeta_images[n].vbmeta_size;
  228. }
  229. if (!cmdline_append_option(
  230. slot_data, "androidboot.vbmeta.hash_alg", "sha256") ||
  231. !cmdline_append_uint64_base10(
  232. slot_data, "androidboot.vbmeta.size", total_size) ||
  233. !cmdline_append_hex(slot_data,
  234. "androidboot.vbmeta.digest",
  235. vbmeta_digest,
  236. AVB_SHA256_DIGEST_SIZE)) {
  237. ret = AVB_SLOT_VERIFY_RESULT_ERROR_OOM;
  238. goto out;
  239. }
  240. } break;
  241. /* Explicit fallthrough. */
  242. case AVB_ALGORITHM_TYPE_SHA512_RSA2048:
  243. case AVB_ALGORITHM_TYPE_SHA512_RSA4096:
  244. case AVB_ALGORITHM_TYPE_SHA512_RSA8192: {
  245. size_t n, total_size = 0;
  246. uint8_t vbmeta_digest[AVB_SHA512_DIGEST_SIZE];
  247. avb_slot_verify_data_calculate_vbmeta_digest(
  248. slot_data, AVB_DIGEST_TYPE_SHA512, vbmeta_digest);
  249. for (n = 0; n < slot_data->num_vbmeta_images; n++) {
  250. total_size += slot_data->vbmeta_images[n].vbmeta_size;
  251. }
  252. if (!cmdline_append_option(
  253. slot_data, "androidboot.vbmeta.hash_alg", "sha512") ||
  254. !cmdline_append_uint64_base10(
  255. slot_data, "androidboot.vbmeta.size", total_size) ||
  256. !cmdline_append_hex(slot_data,
  257. "androidboot.vbmeta.digest",
  258. vbmeta_digest,
  259. AVB_SHA512_DIGEST_SIZE)) {
  260. ret = AVB_SLOT_VERIFY_RESULT_ERROR_OOM;
  261. goto out;
  262. }
  263. } break;
  264. case _AVB_ALGORITHM_NUM_TYPES:
  265. avb_assert_not_reached();
  266. break;
  267. }
  268. /* Set androidboot.veritymode and androidboot.vbmeta.invalidate_on_error */
  269. if (toplevel_vbmeta->flags & AVB_VBMETA_IMAGE_FLAGS_HASHTREE_DISABLED) {
  270. verity_mode = "disabled";
  271. } else {
  272. const char* dm_verity_mode;
  273. char* new_ret;
  274. switch (hashtree_error_mode) {
  275. case AVB_HASHTREE_ERROR_MODE_RESTART_AND_INVALIDATE:
  276. if (!cmdline_append_option(
  277. slot_data, "androidboot.vbmeta.invalidate_on_error", "yes")) {
  278. ret = AVB_SLOT_VERIFY_RESULT_ERROR_OOM;
  279. goto out;
  280. }
  281. verity_mode = "enforcing";
  282. dm_verity_mode = "restart_on_corruption";
  283. break;
  284. case AVB_HASHTREE_ERROR_MODE_RESTART:
  285. verity_mode = "enforcing";
  286. dm_verity_mode = "restart_on_corruption";
  287. break;
  288. case AVB_HASHTREE_ERROR_MODE_EIO:
  289. verity_mode = "eio";
  290. /* For now there's no option to specify the EIO mode. So
  291. * just use 'ignore_zero_blocks' since that's already set
  292. * and dm-verity-target.c supports specifying this multiple
  293. * times.
  294. */
  295. dm_verity_mode = "ignore_zero_blocks";
  296. break;
  297. case AVB_HASHTREE_ERROR_MODE_LOGGING:
  298. verity_mode = "logging";
  299. dm_verity_mode = "ignore_corruption";
  300. break;
  301. default:
  302. ret = AVB_SLOT_VERIFY_RESULT_ERROR_INVALID_ARGUMENT;
  303. goto out;
  304. }
  305. new_ret = avb_replace(
  306. slot_data->cmdline, "$(ANDROID_VERITY_MODE)", dm_verity_mode);
  307. avb_free(slot_data->cmdline);
  308. slot_data->cmdline = new_ret;
  309. if (slot_data->cmdline == NULL) {
  310. ret = AVB_SLOT_VERIFY_RESULT_ERROR_OOM;
  311. goto out;
  312. }
  313. }
  314. if (!cmdline_append_option(
  315. slot_data, "androidboot.veritymode", verity_mode)) {
  316. ret = AVB_SLOT_VERIFY_RESULT_ERROR_OOM;
  317. goto out;
  318. }
  319. ret = AVB_SLOT_VERIFY_RESULT_OK;
  320. out:
  321. return ret;
  322. }
  323. AvbCmdlineSubstList* avb_new_cmdline_subst_list() {
  324. return (AvbCmdlineSubstList*)avb_calloc(sizeof(AvbCmdlineSubstList));
  325. }
  326. void avb_free_cmdline_subst_list(AvbCmdlineSubstList* cmdline_subst) {
  327. size_t i;
  328. for (i = 0; i < cmdline_subst->size; ++i) {
  329. avb_free(cmdline_subst->tokens[i]);
  330. avb_free(cmdline_subst->values[i]);
  331. }
  332. cmdline_subst->size = 0;
  333. avb_free(cmdline_subst);
  334. }
  335. AvbSlotVerifyResult avb_add_root_digest_substitution(
  336. const char* part_name,
  337. const uint8_t* digest,
  338. size_t digest_size,
  339. AvbCmdlineSubstList* out_cmdline_subst) {
  340. const char* kDigestSubPrefix = "$(AVB_";
  341. const char* kDigestSubSuffix = "_ROOT_DIGEST)";
  342. size_t part_name_len = avb_strlen(part_name);
  343. size_t list_index = out_cmdline_subst->size;
  344. avb_assert(part_name_len < AVB_PART_NAME_MAX_SIZE);
  345. avb_assert(digest_size <= AVB_SHA512_DIGEST_SIZE);
  346. if (part_name_len >= AVB_PART_NAME_MAX_SIZE ||
  347. digest_size > AVB_SHA512_DIGEST_SIZE) {
  348. return AVB_SLOT_VERIFY_RESULT_ERROR_INVALID_METADATA;
  349. }
  350. if (out_cmdline_subst->size >= AVB_MAX_NUM_CMDLINE_SUBST) {
  351. /* The list is full. Currently dynamic growth of this list is not supported.
  352. */
  353. return AVB_SLOT_VERIFY_RESULT_ERROR_INVALID_METADATA;
  354. }
  355. /* Construct the token to replace in the command line based on the partition
  356. * name. For partition 'foo', this will be '$(AVB_FOO_ROOT_DIGEST)'.
  357. */
  358. out_cmdline_subst->tokens[list_index] =
  359. avb_strdupv(kDigestSubPrefix, part_name, kDigestSubSuffix, NULL);
  360. if (out_cmdline_subst->tokens[list_index] == NULL) {
  361. goto fail;
  362. }
  363. avb_uppercase(out_cmdline_subst->tokens[list_index]);
  364. /* The digest value is hex encoded when inserted in the command line. */
  365. out_cmdline_subst->values[list_index] = avb_bin2hex(digest, digest_size);
  366. if (out_cmdline_subst->values[list_index] == NULL) {
  367. goto fail;
  368. }
  369. out_cmdline_subst->size++;
  370. return AVB_SLOT_VERIFY_RESULT_OK;
  371. fail:
  372. if (out_cmdline_subst->tokens[list_index]) {
  373. avb_free(out_cmdline_subst->tokens[list_index]);
  374. }
  375. if (out_cmdline_subst->values[list_index]) {
  376. avb_free(out_cmdline_subst->values[list_index]);
  377. }
  378. return AVB_SLOT_VERIFY_RESULT_ERROR_OOM;
  379. }