#endif
 
-static void efi_retrieve_tpm2_eventlog_1_2(efi_system_table_t *sys_table_arg)
+void efi_retrieve_tpm2_eventlog(efi_system_table_t *sys_table_arg)
 {
        efi_guid_t tcg2_guid = EFI_TCG2_PROTOCOL_GUID;
        efi_guid_t linux_eventlog_guid = LINUX_EFI_TPM_EVENT_LOG_GUID;
        unsigned long first_entry_addr, last_entry_addr;
        size_t log_size, last_entry_size;
        efi_bool_t truncated;
+       int version = EFI_TCG2_EVENT_LOG_FORMAT_TCG_2;
        void *tcg2_protocol = NULL;
 
        status = efi_call_early(locate_protocol, &tcg2_guid, NULL,
        if (status != EFI_SUCCESS)
                return;
 
-       status = efi_call_proto(efi_tcg2_protocol, get_event_log, tcg2_protocol,
-                               EFI_TCG2_EVENT_LOG_FORMAT_TCG_1_2,
-                               &log_location, &log_last_entry, &truncated);
-       if (status != EFI_SUCCESS)
-               return;
+       status = efi_call_proto(efi_tcg2_protocol, get_event_log,
+                               tcg2_protocol, version, &log_location,
+                               &log_last_entry, &truncated);
+
+       if (status != EFI_SUCCESS || !log_location) {
+               version = EFI_TCG2_EVENT_LOG_FORMAT_TCG_1_2;
+               status = efi_call_proto(efi_tcg2_protocol, get_event_log,
+                                       tcg2_protocol, version, &log_location,
+                                       &log_last_entry, &truncated);
+               if (status != EFI_SUCCESS || !log_location)
+                       return;
+
+       }
 
-       if (!log_location)
-               return;
        first_entry_addr = (unsigned long) log_location;
 
        /*
                 * We need to calculate its size to deduce the full size of
                 * the logs.
                 */
-               last_entry_size = sizeof(struct tcpa_event) +
-                       ((struct tcpa_event *) last_entry_addr)->event_size;
+               if (version == EFI_TCG2_EVENT_LOG_FORMAT_TCG_2) {
+                       /*
+                        * The TCG2 log format has variable length entries,
+                        * and the information to decode the hash algorithms
+                        * back into a size is contained in the first entry -
+                        * pass a pointer to the final entry (to calculate its
+                        * size) and the first entry (so we know how long each
+                        * digest is)
+                        */
+                       last_entry_size =
+                               __calc_tpm2_event_size((void *)last_entry_addr,
+                                                   (void *)(long)log_location,
+                                                   false);
+               } else {
+                       last_entry_size = sizeof(struct tcpa_event) +
+                          ((struct tcpa_event *) last_entry_addr)->event_size;
+               }
                log_size = log_last_entry - log_location + last_entry_size;
        }
 
 
        memset(log_tbl, 0, sizeof(*log_tbl) + log_size);
        log_tbl->size = log_size;
-       log_tbl->version = EFI_TCG2_EVENT_LOG_FORMAT_TCG_1_2;
+       log_tbl->version = version;
        memcpy(log_tbl->log, (void *) first_entry_addr, log_size);
 
        status = efi_call_early(install_configuration_table,
 err_free:
        efi_call_early(free_pool, log_tbl);
 }
-
-void efi_retrieve_tpm2_eventlog(efi_system_table_t *sys_table_arg)
-{
-       /* Only try to retrieve the logs in 1.2 format. */
-       efi_retrieve_tpm2_eventlog_1_2(sys_table_arg);
-}