summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorRandall Spangler <rspangler@chromium.org>2016-07-21 15:33:31 -0700
committerchrome-bot <chrome-bot@chromium.org>2016-07-22 13:35:56 -0700
commitb3a625f8fef1768d78eab4cfaaea270cb3fbd0c3 (patch)
treed472ca14b51562dc3522d844cbe50e742928e493
parentbea3f7979a4c3088da74accd1b68830214e0934d (diff)
downloadvboot-b3a625f8fef1768d78eab4cfaaea270cb3fbd0c3.tar.gz
vboot: Fix potential alignment issue reading FWMP
RollbackFwmpRead() assumed that a uint8[] array on the stack would be aligned sufficiently for typecasting to struct RollbackSpaceFwmp and accessing its members. This was true on x86 (where unaligned accesses work fine) and probably harmless on other platforms (since RollbackSpaceFwmp is __attribute__(packed). But it's cleaner to switch to using a union of the buffer and struct, since that will provide the proper alignment. BUG=chromium:601492 BRANCH=baytrail and newer platforms TEST=make -j runtests Change-Id: I97077923ab5809c68510cbd382541bf2827aba6b Signed-off-by: Randall Spangler <rspangler@chromium.org> Reviewed-on: https://chromium-review.googlesource.com/362087 Commit-Ready: Dan Shi <dshi@google.com> Reviewed-by: Daisuke Nojiri <dnojiri@chromium.org>
-rw-r--r--firmware/lib/rollback_index.c27
1 files changed, 17 insertions, 10 deletions
diff --git a/firmware/lib/rollback_index.c b/firmware/lib/rollback_index.c
index 5c949dbe..2431e98f 100644
--- a/firmware/lib/rollback_index.c
+++ b/firmware/lib/rollback_index.c
@@ -703,8 +703,15 @@ uint32_t RollbackKernelLock(int recovery_mode)
uint32_t RollbackFwmpRead(struct RollbackSpaceFwmp *fwmp)
{
- uint8_t buf[FWMP_NV_MAX_SIZE];
- struct RollbackSpaceFwmp *bf = (struct RollbackSpaceFwmp *)buf;
+ union {
+ /*
+ * Use a union for buf and bf, rather than making bf a pointer
+ * to a bare uint8_t[] buffer. This ensures bf will be aligned
+ * if necesssary for the target platform.
+ */
+ uint8_t buf[FWMP_NV_MAX_SIZE];
+ struct RollbackSpaceFwmp bf;
+ } u;
uint32_t r;
int attempts = 3;
@@ -713,7 +720,7 @@ uint32_t RollbackFwmpRead(struct RollbackSpaceFwmp *fwmp)
while (attempts--) {
/* Try to read entire 1.0 struct */
- r = TlclRead(FWMP_NV_INDEX, buf, sizeof(*bf));
+ r = TlclRead(FWMP_NV_INDEX, u.buf, sizeof(u.bf));
if (r == TPM_E_BADINDEX) {
/* Missing space is not an error; use defaults */
VBDEBUG(("TPM: %s() - no FWMP space\n", __func__));
@@ -728,28 +735,28 @@ uint32_t RollbackFwmpRead(struct RollbackSpaceFwmp *fwmp)
* Struct must be at least big enough for 1.0, but not bigger
* than our buffer size.
*/
- if (bf->struct_size < sizeof(*bf) ||
- bf->struct_size > sizeof(buf))
+ if (u.bf.struct_size < sizeof(u.bf) ||
+ u.bf.struct_size > sizeof(u.buf))
return TPM_E_STRUCT_SIZE;
/*
* If space is bigger than we expect, re-read so we properly
* compute the CRC.
*/
- if (bf->struct_size > sizeof(*bf)) {
- r = TlclRead(FWMP_NV_INDEX, buf, bf->struct_size);
+ if (u.bf.struct_size > sizeof(u.bf)) {
+ r = TlclRead(FWMP_NV_INDEX, u.buf, u.bf.struct_size);
if (r != TPM_SUCCESS)
return r;
}
/* Verify CRC */
- if (bf->crc != Crc8(buf + 2, bf->struct_size - 2)) {
+ if (u.bf.crc != Crc8(u.buf + 2, u.bf.struct_size - 2)) {
VBDEBUG(("TPM: %s() - bad CRC\n", __func__));
continue;
}
/* Verify major version is compatible */
- if ((bf->struct_version >> 4) !=
+ if ((u.bf.struct_version >> 4) !=
(ROLLBACK_SPACE_FWMP_VERSION >> 4))
return TPM_E_STRUCT_VERSION;
@@ -762,7 +769,7 @@ uint32_t RollbackFwmpRead(struct RollbackSpaceFwmp *fwmp)
* we would need to take care of initializing the extra fields
* added in 1.1+. But that's not an issue yet.
*/
- Memcpy(fwmp, bf, sizeof(*fwmp));
+ Memcpy(fwmp, &u.bf, sizeof(*fwmp));
return TPM_SUCCESS;
}