2023-02-08 16:01:48 -05:00
|
|
|
#!/bin/bash
|
2023-10-30 11:37:31 -04:00
|
|
|
# Retrieve the sealed TOTP secret and initialize a USB Security Dongle with it
|
2018-06-19 12:27:27 -07:00
|
|
|
|
|
|
|
. /etc/functions
|
|
|
|
|
|
|
|
HOTP_SECRET="/tmp/secret/hotp.key"
|
|
|
|
HOTP_COUNTER="/boot/kexec_hotp_counter"
|
2020-06-24 16:11:41 +02:00
|
|
|
HOTP_KEY="/boot/kexec_hotp_key"
|
|
|
|
|
2018-06-19 12:27:27 -07:00
|
|
|
mount_boot()
|
|
|
|
{
|
2023-08-22 14:34:29 -04:00
|
|
|
TRACE "Under /bin/seal-hotpkey:mount_boot"
|
2018-06-19 12:27:27 -07:00
|
|
|
# Mount local disk if it is not already mounted
|
|
|
|
if ! grep -q /boot /proc/mounts ; then
|
|
|
|
mount -o ro /boot \
|
|
|
|
|| recovery "Unable to mount /boot"
|
|
|
|
fi
|
|
|
|
}
|
|
|
|
|
2023-02-20 11:01:17 -05:00
|
|
|
TRACE "Under /bin/seal-hotpkey"
|
2023-02-18 12:58:43 -05:00
|
|
|
|
2019-08-21 15:40:36 -05:00
|
|
|
fatal_error()
|
|
|
|
{
|
|
|
|
echo -e "\nERROR: ${1}; press Enter to continue."
|
|
|
|
read
|
|
|
|
die "$1"
|
|
|
|
}
|
|
|
|
|
2020-06-24 17:40:49 +02:00
|
|
|
# Use stored HOTP key branding (this might be useful after OEM reset)
|
|
|
|
if [ -r /boot/kexec_hotp_key ]; then
|
2020-06-24 18:12:56 +02:00
|
|
|
HOTPKEY_BRANDING="$(cat /boot/kexec_hotp_key)"
|
2020-06-24 17:40:49 +02:00
|
|
|
else
|
2020-06-24 18:12:56 +02:00
|
|
|
HOTPKEY_BRANDING="HOTP USB Security Dongle"
|
2020-06-24 17:40:49 +02:00
|
|
|
fi
|
|
|
|
|
2023-02-28 13:36:11 -05:00
|
|
|
if [ "$CONFIG_TPM" = "y" ]; then
|
2023-03-09 13:28:04 -05:00
|
|
|
DEBUG "Sealing HOTP secret reuses TOTP sealed secret..."
|
2023-03-08 12:39:06 -05:00
|
|
|
tpmr unseal 4d47 0,1,2,3,4,7 312 "$HOTP_SECRET" \
|
2019-08-21 15:40:36 -05:00
|
|
|
|| fatal_error "Unable to unseal HOTP secret"
|
2019-11-25 12:25:11 -06:00
|
|
|
else
|
2023-07-05 10:18:06 -04:00
|
|
|
# without a TPM, generate a secret based on the SHA-256 of the ROM
|
2023-07-03 16:59:23 -04:00
|
|
|
secret_from_rom_hash > "$HOTP_SECRET" || die "Reading ROM failed"
|
2022-08-25 14:43:31 -04:00
|
|
|
fi
|
|
|
|
|
2018-06-20 09:20:39 -07:00
|
|
|
# Store counter in file instead of TPM for now, as it conflicts with Heads
|
|
|
|
# config TPM counter as TPM 1.2 can only increment one counter between reboots
|
2018-06-19 12:27:27 -07:00
|
|
|
# get current value of HOTP counter in TPM, create if absent
|
|
|
|
mount_boot
|
|
|
|
|
2018-06-20 09:20:39 -07:00
|
|
|
#check_tpm_counter $HOTP_COUNTER hotp \
|
|
|
|
#|| die "Unable to find/create TPM counter"
|
|
|
|
#counter="$TPM_COUNTER"
|
|
|
|
#
|
|
|
|
#counter_value=$(read_tpm_counter $counter | cut -f2 -d ' ' | awk 'gsub("^000e","")')
|
|
|
|
#if [ "$counter_value" == "" ]; then
|
|
|
|
# die "Unable to read HOTP counter"
|
|
|
|
#fi
|
2018-06-19 12:27:27 -07:00
|
|
|
|
2018-06-20 09:20:39 -07:00
|
|
|
#counter_value=$(printf "%d" 0x${counter_value})
|
2018-06-19 12:27:27 -07:00
|
|
|
|
2018-06-20 09:20:39 -07:00
|
|
|
counter_value=1
|
2018-06-19 12:27:27 -07:00
|
|
|
|
|
|
|
enable_usb
|
2023-07-06 13:22:40 -04:00
|
|
|
# While making sure the key is inserted, capture the status so we can check how
|
|
|
|
# many PIN attempts remain
|
|
|
|
if ! hotp_token_info="$(hotp_verification info)" ; then
|
2019-08-21 15:40:36 -05:00
|
|
|
echo -e "\nInsert your $HOTPKEY_BRANDING and press Enter to configure it"
|
2018-06-19 12:27:27 -07:00
|
|
|
read
|
2023-07-06 13:22:40 -04:00
|
|
|
if ! hotp_token_info="$(hotp_verification info)" ; then
|
2019-05-24 11:50:27 -05:00
|
|
|
# don't leak key on failure
|
|
|
|
shred -n 10 -z -u "$HOTP_SECRET" 2> /dev/null
|
2019-08-21 15:40:36 -05:00
|
|
|
fatal_error "Unable to find $HOTPKEY_BRANDING"
|
2019-05-24 11:50:27 -05:00
|
|
|
fi
|
2018-06-19 12:27:27 -07:00
|
|
|
fi
|
|
|
|
|
2020-06-24 17:54:39 +02:00
|
|
|
# Set HOTP USB Security Dongle branding based on VID
|
2020-07-24 00:11:33 -05:00
|
|
|
if lsusb | grep -q "20a0:" ; then
|
2020-06-24 18:12:56 +02:00
|
|
|
HOTPKEY_BRANDING="Nitrokey"
|
2020-07-24 00:11:33 -05:00
|
|
|
elif lsusb | grep -q "316d:" ; then
|
2020-06-24 18:12:56 +02:00
|
|
|
HOTPKEY_BRANDING="Librem Key"
|
2020-06-24 17:40:49 +02:00
|
|
|
else
|
2020-06-24 18:12:56 +02:00
|
|
|
HOTPKEY_BRANDING="HOTP USB Security Dongle"
|
2020-06-24 17:40:49 +02:00
|
|
|
fi
|
2020-06-24 16:11:41 +02:00
|
|
|
|
2023-07-05 10:18:06 -04:00
|
|
|
# Truncate the secret if it is longer than the maximum HOTP secret
|
|
|
|
truncate_max_bytes 20 "$HOTP_SECRET"
|
|
|
|
|
2023-07-06 13:22:40 -04:00
|
|
|
# Check when the signing key was created to consider trying the default PIN
|
|
|
|
# (Note: we must avoid using gpg --card-status here as the Nitrokey firmware
|
|
|
|
# locks up, https://github.com/Nitrokey/nitrokey-pro-firmware/issues/54)
|
|
|
|
gpg_key_create_time="$(gpg --list-keys --with-colons | grep -m 1 '^pub:' | cut -d: -f6)"
|
|
|
|
gpg_key_create_time="${gpg_key_create_time:-0}"
|
|
|
|
DEBUG "Signature key was created at $(date -d "@$gpg_key_create_time")"
|
|
|
|
now_date="$(date '+%s')"
|
|
|
|
|
|
|
|
# Get the number of admin PIN retry attempts remaining
|
|
|
|
awk_admin_counter_regex='/^\s*Card counters: Admin (\d),.*$/'
|
|
|
|
awk_get_admin_counter="$awk_admin_counter_regex"' { print gensub('"$awk_admin_counter_regex"', "\\1", "") }'
|
|
|
|
admin_pin_retries="$(echo "$hotp_token_info" | awk "$awk_get_admin_counter")"
|
|
|
|
admin_pin_retries="${admin_pin_retries:-0}"
|
|
|
|
DEBUG "Admin PIN retry counter is $admin_pin_retries"
|
|
|
|
|
|
|
|
# Try using factory default admin PIN for 1 month following OEM reset to ease
|
|
|
|
# initial setup. But don't do it forever to encourage changing the PIN and
|
|
|
|
# so PIN attempts are not consumed by the default attempt.
|
2019-08-21 15:40:36 -05:00
|
|
|
admin_pin="12345678"
|
2023-07-06 13:22:40 -04:00
|
|
|
month_secs="$((30*24*60*60))"
|
|
|
|
admin_pin_status=1
|
|
|
|
if [ "$((now_date - gpg_key_create_time))" -gt "$month_secs" ]; then
|
|
|
|
# Remind what the default PIN was in case it still hasn't been changed
|
|
|
|
echo "Not trying default PIN ($admin_pin)"
|
|
|
|
# Never consume an attempt if there are less than 3 attempts left, otherwise
|
|
|
|
# attempting the default PIN could cause an unexpected lockout before getting a
|
|
|
|
# chance to enter the correct PIN
|
|
|
|
elif [ "$admin_pin_retries" -lt 3 ]; then
|
|
|
|
echo "Not trying default PIN ($admin_pin), only $admin_pin_retries attempt(s) left"
|
|
|
|
else
|
|
|
|
hotp_initialize "$admin_pin" $HOTP_SECRET $counter_value "$HOTPKEY_BRANDING" >/dev/null 2>&1
|
|
|
|
admin_pin_status="$?"
|
|
|
|
fi
|
|
|
|
|
|
|
|
if [ "$admin_pin_status" -ne 0 ]; then
|
2019-08-21 15:40:36 -05:00
|
|
|
# prompt user for PIN and retry
|
|
|
|
echo ""
|
|
|
|
read -s -p "Enter your $HOTPKEY_BRANDING Admin PIN: " admin_pin
|
2019-06-28 23:26:20 -05:00
|
|
|
echo -e "\n"
|
2019-08-21 15:40:36 -05:00
|
|
|
|
|
|
|
hotp_initialize "$admin_pin" $HOTP_SECRET $counter_value "$HOTPKEY_BRANDING"
|
|
|
|
if [ $? -ne 0 ]; then
|
|
|
|
echo -e "\n"
|
|
|
|
read -s -p "Error setting HOTP secret, re-enter Admin PIN and try again: " admin_pin
|
|
|
|
echo -e "\n"
|
|
|
|
if ! hotp_initialize "$admin_pin" $HOTP_SECRET $counter_value "$HOTPKEY_BRANDING" ; then
|
|
|
|
# don't leak key on failure
|
|
|
|
shred -n 10 -z -u "$HOTP_SECRET" 2> /dev/null
|
|
|
|
fatal_error "Setting HOTP secret failed"
|
|
|
|
fi
|
2019-05-24 11:50:27 -05:00
|
|
|
fi
|
2019-08-21 15:40:36 -05:00
|
|
|
else
|
|
|
|
# remind user to change admin password
|
|
|
|
echo -e "\nWARNING: default GPG admin PIN detected: please change this as soon as possible."
|
2018-06-19 12:27:27 -07:00
|
|
|
fi
|
|
|
|
|
2019-05-24 11:50:27 -05:00
|
|
|
# HOTP key no longer needed
|
|
|
|
shred -n 10 -z -u "$HOTP_SECRET" 2> /dev/null
|
2018-06-19 12:27:27 -07:00
|
|
|
|
|
|
|
# Make sure our counter is incremented ahead of the next check
|
2018-06-20 09:20:39 -07:00
|
|
|
#increment_tpm_counter $counter > /dev/null \
|
|
|
|
#|| die "Unable to increment tpm counter"
|
|
|
|
#increment_tpm_counter $counter > /dev/null \
|
|
|
|
#|| die "Unable to increment tpm counter"
|
2018-06-19 12:27:27 -07:00
|
|
|
|
|
|
|
mount -o remount,rw /boot
|
2018-06-20 09:20:39 -07:00
|
|
|
|
|
|
|
counter_value=`expr $counter_value + 1`
|
|
|
|
echo $counter_value > $HOTP_COUNTER \
|
2019-08-21 15:40:36 -05:00
|
|
|
|| fatal_error "Unable to create hotp counter file"
|
2018-06-20 09:20:39 -07:00
|
|
|
|
2020-06-24 17:54:39 +02:00
|
|
|
# Store/overwrite HOTP USB Security Dongle branding found out beforehand
|
2020-06-24 18:12:56 +02:00
|
|
|
echo $HOTPKEY_BRANDING > $HOTP_KEY \
|
2020-06-24 17:40:49 +02:00
|
|
|
|| die "Unable to store hotp key file"
|
|
|
|
|
2018-06-20 09:20:39 -07:00
|
|
|
#sha256sum /tmp/counter-$counter > $HOTP_COUNTER \
|
|
|
|
#|| die "Unable to create hotp counter file"
|
2018-06-19 12:27:27 -07:00
|
|
|
mount -o remount,ro /boot
|
|
|
|
|
2020-06-24 18:12:56 +02:00
|
|
|
echo -e "\n$HOTPKEY_BRANDING initialized successfully. Press Enter to continue."
|
2018-06-19 12:27:27 -07:00
|
|
|
read
|
|
|
|
|
|
|
|
exit 0
|