# Secure Boot Key Generation and Handling
Last updated 2024-03-13.
# Owernship
Certificate and key generation and handling is overseen by ALESCo.
# Key Generation
This is a slow and tedious process. It is important that no corners are cut and the utmost care is taken validating security through every step of the process.
# Prepare a Secure Environment
Download latest AlmaLinux live ISO from https://repo.almalinux.org/
. Using default Gnome mini for this guide as to not have any 3rd-party binaries present (EPEL).
In the folder where you download the ISO, look at the CHECKSUM
and CHECKSUM.asc
files. First let's verify that it is properly signed and that its contents can be authenticated.
wget https://repo.almalinux.org/almalinux/9/live/x86_64/CHECKSUM
wget https://repo.almalinux.org/almalinux/9/live/x86_64/CHECKSUM.asc
wget https://repo.almalinux.org/almalinux/RPM-GPG-KEY-AlmaLinux-9
gpg --import RPM-GPG-KEY-AlmaLinux-9
gpg --verify CHECKSUM.asc CHECKSUM
Now verify the downloaded ISO's sha256 checksum against that contained in the CHECKSUM
file.
Copy the ISO file to a flash drive.
dd if=almalinux.iso of=/dev/sdX
Boot the image. Connect to the internet temporarily to run a single command:
sudo dnf -y install epel-release words
sudo dnf -y install pesign yubikey-manager yubico-piv-tool
# Download GPG Public Keys of Recipients
If you are doing key exports as part of this, grab the public keys of all recipients.
These keys are accurate at the time of this writing. Verify them prior to use, of course.
curl <key> | gpg --import
Disconnect from the internet and do not connect again. Everything from here on out has to be done fully air-gapped and the generated private keys can never be connected to a machine with network access.
# Generate Keys and Certificates
First, create temporary directory and initialize NSS DB inside. Don't set password because of pesign bug.
mkdir -m 0700 -p ~/almalinux-uefi-certs/ca certutil -d ~/almalinux-uefi-certs/ca -N
Generate a 10 years CA certificate. More than 10 years cert is not allowed for shim:
efikeygen -d ~/almalinux-uefi-certs/ca \ --ca --self-sign --not-valid-after=$(date +%s --date='+10 years') \ --nickname='AlmaLinux Secure Boot CA' \ --common-name='CN=AlmaLinux Secure Boot CA,O=AlmaLinux OS Foundation,E=security@almalinux.org'
Generate a 10 years Signing certificate. More than 10 years cert is not allowed for shim:
efikeygen -d ~/almalinux-uefi-certs/ca \ --kernel --not-valid-after=$(date +%s --date='+10 years') --signer='AlmaLinux Secure Boot CA' \ --nickname='AlmaLinux Secure Boot Signing' \ --common-name='CN=AlmaLinux Secure Boot Signing,O=AlmaLinux OS Foundation,E=security@almalinux.com'
Export CA and Signing public and private keys to PKCS#12 files. At this point set a password, and we will store it in Vaultwarden with key holders having access:
pk12util -d ~/almalinux-uefi-certs/ca -o almalinux-secureboot-ca.p12 -n 'AlmaLinux Secure Boot CA' pk12util -d ~/almalinux-uefi-certs/ca -o almalinux-secureboot-signing.p12 -n 'AlmaLinux Secure Boot Signing'
- almalinux-secureboot-ca.p12 file is CA keypair that should be stored. It will be required to issue new Signing keys
- almalinux-secureboot-signing.p12 file is Signing keypair that should be imported to signing node to perform actual binaries signing
Export CA and Signing public keys to DER files. No password required:
certutil -d ~/almalinux-uefi-certs/ca -L -n "AlmaLinux Secure Boot CA" -r > almalinux-secureboot-ca.cer certutil -d ~/almalinux-uefi-certs/ca -L -n "AlmaLinux Secure Boot Signing" -r > almalinux-secureboot.cer
- almalinux-secureboot-ca.cer file is the CA public key we should include into shim. It also should be imported to signing node to perform actual binaries signing.
- almalinux-secureboot.cer is Signing public key that may be recommended to include into some RPM packages (but not necessary)
# Configure Yubikey PIV
# does the system see the key
ykman piv info
Generate PIV Pins (only if new yubikey)
PIVPIN=$(awk '/^\w{6,8}$/' /usr/share/dict/words | shuf -n1)
PIVPUK=$(openssl rand -base64 10 | tr -d +/= | cut -c 1-8)
echo "PIV PIN: $PIVPIN; PIV PUK: $PIVPUK"
Make sure to securely store the PIN and PUK. They are specific to the PIV component of the Yubikey.
Increase PIN/PUK retry limit (only if new yubikey)
ykman piv access set-retries 9 9 --pin 123456 --force
Set our new pins (only if new yubikey)
ykman piv access change-pin --pin 123456 --new-pin $PIVPIN
ykman piv access change-puk --puk 12345678 --new-puk $PIVPUK
Set a non-default management key (only if new yubikey)
ykman piv access change-management-key --algorithm aes128 --protect --pin $PIVPIN
Import keys to Yubikey PIV slots (pay attention to slot number, 82 in this example, if not a new yubikey)
ykman piv keys import --pin-policy always --touch-policy never --pin $PIVPIN 82 almalinux-secureboot-signing.p12
Import certificate to Yubikey PIV slots (pay attention to slot number, 82 in this example, if not a new yubikey)
ykman piv certificates import --pin $PIVPIN --verify 82 almalinux-secureboot.cer
# Copy Keys and Certificates
DO NOT DO NOT DO NOT connect to the internet to copy the files.
# Public Certificates
Copy the .cer (public) files to a blank flash drive. This drive can then be connected to a machine with networking and distributed as necessary - there is nothing secret about them.
# Private Keys
Encrypt the private keys using GPG. The GPG-encrypted files are the only ones that can be distributed to key holders via offline methods. It's recommended that they keys remain encrypted at rest.
gpg -r <recipient-gpg-email> -r <recipient-gpg-email> -e almalinux-secureboot-ca.p12
gpg -r <recipient-gpg-email> -r <recipient-gpg-email> -e almalinux-secureboot-signing.p12
Copy the .p12 (private) keys and GPG-encrypted versions to a blank flash drive. The drive must never be connected to a networked or untrusted system. It should only ever be connected to an air-gapped blank OS install.
# Removal of Custodianship
We need to define how we handle if a keyholder leaves AlmaLinux.
# Distribution of Keys to Key Holders
Keys will be distributed to key holders only via physical medium - never via the internet. Our SOP is for the key generator to encrypt the keys with GPG and place them on a hardware-encrypted (FIPS-140 certified), self-destructing (if brute force or disassembly are attempted) USB key. These may then be mailed to the key holder or delivered in person at, for example, a conference.
The unlock code for the flash drive will be provided by the key generator to the key holder upon their recipient of the flash drive, and confirmation of that receipt via a GPG-signed email.
The hardware unlock code may be provided via the internet using encrypted means - either AlmaLinux's internal password manager (Vaultwarden) or GPG-encrypted email.
# Storage of Keys by Key Holders
Keys should be stored offline and encrypted at rest via GPG and the hardware encryption of the flash drive. It should be stored in a fire resistant lock box/safe or bank safety deposit box.
# Key Generation Log
# Shim 15.8
Secure boot keys generated on 2024-03-13 using policy dated 2024-03-13. These keys and certs will go into use in late 2024 and are valid through 2034.
- https://github.com/rhboot/shim-review/issues/407
- https://github.com/rhboot/shim-review/issues/432
# Changelog
# 2025-01-13
Add Yubikey PIV certificate (public) programming instructions
# 2024-07-03
Add Yubikey PIV programming instructions
# 2024-03-13
Initial ratification