--- /dev/null
+/*
+ * 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 (at your option) 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 );
+
+#include <config/general.h>
+
+/** @file
+ *
+ * Certificate source configuration
+ *
+ */
+
+PROVIDE_REQUIRING_SYMBOL();
+
+#ifdef CERTS_EFI
+REQUIRE_OBJECT ( efi_cacert );
+#endif
--- /dev/null
+/*
+ * 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 ( 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 ( virt_to_user ( 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 ) = {
+ .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,
+};