2023-11-18 19:37:56 +00:00
|
|
|
{ config, options, lib, utils, pkgs, ... }:
|
2010-10-25 00:57:30 +00:00
|
|
|
|
2014-04-14 14:26:48 +00:00
|
|
|
with lib;
|
2010-10-25 00:57:30 +00:00
|
|
|
|
|
|
|
let
|
2012-03-03 16:07:18 +00:00
|
|
|
luks = config.boot.initrd.luks;
|
2023-11-18 19:37:56 +00:00
|
|
|
clevis = config.boot.initrd.clevis;
|
|
|
|
systemd = config.boot.initrd.systemd;
|
2020-01-17 09:39:22 +00:00
|
|
|
kernelPackages = config.boot.kernelPackages;
|
2022-04-13 18:45:29 +00:00
|
|
|
defaultPrio = (mkOptionDefault {}).priority;
|
2012-03-03 16:07:18 +00:00
|
|
|
|
2017-09-15 21:47:36 +00:00
|
|
|
commonFunctions = ''
|
|
|
|
die() {
|
2018-10-11 19:52:11 +00:00
|
|
|
echo "$@" >&2
|
|
|
|
exit 1
|
2017-09-15 21:47:36 +00:00
|
|
|
}
|
2017-04-04 19:19:38 +00:00
|
|
|
|
2018-09-03 17:45:54 +00:00
|
|
|
dev_exist() {
|
2018-10-11 19:52:11 +00:00
|
|
|
local target="$1"
|
|
|
|
if [ -e $target ]; then
|
|
|
|
return 0
|
|
|
|
else
|
|
|
|
local uuid=$(echo -n $target | sed -e 's,UUID=\(.*\),\1,g')
|
2018-10-11 19:52:46 +00:00
|
|
|
blkid --uuid $uuid >/dev/null
|
2018-10-11 19:52:11 +00:00
|
|
|
return $?
|
|
|
|
fi
|
2018-09-03 17:45:54 +00:00
|
|
|
}
|
|
|
|
|
2017-04-04 19:19:38 +00:00
|
|
|
wait_target() {
|
|
|
|
local name="$1"
|
|
|
|
local target="$2"
|
2017-09-15 21:47:36 +00:00
|
|
|
local secs="''${3:-10}"
|
|
|
|
local desc="''${4:-$name $target to appear}"
|
2017-04-04 19:19:38 +00:00
|
|
|
|
2018-09-03 17:45:54 +00:00
|
|
|
if ! dev_exist $target; then
|
2017-09-15 21:47:36 +00:00
|
|
|
echo -n "Waiting $secs seconds for $desc..."
|
2017-04-04 19:19:38 +00:00
|
|
|
local success=false;
|
2017-09-15 21:47:36 +00:00
|
|
|
for try in $(seq $secs); do
|
2017-04-04 19:19:38 +00:00
|
|
|
echo -n "."
|
|
|
|
sleep 1
|
2018-09-03 17:45:54 +00:00
|
|
|
if dev_exist $target; then
|
2017-09-15 21:47:36 +00:00
|
|
|
success=true
|
|
|
|
break
|
|
|
|
fi
|
2017-04-04 19:19:38 +00:00
|
|
|
done
|
2017-09-15 21:47:36 +00:00
|
|
|
if [ $success == true ]; then
|
2017-04-04 19:19:38 +00:00
|
|
|
echo " - success";
|
2017-09-15 21:47:36 +00:00
|
|
|
return 0
|
2017-04-04 19:19:38 +00:00
|
|
|
else
|
|
|
|
echo " - failure";
|
2017-09-15 21:47:36 +00:00
|
|
|
return 1
|
2017-04-04 19:19:38 +00:00
|
|
|
fi
|
|
|
|
fi
|
2017-09-15 21:47:36 +00:00
|
|
|
return 0
|
2017-04-04 19:19:38 +00:00
|
|
|
}
|
|
|
|
|
2017-09-15 21:47:36 +00:00
|
|
|
wait_yubikey() {
|
2018-10-11 19:52:11 +00:00
|
|
|
local secs="''${1:-10}"
|
2017-09-15 21:47:36 +00:00
|
|
|
|
2018-10-11 19:52:11 +00:00
|
|
|
ykinfo -v 1>/dev/null 2>&1
|
|
|
|
if [ $? != 0 ]; then
|
2020-12-20 21:35:59 +00:00
|
|
|
echo -n "Waiting $secs seconds for YubiKey to appear..."
|
2018-10-11 19:52:11 +00:00
|
|
|
local success=false
|
|
|
|
for try in $(seq $secs); do
|
|
|
|
echo -n .
|
|
|
|
sleep 1
|
|
|
|
ykinfo -v 1>/dev/null 2>&1
|
|
|
|
if [ $? == 0 ]; then
|
|
|
|
success=true
|
|
|
|
break
|
|
|
|
fi
|
|
|
|
done
|
|
|
|
if [ $success == true ]; then
|
|
|
|
echo " - success";
|
|
|
|
return 0
|
|
|
|
else
|
|
|
|
echo " - failure";
|
|
|
|
return 1
|
|
|
|
fi
|
|
|
|
fi
|
|
|
|
return 0
|
2017-09-15 21:47:36 +00:00
|
|
|
}
|
2019-05-09 18:15:35 +00:00
|
|
|
|
|
|
|
wait_gpgcard() {
|
|
|
|
local secs="''${1:-10}"
|
|
|
|
|
|
|
|
gpg --card-status > /dev/null 2> /dev/null
|
|
|
|
if [ $? != 0 ]; then
|
|
|
|
echo -n "Waiting $secs seconds for GPG Card to appear"
|
|
|
|
local success=false
|
|
|
|
for try in $(seq $secs); do
|
|
|
|
echo -n .
|
|
|
|
sleep 1
|
|
|
|
gpg --card-status > /dev/null 2> /dev/null
|
|
|
|
if [ $? == 0 ]; then
|
|
|
|
success=true
|
|
|
|
break
|
|
|
|
fi
|
|
|
|
done
|
|
|
|
if [ $success == true ]; then
|
|
|
|
echo " - success";
|
|
|
|
return 0
|
|
|
|
else
|
|
|
|
echo " - failure";
|
|
|
|
return 1
|
|
|
|
fi
|
|
|
|
fi
|
|
|
|
return 0
|
|
|
|
}
|
2017-09-15 21:47:36 +00:00
|
|
|
'';
|
|
|
|
|
2017-09-15 21:48:51 +00:00
|
|
|
preCommands = ''
|
|
|
|
# A place to store crypto things
|
|
|
|
|
|
|
|
# A ramfs is used here to ensure that the file used to update
|
|
|
|
# the key slot with cryptsetup will never get swapped out.
|
|
|
|
# Warning: Do NOT replace with tmpfs!
|
|
|
|
mkdir -p /crypt-ramfs
|
|
|
|
mount -t ramfs none /crypt-ramfs
|
2018-06-10 20:18:21 +00:00
|
|
|
|
2019-04-28 17:49:32 +00:00
|
|
|
# Cryptsetup locking directory
|
|
|
|
mkdir -p /run/cryptsetup
|
|
|
|
|
2020-12-20 21:35:59 +00:00
|
|
|
# For YubiKey salt storage
|
2018-06-10 20:18:21 +00:00
|
|
|
mkdir -p /crypt-storage
|
2018-06-10 20:18:27 +00:00
|
|
|
|
2019-05-09 18:15:35 +00:00
|
|
|
${optionalString luks.gpgSupport ''
|
|
|
|
export GPG_TTY=$(tty)
|
|
|
|
export GNUPGHOME=/crypt-ramfs/.gnupg
|
|
|
|
|
|
|
|
gpg-agent --daemon --scdaemon-program $out/bin/scdaemon > /dev/null 2> /dev/null
|
|
|
|
''}
|
2019-12-10 01:51:19 +00:00
|
|
|
|
2018-06-10 20:18:27 +00:00
|
|
|
# Disable all input echo for the whole stage. We could use read -s
|
2023-05-20 02:11:38 +00:00
|
|
|
# instead but that would occasionally leak characters between read
|
2018-06-10 20:18:27 +00:00
|
|
|
# invocations.
|
|
|
|
stty -echo
|
2017-09-15 21:48:51 +00:00
|
|
|
'';
|
|
|
|
|
|
|
|
postCommands = ''
|
2018-06-10 20:18:27 +00:00
|
|
|
stty echo
|
2018-06-10 20:18:21 +00:00
|
|
|
umount /crypt-storage 2>/dev/null
|
2017-09-15 21:48:51 +00:00
|
|
|
umount /crypt-ramfs 2>/dev/null
|
|
|
|
'';
|
|
|
|
|
2021-05-26 13:43:38 +00:00
|
|
|
openCommand = name: dev: assert name == dev.name;
|
2017-09-15 21:48:51 +00:00
|
|
|
let
|
2021-05-26 13:43:38 +00:00
|
|
|
csopen = "cryptsetup luksOpen ${dev.device} ${dev.name}"
|
|
|
|
+ optionalString dev.allowDiscards " --allow-discards"
|
|
|
|
+ optionalString dev.bypassWorkqueues " --perf-no_read_workqueue --perf-no_write_workqueue"
|
|
|
|
+ optionalString (dev.header != null) " --header=${dev.header}";
|
|
|
|
cschange = "cryptsetup luksChangeKey ${dev.device} ${optionalString (dev.header != null) "--header=${dev.header}"}";
|
2022-08-23 08:27:41 +00:00
|
|
|
fido2luksCredentials = dev.fido2.credentials ++ optional (dev.fido2.credential != null) dev.fido2.credential;
|
2017-09-15 21:48:51 +00:00
|
|
|
in ''
|
2017-04-04 19:19:38 +00:00
|
|
|
# Wait for luksRoot (and optionally keyFile and/or header) to appear, e.g.
|
|
|
|
# if on a USB drive.
|
2021-05-26 13:43:38 +00:00
|
|
|
wait_target "device" ${dev.device} || die "${dev.device} is unavailable"
|
2012-03-03 16:07:18 +00:00
|
|
|
|
2021-05-26 13:43:38 +00:00
|
|
|
${optionalString (dev.header != null) ''
|
|
|
|
wait_target "header" ${dev.header} || die "${dev.header} is unavailable"
|
2017-04-04 19:19:38 +00:00
|
|
|
''}
|
|
|
|
|
2023-03-22 14:17:23 +00:00
|
|
|
try_empty_passphrase() {
|
|
|
|
${if dev.tryEmptyPassphrase then ''
|
|
|
|
echo "Trying empty passphrase!"
|
|
|
|
echo "" | ${csopen}
|
|
|
|
cs_status=$?
|
|
|
|
if [ $cs_status -eq 0 ]; then
|
|
|
|
return 0
|
|
|
|
else
|
|
|
|
return 1
|
|
|
|
fi
|
|
|
|
'' else "return 1"}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2017-09-15 21:48:51 +00:00
|
|
|
do_open_passphrase() {
|
|
|
|
local passphrase
|
|
|
|
|
|
|
|
while true; do
|
2021-05-26 13:43:38 +00:00
|
|
|
echo -n "Passphrase for ${dev.device}: "
|
2017-09-15 21:48:51 +00:00
|
|
|
passphrase=
|
|
|
|
while true; do
|
|
|
|
if [ -e /crypt-ramfs/passphrase ]; then
|
|
|
|
echo "reused"
|
|
|
|
passphrase=$(cat /crypt-ramfs/passphrase)
|
|
|
|
break
|
|
|
|
else
|
|
|
|
# ask cryptsetup-askpass
|
2021-05-26 13:43:38 +00:00
|
|
|
echo -n "${dev.device}" > /crypt-ramfs/device
|
2017-09-15 21:48:51 +00:00
|
|
|
|
2018-06-10 20:18:27 +00:00
|
|
|
# and try reading it from /dev/console with a timeout
|
|
|
|
IFS= read -t 1 -r passphrase
|
2017-09-15 21:48:51 +00:00
|
|
|
if [ -n "$passphrase" ]; then
|
|
|
|
${if luks.reusePassphrases then ''
|
|
|
|
# remember it for the next device
|
|
|
|
echo -n "$passphrase" > /crypt-ramfs/passphrase
|
|
|
|
'' else ''
|
|
|
|
# Don't save it to ramfs. We are very paranoid
|
|
|
|
''}
|
|
|
|
echo
|
|
|
|
break
|
|
|
|
fi
|
|
|
|
fi
|
|
|
|
done
|
2021-05-26 13:43:38 +00:00
|
|
|
echo -n "Verifying passphrase for ${dev.device}..."
|
2017-09-15 21:48:51 +00:00
|
|
|
echo -n "$passphrase" | ${csopen} --key-file=-
|
|
|
|
if [ $? == 0 ]; then
|
|
|
|
echo " - success"
|
|
|
|
${if luks.reusePassphrases then ''
|
|
|
|
# we don't rm here because we might reuse it for the next device
|
|
|
|
'' else ''
|
|
|
|
rm -f /crypt-ramfs/passphrase
|
|
|
|
''}
|
|
|
|
break
|
|
|
|
else
|
|
|
|
echo " - failure"
|
|
|
|
# ask for a different one
|
|
|
|
rm -f /crypt-ramfs/passphrase
|
|
|
|
fi
|
|
|
|
done
|
|
|
|
}
|
2012-05-02 22:37:14 +00:00
|
|
|
|
2017-09-15 21:48:51 +00:00
|
|
|
# LUKS
|
2014-01-28 03:02:51 +00:00
|
|
|
open_normally() {
|
2021-05-26 13:43:38 +00:00
|
|
|
${if (dev.keyFile != null) then ''
|
|
|
|
if wait_target "key file" ${dev.keyFile}; then
|
|
|
|
${csopen} --key-file=${dev.keyFile} \
|
|
|
|
${optionalString (dev.keyFileSize != null) "--keyfile-size=${toString dev.keyFileSize}"} \
|
|
|
|
${optionalString (dev.keyFileOffset != null) "--keyfile-offset=${toString dev.keyFileOffset}"}
|
2023-03-22 14:17:23 +00:00
|
|
|
cs_status=$?
|
|
|
|
if [ $cs_status -ne 0 ]; then
|
|
|
|
echo "Key File ${dev.keyFile} failed!"
|
|
|
|
if ! try_empty_passphrase; then
|
|
|
|
${if dev.fallbackToPassword then "echo" else "die"} "${dev.keyFile} is unavailable"
|
|
|
|
echo " - failing back to interactive password prompt"
|
|
|
|
do_open_passphrase
|
|
|
|
fi
|
|
|
|
fi
|
2017-10-14 16:08:25 +00:00
|
|
|
else
|
2023-03-22 14:17:23 +00:00
|
|
|
# If the key file never shows up we should also try the empty passphrase
|
|
|
|
if ! try_empty_passphrase; then
|
|
|
|
${if dev.fallbackToPassword then "echo" else "die"} "${dev.keyFile} is unavailable"
|
|
|
|
echo " - failing back to interactive password prompt"
|
|
|
|
do_open_passphrase
|
|
|
|
fi
|
2017-10-14 16:08:25 +00:00
|
|
|
fi
|
2017-09-15 21:48:51 +00:00
|
|
|
'' else ''
|
2023-03-22 14:17:23 +00:00
|
|
|
if ! try_empty_passphrase; then
|
|
|
|
do_open_passphrase
|
|
|
|
fi
|
2017-10-14 16:08:25 +00:00
|
|
|
''}
|
2014-01-28 03:02:51 +00:00
|
|
|
}
|
|
|
|
|
2021-05-26 13:43:38 +00:00
|
|
|
${optionalString (luks.yubikeySupport && (dev.yubikey != null)) ''
|
2020-12-20 21:35:59 +00:00
|
|
|
# YubiKey
|
2014-01-28 03:02:51 +00:00
|
|
|
rbtohex() {
|
Replace the current Yubikey PBA implementation with the previous one.
Rationale:
* The main reason for choosing to implement the PBA in accordance
with the Yubico documentation was to prevent a MITM-USB-attack
successfully recovering the new LUKS key.
* However, a MITM-USB-attacker can read user id and password when
they were entered for PBA, which allows him to recover the new
challenge after the PBA is complete, with which he can challenge
the Yubikey, decrypt the new AES blob and recover the LUKS key.
* Additionally, since the Yubikey shared secret is stored in the
same AES blob, after such an attack not only is the LUKS device
compromised, the Yubikey is as well, since the shared secret
has also been recovered by the attacker.
* Furthermore, with this method an attacker could also bruteforce
the AES blob, if he has access to the unencrypted device, which
would again compromise the Yubikey, should he be successful.
* Finally, with this method, once the LUKS key has been recovered
once, the encryption is permanently broken, while with the previous
system, the LUKS key itself it changed at every successful boot,
so recovering it once will not necessarily result in a permanent
breakage and will also not compromise the Yubikey itself (since
its secret is never stored anywhere but on the Yubikey itself).
Summary:
The current implementation opens up up vulnerability to brute-forcing
the AES blob, while retaining the current MITM-USB attack, additionally
making the consequences of this attack permanent and extending it to
the Yubikey itself.
2014-02-03 21:50:17 +00:00
|
|
|
( od -An -vtx1 | tr -d ' \n' )
|
2014-01-28 03:02:51 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
hextorb() {
|
Update to the Yubikey PBA
Security-relevant changes:
* No (salted) passphrase hash send to the yubikey, only hash of the salt (as it was in the original implementation).
* Derive $k_luks with PBKDF2 from the yubikey $response (as the PBKDF2 salt) and the passphrase $k_user
(as the PBKDF2 password), so that if two-factor authentication is enabled
(a) a USB-MITM attack on the yubikey itself is not enough to break the system
(b) the potentially low-entropy $k_user is better protected against brute-force attacks
* Instead of using uuidgen, gather the salt (previously random uuid / uuid_r) directly from /dev/random.
* Length of the new salt in byte added as the parameter "saltLength", defaults to 16 byte.
Note: Length of the challenge is 64 byte, so saltLength > 64 may have no benefit over saltLengh = 64.
* Length of $k_luks derived with PBKDF2 in byte added as the parameter "keyLength", defaults to 64 byte.
Example: For a luks device with a 512-bit key, keyLength should be 64.
* Increase of the PBKDF2 iteration count per successful authentication added as the
parameter "iterationStep", defaults to 0.
Other changes:
* Add optional grace period before trying to find the yubikey, defaults to 2 seconds.
Full overview of the yubikey authentication process:
(1) Read $salt and $iterations from unencrypted device (UD).
(2) Calculate the $challenge from the $salt with a hash function.
Chosen instantiation: SHA-512($salt).
(3) Challenge the yubikey with the $challenge and receive the $response.
(4) Repeat three times:
(a) Prompt for the passphrase $k_user.
(b) Derive the key $k_luks for the luks device with a key derivation function from $k_user and $response.
Chosen instantiation: PBKDF2(HMAC-SHA-512, $k_user, $response, $iterations, keyLength).
(c) Try to open the luks device with $k_luks and escape loop (4) only on success.
(5) Proceed only if luks device was opened successfully, fail otherwise.
(6) Gather $new_salt from a cryptographically secure pseudorandom number generator
Chosen instantiation: /dev/random
(7) Calculate the $new_challenge from the $new_salt with the same hash function as (2).
(8) Challenge the yubikey with the $new_challenge and receive the $new_response.
(9) Derive the new key $new_k_luks for the luks device in the same manner as in (4) (b),
but with more iterations as given by iterationStep.
(10) Try to change the luks device's key $k_luks to $new_k_luks.
(11) If (10) was successful, write the $new_salt and the $new_iterations to the UD.
Note: $new_iterations = $iterations + iterationStep
Known (software) attack vectors:
* A MITM attack on the keyboard can recover $k_user. This, combined with a USB-MITM
attack on the yubikey for the $response (1) or the $new_response (2) will result in
(1) $k_luks being recovered,
(2) $new_k_luks being recovered.
* Any attacker with access to the RAM state of stage-1 at mid- or post-authentication
can recover $k_user, $k_luks, and $new_k_luks
* If an attacker has recovered $response or $new_response, he can perform a brute-force
attack on $k_user with it without the Yubikey needing to be present (using cryptsetup's
"luksOpen --verify-passphrase" oracle. He could even make a copy of the luks device's
luks header and run the brute-force attack without further access to the system.
* A USB-MITM attack on the yubikey will allow an attacker to attempt to brute-force
the yubikey's internal key ("shared secret") without it needing to be present anymore.
Credits:
* Florian Klien,
for the original concept and the reference implementation over at
https://github.com/flowolf/initramfs_ykfde
* Anthony Thysse,
for the reference implementation of accessing OpenSSL's PBKDF2 over at
http://www.ict.griffith.edu.au/anthony/software/pbkdf2.c
2014-02-05 16:10:59 +00:00
|
|
|
( tr '[:lower:]' '[:upper:]' | sed -e 's/\([0-9A-F]\{2\}\)/\\\\\\x\1/gI' | xargs printf )
|
2014-01-28 03:02:51 +00:00
|
|
|
}
|
2014-01-25 02:27:12 +00:00
|
|
|
|
2017-09-15 21:48:51 +00:00
|
|
|
do_open_yubikey() {
|
Update to the Yubikey PBA
Security-relevant changes:
* No (salted) passphrase hash send to the yubikey, only hash of the salt (as it was in the original implementation).
* Derive $k_luks with PBKDF2 from the yubikey $response (as the PBKDF2 salt) and the passphrase $k_user
(as the PBKDF2 password), so that if two-factor authentication is enabled
(a) a USB-MITM attack on the yubikey itself is not enough to break the system
(b) the potentially low-entropy $k_user is better protected against brute-force attacks
* Instead of using uuidgen, gather the salt (previously random uuid / uuid_r) directly from /dev/random.
* Length of the new salt in byte added as the parameter "saltLength", defaults to 16 byte.
Note: Length of the challenge is 64 byte, so saltLength > 64 may have no benefit over saltLengh = 64.
* Length of $k_luks derived with PBKDF2 in byte added as the parameter "keyLength", defaults to 64 byte.
Example: For a luks device with a 512-bit key, keyLength should be 64.
* Increase of the PBKDF2 iteration count per successful authentication added as the
parameter "iterationStep", defaults to 0.
Other changes:
* Add optional grace period before trying to find the yubikey, defaults to 2 seconds.
Full overview of the yubikey authentication process:
(1) Read $salt and $iterations from unencrypted device (UD).
(2) Calculate the $challenge from the $salt with a hash function.
Chosen instantiation: SHA-512($salt).
(3) Challenge the yubikey with the $challenge and receive the $response.
(4) Repeat three times:
(a) Prompt for the passphrase $k_user.
(b) Derive the key $k_luks for the luks device with a key derivation function from $k_user and $response.
Chosen instantiation: PBKDF2(HMAC-SHA-512, $k_user, $response, $iterations, keyLength).
(c) Try to open the luks device with $k_luks and escape loop (4) only on success.
(5) Proceed only if luks device was opened successfully, fail otherwise.
(6) Gather $new_salt from a cryptographically secure pseudorandom number generator
Chosen instantiation: /dev/random
(7) Calculate the $new_challenge from the $new_salt with the same hash function as (2).
(8) Challenge the yubikey with the $new_challenge and receive the $new_response.
(9) Derive the new key $new_k_luks for the luks device in the same manner as in (4) (b),
but with more iterations as given by iterationStep.
(10) Try to change the luks device's key $k_luks to $new_k_luks.
(11) If (10) was successful, write the $new_salt and the $new_iterations to the UD.
Note: $new_iterations = $iterations + iterationStep
Known (software) attack vectors:
* A MITM attack on the keyboard can recover $k_user. This, combined with a USB-MITM
attack on the yubikey for the $response (1) or the $new_response (2) will result in
(1) $k_luks being recovered,
(2) $new_k_luks being recovered.
* Any attacker with access to the RAM state of stage-1 at mid- or post-authentication
can recover $k_user, $k_luks, and $new_k_luks
* If an attacker has recovered $response or $new_response, he can perform a brute-force
attack on $k_user with it without the Yubikey needing to be present (using cryptsetup's
"luksOpen --verify-passphrase" oracle. He could even make a copy of the luks device's
luks header and run the brute-force attack without further access to the system.
* A USB-MITM attack on the yubikey will allow an attacker to attempt to brute-force
the yubikey's internal key ("shared secret") without it needing to be present anymore.
Credits:
* Florian Klien,
for the original concept and the reference implementation over at
https://github.com/flowolf/initramfs_ykfde
* Anthony Thysse,
for the reference implementation of accessing OpenSSL's PBKDF2 over at
http://www.ict.griffith.edu.au/anthony/software/pbkdf2.c
2014-02-05 16:10:59 +00:00
|
|
|
# Make all of these local to this function
|
|
|
|
# to prevent their values being leaked
|
|
|
|
local salt
|
|
|
|
local iterations
|
2014-01-28 03:02:51 +00:00
|
|
|
local k_user
|
|
|
|
local challenge
|
Update to the Yubikey PBA
Security-relevant changes:
* No (salted) passphrase hash send to the yubikey, only hash of the salt (as it was in the original implementation).
* Derive $k_luks with PBKDF2 from the yubikey $response (as the PBKDF2 salt) and the passphrase $k_user
(as the PBKDF2 password), so that if two-factor authentication is enabled
(a) a USB-MITM attack on the yubikey itself is not enough to break the system
(b) the potentially low-entropy $k_user is better protected against brute-force attacks
* Instead of using uuidgen, gather the salt (previously random uuid / uuid_r) directly from /dev/random.
* Length of the new salt in byte added as the parameter "saltLength", defaults to 16 byte.
Note: Length of the challenge is 64 byte, so saltLength > 64 may have no benefit over saltLengh = 64.
* Length of $k_luks derived with PBKDF2 in byte added as the parameter "keyLength", defaults to 64 byte.
Example: For a luks device with a 512-bit key, keyLength should be 64.
* Increase of the PBKDF2 iteration count per successful authentication added as the
parameter "iterationStep", defaults to 0.
Other changes:
* Add optional grace period before trying to find the yubikey, defaults to 2 seconds.
Full overview of the yubikey authentication process:
(1) Read $salt and $iterations from unencrypted device (UD).
(2) Calculate the $challenge from the $salt with a hash function.
Chosen instantiation: SHA-512($salt).
(3) Challenge the yubikey with the $challenge and receive the $response.
(4) Repeat three times:
(a) Prompt for the passphrase $k_user.
(b) Derive the key $k_luks for the luks device with a key derivation function from $k_user and $response.
Chosen instantiation: PBKDF2(HMAC-SHA-512, $k_user, $response, $iterations, keyLength).
(c) Try to open the luks device with $k_luks and escape loop (4) only on success.
(5) Proceed only if luks device was opened successfully, fail otherwise.
(6) Gather $new_salt from a cryptographically secure pseudorandom number generator
Chosen instantiation: /dev/random
(7) Calculate the $new_challenge from the $new_salt with the same hash function as (2).
(8) Challenge the yubikey with the $new_challenge and receive the $new_response.
(9) Derive the new key $new_k_luks for the luks device in the same manner as in (4) (b),
but with more iterations as given by iterationStep.
(10) Try to change the luks device's key $k_luks to $new_k_luks.
(11) If (10) was successful, write the $new_salt and the $new_iterations to the UD.
Note: $new_iterations = $iterations + iterationStep
Known (software) attack vectors:
* A MITM attack on the keyboard can recover $k_user. This, combined with a USB-MITM
attack on the yubikey for the $response (1) or the $new_response (2) will result in
(1) $k_luks being recovered,
(2) $new_k_luks being recovered.
* Any attacker with access to the RAM state of stage-1 at mid- or post-authentication
can recover $k_user, $k_luks, and $new_k_luks
* If an attacker has recovered $response or $new_response, he can perform a brute-force
attack on $k_user with it without the Yubikey needing to be present (using cryptsetup's
"luksOpen --verify-passphrase" oracle. He could even make a copy of the luks device's
luks header and run the brute-force attack without further access to the system.
* A USB-MITM attack on the yubikey will allow an attacker to attempt to brute-force
the yubikey's internal key ("shared secret") without it needing to be present anymore.
Credits:
* Florian Klien,
for the original concept and the reference implementation over at
https://github.com/flowolf/initramfs_ykfde
* Anthony Thysse,
for the reference implementation of accessing OpenSSL's PBKDF2 over at
http://www.ict.griffith.edu.au/anthony/software/pbkdf2.c
2014-02-05 16:10:59 +00:00
|
|
|
local response
|
|
|
|
local k_luks
|
Replace the current Yubikey PBA implementation with the previous one.
Rationale:
* The main reason for choosing to implement the PBA in accordance
with the Yubico documentation was to prevent a MITM-USB-attack
successfully recovering the new LUKS key.
* However, a MITM-USB-attacker can read user id and password when
they were entered for PBA, which allows him to recover the new
challenge after the PBA is complete, with which he can challenge
the Yubikey, decrypt the new AES blob and recover the LUKS key.
* Additionally, since the Yubikey shared secret is stored in the
same AES blob, after such an attack not only is the LUKS device
compromised, the Yubikey is as well, since the shared secret
has also been recovered by the attacker.
* Furthermore, with this method an attacker could also bruteforce
the AES blob, if he has access to the unencrypted device, which
would again compromise the Yubikey, should he be successful.
* Finally, with this method, once the LUKS key has been recovered
once, the encryption is permanently broken, while with the previous
system, the LUKS key itself it changed at every successful boot,
so recovering it once will not necessarily result in a permanent
breakage and will also not compromise the Yubikey itself (since
its secret is never stored anywhere but on the Yubikey itself).
Summary:
The current implementation opens up up vulnerability to brute-forcing
the AES blob, while retaining the current MITM-USB attack, additionally
making the consequences of this attack permanent and extending it to
the Yubikey itself.
2014-02-03 21:50:17 +00:00
|
|
|
local opened
|
Update to the Yubikey PBA
Security-relevant changes:
* No (salted) passphrase hash send to the yubikey, only hash of the salt (as it was in the original implementation).
* Derive $k_luks with PBKDF2 from the yubikey $response (as the PBKDF2 salt) and the passphrase $k_user
(as the PBKDF2 password), so that if two-factor authentication is enabled
(a) a USB-MITM attack on the yubikey itself is not enough to break the system
(b) the potentially low-entropy $k_user is better protected against brute-force attacks
* Instead of using uuidgen, gather the salt (previously random uuid / uuid_r) directly from /dev/random.
* Length of the new salt in byte added as the parameter "saltLength", defaults to 16 byte.
Note: Length of the challenge is 64 byte, so saltLength > 64 may have no benefit over saltLengh = 64.
* Length of $k_luks derived with PBKDF2 in byte added as the parameter "keyLength", defaults to 64 byte.
Example: For a luks device with a 512-bit key, keyLength should be 64.
* Increase of the PBKDF2 iteration count per successful authentication added as the
parameter "iterationStep", defaults to 0.
Other changes:
* Add optional grace period before trying to find the yubikey, defaults to 2 seconds.
Full overview of the yubikey authentication process:
(1) Read $salt and $iterations from unencrypted device (UD).
(2) Calculate the $challenge from the $salt with a hash function.
Chosen instantiation: SHA-512($salt).
(3) Challenge the yubikey with the $challenge and receive the $response.
(4) Repeat three times:
(a) Prompt for the passphrase $k_user.
(b) Derive the key $k_luks for the luks device with a key derivation function from $k_user and $response.
Chosen instantiation: PBKDF2(HMAC-SHA-512, $k_user, $response, $iterations, keyLength).
(c) Try to open the luks device with $k_luks and escape loop (4) only on success.
(5) Proceed only if luks device was opened successfully, fail otherwise.
(6) Gather $new_salt from a cryptographically secure pseudorandom number generator
Chosen instantiation: /dev/random
(7) Calculate the $new_challenge from the $new_salt with the same hash function as (2).
(8) Challenge the yubikey with the $new_challenge and receive the $new_response.
(9) Derive the new key $new_k_luks for the luks device in the same manner as in (4) (b),
but with more iterations as given by iterationStep.
(10) Try to change the luks device's key $k_luks to $new_k_luks.
(11) If (10) was successful, write the $new_salt and the $new_iterations to the UD.
Note: $new_iterations = $iterations + iterationStep
Known (software) attack vectors:
* A MITM attack on the keyboard can recover $k_user. This, combined with a USB-MITM
attack on the yubikey for the $response (1) or the $new_response (2) will result in
(1) $k_luks being recovered,
(2) $new_k_luks being recovered.
* Any attacker with access to the RAM state of stage-1 at mid- or post-authentication
can recover $k_user, $k_luks, and $new_k_luks
* If an attacker has recovered $response or $new_response, he can perform a brute-force
attack on $k_user with it without the Yubikey needing to be present (using cryptsetup's
"luksOpen --verify-passphrase" oracle. He could even make a copy of the luks device's
luks header and run the brute-force attack without further access to the system.
* A USB-MITM attack on the yubikey will allow an attacker to attempt to brute-force
the yubikey's internal key ("shared secret") without it needing to be present anymore.
Credits:
* Florian Klien,
for the original concept and the reference implementation over at
https://github.com/flowolf/initramfs_ykfde
* Anthony Thysse,
for the reference implementation of accessing OpenSSL's PBKDF2 over at
http://www.ict.griffith.edu.au/anthony/software/pbkdf2.c
2014-02-05 16:10:59 +00:00
|
|
|
local new_salt
|
|
|
|
local new_iterations
|
|
|
|
local new_challenge
|
|
|
|
local new_response
|
|
|
|
local new_k_luks
|
2014-01-29 16:20:05 +00:00
|
|
|
|
2021-05-26 13:43:38 +00:00
|
|
|
mount -t ${dev.yubikey.storage.fsType} ${dev.yubikey.storage.device} /crypt-storage || \
|
2020-12-20 21:35:59 +00:00
|
|
|
die "Failed to mount YubiKey salt storage device"
|
2014-01-29 16:20:05 +00:00
|
|
|
|
2021-05-26 13:43:38 +00:00
|
|
|
salt="$(cat /crypt-storage${dev.yubikey.storage.path} | sed -n 1p | tr -d '\n')"
|
|
|
|
iterations="$(cat /crypt-storage${dev.yubikey.storage.path} | sed -n 2p | tr -d '\n')"
|
Update to the Yubikey PBA
Security-relevant changes:
* No (salted) passphrase hash send to the yubikey, only hash of the salt (as it was in the original implementation).
* Derive $k_luks with PBKDF2 from the yubikey $response (as the PBKDF2 salt) and the passphrase $k_user
(as the PBKDF2 password), so that if two-factor authentication is enabled
(a) a USB-MITM attack on the yubikey itself is not enough to break the system
(b) the potentially low-entropy $k_user is better protected against brute-force attacks
* Instead of using uuidgen, gather the salt (previously random uuid / uuid_r) directly from /dev/random.
* Length of the new salt in byte added as the parameter "saltLength", defaults to 16 byte.
Note: Length of the challenge is 64 byte, so saltLength > 64 may have no benefit over saltLengh = 64.
* Length of $k_luks derived with PBKDF2 in byte added as the parameter "keyLength", defaults to 64 byte.
Example: For a luks device with a 512-bit key, keyLength should be 64.
* Increase of the PBKDF2 iteration count per successful authentication added as the
parameter "iterationStep", defaults to 0.
Other changes:
* Add optional grace period before trying to find the yubikey, defaults to 2 seconds.
Full overview of the yubikey authentication process:
(1) Read $salt and $iterations from unencrypted device (UD).
(2) Calculate the $challenge from the $salt with a hash function.
Chosen instantiation: SHA-512($salt).
(3) Challenge the yubikey with the $challenge and receive the $response.
(4) Repeat three times:
(a) Prompt for the passphrase $k_user.
(b) Derive the key $k_luks for the luks device with a key derivation function from $k_user and $response.
Chosen instantiation: PBKDF2(HMAC-SHA-512, $k_user, $response, $iterations, keyLength).
(c) Try to open the luks device with $k_luks and escape loop (4) only on success.
(5) Proceed only if luks device was opened successfully, fail otherwise.
(6) Gather $new_salt from a cryptographically secure pseudorandom number generator
Chosen instantiation: /dev/random
(7) Calculate the $new_challenge from the $new_salt with the same hash function as (2).
(8) Challenge the yubikey with the $new_challenge and receive the $new_response.
(9) Derive the new key $new_k_luks for the luks device in the same manner as in (4) (b),
but with more iterations as given by iterationStep.
(10) Try to change the luks device's key $k_luks to $new_k_luks.
(11) If (10) was successful, write the $new_salt and the $new_iterations to the UD.
Note: $new_iterations = $iterations + iterationStep
Known (software) attack vectors:
* A MITM attack on the keyboard can recover $k_user. This, combined with a USB-MITM
attack on the yubikey for the $response (1) or the $new_response (2) will result in
(1) $k_luks being recovered,
(2) $new_k_luks being recovered.
* Any attacker with access to the RAM state of stage-1 at mid- or post-authentication
can recover $k_user, $k_luks, and $new_k_luks
* If an attacker has recovered $response or $new_response, he can perform a brute-force
attack on $k_user with it without the Yubikey needing to be present (using cryptsetup's
"luksOpen --verify-passphrase" oracle. He could even make a copy of the luks device's
luks header and run the brute-force attack without further access to the system.
* A USB-MITM attack on the yubikey will allow an attacker to attempt to brute-force
the yubikey's internal key ("shared secret") without it needing to be present anymore.
Credits:
* Florian Klien,
for the original concept and the reference implementation over at
https://github.com/flowolf/initramfs_ykfde
* Anthony Thysse,
for the reference implementation of accessing OpenSSL's PBKDF2 over at
http://www.ict.griffith.edu.au/anthony/software/pbkdf2.c
2014-02-05 16:10:59 +00:00
|
|
|
challenge="$(echo -n $salt | openssl-wrap dgst -binary -sha512 | rbtohex)"
|
2021-05-26 13:43:38 +00:00
|
|
|
response="$(ykchalresp -${toString dev.yubikey.slot} -x $challenge 2>/dev/null)"
|
2014-01-25 02:27:12 +00:00
|
|
|
|
2014-01-28 03:02:51 +00:00
|
|
|
for try in $(seq 3); do
|
2021-05-26 13:43:38 +00:00
|
|
|
${optionalString dev.yubikey.twoFactor ''
|
2016-07-17 10:25:34 +00:00
|
|
|
echo -n "Enter two-factor passphrase: "
|
2020-12-20 21:24:14 +00:00
|
|
|
k_user=
|
|
|
|
while true; do
|
|
|
|
if [ -e /crypt-ramfs/passphrase ]; then
|
|
|
|
echo "reused"
|
|
|
|
k_user=$(cat /crypt-ramfs/passphrase)
|
|
|
|
break
|
|
|
|
else
|
|
|
|
# Try reading it from /dev/console with a timeout
|
|
|
|
IFS= read -t 1 -r k_user
|
|
|
|
if [ -n "$k_user" ]; then
|
|
|
|
${if luks.reusePassphrases then ''
|
|
|
|
# Remember it for the next device
|
|
|
|
echo -n "$k_user" > /crypt-ramfs/passphrase
|
|
|
|
'' else ''
|
|
|
|
# Don't save it to ramfs. We are very paranoid
|
|
|
|
''}
|
|
|
|
echo
|
|
|
|
break
|
|
|
|
fi
|
|
|
|
fi
|
|
|
|
done
|
2014-01-28 03:02:51 +00:00
|
|
|
''}
|
|
|
|
|
Update to the Yubikey PBA
Security-relevant changes:
* No (salted) passphrase hash send to the yubikey, only hash of the salt (as it was in the original implementation).
* Derive $k_luks with PBKDF2 from the yubikey $response (as the PBKDF2 salt) and the passphrase $k_user
(as the PBKDF2 password), so that if two-factor authentication is enabled
(a) a USB-MITM attack on the yubikey itself is not enough to break the system
(b) the potentially low-entropy $k_user is better protected against brute-force attacks
* Instead of using uuidgen, gather the salt (previously random uuid / uuid_r) directly from /dev/random.
* Length of the new salt in byte added as the parameter "saltLength", defaults to 16 byte.
Note: Length of the challenge is 64 byte, so saltLength > 64 may have no benefit over saltLengh = 64.
* Length of $k_luks derived with PBKDF2 in byte added as the parameter "keyLength", defaults to 64 byte.
Example: For a luks device with a 512-bit key, keyLength should be 64.
* Increase of the PBKDF2 iteration count per successful authentication added as the
parameter "iterationStep", defaults to 0.
Other changes:
* Add optional grace period before trying to find the yubikey, defaults to 2 seconds.
Full overview of the yubikey authentication process:
(1) Read $salt and $iterations from unencrypted device (UD).
(2) Calculate the $challenge from the $salt with a hash function.
Chosen instantiation: SHA-512($salt).
(3) Challenge the yubikey with the $challenge and receive the $response.
(4) Repeat three times:
(a) Prompt for the passphrase $k_user.
(b) Derive the key $k_luks for the luks device with a key derivation function from $k_user and $response.
Chosen instantiation: PBKDF2(HMAC-SHA-512, $k_user, $response, $iterations, keyLength).
(c) Try to open the luks device with $k_luks and escape loop (4) only on success.
(5) Proceed only if luks device was opened successfully, fail otherwise.
(6) Gather $new_salt from a cryptographically secure pseudorandom number generator
Chosen instantiation: /dev/random
(7) Calculate the $new_challenge from the $new_salt with the same hash function as (2).
(8) Challenge the yubikey with the $new_challenge and receive the $new_response.
(9) Derive the new key $new_k_luks for the luks device in the same manner as in (4) (b),
but with more iterations as given by iterationStep.
(10) Try to change the luks device's key $k_luks to $new_k_luks.
(11) If (10) was successful, write the $new_salt and the $new_iterations to the UD.
Note: $new_iterations = $iterations + iterationStep
Known (software) attack vectors:
* A MITM attack on the keyboard can recover $k_user. This, combined with a USB-MITM
attack on the yubikey for the $response (1) or the $new_response (2) will result in
(1) $k_luks being recovered,
(2) $new_k_luks being recovered.
* Any attacker with access to the RAM state of stage-1 at mid- or post-authentication
can recover $k_user, $k_luks, and $new_k_luks
* If an attacker has recovered $response or $new_response, he can perform a brute-force
attack on $k_user with it without the Yubikey needing to be present (using cryptsetup's
"luksOpen --verify-passphrase" oracle. He could even make a copy of the luks device's
luks header and run the brute-force attack without further access to the system.
* A USB-MITM attack on the yubikey will allow an attacker to attempt to brute-force
the yubikey's internal key ("shared secret") without it needing to be present anymore.
Credits:
* Florian Klien,
for the original concept and the reference implementation over at
https://github.com/flowolf/initramfs_ykfde
* Anthony Thysse,
for the reference implementation of accessing OpenSSL's PBKDF2 over at
http://www.ict.griffith.edu.au/anthony/software/pbkdf2.c
2014-02-05 16:10:59 +00:00
|
|
|
if [ ! -z "$k_user" ]; then
|
2021-05-26 13:43:38 +00:00
|
|
|
k_luks="$(echo -n $k_user | pbkdf2-sha512 ${toString dev.yubikey.keyLength} $iterations $response | rbtohex)"
|
Update to the Yubikey PBA
Security-relevant changes:
* No (salted) passphrase hash send to the yubikey, only hash of the salt (as it was in the original implementation).
* Derive $k_luks with PBKDF2 from the yubikey $response (as the PBKDF2 salt) and the passphrase $k_user
(as the PBKDF2 password), so that if two-factor authentication is enabled
(a) a USB-MITM attack on the yubikey itself is not enough to break the system
(b) the potentially low-entropy $k_user is better protected against brute-force attacks
* Instead of using uuidgen, gather the salt (previously random uuid / uuid_r) directly from /dev/random.
* Length of the new salt in byte added as the parameter "saltLength", defaults to 16 byte.
Note: Length of the challenge is 64 byte, so saltLength > 64 may have no benefit over saltLengh = 64.
* Length of $k_luks derived with PBKDF2 in byte added as the parameter "keyLength", defaults to 64 byte.
Example: For a luks device with a 512-bit key, keyLength should be 64.
* Increase of the PBKDF2 iteration count per successful authentication added as the
parameter "iterationStep", defaults to 0.
Other changes:
* Add optional grace period before trying to find the yubikey, defaults to 2 seconds.
Full overview of the yubikey authentication process:
(1) Read $salt and $iterations from unencrypted device (UD).
(2) Calculate the $challenge from the $salt with a hash function.
Chosen instantiation: SHA-512($salt).
(3) Challenge the yubikey with the $challenge and receive the $response.
(4) Repeat three times:
(a) Prompt for the passphrase $k_user.
(b) Derive the key $k_luks for the luks device with a key derivation function from $k_user and $response.
Chosen instantiation: PBKDF2(HMAC-SHA-512, $k_user, $response, $iterations, keyLength).
(c) Try to open the luks device with $k_luks and escape loop (4) only on success.
(5) Proceed only if luks device was opened successfully, fail otherwise.
(6) Gather $new_salt from a cryptographically secure pseudorandom number generator
Chosen instantiation: /dev/random
(7) Calculate the $new_challenge from the $new_salt with the same hash function as (2).
(8) Challenge the yubikey with the $new_challenge and receive the $new_response.
(9) Derive the new key $new_k_luks for the luks device in the same manner as in (4) (b),
but with more iterations as given by iterationStep.
(10) Try to change the luks device's key $k_luks to $new_k_luks.
(11) If (10) was successful, write the $new_salt and the $new_iterations to the UD.
Note: $new_iterations = $iterations + iterationStep
Known (software) attack vectors:
* A MITM attack on the keyboard can recover $k_user. This, combined with a USB-MITM
attack on the yubikey for the $response (1) or the $new_response (2) will result in
(1) $k_luks being recovered,
(2) $new_k_luks being recovered.
* Any attacker with access to the RAM state of stage-1 at mid- or post-authentication
can recover $k_user, $k_luks, and $new_k_luks
* If an attacker has recovered $response or $new_response, he can perform a brute-force
attack on $k_user with it without the Yubikey needing to be present (using cryptsetup's
"luksOpen --verify-passphrase" oracle. He could even make a copy of the luks device's
luks header and run the brute-force attack without further access to the system.
* A USB-MITM attack on the yubikey will allow an attacker to attempt to brute-force
the yubikey's internal key ("shared secret") without it needing to be present anymore.
Credits:
* Florian Klien,
for the original concept and the reference implementation over at
https://github.com/flowolf/initramfs_ykfde
* Anthony Thysse,
for the reference implementation of accessing OpenSSL's PBKDF2 over at
http://www.ict.griffith.edu.au/anthony/software/pbkdf2.c
2014-02-05 16:10:59 +00:00
|
|
|
else
|
2021-05-26 13:43:38 +00:00
|
|
|
k_luks="$(echo | pbkdf2-sha512 ${toString dev.yubikey.keyLength} $iterations $response | rbtohex)"
|
Update to the Yubikey PBA
Security-relevant changes:
* No (salted) passphrase hash send to the yubikey, only hash of the salt (as it was in the original implementation).
* Derive $k_luks with PBKDF2 from the yubikey $response (as the PBKDF2 salt) and the passphrase $k_user
(as the PBKDF2 password), so that if two-factor authentication is enabled
(a) a USB-MITM attack on the yubikey itself is not enough to break the system
(b) the potentially low-entropy $k_user is better protected against brute-force attacks
* Instead of using uuidgen, gather the salt (previously random uuid / uuid_r) directly from /dev/random.
* Length of the new salt in byte added as the parameter "saltLength", defaults to 16 byte.
Note: Length of the challenge is 64 byte, so saltLength > 64 may have no benefit over saltLengh = 64.
* Length of $k_luks derived with PBKDF2 in byte added as the parameter "keyLength", defaults to 64 byte.
Example: For a luks device with a 512-bit key, keyLength should be 64.
* Increase of the PBKDF2 iteration count per successful authentication added as the
parameter "iterationStep", defaults to 0.
Other changes:
* Add optional grace period before trying to find the yubikey, defaults to 2 seconds.
Full overview of the yubikey authentication process:
(1) Read $salt and $iterations from unencrypted device (UD).
(2) Calculate the $challenge from the $salt with a hash function.
Chosen instantiation: SHA-512($salt).
(3) Challenge the yubikey with the $challenge and receive the $response.
(4) Repeat three times:
(a) Prompt for the passphrase $k_user.
(b) Derive the key $k_luks for the luks device with a key derivation function from $k_user and $response.
Chosen instantiation: PBKDF2(HMAC-SHA-512, $k_user, $response, $iterations, keyLength).
(c) Try to open the luks device with $k_luks and escape loop (4) only on success.
(5) Proceed only if luks device was opened successfully, fail otherwise.
(6) Gather $new_salt from a cryptographically secure pseudorandom number generator
Chosen instantiation: /dev/random
(7) Calculate the $new_challenge from the $new_salt with the same hash function as (2).
(8) Challenge the yubikey with the $new_challenge and receive the $new_response.
(9) Derive the new key $new_k_luks for the luks device in the same manner as in (4) (b),
but with more iterations as given by iterationStep.
(10) Try to change the luks device's key $k_luks to $new_k_luks.
(11) If (10) was successful, write the $new_salt and the $new_iterations to the UD.
Note: $new_iterations = $iterations + iterationStep
Known (software) attack vectors:
* A MITM attack on the keyboard can recover $k_user. This, combined with a USB-MITM
attack on the yubikey for the $response (1) or the $new_response (2) will result in
(1) $k_luks being recovered,
(2) $new_k_luks being recovered.
* Any attacker with access to the RAM state of stage-1 at mid- or post-authentication
can recover $k_user, $k_luks, and $new_k_luks
* If an attacker has recovered $response or $new_response, he can perform a brute-force
attack on $k_user with it without the Yubikey needing to be present (using cryptsetup's
"luksOpen --verify-passphrase" oracle. He could even make a copy of the luks device's
luks header and run the brute-force attack without further access to the system.
* A USB-MITM attack on the yubikey will allow an attacker to attempt to brute-force
the yubikey's internal key ("shared secret") without it needing to be present anymore.
Credits:
* Florian Klien,
for the original concept and the reference implementation over at
https://github.com/flowolf/initramfs_ykfde
* Anthony Thysse,
for the reference implementation of accessing OpenSSL's PBKDF2 over at
http://www.ict.griffith.edu.au/anthony/software/pbkdf2.c
2014-02-05 16:10:59 +00:00
|
|
|
fi
|
2014-01-29 16:20:05 +00:00
|
|
|
|
2017-09-15 21:48:51 +00:00
|
|
|
echo -n "$k_luks" | hextorb | ${csopen} --key-file=-
|
2014-01-29 16:20:05 +00:00
|
|
|
|
2017-09-15 21:47:36 +00:00
|
|
|
if [ $? == 0 ]; then
|
Replace the current Yubikey PBA implementation with the previous one.
Rationale:
* The main reason for choosing to implement the PBA in accordance
with the Yubico documentation was to prevent a MITM-USB-attack
successfully recovering the new LUKS key.
* However, a MITM-USB-attacker can read user id and password when
they were entered for PBA, which allows him to recover the new
challenge after the PBA is complete, with which he can challenge
the Yubikey, decrypt the new AES blob and recover the LUKS key.
* Additionally, since the Yubikey shared secret is stored in the
same AES blob, after such an attack not only is the LUKS device
compromised, the Yubikey is as well, since the shared secret
has also been recovered by the attacker.
* Furthermore, with this method an attacker could also bruteforce
the AES blob, if he has access to the unencrypted device, which
would again compromise the Yubikey, should he be successful.
* Finally, with this method, once the LUKS key has been recovered
once, the encryption is permanently broken, while with the previous
system, the LUKS key itself it changed at every successful boot,
so recovering it once will not necessarily result in a permanent
breakage and will also not compromise the Yubikey itself (since
its secret is never stored anywhere but on the Yubikey itself).
Summary:
The current implementation opens up up vulnerability to brute-forcing
the AES blob, while retaining the current MITM-USB attack, additionally
making the consequences of this attack permanent and extending it to
the Yubikey itself.
2014-02-03 21:50:17 +00:00
|
|
|
opened=true
|
2020-12-20 21:24:14 +00:00
|
|
|
${if luks.reusePassphrases then ''
|
|
|
|
# We don't rm here because we might reuse it for the next device
|
|
|
|
'' else ''
|
|
|
|
rm -f /crypt-ramfs/passphrase
|
|
|
|
''}
|
Replace the current Yubikey PBA implementation with the previous one.
Rationale:
* The main reason for choosing to implement the PBA in accordance
with the Yubico documentation was to prevent a MITM-USB-attack
successfully recovering the new LUKS key.
* However, a MITM-USB-attacker can read user id and password when
they were entered for PBA, which allows him to recover the new
challenge after the PBA is complete, with which he can challenge
the Yubikey, decrypt the new AES blob and recover the LUKS key.
* Additionally, since the Yubikey shared secret is stored in the
same AES blob, after such an attack not only is the LUKS device
compromised, the Yubikey is as well, since the shared secret
has also been recovered by the attacker.
* Furthermore, with this method an attacker could also bruteforce
the AES blob, if he has access to the unencrypted device, which
would again compromise the Yubikey, should he be successful.
* Finally, with this method, once the LUKS key has been recovered
once, the encryption is permanently broken, while with the previous
system, the LUKS key itself it changed at every successful boot,
so recovering it once will not necessarily result in a permanent
breakage and will also not compromise the Yubikey itself (since
its secret is never stored anywhere but on the Yubikey itself).
Summary:
The current implementation opens up up vulnerability to brute-forcing
the AES blob, while retaining the current MITM-USB attack, additionally
making the consequences of this attack permanent and extending it to
the Yubikey itself.
2014-02-03 21:50:17 +00:00
|
|
|
break
|
2014-01-28 03:02:51 +00:00
|
|
|
else
|
Replace the current Yubikey PBA implementation with the previous one.
Rationale:
* The main reason for choosing to implement the PBA in accordance
with the Yubico documentation was to prevent a MITM-USB-attack
successfully recovering the new LUKS key.
* However, a MITM-USB-attacker can read user id and password when
they were entered for PBA, which allows him to recover the new
challenge after the PBA is complete, with which he can challenge
the Yubikey, decrypt the new AES blob and recover the LUKS key.
* Additionally, since the Yubikey shared secret is stored in the
same AES blob, after such an attack not only is the LUKS device
compromised, the Yubikey is as well, since the shared secret
has also been recovered by the attacker.
* Furthermore, with this method an attacker could also bruteforce
the AES blob, if he has access to the unencrypted device, which
would again compromise the Yubikey, should he be successful.
* Finally, with this method, once the LUKS key has been recovered
once, the encryption is permanently broken, while with the previous
system, the LUKS key itself it changed at every successful boot,
so recovering it once will not necessarily result in a permanent
breakage and will also not compromise the Yubikey itself (since
its secret is never stored anywhere but on the Yubikey itself).
Summary:
The current implementation opens up up vulnerability to brute-forcing
the AES blob, while retaining the current MITM-USB attack, additionally
making the consequences of this attack permanent and extending it to
the Yubikey itself.
2014-02-03 21:50:17 +00:00
|
|
|
opened=false
|
2014-01-29 16:20:05 +00:00
|
|
|
echo "Authentication failed!"
|
2014-01-28 03:02:51 +00:00
|
|
|
fi
|
|
|
|
done
|
|
|
|
|
2018-06-10 20:18:21 +00:00
|
|
|
[ "$opened" == false ] && die "Maximum authentication errors reached"
|
2014-01-25 02:27:12 +00:00
|
|
|
|
Update to the Yubikey PBA
Security-relevant changes:
* No (salted) passphrase hash send to the yubikey, only hash of the salt (as it was in the original implementation).
* Derive $k_luks with PBKDF2 from the yubikey $response (as the PBKDF2 salt) and the passphrase $k_user
(as the PBKDF2 password), so that if two-factor authentication is enabled
(a) a USB-MITM attack on the yubikey itself is not enough to break the system
(b) the potentially low-entropy $k_user is better protected against brute-force attacks
* Instead of using uuidgen, gather the salt (previously random uuid / uuid_r) directly from /dev/random.
* Length of the new salt in byte added as the parameter "saltLength", defaults to 16 byte.
Note: Length of the challenge is 64 byte, so saltLength > 64 may have no benefit over saltLengh = 64.
* Length of $k_luks derived with PBKDF2 in byte added as the parameter "keyLength", defaults to 64 byte.
Example: For a luks device with a 512-bit key, keyLength should be 64.
* Increase of the PBKDF2 iteration count per successful authentication added as the
parameter "iterationStep", defaults to 0.
Other changes:
* Add optional grace period before trying to find the yubikey, defaults to 2 seconds.
Full overview of the yubikey authentication process:
(1) Read $salt and $iterations from unencrypted device (UD).
(2) Calculate the $challenge from the $salt with a hash function.
Chosen instantiation: SHA-512($salt).
(3) Challenge the yubikey with the $challenge and receive the $response.
(4) Repeat three times:
(a) Prompt for the passphrase $k_user.
(b) Derive the key $k_luks for the luks device with a key derivation function from $k_user and $response.
Chosen instantiation: PBKDF2(HMAC-SHA-512, $k_user, $response, $iterations, keyLength).
(c) Try to open the luks device with $k_luks and escape loop (4) only on success.
(5) Proceed only if luks device was opened successfully, fail otherwise.
(6) Gather $new_salt from a cryptographically secure pseudorandom number generator
Chosen instantiation: /dev/random
(7) Calculate the $new_challenge from the $new_salt with the same hash function as (2).
(8) Challenge the yubikey with the $new_challenge and receive the $new_response.
(9) Derive the new key $new_k_luks for the luks device in the same manner as in (4) (b),
but with more iterations as given by iterationStep.
(10) Try to change the luks device's key $k_luks to $new_k_luks.
(11) If (10) was successful, write the $new_salt and the $new_iterations to the UD.
Note: $new_iterations = $iterations + iterationStep
Known (software) attack vectors:
* A MITM attack on the keyboard can recover $k_user. This, combined with a USB-MITM
attack on the yubikey for the $response (1) or the $new_response (2) will result in
(1) $k_luks being recovered,
(2) $new_k_luks being recovered.
* Any attacker with access to the RAM state of stage-1 at mid- or post-authentication
can recover $k_user, $k_luks, and $new_k_luks
* If an attacker has recovered $response or $new_response, he can perform a brute-force
attack on $k_user with it without the Yubikey needing to be present (using cryptsetup's
"luksOpen --verify-passphrase" oracle. He could even make a copy of the luks device's
luks header and run the brute-force attack without further access to the system.
* A USB-MITM attack on the yubikey will allow an attacker to attempt to brute-force
the yubikey's internal key ("shared secret") without it needing to be present anymore.
Credits:
* Florian Klien,
for the original concept and the reference implementation over at
https://github.com/flowolf/initramfs_ykfde
* Anthony Thysse,
for the reference implementation of accessing OpenSSL's PBKDF2 over at
http://www.ict.griffith.edu.au/anthony/software/pbkdf2.c
2014-02-05 16:10:59 +00:00
|
|
|
echo -n "Gathering entropy for new salt (please enter random keys to generate entropy if this blocks for long)..."
|
2021-05-26 13:43:38 +00:00
|
|
|
for i in $(seq ${toString dev.yubikey.saltLength}); do
|
Update to the Yubikey PBA
Security-relevant changes:
* No (salted) passphrase hash send to the yubikey, only hash of the salt (as it was in the original implementation).
* Derive $k_luks with PBKDF2 from the yubikey $response (as the PBKDF2 salt) and the passphrase $k_user
(as the PBKDF2 password), so that if two-factor authentication is enabled
(a) a USB-MITM attack on the yubikey itself is not enough to break the system
(b) the potentially low-entropy $k_user is better protected against brute-force attacks
* Instead of using uuidgen, gather the salt (previously random uuid / uuid_r) directly from /dev/random.
* Length of the new salt in byte added as the parameter "saltLength", defaults to 16 byte.
Note: Length of the challenge is 64 byte, so saltLength > 64 may have no benefit over saltLengh = 64.
* Length of $k_luks derived with PBKDF2 in byte added as the parameter "keyLength", defaults to 64 byte.
Example: For a luks device with a 512-bit key, keyLength should be 64.
* Increase of the PBKDF2 iteration count per successful authentication added as the
parameter "iterationStep", defaults to 0.
Other changes:
* Add optional grace period before trying to find the yubikey, defaults to 2 seconds.
Full overview of the yubikey authentication process:
(1) Read $salt and $iterations from unencrypted device (UD).
(2) Calculate the $challenge from the $salt with a hash function.
Chosen instantiation: SHA-512($salt).
(3) Challenge the yubikey with the $challenge and receive the $response.
(4) Repeat three times:
(a) Prompt for the passphrase $k_user.
(b) Derive the key $k_luks for the luks device with a key derivation function from $k_user and $response.
Chosen instantiation: PBKDF2(HMAC-SHA-512, $k_user, $response, $iterations, keyLength).
(c) Try to open the luks device with $k_luks and escape loop (4) only on success.
(5) Proceed only if luks device was opened successfully, fail otherwise.
(6) Gather $new_salt from a cryptographically secure pseudorandom number generator
Chosen instantiation: /dev/random
(7) Calculate the $new_challenge from the $new_salt with the same hash function as (2).
(8) Challenge the yubikey with the $new_challenge and receive the $new_response.
(9) Derive the new key $new_k_luks for the luks device in the same manner as in (4) (b),
but with more iterations as given by iterationStep.
(10) Try to change the luks device's key $k_luks to $new_k_luks.
(11) If (10) was successful, write the $new_salt and the $new_iterations to the UD.
Note: $new_iterations = $iterations + iterationStep
Known (software) attack vectors:
* A MITM attack on the keyboard can recover $k_user. This, combined with a USB-MITM
attack on the yubikey for the $response (1) or the $new_response (2) will result in
(1) $k_luks being recovered,
(2) $new_k_luks being recovered.
* Any attacker with access to the RAM state of stage-1 at mid- or post-authentication
can recover $k_user, $k_luks, and $new_k_luks
* If an attacker has recovered $response or $new_response, he can perform a brute-force
attack on $k_user with it without the Yubikey needing to be present (using cryptsetup's
"luksOpen --verify-passphrase" oracle. He could even make a copy of the luks device's
luks header and run the brute-force attack without further access to the system.
* A USB-MITM attack on the yubikey will allow an attacker to attempt to brute-force
the yubikey's internal key ("shared secret") without it needing to be present anymore.
Credits:
* Florian Klien,
for the original concept and the reference implementation over at
https://github.com/flowolf/initramfs_ykfde
* Anthony Thysse,
for the reference implementation of accessing OpenSSL's PBKDF2 over at
http://www.ict.griffith.edu.au/anthony/software/pbkdf2.c
2014-02-05 16:10:59 +00:00
|
|
|
byte="$(dd if=/dev/random bs=1 count=1 2>/dev/null | rbtohex)";
|
|
|
|
new_salt="$new_salt$byte";
|
|
|
|
echo -n .
|
|
|
|
done;
|
|
|
|
echo "ok"
|
2014-01-25 02:27:12 +00:00
|
|
|
|
Update to the Yubikey PBA
Security-relevant changes:
* No (salted) passphrase hash send to the yubikey, only hash of the salt (as it was in the original implementation).
* Derive $k_luks with PBKDF2 from the yubikey $response (as the PBKDF2 salt) and the passphrase $k_user
(as the PBKDF2 password), so that if two-factor authentication is enabled
(a) a USB-MITM attack on the yubikey itself is not enough to break the system
(b) the potentially low-entropy $k_user is better protected against brute-force attacks
* Instead of using uuidgen, gather the salt (previously random uuid / uuid_r) directly from /dev/random.
* Length of the new salt in byte added as the parameter "saltLength", defaults to 16 byte.
Note: Length of the challenge is 64 byte, so saltLength > 64 may have no benefit over saltLengh = 64.
* Length of $k_luks derived with PBKDF2 in byte added as the parameter "keyLength", defaults to 64 byte.
Example: For a luks device with a 512-bit key, keyLength should be 64.
* Increase of the PBKDF2 iteration count per successful authentication added as the
parameter "iterationStep", defaults to 0.
Other changes:
* Add optional grace period before trying to find the yubikey, defaults to 2 seconds.
Full overview of the yubikey authentication process:
(1) Read $salt and $iterations from unencrypted device (UD).
(2) Calculate the $challenge from the $salt with a hash function.
Chosen instantiation: SHA-512($salt).
(3) Challenge the yubikey with the $challenge and receive the $response.
(4) Repeat three times:
(a) Prompt for the passphrase $k_user.
(b) Derive the key $k_luks for the luks device with a key derivation function from $k_user and $response.
Chosen instantiation: PBKDF2(HMAC-SHA-512, $k_user, $response, $iterations, keyLength).
(c) Try to open the luks device with $k_luks and escape loop (4) only on success.
(5) Proceed only if luks device was opened successfully, fail otherwise.
(6) Gather $new_salt from a cryptographically secure pseudorandom number generator
Chosen instantiation: /dev/random
(7) Calculate the $new_challenge from the $new_salt with the same hash function as (2).
(8) Challenge the yubikey with the $new_challenge and receive the $new_response.
(9) Derive the new key $new_k_luks for the luks device in the same manner as in (4) (b),
but with more iterations as given by iterationStep.
(10) Try to change the luks device's key $k_luks to $new_k_luks.
(11) If (10) was successful, write the $new_salt and the $new_iterations to the UD.
Note: $new_iterations = $iterations + iterationStep
Known (software) attack vectors:
* A MITM attack on the keyboard can recover $k_user. This, combined with a USB-MITM
attack on the yubikey for the $response (1) or the $new_response (2) will result in
(1) $k_luks being recovered,
(2) $new_k_luks being recovered.
* Any attacker with access to the RAM state of stage-1 at mid- or post-authentication
can recover $k_user, $k_luks, and $new_k_luks
* If an attacker has recovered $response or $new_response, he can perform a brute-force
attack on $k_user with it without the Yubikey needing to be present (using cryptsetup's
"luksOpen --verify-passphrase" oracle. He could even make a copy of the luks device's
luks header and run the brute-force attack without further access to the system.
* A USB-MITM attack on the yubikey will allow an attacker to attempt to brute-force
the yubikey's internal key ("shared secret") without it needing to be present anymore.
Credits:
* Florian Klien,
for the original concept and the reference implementation over at
https://github.com/flowolf/initramfs_ykfde
* Anthony Thysse,
for the reference implementation of accessing OpenSSL's PBKDF2 over at
http://www.ict.griffith.edu.au/anthony/software/pbkdf2.c
2014-02-05 16:10:59 +00:00
|
|
|
new_iterations="$iterations"
|
2021-05-26 13:43:38 +00:00
|
|
|
${optionalString (dev.yubikey.iterationStep > 0) ''
|
|
|
|
new_iterations="$(($new_iterations + ${toString dev.yubikey.iterationStep}))"
|
Update to the Yubikey PBA
Security-relevant changes:
* No (salted) passphrase hash send to the yubikey, only hash of the salt (as it was in the original implementation).
* Derive $k_luks with PBKDF2 from the yubikey $response (as the PBKDF2 salt) and the passphrase $k_user
(as the PBKDF2 password), so that if two-factor authentication is enabled
(a) a USB-MITM attack on the yubikey itself is not enough to break the system
(b) the potentially low-entropy $k_user is better protected against brute-force attacks
* Instead of using uuidgen, gather the salt (previously random uuid / uuid_r) directly from /dev/random.
* Length of the new salt in byte added as the parameter "saltLength", defaults to 16 byte.
Note: Length of the challenge is 64 byte, so saltLength > 64 may have no benefit over saltLengh = 64.
* Length of $k_luks derived with PBKDF2 in byte added as the parameter "keyLength", defaults to 64 byte.
Example: For a luks device with a 512-bit key, keyLength should be 64.
* Increase of the PBKDF2 iteration count per successful authentication added as the
parameter "iterationStep", defaults to 0.
Other changes:
* Add optional grace period before trying to find the yubikey, defaults to 2 seconds.
Full overview of the yubikey authentication process:
(1) Read $salt and $iterations from unencrypted device (UD).
(2) Calculate the $challenge from the $salt with a hash function.
Chosen instantiation: SHA-512($salt).
(3) Challenge the yubikey with the $challenge and receive the $response.
(4) Repeat three times:
(a) Prompt for the passphrase $k_user.
(b) Derive the key $k_luks for the luks device with a key derivation function from $k_user and $response.
Chosen instantiation: PBKDF2(HMAC-SHA-512, $k_user, $response, $iterations, keyLength).
(c) Try to open the luks device with $k_luks and escape loop (4) only on success.
(5) Proceed only if luks device was opened successfully, fail otherwise.
(6) Gather $new_salt from a cryptographically secure pseudorandom number generator
Chosen instantiation: /dev/random
(7) Calculate the $new_challenge from the $new_salt with the same hash function as (2).
(8) Challenge the yubikey with the $new_challenge and receive the $new_response.
(9) Derive the new key $new_k_luks for the luks device in the same manner as in (4) (b),
but with more iterations as given by iterationStep.
(10) Try to change the luks device's key $k_luks to $new_k_luks.
(11) If (10) was successful, write the $new_salt and the $new_iterations to the UD.
Note: $new_iterations = $iterations + iterationStep
Known (software) attack vectors:
* A MITM attack on the keyboard can recover $k_user. This, combined with a USB-MITM
attack on the yubikey for the $response (1) or the $new_response (2) will result in
(1) $k_luks being recovered,
(2) $new_k_luks being recovered.
* Any attacker with access to the RAM state of stage-1 at mid- or post-authentication
can recover $k_user, $k_luks, and $new_k_luks
* If an attacker has recovered $response or $new_response, he can perform a brute-force
attack on $k_user with it without the Yubikey needing to be present (using cryptsetup's
"luksOpen --verify-passphrase" oracle. He could even make a copy of the luks device's
luks header and run the brute-force attack without further access to the system.
* A USB-MITM attack on the yubikey will allow an attacker to attempt to brute-force
the yubikey's internal key ("shared secret") without it needing to be present anymore.
Credits:
* Florian Klien,
for the original concept and the reference implementation over at
https://github.com/flowolf/initramfs_ykfde
* Anthony Thysse,
for the reference implementation of accessing OpenSSL's PBKDF2 over at
http://www.ict.griffith.edu.au/anthony/software/pbkdf2.c
2014-02-05 16:10:59 +00:00
|
|
|
''}
|
2014-01-25 02:27:12 +00:00
|
|
|
|
Update to the Yubikey PBA
Security-relevant changes:
* No (salted) passphrase hash send to the yubikey, only hash of the salt (as it was in the original implementation).
* Derive $k_luks with PBKDF2 from the yubikey $response (as the PBKDF2 salt) and the passphrase $k_user
(as the PBKDF2 password), so that if two-factor authentication is enabled
(a) a USB-MITM attack on the yubikey itself is not enough to break the system
(b) the potentially low-entropy $k_user is better protected against brute-force attacks
* Instead of using uuidgen, gather the salt (previously random uuid / uuid_r) directly from /dev/random.
* Length of the new salt in byte added as the parameter "saltLength", defaults to 16 byte.
Note: Length of the challenge is 64 byte, so saltLength > 64 may have no benefit over saltLengh = 64.
* Length of $k_luks derived with PBKDF2 in byte added as the parameter "keyLength", defaults to 64 byte.
Example: For a luks device with a 512-bit key, keyLength should be 64.
* Increase of the PBKDF2 iteration count per successful authentication added as the
parameter "iterationStep", defaults to 0.
Other changes:
* Add optional grace period before trying to find the yubikey, defaults to 2 seconds.
Full overview of the yubikey authentication process:
(1) Read $salt and $iterations from unencrypted device (UD).
(2) Calculate the $challenge from the $salt with a hash function.
Chosen instantiation: SHA-512($salt).
(3) Challenge the yubikey with the $challenge and receive the $response.
(4) Repeat three times:
(a) Prompt for the passphrase $k_user.
(b) Derive the key $k_luks for the luks device with a key derivation function from $k_user and $response.
Chosen instantiation: PBKDF2(HMAC-SHA-512, $k_user, $response, $iterations, keyLength).
(c) Try to open the luks device with $k_luks and escape loop (4) only on success.
(5) Proceed only if luks device was opened successfully, fail otherwise.
(6) Gather $new_salt from a cryptographically secure pseudorandom number generator
Chosen instantiation: /dev/random
(7) Calculate the $new_challenge from the $new_salt with the same hash function as (2).
(8) Challenge the yubikey with the $new_challenge and receive the $new_response.
(9) Derive the new key $new_k_luks for the luks device in the same manner as in (4) (b),
but with more iterations as given by iterationStep.
(10) Try to change the luks device's key $k_luks to $new_k_luks.
(11) If (10) was successful, write the $new_salt and the $new_iterations to the UD.
Note: $new_iterations = $iterations + iterationStep
Known (software) attack vectors:
* A MITM attack on the keyboard can recover $k_user. This, combined with a USB-MITM
attack on the yubikey for the $response (1) or the $new_response (2) will result in
(1) $k_luks being recovered,
(2) $new_k_luks being recovered.
* Any attacker with access to the RAM state of stage-1 at mid- or post-authentication
can recover $k_user, $k_luks, and $new_k_luks
* If an attacker has recovered $response or $new_response, he can perform a brute-force
attack on $k_user with it without the Yubikey needing to be present (using cryptsetup's
"luksOpen --verify-passphrase" oracle. He could even make a copy of the luks device's
luks header and run the brute-force attack without further access to the system.
* A USB-MITM attack on the yubikey will allow an attacker to attempt to brute-force
the yubikey's internal key ("shared secret") without it needing to be present anymore.
Credits:
* Florian Klien,
for the original concept and the reference implementation over at
https://github.com/flowolf/initramfs_ykfde
* Anthony Thysse,
for the reference implementation of accessing OpenSSL's PBKDF2 over at
http://www.ict.griffith.edu.au/anthony/software/pbkdf2.c
2014-02-05 16:10:59 +00:00
|
|
|
new_challenge="$(echo -n $new_salt | openssl-wrap dgst -binary -sha512 | rbtohex)"
|
2014-01-25 02:27:12 +00:00
|
|
|
|
2021-05-26 13:43:38 +00:00
|
|
|
new_response="$(ykchalresp -${toString dev.yubikey.slot} -x $new_challenge 2>/dev/null)"
|
2014-01-25 02:27:12 +00:00
|
|
|
|
2023-04-25 13:48:54 +00:00
|
|
|
if [ -z "$new_response" ]; then
|
|
|
|
echo "Warning: Unable to generate new challenge response, current challenge persists!"
|
|
|
|
umount /crypt-storage
|
|
|
|
return
|
|
|
|
fi
|
|
|
|
|
Update to the Yubikey PBA
Security-relevant changes:
* No (salted) passphrase hash send to the yubikey, only hash of the salt (as it was in the original implementation).
* Derive $k_luks with PBKDF2 from the yubikey $response (as the PBKDF2 salt) and the passphrase $k_user
(as the PBKDF2 password), so that if two-factor authentication is enabled
(a) a USB-MITM attack on the yubikey itself is not enough to break the system
(b) the potentially low-entropy $k_user is better protected against brute-force attacks
* Instead of using uuidgen, gather the salt (previously random uuid / uuid_r) directly from /dev/random.
* Length of the new salt in byte added as the parameter "saltLength", defaults to 16 byte.
Note: Length of the challenge is 64 byte, so saltLength > 64 may have no benefit over saltLengh = 64.
* Length of $k_luks derived with PBKDF2 in byte added as the parameter "keyLength", defaults to 64 byte.
Example: For a luks device with a 512-bit key, keyLength should be 64.
* Increase of the PBKDF2 iteration count per successful authentication added as the
parameter "iterationStep", defaults to 0.
Other changes:
* Add optional grace period before trying to find the yubikey, defaults to 2 seconds.
Full overview of the yubikey authentication process:
(1) Read $salt and $iterations from unencrypted device (UD).
(2) Calculate the $challenge from the $salt with a hash function.
Chosen instantiation: SHA-512($salt).
(3) Challenge the yubikey with the $challenge and receive the $response.
(4) Repeat three times:
(a) Prompt for the passphrase $k_user.
(b) Derive the key $k_luks for the luks device with a key derivation function from $k_user and $response.
Chosen instantiation: PBKDF2(HMAC-SHA-512, $k_user, $response, $iterations, keyLength).
(c) Try to open the luks device with $k_luks and escape loop (4) only on success.
(5) Proceed only if luks device was opened successfully, fail otherwise.
(6) Gather $new_salt from a cryptographically secure pseudorandom number generator
Chosen instantiation: /dev/random
(7) Calculate the $new_challenge from the $new_salt with the same hash function as (2).
(8) Challenge the yubikey with the $new_challenge and receive the $new_response.
(9) Derive the new key $new_k_luks for the luks device in the same manner as in (4) (b),
but with more iterations as given by iterationStep.
(10) Try to change the luks device's key $k_luks to $new_k_luks.
(11) If (10) was successful, write the $new_salt and the $new_iterations to the UD.
Note: $new_iterations = $iterations + iterationStep
Known (software) attack vectors:
* A MITM attack on the keyboard can recover $k_user. This, combined with a USB-MITM
attack on the yubikey for the $response (1) or the $new_response (2) will result in
(1) $k_luks being recovered,
(2) $new_k_luks being recovered.
* Any attacker with access to the RAM state of stage-1 at mid- or post-authentication
can recover $k_user, $k_luks, and $new_k_luks
* If an attacker has recovered $response or $new_response, he can perform a brute-force
attack on $k_user with it without the Yubikey needing to be present (using cryptsetup's
"luksOpen --verify-passphrase" oracle. He could even make a copy of the luks device's
luks header and run the brute-force attack without further access to the system.
* A USB-MITM attack on the yubikey will allow an attacker to attempt to brute-force
the yubikey's internal key ("shared secret") without it needing to be present anymore.
Credits:
* Florian Klien,
for the original concept and the reference implementation over at
https://github.com/flowolf/initramfs_ykfde
* Anthony Thysse,
for the reference implementation of accessing OpenSSL's PBKDF2 over at
http://www.ict.griffith.edu.au/anthony/software/pbkdf2.c
2014-02-05 16:10:59 +00:00
|
|
|
if [ ! -z "$k_user" ]; then
|
2021-05-26 13:43:38 +00:00
|
|
|
new_k_luks="$(echo -n $k_user | pbkdf2-sha512 ${toString dev.yubikey.keyLength} $new_iterations $new_response | rbtohex)"
|
Update to the Yubikey PBA
Security-relevant changes:
* No (salted) passphrase hash send to the yubikey, only hash of the salt (as it was in the original implementation).
* Derive $k_luks with PBKDF2 from the yubikey $response (as the PBKDF2 salt) and the passphrase $k_user
(as the PBKDF2 password), so that if two-factor authentication is enabled
(a) a USB-MITM attack on the yubikey itself is not enough to break the system
(b) the potentially low-entropy $k_user is better protected against brute-force attacks
* Instead of using uuidgen, gather the salt (previously random uuid / uuid_r) directly from /dev/random.
* Length of the new salt in byte added as the parameter "saltLength", defaults to 16 byte.
Note: Length of the challenge is 64 byte, so saltLength > 64 may have no benefit over saltLengh = 64.
* Length of $k_luks derived with PBKDF2 in byte added as the parameter "keyLength", defaults to 64 byte.
Example: For a luks device with a 512-bit key, keyLength should be 64.
* Increase of the PBKDF2 iteration count per successful authentication added as the
parameter "iterationStep", defaults to 0.
Other changes:
* Add optional grace period before trying to find the yubikey, defaults to 2 seconds.
Full overview of the yubikey authentication process:
(1) Read $salt and $iterations from unencrypted device (UD).
(2) Calculate the $challenge from the $salt with a hash function.
Chosen instantiation: SHA-512($salt).
(3) Challenge the yubikey with the $challenge and receive the $response.
(4) Repeat three times:
(a) Prompt for the passphrase $k_user.
(b) Derive the key $k_luks for the luks device with a key derivation function from $k_user and $response.
Chosen instantiation: PBKDF2(HMAC-SHA-512, $k_user, $response, $iterations, keyLength).
(c) Try to open the luks device with $k_luks and escape loop (4) only on success.
(5) Proceed only if luks device was opened successfully, fail otherwise.
(6) Gather $new_salt from a cryptographically secure pseudorandom number generator
Chosen instantiation: /dev/random
(7) Calculate the $new_challenge from the $new_salt with the same hash function as (2).
(8) Challenge the yubikey with the $new_challenge and receive the $new_response.
(9) Derive the new key $new_k_luks for the luks device in the same manner as in (4) (b),
but with more iterations as given by iterationStep.
(10) Try to change the luks device's key $k_luks to $new_k_luks.
(11) If (10) was successful, write the $new_salt and the $new_iterations to the UD.
Note: $new_iterations = $iterations + iterationStep
Known (software) attack vectors:
* A MITM attack on the keyboard can recover $k_user. This, combined with a USB-MITM
attack on the yubikey for the $response (1) or the $new_response (2) will result in
(1) $k_luks being recovered,
(2) $new_k_luks being recovered.
* Any attacker with access to the RAM state of stage-1 at mid- or post-authentication
can recover $k_user, $k_luks, and $new_k_luks
* If an attacker has recovered $response or $new_response, he can perform a brute-force
attack on $k_user with it without the Yubikey needing to be present (using cryptsetup's
"luksOpen --verify-passphrase" oracle. He could even make a copy of the luks device's
luks header and run the brute-force attack without further access to the system.
* A USB-MITM attack on the yubikey will allow an attacker to attempt to brute-force
the yubikey's internal key ("shared secret") without it needing to be present anymore.
Credits:
* Florian Klien,
for the original concept and the reference implementation over at
https://github.com/flowolf/initramfs_ykfde
* Anthony Thysse,
for the reference implementation of accessing OpenSSL's PBKDF2 over at
http://www.ict.griffith.edu.au/anthony/software/pbkdf2.c
2014-02-05 16:10:59 +00:00
|
|
|
else
|
2021-05-26 13:43:38 +00:00
|
|
|
new_k_luks="$(echo | pbkdf2-sha512 ${toString dev.yubikey.keyLength} $new_iterations $new_response | rbtohex)"
|
Update to the Yubikey PBA
Security-relevant changes:
* No (salted) passphrase hash send to the yubikey, only hash of the salt (as it was in the original implementation).
* Derive $k_luks with PBKDF2 from the yubikey $response (as the PBKDF2 salt) and the passphrase $k_user
(as the PBKDF2 password), so that if two-factor authentication is enabled
(a) a USB-MITM attack on the yubikey itself is not enough to break the system
(b) the potentially low-entropy $k_user is better protected against brute-force attacks
* Instead of using uuidgen, gather the salt (previously random uuid / uuid_r) directly from /dev/random.
* Length of the new salt in byte added as the parameter "saltLength", defaults to 16 byte.
Note: Length of the challenge is 64 byte, so saltLength > 64 may have no benefit over saltLengh = 64.
* Length of $k_luks derived with PBKDF2 in byte added as the parameter "keyLength", defaults to 64 byte.
Example: For a luks device with a 512-bit key, keyLength should be 64.
* Increase of the PBKDF2 iteration count per successful authentication added as the
parameter "iterationStep", defaults to 0.
Other changes:
* Add optional grace period before trying to find the yubikey, defaults to 2 seconds.
Full overview of the yubikey authentication process:
(1) Read $salt and $iterations from unencrypted device (UD).
(2) Calculate the $challenge from the $salt with a hash function.
Chosen instantiation: SHA-512($salt).
(3) Challenge the yubikey with the $challenge and receive the $response.
(4) Repeat three times:
(a) Prompt for the passphrase $k_user.
(b) Derive the key $k_luks for the luks device with a key derivation function from $k_user and $response.
Chosen instantiation: PBKDF2(HMAC-SHA-512, $k_user, $response, $iterations, keyLength).
(c) Try to open the luks device with $k_luks and escape loop (4) only on success.
(5) Proceed only if luks device was opened successfully, fail otherwise.
(6) Gather $new_salt from a cryptographically secure pseudorandom number generator
Chosen instantiation: /dev/random
(7) Calculate the $new_challenge from the $new_salt with the same hash function as (2).
(8) Challenge the yubikey with the $new_challenge and receive the $new_response.
(9) Derive the new key $new_k_luks for the luks device in the same manner as in (4) (b),
but with more iterations as given by iterationStep.
(10) Try to change the luks device's key $k_luks to $new_k_luks.
(11) If (10) was successful, write the $new_salt and the $new_iterations to the UD.
Note: $new_iterations = $iterations + iterationStep
Known (software) attack vectors:
* A MITM attack on the keyboard can recover $k_user. This, combined with a USB-MITM
attack on the yubikey for the $response (1) or the $new_response (2) will result in
(1) $k_luks being recovered,
(2) $new_k_luks being recovered.
* Any attacker with access to the RAM state of stage-1 at mid- or post-authentication
can recover $k_user, $k_luks, and $new_k_luks
* If an attacker has recovered $response or $new_response, he can perform a brute-force
attack on $k_user with it without the Yubikey needing to be present (using cryptsetup's
"luksOpen --verify-passphrase" oracle. He could even make a copy of the luks device's
luks header and run the brute-force attack without further access to the system.
* A USB-MITM attack on the yubikey will allow an attacker to attempt to brute-force
the yubikey's internal key ("shared secret") without it needing to be present anymore.
Credits:
* Florian Klien,
for the original concept and the reference implementation over at
https://github.com/flowolf/initramfs_ykfde
* Anthony Thysse,
for the reference implementation of accessing OpenSSL's PBKDF2 over at
http://www.ict.griffith.edu.au/anthony/software/pbkdf2.c
2014-02-05 16:10:59 +00:00
|
|
|
fi
|
2014-01-29 16:20:05 +00:00
|
|
|
|
2018-06-10 20:18:21 +00:00
|
|
|
echo -n "$new_k_luks" | hextorb > /crypt-ramfs/new_key
|
|
|
|
echo -n "$k_luks" | hextorb | ${cschange} --key-file=- /crypt-ramfs/new_key
|
Replace the current Yubikey PBA implementation with the previous one.
Rationale:
* The main reason for choosing to implement the PBA in accordance
with the Yubico documentation was to prevent a MITM-USB-attack
successfully recovering the new LUKS key.
* However, a MITM-USB-attacker can read user id and password when
they were entered for PBA, which allows him to recover the new
challenge after the PBA is complete, with which he can challenge
the Yubikey, decrypt the new AES blob and recover the LUKS key.
* Additionally, since the Yubikey shared secret is stored in the
same AES blob, after such an attack not only is the LUKS device
compromised, the Yubikey is as well, since the shared secret
has also been recovered by the attacker.
* Furthermore, with this method an attacker could also bruteforce
the AES blob, if he has access to the unencrypted device, which
would again compromise the Yubikey, should he be successful.
* Finally, with this method, once the LUKS key has been recovered
once, the encryption is permanently broken, while with the previous
system, the LUKS key itself it changed at every successful boot,
so recovering it once will not necessarily result in a permanent
breakage and will also not compromise the Yubikey itself (since
its secret is never stored anywhere but on the Yubikey itself).
Summary:
The current implementation opens up up vulnerability to brute-forcing
the AES blob, while retaining the current MITM-USB attack, additionally
making the consequences of this attack permanent and extending it to
the Yubikey itself.
2014-02-03 21:50:17 +00:00
|
|
|
|
2017-09-15 21:47:36 +00:00
|
|
|
if [ $? == 0 ]; then
|
2021-05-26 13:43:38 +00:00
|
|
|
echo -ne "$new_salt\n$new_iterations" > /crypt-storage${dev.yubikey.storage.path}
|
2021-03-08 20:17:42 +00:00
|
|
|
sync /crypt-storage${dev.yubikey.storage.path}
|
2014-01-25 02:27:12 +00:00
|
|
|
else
|
Update to the Yubikey PBA
Security-relevant changes:
* No (salted) passphrase hash send to the yubikey, only hash of the salt (as it was in the original implementation).
* Derive $k_luks with PBKDF2 from the yubikey $response (as the PBKDF2 salt) and the passphrase $k_user
(as the PBKDF2 password), so that if two-factor authentication is enabled
(a) a USB-MITM attack on the yubikey itself is not enough to break the system
(b) the potentially low-entropy $k_user is better protected against brute-force attacks
* Instead of using uuidgen, gather the salt (previously random uuid / uuid_r) directly from /dev/random.
* Length of the new salt in byte added as the parameter "saltLength", defaults to 16 byte.
Note: Length of the challenge is 64 byte, so saltLength > 64 may have no benefit over saltLengh = 64.
* Length of $k_luks derived with PBKDF2 in byte added as the parameter "keyLength", defaults to 64 byte.
Example: For a luks device with a 512-bit key, keyLength should be 64.
* Increase of the PBKDF2 iteration count per successful authentication added as the
parameter "iterationStep", defaults to 0.
Other changes:
* Add optional grace period before trying to find the yubikey, defaults to 2 seconds.
Full overview of the yubikey authentication process:
(1) Read $salt and $iterations from unencrypted device (UD).
(2) Calculate the $challenge from the $salt with a hash function.
Chosen instantiation: SHA-512($salt).
(3) Challenge the yubikey with the $challenge and receive the $response.
(4) Repeat three times:
(a) Prompt for the passphrase $k_user.
(b) Derive the key $k_luks for the luks device with a key derivation function from $k_user and $response.
Chosen instantiation: PBKDF2(HMAC-SHA-512, $k_user, $response, $iterations, keyLength).
(c) Try to open the luks device with $k_luks and escape loop (4) only on success.
(5) Proceed only if luks device was opened successfully, fail otherwise.
(6) Gather $new_salt from a cryptographically secure pseudorandom number generator
Chosen instantiation: /dev/random
(7) Calculate the $new_challenge from the $new_salt with the same hash function as (2).
(8) Challenge the yubikey with the $new_challenge and receive the $new_response.
(9) Derive the new key $new_k_luks for the luks device in the same manner as in (4) (b),
but with more iterations as given by iterationStep.
(10) Try to change the luks device's key $k_luks to $new_k_luks.
(11) If (10) was successful, write the $new_salt and the $new_iterations to the UD.
Note: $new_iterations = $iterations + iterationStep
Known (software) attack vectors:
* A MITM attack on the keyboard can recover $k_user. This, combined with a USB-MITM
attack on the yubikey for the $response (1) or the $new_response (2) will result in
(1) $k_luks being recovered,
(2) $new_k_luks being recovered.
* Any attacker with access to the RAM state of stage-1 at mid- or post-authentication
can recover $k_user, $k_luks, and $new_k_luks
* If an attacker has recovered $response or $new_response, he can perform a brute-force
attack on $k_user with it without the Yubikey needing to be present (using cryptsetup's
"luksOpen --verify-passphrase" oracle. He could even make a copy of the luks device's
luks header and run the brute-force attack without further access to the system.
* A USB-MITM attack on the yubikey will allow an attacker to attempt to brute-force
the yubikey's internal key ("shared secret") without it needing to be present anymore.
Credits:
* Florian Klien,
for the original concept and the reference implementation over at
https://github.com/flowolf/initramfs_ykfde
* Anthony Thysse,
for the reference implementation of accessing OpenSSL's PBKDF2 over at
http://www.ict.griffith.edu.au/anthony/software/pbkdf2.c
2014-02-05 16:10:59 +00:00
|
|
|
echo "Warning: Could not update LUKS key, current challenge persists!"
|
2014-01-25 02:27:12 +00:00
|
|
|
fi
|
|
|
|
|
2018-06-10 20:18:21 +00:00
|
|
|
rm -f /crypt-ramfs/new_key
|
|
|
|
umount /crypt-storage
|
2014-01-28 03:02:51 +00:00
|
|
|
}
|
|
|
|
|
2019-05-09 18:15:35 +00:00
|
|
|
open_with_hardware() {
|
2021-05-26 13:43:38 +00:00
|
|
|
if wait_yubikey ${toString dev.yubikey.gracePeriod}; then
|
2017-09-15 21:48:51 +00:00
|
|
|
do_open_yubikey
|
|
|
|
else
|
2020-12-20 21:35:59 +00:00
|
|
|
echo "No YubiKey found, falling back to non-YubiKey open procedure"
|
2017-09-15 21:48:51 +00:00
|
|
|
open_normally
|
|
|
|
fi
|
|
|
|
}
|
2019-05-09 18:15:35 +00:00
|
|
|
''}
|
2014-01-28 03:02:51 +00:00
|
|
|
|
2021-05-26 13:43:38 +00:00
|
|
|
${optionalString (luks.gpgSupport && (dev.gpgCard != null)) ''
|
2019-05-09 18:15:35 +00:00
|
|
|
|
|
|
|
do_open_gpg_card() {
|
|
|
|
# Make all of these local to this function
|
|
|
|
# to prevent their values being leaked
|
|
|
|
local pin
|
|
|
|
local opened
|
|
|
|
|
2021-05-26 13:43:38 +00:00
|
|
|
gpg --import /gpg-keys/${dev.device}/pubkey.asc > /dev/null 2> /dev/null
|
2019-05-09 18:15:35 +00:00
|
|
|
|
|
|
|
gpg --card-status > /dev/null 2> /dev/null
|
|
|
|
|
|
|
|
for try in $(seq 3); do
|
2021-05-26 13:43:38 +00:00
|
|
|
echo -n "PIN for GPG Card associated with device ${dev.device}: "
|
2019-05-09 18:15:35 +00:00
|
|
|
pin=
|
|
|
|
while true; do
|
|
|
|
if [ -e /crypt-ramfs/passphrase ]; then
|
|
|
|
echo "reused"
|
|
|
|
pin=$(cat /crypt-ramfs/passphrase)
|
|
|
|
break
|
|
|
|
else
|
|
|
|
# and try reading it from /dev/console with a timeout
|
|
|
|
IFS= read -t 1 -r pin
|
|
|
|
if [ -n "$pin" ]; then
|
|
|
|
${if luks.reusePassphrases then ''
|
|
|
|
# remember it for the next device
|
|
|
|
echo -n "$pin" > /crypt-ramfs/passphrase
|
|
|
|
'' else ''
|
|
|
|
# Don't save it to ramfs. We are very paranoid
|
|
|
|
''}
|
|
|
|
echo
|
|
|
|
break
|
|
|
|
fi
|
|
|
|
fi
|
|
|
|
done
|
2021-05-26 13:43:38 +00:00
|
|
|
echo -n "Verifying passphrase for ${dev.device}..."
|
|
|
|
echo -n "$pin" | gpg -q --batch --passphrase-fd 0 --pinentry-mode loopback -d /gpg-keys/${dev.device}/cryptkey.gpg 2> /dev/null | ${csopen} --key-file=- > /dev/null 2> /dev/null
|
2019-05-09 18:15:35 +00:00
|
|
|
if [ $? == 0 ]; then
|
|
|
|
echo " - success"
|
|
|
|
${if luks.reusePassphrases then ''
|
|
|
|
# we don't rm here because we might reuse it for the next device
|
|
|
|
'' else ''
|
|
|
|
rm -f /crypt-ramfs/passphrase
|
|
|
|
''}
|
|
|
|
break
|
|
|
|
else
|
|
|
|
echo " - failure"
|
|
|
|
# ask for a different one
|
|
|
|
rm -f /crypt-ramfs/passphrase
|
|
|
|
fi
|
|
|
|
done
|
|
|
|
|
|
|
|
[ "$opened" == false ] && die "Maximum authentication errors reached"
|
|
|
|
}
|
|
|
|
|
|
|
|
open_with_hardware() {
|
2021-05-26 13:43:38 +00:00
|
|
|
if wait_gpgcard ${toString dev.gpgCard.gracePeriod}; then
|
2019-05-09 18:15:35 +00:00
|
|
|
do_open_gpg_card
|
|
|
|
else
|
|
|
|
echo "No GPG Card found, falling back to normal open procedure"
|
|
|
|
open_normally
|
|
|
|
fi
|
|
|
|
}
|
|
|
|
''}
|
|
|
|
|
2022-08-23 08:27:41 +00:00
|
|
|
${optionalString (luks.fido2Support && fido2luksCredentials != []) ''
|
2020-01-13 11:07:48 +00:00
|
|
|
|
|
|
|
open_with_hardware() {
|
|
|
|
local passsphrase
|
|
|
|
|
2021-05-26 13:43:38 +00:00
|
|
|
${if dev.fido2.passwordLess then ''
|
2020-01-17 09:39:22 +00:00
|
|
|
export passphrase=""
|
|
|
|
'' else ''
|
2021-05-26 13:43:38 +00:00
|
|
|
read -rsp "FIDO2 salt for ${dev.device}: " passphrase
|
2020-01-17 09:39:22 +00:00
|
|
|
echo
|
|
|
|
''}
|
|
|
|
${optionalString (lib.versionOlder kernelPackages.kernel.version "5.4") ''
|
|
|
|
echo "On systems with Linux Kernel < 5.4, it might take a while to initialize the CRNG, you might want to use linuxPackages_latest."
|
|
|
|
echo "Please move your mouse to create needed randomness."
|
|
|
|
''}
|
|
|
|
echo "Waiting for your FIDO2 device..."
|
2022-08-23 08:27:41 +00:00
|
|
|
fido2luks open${optionalString dev.allowDiscards " --allow-discards"} ${dev.device} ${dev.name} "${builtins.concatStringsSep "," fido2luksCredentials}" --await-dev ${toString dev.fido2.gracePeriod} --salt string:$passphrase
|
2020-01-17 09:39:22 +00:00
|
|
|
if [ $? -ne 0 ]; then
|
|
|
|
echo "No FIDO2 key found, falling back to normal open procedure"
|
|
|
|
open_normally
|
2020-01-13 11:07:48 +00:00
|
|
|
fi
|
|
|
|
}
|
|
|
|
''}
|
|
|
|
|
2020-06-14 10:03:00 +00:00
|
|
|
# commands to run right before we mount our device
|
2021-05-26 13:43:38 +00:00
|
|
|
${dev.preOpenCommands}
|
2020-06-14 10:03:00 +00:00
|
|
|
|
2022-08-23 08:27:41 +00:00
|
|
|
${if (luks.yubikeySupport && (dev.yubikey != null)) || (luks.gpgSupport && (dev.gpgCard != null)) || (luks.fido2Support && fido2luksCredentials != []) then ''
|
2019-05-09 18:15:35 +00:00
|
|
|
open_with_hardware
|
2017-09-15 21:48:51 +00:00
|
|
|
'' else ''
|
2014-01-28 03:02:51 +00:00
|
|
|
open_normally
|
2014-01-25 02:27:12 +00:00
|
|
|
''}
|
2020-06-14 10:03:00 +00:00
|
|
|
|
|
|
|
# commands to run right after we mounted our device
|
2021-05-26 13:43:38 +00:00
|
|
|
${dev.postOpenCommands}
|
2012-03-03 16:07:18 +00:00
|
|
|
'';
|
|
|
|
|
2017-09-15 21:48:51 +00:00
|
|
|
askPass = pkgs.writeScriptBin "cryptsetup-askpass" ''
|
|
|
|
#!/bin/sh
|
|
|
|
|
|
|
|
${commonFunctions}
|
|
|
|
|
|
|
|
while true; do
|
|
|
|
wait_target "luks" /crypt-ramfs/device 10 "LUKS to request a passphrase" || die "Passphrase is not requested now"
|
|
|
|
device=$(cat /crypt-ramfs/device)
|
|
|
|
|
|
|
|
echo -n "Passphrase for $device: "
|
|
|
|
IFS= read -rs passphrase
|
|
|
|
echo
|
|
|
|
|
|
|
|
rm /crypt-ramfs/device
|
|
|
|
echo -n "$passphrase" > /crypt-ramfs/passphrase
|
|
|
|
done
|
|
|
|
'';
|
|
|
|
|
2016-05-25 11:23:32 +00:00
|
|
|
preLVM = filterAttrs (n: v: v.preLVM) luks.devices;
|
|
|
|
postLVM = filterAttrs (n: v: !v.preLVM) luks.devices;
|
2012-03-04 21:00:35 +00:00
|
|
|
|
2023-03-22 14:17:23 +00:00
|
|
|
|
2023-12-18 00:55:20 +00:00
|
|
|
stage1Crypttab = pkgs.writeText "initrd-crypttab" (lib.concatLines (lib.mapAttrsToList (n: v: let
|
2022-04-13 18:45:29 +00:00
|
|
|
opts = v.crypttabExtraOpts
|
|
|
|
++ optional v.allowDiscards "discard"
|
|
|
|
++ optionals v.bypassWorkqueues [ "no-read-workqueue" "no-write-workqueue" ]
|
|
|
|
++ optional (v.header != null) "header=${v.header}"
|
2022-10-05 03:11:12 +00:00
|
|
|
++ optional (v.keyFileOffset != null) "keyfile-offset=${toString v.keyFileOffset}"
|
|
|
|
++ optional (v.keyFileSize != null) "keyfile-size=${toString v.keyFileSize}"
|
2023-03-22 14:17:23 +00:00
|
|
|
++ optional (v.keyFileTimeout != null) "keyfile-timeout=${builtins.toString v.keyFileTimeout}s"
|
|
|
|
++ optional (v.tryEmptyPassphrase) "try-empty-password=true"
|
2022-04-13 18:45:29 +00:00
|
|
|
;
|
|
|
|
in "${n} ${v.device} ${if v.keyFile == null then "-" else v.keyFile} ${lib.concatStringsSep "," opts}") luks.devices));
|
|
|
|
|
2010-10-25 00:57:30 +00:00
|
|
|
in
|
|
|
|
{
|
2019-12-10 01:51:19 +00:00
|
|
|
imports = [
|
|
|
|
(mkRemovedOptionModule [ "boot" "initrd" "luks" "enable" ] "")
|
|
|
|
];
|
2010-10-25 00:57:30 +00:00
|
|
|
|
|
|
|
options = {
|
|
|
|
|
2013-01-13 09:04:26 +00:00
|
|
|
boot.initrd.luks.mitigateDMAAttacks = mkOption {
|
2013-10-23 16:22:26 +00:00
|
|
|
type = types.bool;
|
2013-01-13 09:04:26 +00:00
|
|
|
default = true;
|
|
|
|
description = ''
|
2013-07-23 16:56:12 +00:00
|
|
|
Unless enabled, encryption keys can be easily recovered by an attacker with physical
|
|
|
|
access to any machine with PCMCIA, ExpressCard, ThunderBolt or FireWire port.
|
2023-10-30 20:41:44 +00:00
|
|
|
More information is available at <https://en.wikipedia.org/wiki/DMA_attack>.
|
2013-01-13 09:04:26 +00:00
|
|
|
|
2013-07-23 16:56:12 +00:00
|
|
|
This option blacklists FireWire drivers, but doesn't remove them. You can manually
|
|
|
|
load the drivers if you need to use a FireWire device, but don't forget to unload them!
|
2013-01-13 09:04:26 +00:00
|
|
|
'';
|
|
|
|
};
|
|
|
|
|
2013-01-13 10:44:16 +00:00
|
|
|
boot.initrd.luks.cryptoModules = mkOption {
|
2015-06-15 16:18:46 +00:00
|
|
|
type = types.listOf types.str;
|
2013-07-23 16:56:12 +00:00
|
|
|
default =
|
2013-10-09 17:11:47 +00:00
|
|
|
[ "aes" "aes_generic" "blowfish" "twofish"
|
|
|
|
"serpent" "cbc" "xts" "lrw" "sha1" "sha256" "sha512"
|
2019-06-07 20:15:35 +00:00
|
|
|
"af_alg" "algif_skcipher"
|
2013-07-23 16:56:12 +00:00
|
|
|
];
|
2013-01-13 10:44:16 +00:00
|
|
|
description = ''
|
2013-07-23 16:56:12 +00:00
|
|
|
A list of cryptographic kernel modules needed to decrypt the root device(s).
|
|
|
|
The default includes all common modules.
|
2013-01-13 10:44:16 +00:00
|
|
|
'';
|
|
|
|
};
|
|
|
|
|
2017-09-14 02:44:14 +00:00
|
|
|
boot.initrd.luks.forceLuksSupportInInitrd = mkOption {
|
|
|
|
type = types.bool;
|
|
|
|
default = false;
|
|
|
|
internal = true;
|
|
|
|
description = ''
|
|
|
|
Whether to configure luks support in the initrd, when no luks
|
|
|
|
devices are configured.
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
|
2017-09-15 21:48:51 +00:00
|
|
|
boot.initrd.luks.reusePassphrases = mkOption {
|
|
|
|
type = types.bool;
|
|
|
|
default = true;
|
|
|
|
description = ''
|
|
|
|
When opening a new LUKS device try reusing last successful
|
|
|
|
passphrase.
|
|
|
|
|
|
|
|
Useful for mounting a number of devices that use the same
|
|
|
|
passphrase without retyping it several times.
|
|
|
|
|
2022-08-13 09:35:46 +00:00
|
|
|
Such setup can be useful if you use {command}`cryptsetup luksSuspend`.
|
|
|
|
Different LUKS devices will still have
|
2017-09-15 21:48:51 +00:00
|
|
|
different master keys even when using the same passphrase.
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
|
2012-03-03 16:07:18 +00:00
|
|
|
boot.initrd.luks.devices = mkOption {
|
2016-05-25 11:23:32 +00:00
|
|
|
default = { };
|
2019-08-13 21:52:01 +00:00
|
|
|
example = { luksroot.device = "/dev/disk/by-uuid/430e9eff-d852-4f68-aa3b-2fa3599ebe08"; };
|
2012-05-15 20:45:01 +00:00
|
|
|
description = ''
|
2016-05-25 11:23:32 +00:00
|
|
|
The encrypted disk that should be opened before the root
|
|
|
|
filesystem is mounted. Both LVM-over-LUKS and LUKS-over-LVM
|
2017-07-02 02:37:51 +00:00
|
|
|
setups are supported. The unencrypted devices can be accessed as
|
2022-08-02 23:57:59 +00:00
|
|
|
{file}`/dev/mapper/«name»`.
|
2010-10-25 00:57:30 +00:00
|
|
|
'';
|
2012-03-04 21:00:35 +00:00
|
|
|
|
2020-08-22 23:28:45 +00:00
|
|
|
type = with types; attrsOf (submodule (
|
2023-11-18 19:37:56 +00:00
|
|
|
{ config, name, ... }: { options = {
|
2016-09-11 09:58:26 +00:00
|
|
|
|
|
|
|
name = mkOption {
|
|
|
|
visible = false;
|
|
|
|
default = name;
|
|
|
|
example = "luksroot";
|
|
|
|
type = types.str;
|
|
|
|
description = "Name of the unencrypted device in {file}`/dev/mapper`.";
|
|
|
|
};
|
2014-01-25 02:27:12 +00:00
|
|
|
|
2016-09-11 09:58:26 +00:00
|
|
|
device = mkOption {
|
|
|
|
example = "/dev/disk/by-uuid/430e9eff-d852-4f68-aa3b-2fa3599ebe08";
|
|
|
|
type = types.str;
|
|
|
|
description = "Path of the underlying encrypted block device.";
|
|
|
|
};
|
2014-08-29 16:43:03 +00:00
|
|
|
|
2016-09-11 09:58:26 +00:00
|
|
|
header = mkOption {
|
|
|
|
default = null;
|
|
|
|
example = "/root/header.img";
|
|
|
|
type = types.nullOr types.str;
|
|
|
|
description = ''
|
|
|
|
The name of the file or block device that
|
|
|
|
should be used as header for the encrypted device.
|
|
|
|
'';
|
|
|
|
};
|
2014-08-29 16:43:03 +00:00
|
|
|
|
2016-09-11 09:58:26 +00:00
|
|
|
keyFile = mkOption {
|
|
|
|
default = null;
|
|
|
|
example = "/dev/sdb1";
|
|
|
|
type = types.nullOr types.str;
|
|
|
|
description = ''
|
|
|
|
The name of the file (can be a raw device or a partition) that
|
|
|
|
should be used as the decryption key for the encrypted device. If
|
|
|
|
not specified, you will be prompted for a passphrase instead.
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
|
2023-03-22 14:17:23 +00:00
|
|
|
tryEmptyPassphrase = mkOption {
|
|
|
|
default = false;
|
|
|
|
type = types.bool;
|
|
|
|
description = ''
|
|
|
|
If keyFile fails then try an empty passphrase first before
|
|
|
|
prompting for password.
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
|
|
|
|
keyFileTimeout = mkOption {
|
|
|
|
default = null;
|
|
|
|
example = 5;
|
|
|
|
type = types.nullOr types.int;
|
|
|
|
description = ''
|
|
|
|
The amount of time in seconds for a keyFile to appear before
|
|
|
|
timing out and trying passwords.
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
|
2016-09-11 09:58:26 +00:00
|
|
|
keyFileSize = mkOption {
|
|
|
|
default = null;
|
|
|
|
example = 4096;
|
|
|
|
type = types.nullOr types.int;
|
|
|
|
description = ''
|
|
|
|
The size of the key file. Use this if only the beginning of the
|
|
|
|
key file should be used as a key (often the case if a raw device
|
|
|
|
or partition is used as key file). If not specified, the whole
|
|
|
|
`keyFile` will be used decryption, instead of just
|
|
|
|
the first `keyFileSize` bytes.
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
|
2018-06-03 19:25:06 +00:00
|
|
|
keyFileOffset = mkOption {
|
|
|
|
default = null;
|
|
|
|
example = 4096;
|
|
|
|
type = types.nullOr types.int;
|
|
|
|
description = ''
|
|
|
|
The offset of the key file. Use this in combination with
|
|
|
|
`keyFileSize` to use part of a file as key file
|
|
|
|
(often the case if a raw device or partition is used as a key file).
|
|
|
|
If not specified, the key begins at the first byte of
|
|
|
|
`keyFile`.
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
|
2016-09-11 09:58:26 +00:00
|
|
|
# FIXME: get rid of this option.
|
|
|
|
preLVM = mkOption {
|
|
|
|
default = true;
|
|
|
|
type = types.bool;
|
|
|
|
description = "Whether the luksOpen will be attempted before LVM scan or after it.";
|
|
|
|
};
|
|
|
|
|
|
|
|
allowDiscards = mkOption {
|
|
|
|
default = false;
|
|
|
|
type = types.bool;
|
|
|
|
description = ''
|
|
|
|
Whether to allow TRIM requests to the underlying device. This option
|
|
|
|
has security implications; please read the LUKS documentation before
|
|
|
|
activating it.
|
2021-07-01 21:37:07 +00:00
|
|
|
This option is incompatible with authenticated encryption (dm-crypt
|
|
|
|
stacked over dm-integrity).
|
2016-09-11 09:58:26 +00:00
|
|
|
'';
|
|
|
|
};
|
2014-08-29 16:43:03 +00:00
|
|
|
|
2021-05-26 13:43:38 +00:00
|
|
|
bypassWorkqueues = mkOption {
|
|
|
|
default = false;
|
|
|
|
type = types.bool;
|
|
|
|
description = ''
|
|
|
|
Whether to bypass dm-crypt's internal read and write workqueues.
|
|
|
|
Enabling this should improve performance on SSDs; see
|
|
|
|
[here](https://wiki.archlinux.org/index.php/Dm-crypt/Specialties#Disable_workqueue_for_increased_solid_state_drive_(SSD)_performance)
|
|
|
|
for more information. Needs Linux 5.9 or later.
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
|
2017-12-14 12:43:14 +00:00
|
|
|
fallbackToPassword = mkOption {
|
2017-10-23 20:22:26 +00:00
|
|
|
default = false;
|
|
|
|
type = types.bool;
|
|
|
|
description = ''
|
|
|
|
Whether to fallback to interactive passphrase prompt if the keyfile
|
|
|
|
cannot be found. This will prevent unattended boot should the keyfile
|
|
|
|
go missing.
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
|
2019-05-09 18:15:35 +00:00
|
|
|
gpgCard = mkOption {
|
|
|
|
default = null;
|
|
|
|
description = ''
|
|
|
|
The option to use this LUKS device with a GPG encrypted luks password by the GPG Smartcard.
|
|
|
|
If null (the default), GPG-Smartcard will be disabled for this device.
|
|
|
|
'';
|
|
|
|
|
|
|
|
type = with types; nullOr (submodule {
|
|
|
|
options = {
|
|
|
|
gracePeriod = mkOption {
|
|
|
|
default = 10;
|
|
|
|
type = types.int;
|
|
|
|
description = "Time in seconds to wait for the GPG Smartcard.";
|
|
|
|
};
|
|
|
|
|
|
|
|
encryptedPass = mkOption {
|
|
|
|
type = types.path;
|
|
|
|
description = "Path to the GPG encrypted passphrase.";
|
|
|
|
};
|
|
|
|
|
|
|
|
publicKey = mkOption {
|
|
|
|
type = types.path;
|
|
|
|
description = "Path to the Public Key.";
|
|
|
|
};
|
|
|
|
};
|
|
|
|
});
|
|
|
|
};
|
|
|
|
|
2020-01-13 11:07:48 +00:00
|
|
|
fido2 = {
|
|
|
|
credential = mkOption {
|
|
|
|
default = null;
|
|
|
|
example = "f1d00200d8dc783f7fb1e10ace8da27f8312d72692abfca2f7e4960a73f48e82e1f7571f6ebfcee9fb434f9886ccc8fcc52a6614d8d2";
|
2020-04-28 17:13:21 +00:00
|
|
|
type = types.nullOr types.str;
|
2020-01-13 11:07:48 +00:00
|
|
|
description = "The FIDO2 credential ID.";
|
|
|
|
};
|
|
|
|
|
2022-08-23 08:27:41 +00:00
|
|
|
credentials = mkOption {
|
|
|
|
default = [];
|
|
|
|
example = [ "f1d00200d8dc783f7fb1e10ace8da27f8312d72692abfca2f7e4960a73f48e82e1f7571f6ebfcee9fb434f9886ccc8fcc52a6614d8d2" ];
|
|
|
|
type = types.listOf types.str;
|
|
|
|
description = ''
|
|
|
|
List of FIDO2 credential IDs.
|
|
|
|
|
|
|
|
Use this if you have multiple FIDO2 keys you want to use for the same luks device.
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
|
2020-01-13 11:07:48 +00:00
|
|
|
gracePeriod = mkOption {
|
|
|
|
default = 10;
|
|
|
|
type = types.int;
|
|
|
|
description = "Time in seconds to wait for the FIDO2 key.";
|
|
|
|
};
|
|
|
|
|
|
|
|
passwordLess = mkOption {
|
|
|
|
default = false;
|
|
|
|
type = types.bool;
|
|
|
|
description = ''
|
|
|
|
Defines whatever to use an empty string as a default salt.
|
|
|
|
|
|
|
|
Enable only when your device is PIN protected, such as [Trezor](https://trezor.io/).
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
};
|
|
|
|
|
2016-09-11 09:58:26 +00:00
|
|
|
yubikey = mkOption {
|
|
|
|
default = null;
|
|
|
|
description = ''
|
2020-12-20 21:35:59 +00:00
|
|
|
The options to use for this LUKS device in YubiKey-PBA.
|
|
|
|
If null (the default), YubiKey-PBA will be disabled for this device.
|
2016-09-11 09:58:26 +00:00
|
|
|
'';
|
|
|
|
|
|
|
|
type = with types; nullOr (submodule {
|
|
|
|
options = {
|
|
|
|
twoFactor = mkOption {
|
|
|
|
default = true;
|
|
|
|
type = types.bool;
|
2020-12-20 21:35:59 +00:00
|
|
|
description = "Whether to use a passphrase and a YubiKey (true), or only a YubiKey (false).";
|
2016-09-11 09:58:26 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
slot = mkOption {
|
|
|
|
default = 2;
|
|
|
|
type = types.int;
|
2020-12-20 21:35:59 +00:00
|
|
|
description = "Which slot on the YubiKey to challenge.";
|
2016-09-11 09:58:26 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
saltLength = mkOption {
|
|
|
|
default = 16;
|
|
|
|
type = types.int;
|
|
|
|
description = "Length of the new salt in byte (64 is the effective maximum).";
|
|
|
|
};
|
|
|
|
|
|
|
|
keyLength = mkOption {
|
|
|
|
default = 64;
|
|
|
|
type = types.int;
|
|
|
|
description = "Length of the LUKS slot key derived with PBKDF2 in byte.";
|
|
|
|
};
|
|
|
|
|
|
|
|
iterationStep = mkOption {
|
|
|
|
default = 0;
|
|
|
|
type = types.int;
|
|
|
|
description = "How much the iteration count for PBKDF2 is increased at each successful authentication.";
|
|
|
|
};
|
|
|
|
|
|
|
|
gracePeriod = mkOption {
|
2017-09-15 21:47:36 +00:00
|
|
|
default = 10;
|
2016-09-11 09:58:26 +00:00
|
|
|
type = types.int;
|
2020-12-20 21:35:59 +00:00
|
|
|
description = "Time in seconds to wait for the YubiKey.";
|
2016-09-11 09:58:26 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
/* TODO: Add to the documentation of the current module:
|
|
|
|
|
|
|
|
Options related to the storing the salt.
|
|
|
|
*/
|
|
|
|
storage = {
|
|
|
|
device = mkOption {
|
|
|
|
default = "/dev/sda1";
|
|
|
|
type = types.path;
|
|
|
|
description = ''
|
|
|
|
An unencrypted device that will temporarily be mounted in stage-1.
|
|
|
|
Must contain the current salt to create the challenge for this LUKS device.
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
|
|
|
|
fsType = mkOption {
|
|
|
|
default = "vfat";
|
|
|
|
type = types.str;
|
|
|
|
description = "The filesystem of the unencrypted device.";
|
|
|
|
};
|
|
|
|
|
|
|
|
path = mkOption {
|
|
|
|
default = "/crypt-storage/default";
|
|
|
|
type = types.str;
|
|
|
|
description = ''
|
|
|
|
Absolute path of the salt on the unencrypted device with
|
|
|
|
that device's root directory as "/".
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
};
|
2014-01-25 02:27:12 +00:00
|
|
|
};
|
2016-09-11 09:58:26 +00:00
|
|
|
});
|
2014-01-25 02:27:12 +00:00
|
|
|
};
|
2020-06-14 10:03:00 +00:00
|
|
|
|
|
|
|
preOpenCommands = mkOption {
|
|
|
|
type = types.lines;
|
|
|
|
default = "";
|
|
|
|
example = ''
|
|
|
|
mkdir -p /tmp/persistent
|
|
|
|
mount -t zfs rpool/safe/persistent /tmp/persistent
|
|
|
|
'';
|
|
|
|
description = ''
|
|
|
|
Commands that should be run right before we try to mount our LUKS device.
|
2023-05-20 02:11:38 +00:00
|
|
|
This can be useful, if the keys needed to open the drive is on another partition.
|
2020-06-14 10:03:00 +00:00
|
|
|
'';
|
|
|
|
};
|
|
|
|
|
|
|
|
postOpenCommands = mkOption {
|
|
|
|
type = types.lines;
|
|
|
|
default = "";
|
|
|
|
example = ''
|
|
|
|
umount /tmp/persistent
|
|
|
|
'';
|
|
|
|
description = ''
|
|
|
|
Commands that should be run right after we have mounted our LUKS device.
|
|
|
|
'';
|
|
|
|
};
|
2022-04-13 18:45:29 +00:00
|
|
|
|
|
|
|
crypttabExtraOpts = mkOption {
|
|
|
|
type = with types; listOf singleLineStr;
|
|
|
|
default = [];
|
|
|
|
example = [ "_netdev" ];
|
|
|
|
visible = false;
|
|
|
|
description = ''
|
|
|
|
Only used with systemd stage 1.
|
|
|
|
|
|
|
|
Extra options to append to the last column of the generated crypttab file.
|
|
|
|
'';
|
|
|
|
};
|
2018-06-10 20:18:21 +00:00
|
|
|
};
|
2023-11-18 19:37:56 +00:00
|
|
|
|
|
|
|
config = mkIf (clevis.enable && (hasAttr name clevis.devices)) {
|
|
|
|
preOpenCommands = mkIf (!systemd.enable) ''
|
|
|
|
mkdir -p /clevis-${name}
|
|
|
|
mount -t ramfs none /clevis-${name}
|
|
|
|
clevis decrypt < /etc/clevis/${name}.jwe > /clevis-${name}/decrypted
|
|
|
|
'';
|
|
|
|
keyFile = "/clevis-${name}/decrypted";
|
|
|
|
fallbackToPassword = !systemd.enable;
|
|
|
|
postOpenCommands = mkIf (!systemd.enable) ''
|
|
|
|
umount /clevis-${name}
|
|
|
|
'';
|
|
|
|
};
|
2018-06-10 20:18:21 +00:00
|
|
|
}));
|
2010-10-25 00:57:30 +00:00
|
|
|
};
|
2013-10-23 16:22:26 +00:00
|
|
|
|
2019-05-09 18:15:35 +00:00
|
|
|
boot.initrd.luks.gpgSupport = mkOption {
|
|
|
|
default = false;
|
|
|
|
type = types.bool;
|
|
|
|
description = ''
|
|
|
|
Enables support for authenticating with a GPG encrypted password.
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
|
2014-01-25 02:27:12 +00:00
|
|
|
boot.initrd.luks.yubikeySupport = mkOption {
|
|
|
|
default = false;
|
|
|
|
type = types.bool;
|
2014-01-28 22:45:16 +00:00
|
|
|
description = ''
|
2020-12-20 21:35:59 +00:00
|
|
|
Enables support for authenticating with a YubiKey on LUKS devices.
|
2014-01-28 22:45:16 +00:00
|
|
|
See the NixOS wiki for information on how to properly setup a LUKS device
|
2020-12-20 21:35:59 +00:00
|
|
|
and a YubiKey to work with this feature.
|
2014-01-28 22:45:16 +00:00
|
|
|
'';
|
2014-01-25 02:27:12 +00:00
|
|
|
};
|
2020-01-13 11:07:48 +00:00
|
|
|
|
|
|
|
boot.initrd.luks.fido2Support = mkOption {
|
|
|
|
default = false;
|
|
|
|
type = types.bool;
|
|
|
|
description = ''
|
|
|
|
Enables support for authenticating with FIDO2 devices.
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
|
2010-10-25 00:57:30 +00:00
|
|
|
};
|
|
|
|
|
2017-09-14 02:44:14 +00:00
|
|
|
config = mkIf (luks.devices != {} || luks.forceLuksSupportInInitrd) {
|
2010-10-25 00:57:30 +00:00
|
|
|
|
2019-05-09 18:15:35 +00:00
|
|
|
assertions =
|
|
|
|
[ { assertion = !(luks.gpgSupport && luks.yubikeySupport);
|
2020-12-20 21:35:59 +00:00
|
|
|
message = "YubiKey and GPG Card may not be used at the same time.";
|
2019-05-09 18:15:35 +00:00
|
|
|
}
|
2020-01-13 11:07:48 +00:00
|
|
|
|
|
|
|
{ assertion = !(luks.gpgSupport && luks.fido2Support);
|
|
|
|
message = "FIDO2 and GPG Card may not be used at the same time.";
|
|
|
|
}
|
|
|
|
|
|
|
|
{ assertion = !(luks.fido2Support && luks.yubikeySupport);
|
2020-12-20 21:35:59 +00:00
|
|
|
message = "FIDO2 and YubiKey may not be used at the same time.";
|
2020-01-13 11:07:48 +00:00
|
|
|
}
|
2021-05-26 13:43:38 +00:00
|
|
|
|
|
|
|
{ assertion = any (dev: dev.bypassWorkqueues) (attrValues luks.devices)
|
|
|
|
-> versionAtLeast kernelPackages.kernel.version "5.9";
|
|
|
|
message = "boot.initrd.luks.devices.<name>.bypassWorkqueues is not supported for kernels older than 5.9";
|
|
|
|
}
|
2022-04-13 18:45:29 +00:00
|
|
|
|
2023-03-22 14:17:23 +00:00
|
|
|
{ assertion = !config.boot.initrd.systemd.enable -> all (x: x.keyFileTimeout == null) (attrValues luks.devices);
|
|
|
|
message = "boot.initrd.luks.devices.<name>.keyFileTimeout is only supported for systemd initrd";
|
|
|
|
}
|
|
|
|
|
2022-04-13 18:45:29 +00:00
|
|
|
{ assertion = config.boot.initrd.systemd.enable -> all (dev: !dev.fallbackToPassword) (attrValues luks.devices);
|
|
|
|
message = "boot.initrd.luks.devices.<name>.fallbackToPassword is implied by systemd stage 1.";
|
|
|
|
}
|
|
|
|
{ assertion = config.boot.initrd.systemd.enable -> all (dev: dev.preLVM) (attrValues luks.devices);
|
|
|
|
message = "boot.initrd.luks.devices.<name>.preLVM is not used by systemd stage 1.";
|
|
|
|
}
|
|
|
|
{ assertion = config.boot.initrd.systemd.enable -> options.boot.initrd.luks.reusePassphrases.highestPrio == defaultPrio;
|
|
|
|
message = "boot.initrd.luks.reusePassphrases has no effect with systemd stage 1.";
|
|
|
|
}
|
|
|
|
{ assertion = config.boot.initrd.systemd.enable -> all (dev: dev.preOpenCommands == "" && dev.postOpenCommands == "") (attrValues luks.devices);
|
|
|
|
message = "boot.initrd.luks.devices.<name>.preOpenCommands and postOpenCommands is not supported by systemd stage 1. Please bind a service to cryptsetup.target or cryptsetup-pre.target instead.";
|
|
|
|
}
|
|
|
|
# TODO
|
|
|
|
{ assertion = config.boot.initrd.systemd.enable -> !luks.gpgSupport;
|
|
|
|
message = "systemd stage 1 does not support GPG smartcards yet.";
|
|
|
|
}
|
|
|
|
{ assertion = config.boot.initrd.systemd.enable -> !luks.fido2Support;
|
2022-09-23 21:47:05 +00:00
|
|
|
message = ''
|
2024-03-05 21:38:38 +00:00
|
|
|
systemd stage 1 does not support configuring FIDO2 unlocking through `boot.initrd.luks.fido2Support`.
|
|
|
|
Use systemd-cryptenroll(1) to configure FIDO2 support, and set
|
|
|
|
`boot.initrd.luks.devices.''${DEVICE}.crypttabExtraOpts` as appropriate per crypttab(5)
|
|
|
|
(e.g. `fido2-device=auto`).
|
2022-09-23 21:47:05 +00:00
|
|
|
'';
|
2022-04-13 18:45:29 +00:00
|
|
|
}
|
|
|
|
# TODO
|
|
|
|
{ assertion = config.boot.initrd.systemd.enable -> !luks.yubikeySupport;
|
|
|
|
message = "systemd stage 1 does not support Yubikeys yet.";
|
|
|
|
}
|
2019-05-09 18:15:35 +00:00
|
|
|
];
|
|
|
|
|
2013-01-13 09:04:26 +00:00
|
|
|
# actually, sbp2 driver is the one enabling the DMA attack, but this needs to be tested
|
|
|
|
boot.blacklistedKernelModules = optionals luks.mitigateDMAAttacks
|
|
|
|
["firewire_ohci" "firewire_core" "firewire_sbp2"];
|
|
|
|
|
2012-03-03 16:07:18 +00:00
|
|
|
# Some modules that may be needed for mounting anything ciphered
|
2017-12-18 23:57:45 +00:00
|
|
|
boot.initrd.availableKernelModules = [ "dm_mod" "dm_crypt" "cryptd" "input_leds" ]
|
2018-02-25 20:36:19 +00:00
|
|
|
++ luks.cryptoModules
|
|
|
|
# workaround until https://marc.info/?l=linux-crypto-vger&m=148783562211457&w=4 is merged
|
|
|
|
# remove once 'modprobe --show-depends xts' shows ecb as a dependency
|
2023-06-25 09:47:43 +00:00
|
|
|
++ (optional (builtins.elem "xts" luks.cryptoModules) "ecb");
|
2010-10-25 00:57:30 +00:00
|
|
|
|
2011-12-28 21:46:40 +00:00
|
|
|
# copy the cryptsetup binary and it's dependencies
|
2023-02-08 01:35:32 +00:00
|
|
|
boot.initrd.extraUtilsCommands = let
|
|
|
|
pbkdf2-sha512 = pkgs.runCommandCC "pbkdf2-sha512" { buildInputs = [ pkgs.openssl ]; } ''
|
|
|
|
mkdir -p "$out/bin"
|
|
|
|
cc -O3 -lcrypto ${./pbkdf2-sha512.c} -o "$out/bin/pbkdf2-sha512"
|
|
|
|
strip -s "$out/bin/pbkdf2-sha512"
|
|
|
|
'';
|
|
|
|
in
|
|
|
|
mkIf (!config.boot.initrd.systemd.enable) ''
|
2015-03-29 00:15:41 +00:00
|
|
|
copy_bin_and_libs ${pkgs.cryptsetup}/bin/cryptsetup
|
2017-09-15 21:48:51 +00:00
|
|
|
copy_bin_and_libs ${askPass}/bin/cryptsetup-askpass
|
|
|
|
sed -i s,/bin/sh,$out/bin/sh, $out/bin/cryptsetup-askpass
|
2015-10-18 10:50:36 +00:00
|
|
|
|
2014-01-25 02:27:12 +00:00
|
|
|
${optionalString luks.yubikeySupport ''
|
2017-03-11 21:23:00 +00:00
|
|
|
copy_bin_and_libs ${pkgs.yubikey-personalization}/bin/ykchalresp
|
|
|
|
copy_bin_and_libs ${pkgs.yubikey-personalization}/bin/ykinfo
|
2016-02-01 18:46:16 +00:00
|
|
|
copy_bin_and_libs ${pkgs.openssl.bin}/bin/openssl
|
Update to the Yubikey PBA
Security-relevant changes:
* No (salted) passphrase hash send to the yubikey, only hash of the salt (as it was in the original implementation).
* Derive $k_luks with PBKDF2 from the yubikey $response (as the PBKDF2 salt) and the passphrase $k_user
(as the PBKDF2 password), so that if two-factor authentication is enabled
(a) a USB-MITM attack on the yubikey itself is not enough to break the system
(b) the potentially low-entropy $k_user is better protected against brute-force attacks
* Instead of using uuidgen, gather the salt (previously random uuid / uuid_r) directly from /dev/random.
* Length of the new salt in byte added as the parameter "saltLength", defaults to 16 byte.
Note: Length of the challenge is 64 byte, so saltLength > 64 may have no benefit over saltLengh = 64.
* Length of $k_luks derived with PBKDF2 in byte added as the parameter "keyLength", defaults to 64 byte.
Example: For a luks device with a 512-bit key, keyLength should be 64.
* Increase of the PBKDF2 iteration count per successful authentication added as the
parameter "iterationStep", defaults to 0.
Other changes:
* Add optional grace period before trying to find the yubikey, defaults to 2 seconds.
Full overview of the yubikey authentication process:
(1) Read $salt and $iterations from unencrypted device (UD).
(2) Calculate the $challenge from the $salt with a hash function.
Chosen instantiation: SHA-512($salt).
(3) Challenge the yubikey with the $challenge and receive the $response.
(4) Repeat three times:
(a) Prompt for the passphrase $k_user.
(b) Derive the key $k_luks for the luks device with a key derivation function from $k_user and $response.
Chosen instantiation: PBKDF2(HMAC-SHA-512, $k_user, $response, $iterations, keyLength).
(c) Try to open the luks device with $k_luks and escape loop (4) only on success.
(5) Proceed only if luks device was opened successfully, fail otherwise.
(6) Gather $new_salt from a cryptographically secure pseudorandom number generator
Chosen instantiation: /dev/random
(7) Calculate the $new_challenge from the $new_salt with the same hash function as (2).
(8) Challenge the yubikey with the $new_challenge and receive the $new_response.
(9) Derive the new key $new_k_luks for the luks device in the same manner as in (4) (b),
but with more iterations as given by iterationStep.
(10) Try to change the luks device's key $k_luks to $new_k_luks.
(11) If (10) was successful, write the $new_salt and the $new_iterations to the UD.
Note: $new_iterations = $iterations + iterationStep
Known (software) attack vectors:
* A MITM attack on the keyboard can recover $k_user. This, combined with a USB-MITM
attack on the yubikey for the $response (1) or the $new_response (2) will result in
(1) $k_luks being recovered,
(2) $new_k_luks being recovered.
* Any attacker with access to the RAM state of stage-1 at mid- or post-authentication
can recover $k_user, $k_luks, and $new_k_luks
* If an attacker has recovered $response or $new_response, he can perform a brute-force
attack on $k_user with it without the Yubikey needing to be present (using cryptsetup's
"luksOpen --verify-passphrase" oracle. He could even make a copy of the luks device's
luks header and run the brute-force attack without further access to the system.
* A USB-MITM attack on the yubikey will allow an attacker to attempt to brute-force
the yubikey's internal key ("shared secret") without it needing to be present anymore.
Credits:
* Florian Klien,
for the original concept and the reference implementation over at
https://github.com/flowolf/initramfs_ykfde
* Anthony Thysse,
for the reference implementation of accessing OpenSSL's PBKDF2 over at
http://www.ict.griffith.edu.au/anthony/software/pbkdf2.c
2014-02-05 16:10:59 +00:00
|
|
|
|
2023-02-08 01:35:32 +00:00
|
|
|
copy_bin_and_libs ${pbkdf2-sha512}/bin/pbkdf2-sha512
|
2014-01-28 03:02:51 +00:00
|
|
|
|
2015-03-29 00:15:41 +00:00
|
|
|
mkdir -p $out/etc/ssl
|
2016-08-04 20:12:39 +00:00
|
|
|
cp -pdv ${pkgs.openssl.out}/etc/ssl/openssl.cnf $out/etc/ssl
|
2014-01-28 03:02:51 +00:00
|
|
|
|
2015-05-04 12:16:03 +00:00
|
|
|
cat > $out/bin/openssl-wrap <<EOF
|
|
|
|
#!$out/bin/sh
|
2015-10-18 14:55:28 +00:00
|
|
|
export OPENSSL_CONF=$out/etc/ssl/openssl.cnf
|
|
|
|
$out/bin/openssl "\$@"
|
2015-05-04 12:16:03 +00:00
|
|
|
EOF
|
|
|
|
chmod +x $out/bin/openssl-wrap
|
2014-01-25 02:27:12 +00:00
|
|
|
''}
|
2019-05-09 18:15:35 +00:00
|
|
|
|
2020-01-13 11:07:48 +00:00
|
|
|
${optionalString luks.fido2Support ''
|
|
|
|
copy_bin_and_libs ${pkgs.fido2luks}/bin/fido2luks
|
|
|
|
''}
|
|
|
|
|
|
|
|
|
2019-05-09 18:15:35 +00:00
|
|
|
${optionalString luks.gpgSupport ''
|
|
|
|
copy_bin_and_libs ${pkgs.gnupg}/bin/gpg
|
|
|
|
copy_bin_and_libs ${pkgs.gnupg}/bin/gpg-agent
|
|
|
|
copy_bin_and_libs ${pkgs.gnupg}/libexec/scdaemon
|
|
|
|
|
|
|
|
${concatMapStringsSep "\n" (x:
|
2023-03-19 20:44:31 +00:00
|
|
|
optionalString (x.gpgCard != null)
|
2019-05-09 18:15:35 +00:00
|
|
|
''
|
|
|
|
mkdir -p $out/secrets/gpg-keys/${x.device}
|
|
|
|
cp -a ${x.gpgCard.encryptedPass} $out/secrets/gpg-keys/${x.device}/cryptkey.gpg
|
|
|
|
cp -a ${x.gpgCard.publicKey} $out/secrets/gpg-keys/${x.device}/pubkey.asc
|
|
|
|
''
|
|
|
|
) (attrValues luks.devices)
|
|
|
|
}
|
|
|
|
''}
|
2011-12-28 21:46:40 +00:00
|
|
|
'';
|
|
|
|
|
2022-04-13 18:45:29 +00:00
|
|
|
boot.initrd.extraUtilsCommandsTest = mkIf (!config.boot.initrd.systemd.enable) ''
|
2011-12-28 22:37:38 +00:00
|
|
|
$out/bin/cryptsetup --version
|
2014-01-25 02:27:12 +00:00
|
|
|
${optionalString luks.yubikeySupport ''
|
|
|
|
$out/bin/ykchalresp -V
|
2014-01-28 03:02:51 +00:00
|
|
|
$out/bin/ykinfo -V
|
|
|
|
$out/bin/openssl-wrap version
|
2014-01-25 02:27:12 +00:00
|
|
|
''}
|
2019-05-09 18:15:35 +00:00
|
|
|
${optionalString luks.gpgSupport ''
|
|
|
|
$out/bin/gpg --version
|
|
|
|
$out/bin/gpg-agent --version
|
|
|
|
$out/bin/scdaemon --version
|
|
|
|
''}
|
2020-01-13 11:07:48 +00:00
|
|
|
${optionalString luks.fido2Support ''
|
|
|
|
$out/bin/fido2luks --version
|
|
|
|
''}
|
2010-10-25 00:57:30 +00:00
|
|
|
'';
|
|
|
|
|
2022-04-13 18:45:29 +00:00
|
|
|
boot.initrd.systemd = {
|
|
|
|
contents."/etc/crypttab".source = stage1Crypttab;
|
|
|
|
|
2023-11-19 01:35:57 +00:00
|
|
|
extraBin.systemd-cryptsetup = "${config.boot.initrd.systemd.package}/bin/systemd-cryptsetup";
|
2022-04-13 18:45:29 +00:00
|
|
|
|
|
|
|
additionalUpstreamUnits = [
|
|
|
|
"cryptsetup-pre.target"
|
|
|
|
"cryptsetup.target"
|
|
|
|
"remote-cryptsetup.target"
|
|
|
|
];
|
|
|
|
storePaths = [
|
2023-11-19 01:35:57 +00:00
|
|
|
"${config.boot.initrd.systemd.package}/bin/systemd-cryptsetup"
|
2022-04-30 12:44:01 +00:00
|
|
|
"${config.boot.initrd.systemd.package}/lib/systemd/system-generators/systemd-cryptsetup-generator"
|
2024-09-19 08:10:56 +00:00
|
|
|
] ++ lib.optionals config.boot.initrd.systemd.tpm2.enable [
|
|
|
|
"${config.boot.initrd.systemd.package}/lib/cryptsetup/libcryptsetup-token-systemd-tpm2.so"
|
2022-04-13 18:45:29 +00:00
|
|
|
];
|
|
|
|
|
|
|
|
};
|
|
|
|
# We do this because we need the udev rules from the package
|
2024-11-12 15:53:46 +00:00
|
|
|
services.lvm.enable = true;
|
2022-04-13 18:45:29 +00:00
|
|
|
boot.initrd.services.lvm.enable = true;
|
|
|
|
|
|
|
|
boot.initrd.preFailCommands = mkIf (!config.boot.initrd.systemd.enable) postCommands;
|
|
|
|
boot.initrd.preLVMCommands = mkIf (!config.boot.initrd.systemd.enable) (commonFunctions + preCommands + concatStrings (mapAttrsToList openCommand preLVM) + postCommands);
|
|
|
|
boot.initrd.postDeviceCommands = mkIf (!config.boot.initrd.systemd.enable) (commonFunctions + preCommands + concatStrings (mapAttrsToList openCommand postLVM) + postCommands);
|
2012-03-08 20:49:26 +00:00
|
|
|
|
2023-11-18 19:37:56 +00:00
|
|
|
boot.initrd.systemd.services = let devicesWithClevis = filterAttrs (device: _: (hasAttr device clevis.devices)) luks.devices; in
|
|
|
|
mkIf (clevis.enable && systemd.enable) (
|
|
|
|
(mapAttrs'
|
|
|
|
(name: _: nameValuePair "cryptsetup-clevis-${name}" {
|
|
|
|
wantedBy = [ "systemd-cryptsetup@${utils.escapeSystemdPath name}.service" ];
|
|
|
|
before = [
|
|
|
|
"systemd-cryptsetup@${utils.escapeSystemdPath name}.service"
|
|
|
|
"initrd-switch-root.target"
|
|
|
|
"shutdown.target"
|
|
|
|
];
|
|
|
|
wants = [ "systemd-udev-settle.service" ] ++ optional clevis.useTang "network-online.target";
|
|
|
|
after = [ "systemd-modules-load.service" "systemd-udev-settle.service" ] ++ optional clevis.useTang "network-online.target";
|
|
|
|
script = ''
|
|
|
|
mkdir -p /clevis-${name}
|
|
|
|
mount -t ramfs none /clevis-${name}
|
|
|
|
umask 277
|
|
|
|
clevis decrypt < /etc/clevis/${name}.jwe > /clevis-${name}/decrypted
|
|
|
|
'';
|
|
|
|
conflicts = [ "initrd-switch-root.target" "shutdown.target" ];
|
|
|
|
unitConfig.DefaultDependencies = "no";
|
|
|
|
serviceConfig = {
|
|
|
|
Type = "oneshot";
|
|
|
|
RemainAfterExit = true;
|
|
|
|
ExecStop = "${config.boot.initrd.systemd.package.util-linux}/bin/umount /clevis-${name}";
|
|
|
|
};
|
|
|
|
})
|
|
|
|
devicesWithClevis)
|
|
|
|
);
|
|
|
|
|
2012-03-08 20:49:26 +00:00
|
|
|
environment.systemPackages = [ pkgs.cryptsetup ];
|
2010-10-25 00:57:30 +00:00
|
|
|
};
|
2011-09-14 18:20:50 +00:00
|
|
|
}
|