Commit 166a2809 authored by Matthew Garrett's avatar Matthew Garrett Committed by Jarkko Sakkinen

tpm: Don't duplicate events from the final event log in the TCG2 log

After the first call to GetEventLog() on UEFI systems using the TCG2
crypto agile log format, any further log events (other than those
triggered by ExitBootServices()) will be logged in both the main log and
also in the Final Events Log. While the kernel only calls GetEventLog()
immediately before ExitBootServices(), we can't control whether earlier
parts of the boot process have done so. This will result in log entries
that exist in both logs, and so the current approach of simply appending
the Final Event Log to the main log will result in events being
duplicated.

We can avoid this problem by looking at the size of the Final Event Log
just before we call ExitBootServices() and exporting this to the main
kernel. The kernel can then skip over all events that occured before
ExitBootServices() and only append events that were not also logged to
the main log.
Signed-off-by: default avatarMatthew Garrett <mjg59@google.com>
Reported-by: default avatarJoe Richey <joerichey@google.com>
Suggested-by: default avatarJoe Richey <joerichey@google.com>
Acked-by: default avatarArd Biesheuvel <ard.biesheuvel@linaro.org>
Reviewed-by: default avatarJarkko Sakkinen <jarkko.sakkinen@linux.intel.com>
Tested-by: default avatarJarkko Sakkinen <jarkko.sakkinen@linux.intel.com>
Signed-off-by: default avatarJarkko Sakkinen <jarkko.sakkinen@linux.intel.com>
parent 82d736ac
...@@ -75,6 +75,8 @@ int tpm_read_log_efi(struct tpm_chip *chip) ...@@ -75,6 +75,8 @@ int tpm_read_log_efi(struct tpm_chip *chip)
goto out; goto out;
} }
efi_tpm_final_log_size -= log_tbl->final_events_preboot_size;
tmp = krealloc(log->bios_event_log, tmp = krealloc(log->bios_event_log,
log_size + efi_tpm_final_log_size, log_size + efi_tpm_final_log_size,
GFP_KERNEL); GFP_KERNEL);
...@@ -85,8 +87,15 @@ int tpm_read_log_efi(struct tpm_chip *chip) ...@@ -85,8 +87,15 @@ int tpm_read_log_efi(struct tpm_chip *chip)
} }
log->bios_event_log = tmp; log->bios_event_log = tmp;
/*
* Copy any of the final events log that didn't also end up in the
* main log. Events can be logged in both if events are generated
* between GetEventLog() and ExitBootServices().
*/
memcpy((void *)log->bios_event_log + log_size, memcpy((void *)log->bios_event_log + log_size,
final_tbl->events, efi_tpm_final_log_size); final_tbl->events + log_tbl->final_events_preboot_size,
efi_tpm_final_log_size);
log->bios_event_log_end = log->bios_event_log + log->bios_event_log_end = log->bios_event_log +
log_size + efi_tpm_final_log_size; log_size + efi_tpm_final_log_size;
......
...@@ -64,11 +64,13 @@ void efi_retrieve_tpm2_eventlog(efi_system_table_t *sys_table_arg) ...@@ -64,11 +64,13 @@ void efi_retrieve_tpm2_eventlog(efi_system_table_t *sys_table_arg)
efi_status_t status; efi_status_t status;
efi_physical_addr_t log_location = 0, log_last_entry = 0; efi_physical_addr_t log_location = 0, log_last_entry = 0;
struct linux_efi_tpm_eventlog *log_tbl = NULL; struct linux_efi_tpm_eventlog *log_tbl = NULL;
struct efi_tcg2_final_events_table *final_events_table;
unsigned long first_entry_addr, last_entry_addr; unsigned long first_entry_addr, last_entry_addr;
size_t log_size, last_entry_size; size_t log_size, last_entry_size;
efi_bool_t truncated; efi_bool_t truncated;
int version = EFI_TCG2_EVENT_LOG_FORMAT_TCG_2; int version = EFI_TCG2_EVENT_LOG_FORMAT_TCG_2;
void *tcg2_protocol = NULL; void *tcg2_protocol = NULL;
int final_events_size = 0;
status = efi_call_early(locate_protocol, &tcg2_guid, NULL, status = efi_call_early(locate_protocol, &tcg2_guid, NULL,
&tcg2_protocol); &tcg2_protocol);
...@@ -134,8 +136,36 @@ void efi_retrieve_tpm2_eventlog(efi_system_table_t *sys_table_arg) ...@@ -134,8 +136,36 @@ void efi_retrieve_tpm2_eventlog(efi_system_table_t *sys_table_arg)
return; return;
} }
/*
* Figure out whether any events have already been logged to the
* final events structure, and if so how much space they take up
*/
final_events_table = get_efi_config_table(sys_table_arg,
LINUX_EFI_TPM_FINAL_LOG_GUID);
if (final_events_table && final_events_table->nr_events) {
struct tcg_pcr_event2_head *header;
int offset;
void *data;
int event_size;
int i = final_events_table->nr_events;
data = (void *)final_events_table;
offset = sizeof(final_events_table->version) +
sizeof(final_events_table->nr_events);
while (i > 0) {
header = data + offset + final_events_size;
event_size = __calc_tpm2_event_size(header,
(void *)(long)log_location,
false);
final_events_size += event_size;
i--;
}
}
memset(log_tbl, 0, sizeof(*log_tbl) + log_size); memset(log_tbl, 0, sizeof(*log_tbl) + log_size);
log_tbl->size = log_size; log_tbl->size = log_size;
log_tbl->final_events_preboot_size = final_events_size;
log_tbl->version = version; log_tbl->version = version;
memcpy(log_tbl->log, (void *) first_entry_addr, log_size); memcpy(log_tbl->log, (void *) first_entry_addr, log_size);
......
...@@ -1708,6 +1708,7 @@ struct linux_efi_random_seed { ...@@ -1708,6 +1708,7 @@ struct linux_efi_random_seed {
struct linux_efi_tpm_eventlog { struct linux_efi_tpm_eventlog {
u32 size; u32 size;
u32 final_events_preboot_size;
u8 version; u8 version;
u8 log[]; u8 log[];
}; };
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment