avb_descriptor.c 3.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142
  1. /*
  2. * Copyright (C) 2016 The Android Open Source Project
  3. *
  4. * SPDX-License-Identifier: MIT
  5. */
  6. #include "avb_descriptor.h"
  7. #include "avb_util.h"
  8. #include "avb_vbmeta_image.h"
  9. bool avb_descriptor_validate_and_byteswap(const AvbDescriptor* src,
  10. AvbDescriptor* dest) {
  11. dest->tag = avb_be64toh(src->tag);
  12. dest->num_bytes_following = avb_be64toh(src->num_bytes_following);
  13. if ((dest->num_bytes_following & 0x07) != 0) {
  14. avb_error("Descriptor size is not divisible by 8.\n");
  15. return false;
  16. }
  17. return true;
  18. }
  19. bool avb_descriptor_foreach(const uint8_t* image_data,
  20. size_t image_size,
  21. AvbDescriptorForeachFunc foreach_func,
  22. void* user_data) {
  23. const AvbVBMetaImageHeader* header = NULL;
  24. bool ret = false;
  25. const uint8_t* image_end;
  26. const uint8_t* desc_start;
  27. const uint8_t* desc_end;
  28. const uint8_t* p;
  29. if (image_data == NULL) {
  30. avb_error("image_data is NULL\n.");
  31. goto out;
  32. }
  33. if (foreach_func == NULL) {
  34. avb_error("foreach_func is NULL\n.");
  35. goto out;
  36. }
  37. if (image_size < sizeof(AvbVBMetaImageHeader)) {
  38. avb_error("Length is smaller than header.\n");
  39. goto out;
  40. }
  41. /* Ensure magic is correct. */
  42. if (avb_memcmp(image_data, AVB_MAGIC, AVB_MAGIC_LEN) != 0) {
  43. avb_error("Magic is incorrect.\n");
  44. goto out;
  45. }
  46. /* Careful, not byteswapped - also ensure it's aligned properly. */
  47. avb_assert_aligned(image_data);
  48. header = (const AvbVBMetaImageHeader*)image_data;
  49. image_end = image_data + image_size;
  50. desc_start = image_data + sizeof(AvbVBMetaImageHeader) +
  51. avb_be64toh(header->authentication_data_block_size) +
  52. avb_be64toh(header->descriptors_offset);
  53. desc_end = desc_start + avb_be64toh(header->descriptors_size);
  54. if (desc_start < image_data || desc_start > image_end ||
  55. desc_end < image_data || desc_end > image_end || desc_end < desc_start) {
  56. avb_error("Descriptors not inside passed-in data.\n");
  57. goto out;
  58. }
  59. for (p = desc_start; p < desc_end;) {
  60. const AvbDescriptor* dh = (const AvbDescriptor*)p;
  61. avb_assert_aligned(dh);
  62. uint64_t nb_following = avb_be64toh(dh->num_bytes_following);
  63. uint64_t nb_total = sizeof(AvbDescriptor) + nb_following;
  64. if ((nb_total & 7) != 0) {
  65. avb_error("Invalid descriptor length.\n");
  66. goto out;
  67. }
  68. if (nb_total + p < desc_start || nb_total + p > desc_end) {
  69. avb_error("Invalid data in descriptors array.\n");
  70. goto out;
  71. }
  72. if (foreach_func(dh, user_data) == 0) {
  73. goto out;
  74. }
  75. p += nb_total;
  76. }
  77. ret = true;
  78. out:
  79. return ret;
  80. }
  81. static bool count_descriptors(const AvbDescriptor* descriptor,
  82. void* user_data) {
  83. size_t* num_descriptors = user_data;
  84. *num_descriptors += 1;
  85. return true;
  86. }
  87. typedef struct {
  88. size_t descriptor_number;
  89. const AvbDescriptor** descriptors;
  90. } SetDescriptorData;
  91. static bool set_descriptors(const AvbDescriptor* descriptor, void* user_data) {
  92. SetDescriptorData* data = user_data;
  93. data->descriptors[data->descriptor_number++] = descriptor;
  94. return true;
  95. }
  96. const AvbDescriptor** avb_descriptor_get_all(const uint8_t* image_data,
  97. size_t image_size,
  98. size_t* out_num_descriptors) {
  99. size_t num_descriptors = 0;
  100. SetDescriptorData data;
  101. avb_descriptor_foreach(
  102. image_data, image_size, count_descriptors, &num_descriptors);
  103. data.descriptor_number = 0;
  104. data.descriptors =
  105. avb_calloc(sizeof(const AvbDescriptor*) * (num_descriptors + 1));
  106. if (data.descriptors == NULL) {
  107. return NULL;
  108. }
  109. avb_descriptor_foreach(image_data, image_size, set_descriptors, &data);
  110. avb_assert(data.descriptor_number == num_descriptors);
  111. if (out_num_descriptors != NULL) {
  112. *out_num_descriptors = num_descriptors;
  113. }
  114. return data.descriptors;
  115. }