Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

bootutil: Add compressed image flags and TLV, fix misc. bugs in bootutil #2028

Merged
merged 7 commits into from
Aug 27, 2024

Conversation

nordicjm
Copy link
Collaborator

@nordicjm nordicjm commented Jul 31, 2024

Adds some flags to indicate if the data of an image is compressed (lzma1 and lzma2) and adds a new TLV for a sha256 hash of the decompressed/decrypted image data, this would (optionally) be used for images where one or both options are enabled to be able to verify the data is correct and that is can be written/booted successfully

Note: no code additions relating to actual compression are present here, just addition of TLVs and flags

Also includes some bug fixes to bootutil's image header checks.

Requires #2045 be merged

@nordicjm nordicjm added the area: core Affects core functionality label Jul 31, 2024
@de-nordic de-nordic requested a review from michalek-no July 31, 2024 08:08
Comment on lines 99 to 101
* SHA256 of compressed/encrypted image data in the slot
* which should be checked to ensure image integrity
* before decrypting/decompressing data
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Isn't this little confusing? In case when image is encrypted actually of the other SHA TLV are used but the image is first decrypted and then sha is calculated on the raw image.
So if only the existing encryption is used the other TLV serve as the RAW sha.
Shouldn't we have opposite?

Currently what happens is the bootutil_img_validate will first call bootutil_img_hash that will decrypt image into RAM and calculate the sha with pre-selected, during compilation, sha algorithm and then compare the resulting sha with expected TLV (EXPECTED_HASH_TLV ), which would be the IMAGE_TLV_SHA256 by default.

This also means that if we validate slot, that has image already decrypted and decompressed we will check it against EXPECTED_HASH_TLV stored sha, which currently serves as the "RAW" sha.

Copy link
Collaborator Author

@nordicjm nordicjm Aug 5, 2024

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

There needs to be a way to validate the image post-decompression which the existing tag does, however you also need to check the image before doing any sort of decompression on it, it's not an issue for decryption because you have a block size and the output size is the same as the padded input size, but that is not the case with decompression, a 4000 byte input can have a megabyte output, and you need to know that the image you have is actually a valid image, so a hash of it in the protected TLV region needs to be used.
(both TLVs would be used, decompressed image will still continue to use the original tag, new tag is for the compressed data itself, can optionally be used with encrypted data as well since it doesn't make sense to decrypt data that is wrong and delay device boot up by a long period of time)

@nordicjm
Copy link
Collaborator Author

nordicjm commented Aug 7, 2024

Flags:
IMAGE_F_COMPRESSED_LZMA1 = image is compressed, using lzma1 algorithm
IMAGE_F_COMPRESSED_LZMA2 = image is compressed, using lzma2 algorithm

TLVs:
IMAGE_TLV_COMP_SIZE = size of image once decompressed
IMAGE_TLV_COMP_SHA = sha(256/384) of decompressed image data
IMAGE_TLV_COMP_SIGNATURE = signature of decompressed image data (with MCUboot header and protected TLV section)

The image header field for the image size should be the compressed size, and the hash there should be the hash of the compressed data, since the only current supported route will be to load a compressed update to the secondary slot and decompress it into the primary slot, the image can be verified first, then decompression can run to generate the complete decompressed image which can be written back to the primary slot. Some fields in the secondary slot will be changed when the primary slot data is written, the unprotected fields for hash and signature will be replaced with those from the decompressed TLVs above and the image size will also be replaced with the decompressed TLV, then those fields will be removed from the protected TLV section and the size of that section will be reduced to account for it. The compressed flag will also be removed from the image as it is no longer compressed.

Benefit of this is to not store additional fields in the primary slot that are not needed.

@nordicjm nordicjm requested a review from de-nordic August 7, 2024 06:55
@nvlsianpu nvlsianpu self-assigned this Aug 8, 2024
@@ -101,6 +112,7 @@ struct flash_area;
#define IMAGE_TLV_DEPENDENCY 0x40 /* Image depends on other image */
#define IMAGE_TLV_SEC_CNT 0x50 /* security counter */
#define IMAGE_TLV_BOOT_RECORD 0x60 /* measured boot record */
#define IMAGE_TLV_COMP_SIZE 0x70 /* Compressed image size */
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Additionally, it is a little unclear what the sizes mean. Does this mean the has the uncompressed size? How would we find the TLV if the actual data stored is different than the size in the header?

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

For some reason I thought it went through looking for the magic bytes, apparently not... Will have a rethink

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Have updated the PR to instead have 3 TLVs, the hash and signature would be equivalent to the compressed image fields but on the decompressed image data instead (this is so that the data can be validated before decompressing it)

Adds some flags to indicate if the data of an image is compressed
(lzma1 and lzma2) and adds new TLVs for compressed images relating
to the hash, signature and size of the decompressed image data,
this allows the image to be validated before decompressing, then
validated after decompression to ensure an image is always valid
for a device

Signed-off-by: Jamie McCrae <[email protected]>
Adds a Kconfig allowing the decompression option to be selected

Signed-off-by: Jamie McCrae <[email protected]>
Checks if images have compressed or encrypted image flags and, if
so, and those options are not enabled in that MCUboot build, will
class the images as invalid and delete them (these images cannot
be used without support anyway)

Signed-off-by: Jamie McCrae <[email protected]>
Marks images as invalid if they have conflicting flags, e.g. more
than one type of LZMA compression or more than one type of
encryption

Signed-off-by: Jamie McCrae <[email protected]>
The protected TLV section was not included in the size check of
if an image could fit into a slot, which means that it was
possible for file to be deemed as OK for storing but then failing
due to insufficient flash space during the update

Signed-off-by: Jamie McCrae <[email protected]>
@nordicjm nordicjm changed the title bootutil: Add compressed image flags and TLV bootutil: Add compressed image flags and TLV, fix misc. bugs in bootutil Aug 23, 2024
When compression is used, it allows for the secondary slot to be
smaller than the primary slot, therefore do not ensure that the
number of sectors in each slot are the same

Signed-off-by: Jamie McCrae <[email protected]>
Adds release notes based on these changes

Signed-off-by: Jamie McCrae <[email protected]>
Copy link
Collaborator

@nvlsianpu nvlsianpu left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I'm fine with additional commits.
@d3zd3z Can you give us a review?

@nordicjm nordicjm requested a review from davidvincze August 27, 2024 11:52
@nvlsianpu
Copy link
Collaborator

@davidvincze Can you review this?

@nordicjm nordicjm merged commit ebf60e0 into mcu-tools:main Aug 27, 2024
58 checks passed
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
area: core Affects core functionality
Projects
None yet
Development

Successfully merging this pull request may close these issues.

5 participants