|  | /* | 
|  | * Copyright (C) 2025 Michael Brown <mbrown@fensystems.co.uk>. | 
|  | * | 
|  | * This program is free software; you can redistribute it and/or | 
|  | * modify it under the terms of the GNU General Public License as | 
|  | * published by the Free Software Foundation; either version 2 of the | 
|  | * License, or any later version. | 
|  | * | 
|  | * This program is distributed in the hope that it will be useful, but | 
|  | * WITHOUT ANY WARRANTY; without even the implied warranty of | 
|  | * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU | 
|  | * General Public License for more details. | 
|  | * | 
|  | * You should have received a copy of the GNU General Public License | 
|  | * along with this program; if not, write to the Free Software | 
|  | * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA | 
|  | * 02110-1301, USA. | 
|  | * | 
|  | * You can also choose to distribute this program under the terms of | 
|  | * the Unmodified Binary Distribution Licence (as given in the file | 
|  | * COPYING.UBDL), provided that you have satisfied its requirements. | 
|  | */ | 
|  |  | 
|  | FILE_LICENCE ( GPL2_OR_LATER_OR_UBDL ); | 
|  |  | 
|  | /** @file | 
|  | * | 
|  | * EFI CA certificates | 
|  | * | 
|  | */ | 
|  |  | 
|  | #include <stdlib.h> | 
|  | #include <string.h> | 
|  | #include <assert.h> | 
|  | #include <errno.h> | 
|  | #include <ipxe/init.h> | 
|  | #include <ipxe/x509.h> | 
|  | #include <ipxe/rootcert.h> | 
|  | #include <ipxe/efi/efi.h> | 
|  | #include <ipxe/efi/efi_siglist.h> | 
|  | #include <ipxe/efi/Guid/TlsAuthentication.h> | 
|  |  | 
|  | /** List of EFI CA certificates */ | 
|  | static struct x509_chain efi_cacerts = { | 
|  | .refcnt = REF_INIT ( ref_no_free ), | 
|  | .links = LIST_HEAD_INIT ( efi_cacerts.links ), | 
|  | }; | 
|  |  | 
|  | /** | 
|  | * Retrieve EFI CA certificate | 
|  | * | 
|  | * @v data		TlsCaCertificate variable data | 
|  | * @v len		Length of TlsCaCertificate | 
|  | * @v offset		Offset within data | 
|  | * @v next		Next offset, or negative error | 
|  | */ | 
|  | static int efi_cacert ( const void *data, size_t len, size_t offset ) { | 
|  | struct asn1_cursor *cursor; | 
|  | struct x509_certificate *cert; | 
|  | int next; | 
|  | int rc; | 
|  |  | 
|  | /* Extract ASN.1 object */ | 
|  | next = efisig_asn1 ( data, len, offset, &cursor ); | 
|  | if ( next < 0 ) { | 
|  | rc = next; | 
|  | DBGC ( &efi_cacerts, "EFICA could not parse at +%#zx: %s\n", | 
|  | offset, strerror ( rc ) ); | 
|  | goto err_asn1; | 
|  | } | 
|  |  | 
|  | /* Append to list of EFI CA certificates */ | 
|  | if ( ( rc = x509_append_raw ( &efi_cacerts, cursor->data, | 
|  | cursor->len ) ) != 0 ) { | 
|  | DBGC ( &efi_cacerts, "EFICA could not append at +%#zx: %s\n", | 
|  | offset, strerror ( rc ) ); | 
|  | goto err_append; | 
|  | } | 
|  | cert = x509_last ( &efi_cacerts ); | 
|  | DBGC ( &efi_cacerts, "EFICA found certificate %s\n", | 
|  | x509_name ( cert ) ); | 
|  |  | 
|  | /* Mark certificate as valid (i.e. trusted) if permitted */ | 
|  | if ( allow_trust_override ) { | 
|  | DBGC ( &efi_cacerts, "EFICA trusting certificate %s\n", | 
|  | x509_name ( cert ) ); | 
|  | x509_set_valid ( cert, NULL, &root_certificates ); | 
|  | } | 
|  |  | 
|  | /* Free ASN.1 object */ | 
|  | free ( cursor ); | 
|  |  | 
|  | return next; | 
|  |  | 
|  | err_append: | 
|  | free ( cursor ); | 
|  | err_asn1: | 
|  | return rc; | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Retrieve all EFI CA certificates | 
|  | * | 
|  | * @ret rc		Return status code | 
|  | */ | 
|  | static int efi_cacert_all ( void ) { | 
|  | EFI_RUNTIME_SERVICES *rs = efi_systab->RuntimeServices; | 
|  | EFI_GUID *guid = &efi_tls_ca_certificate_guid; | 
|  | static CHAR16 *wname = EFI_TLS_CA_CERTIFICATE_VARIABLE; | 
|  | int offset = 0; | 
|  | UINT32 attrs; | 
|  | UINTN size; | 
|  | void *data; | 
|  | EFI_STATUS efirc; | 
|  | int rc; | 
|  |  | 
|  | /* Get variable length */ | 
|  | size = 0; | 
|  | if ( ( efirc = rs->GetVariable ( wname, guid, &attrs, &size, | 
|  | NULL ) ) != EFI_BUFFER_TOO_SMALL ) { | 
|  | rc = -EEFI ( efirc ); | 
|  | DBGC ( &efi_cacerts, "EFICA could not get %ls size: %s\n", | 
|  | wname, strerror ( rc ) ); | 
|  | goto err_len; | 
|  | } | 
|  |  | 
|  | /* Allocate temporary buffer */ | 
|  | data = malloc ( size ); | 
|  | if ( ! data ) { | 
|  | rc = -ENOMEM; | 
|  | goto err_alloc; | 
|  | } | 
|  |  | 
|  | /* Read variable */ | 
|  | if ( ( efirc = rs->GetVariable ( wname, guid, &attrs, &size, | 
|  | data ) ) != 0 ) { | 
|  | rc = -EEFI ( efirc ); | 
|  | DBGC ( &efi_cacerts, "EFICA could not read %ls: %s\n", | 
|  | wname, strerror ( rc ) ); | 
|  | goto err_get; | 
|  | } | 
|  |  | 
|  | /* Parse certificates */ | 
|  | while ( ( ( size_t ) offset ) < size ) { | 
|  | offset = efi_cacert ( data, size, offset ); | 
|  | if ( offset < 0 ) { | 
|  | rc = offset; | 
|  | goto err_cacert; | 
|  | } | 
|  | } | 
|  |  | 
|  | /* Success */ | 
|  | rc = 0; | 
|  |  | 
|  | err_cacert: | 
|  | err_get: | 
|  | free ( data ); | 
|  | err_alloc: | 
|  | err_len: | 
|  | return rc; | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Initialise EFI CA certificates | 
|  | * | 
|  | */ | 
|  | static void efi_cacert_init ( void ) { | 
|  | int rc; | 
|  |  | 
|  | /* Initialise all certificates */ | 
|  | if ( ( rc = efi_cacert_all() ) != 0 ) { | 
|  | DBGC ( &efi_cacert, "EFICA could not initialise: %s\n", | 
|  | strerror ( rc ) ); | 
|  | /* Nothing we can do at this point */ | 
|  | return; | 
|  | } | 
|  | } | 
|  |  | 
|  | /** EFI CA certificates initialisation function */ | 
|  | struct init_fn efi_cacert_init_fn __init_fn ( INIT_LATE ) = { | 
|  | .name = "eficacert", | 
|  | .initialise = efi_cacert_init, | 
|  | }; | 
|  |  | 
|  | /** | 
|  | * Discard any EFI CA certificates | 
|  | * | 
|  | */ | 
|  | static void efi_cacert_shutdown ( int booting __unused ) { | 
|  |  | 
|  | /* Drop our references to the certificates */ | 
|  | DBGC ( &efi_cacert, "EFICA discarding certificates\n" ); | 
|  | x509_truncate ( &efi_cacerts, NULL ); | 
|  | assert ( list_empty ( &efi_cacerts.links ) ); | 
|  | } | 
|  |  | 
|  | /** EFI CA certificates shutdown function */ | 
|  | struct startup_fn efi_cacert_shutdown_fn __startup_fn ( STARTUP_NORMAL ) = { | 
|  | .name = "efi_cacert", | 
|  | .shutdown = efi_cacert_shutdown, | 
|  | }; |