| // SPDX-License-Identifier: MIT |
| /* |
| * Copyright (C) 2016 The Android Open Source Project |
| */ |
| |
| #include "avb_property_descriptor.h" |
| #include "avb_util.h" |
| |
| bool avb_property_descriptor_validate_and_byteswap( |
| const AvbPropertyDescriptor* src, AvbPropertyDescriptor* dest) { |
| uint64_t expected_size; |
| |
| avb_memcpy(dest, src, sizeof(AvbPropertyDescriptor)); |
| |
| if (!avb_descriptor_validate_and_byteswap((const AvbDescriptor*)src, |
| (AvbDescriptor*)dest)) |
| return false; |
| |
| if (dest->parent_descriptor.tag != AVB_DESCRIPTOR_TAG_PROPERTY) { |
| avb_error("Invalid tag for property descriptor.\n"); |
| return false; |
| } |
| |
| dest->key_num_bytes = avb_be64toh(dest->key_num_bytes); |
| dest->value_num_bytes = avb_be64toh(dest->value_num_bytes); |
| |
| /* Check that key and value are fully contained. */ |
| expected_size = sizeof(AvbPropertyDescriptor) - sizeof(AvbDescriptor) + 2; |
| if (!avb_safe_add_to(&expected_size, dest->key_num_bytes) || |
| !avb_safe_add_to(&expected_size, dest->value_num_bytes)) { |
| avb_error("Overflow while adding up sizes.\n"); |
| return false; |
| } |
| if (expected_size > dest->parent_descriptor.num_bytes_following) { |
| avb_error("Descriptor payload size overflow.\n"); |
| return false; |
| } |
| |
| return true; |
| } |
| |
| typedef struct { |
| const char* key; |
| size_t key_size; |
| const char* ret_value; |
| size_t ret_value_size; |
| } PropertyIteratorData; |
| |
| static bool property_lookup_desc_foreach(const AvbDescriptor* header, |
| void* user_data) { |
| PropertyIteratorData* data = (PropertyIteratorData*)user_data; |
| AvbPropertyDescriptor prop_desc; |
| const uint8_t* p; |
| bool ret = true; |
| |
| if (header->tag != AVB_DESCRIPTOR_TAG_PROPERTY) { |
| goto out; |
| } |
| |
| if (!avb_property_descriptor_validate_and_byteswap( |
| (const AvbPropertyDescriptor*)header, &prop_desc)) { |
| goto out; |
| } |
| |
| p = (const uint8_t*)header; |
| if (p[sizeof(AvbPropertyDescriptor) + prop_desc.key_num_bytes] != 0) { |
| avb_error("No terminating NUL byte in key.\n"); |
| goto out; |
| } |
| |
| if (data->key_size == prop_desc.key_num_bytes) { |
| if (avb_memcmp(p + sizeof(AvbPropertyDescriptor), |
| data->key, |
| data->key_size) == 0) { |
| data->ret_value = (const char*)(p + sizeof(AvbPropertyDescriptor) + |
| prop_desc.key_num_bytes + 1); |
| data->ret_value_size = prop_desc.value_num_bytes; |
| /* Stop iterating. */ |
| ret = false; |
| goto out; |
| } |
| } |
| |
| out: |
| return ret; |
| } |
| |
| const char* avb_property_lookup(const uint8_t* image_data, |
| size_t image_size, |
| const char* key, |
| size_t key_size, |
| size_t* out_value_size) { |
| PropertyIteratorData data; |
| |
| if (key_size == 0) { |
| key_size = avb_strlen(key); |
| } |
| |
| data.key = key; |
| data.key_size = key_size; |
| |
| if (avb_descriptor_foreach( |
| image_data, image_size, property_lookup_desc_foreach, &data) == 0) { |
| if (out_value_size != NULL) { |
| *out_value_size = data.ret_value_size; |
| } |
| return data.ret_value; |
| } |
| |
| if (out_value_size != NULL) { |
| *out_value_size = 0; |
| } |
| return NULL; |
| } |
| |
| bool avb_property_lookup_uint64(const uint8_t* image_data, |
| size_t image_size, |
| const char* key, |
| size_t key_size, |
| uint64_t* out_value) { |
| const char* value; |
| bool ret = false; |
| uint64_t parsed_val; |
| int base; |
| int n; |
| |
| value = avb_property_lookup(image_data, image_size, key, key_size, NULL); |
| if (value == NULL) { |
| goto out; |
| } |
| |
| base = 10; |
| if (avb_memcmp(value, "0x", 2) == 0) { |
| base = 16; |
| value += 2; |
| } |
| |
| parsed_val = 0; |
| for (n = 0; value[n] != '\0'; n++) { |
| int c = value[n]; |
| int digit; |
| |
| parsed_val *= base; |
| |
| if (c >= '0' && c <= '9') { |
| digit = c - '0'; |
| } else if (base == 16 && c >= 'a' && c <= 'f') { |
| digit = c - 'a' + 10; |
| } else if (base == 16 && c >= 'A' && c <= 'F') { |
| digit = c - 'A' + 10; |
| } else { |
| avb_error("Invalid digit.\n"); |
| goto out; |
| } |
| |
| parsed_val += digit; |
| } |
| |
| ret = true; |
| if (out_value != NULL) { |
| *out_value = parsed_val; |
| } |
| |
| out: |
| return ret; |
| } |