dm-crypt/加密整个系统
本文介绍了使用“dm-crypt”加密整个系统的几种常见方式及相比正常安装过程所需的改动。官方安装镜像包含了加密所需的所有工具。
若要加密现存的未加密的文件系统,见 dm-crypt/设备加密#加密现有的未加密文件系统。
概览[编辑 | 编辑源代码]
为根文件系统提供安全保护是“dm-crypt”的特色功能,且相比其它方案“dm-crypt”的性能更加优秀。与仅仅加密部分非根文件系统不同,加密根文件系统可以隐藏如已安装的程序列表、用户名等信息,并能避免通过mlocate、/var/log/
等常见方式泄露数据。此外,由于除了引导加载程序和内核(通常情况下),所有数据都被加密,对系统进行未授权的改动将更加困难。
下面所述的所有方案都有上述优点。此外,各种方案的优缺点总结如下:
方案 | 优点 | 缺点 |
---|---|---|
#在单一分区上配置LUKS 展示了设置使用LUKS完全加密的根文件系统的基本方式,最直接。 |
|
|
#在LUKS上配置LVM
通过在LUKS加密的单一分区上配置LVM来达到分区灵活性。 |
|
|
#在LVM上配置LUKS
只在LVM上层使用LUKS。 |
|
|
#在软RAID上配置LUKS
只在软RAID上层使用LUKS。 |
|
|
#Plain dm-crypt
使用dm-crypt的plain模式,即不使用LUKS头及LUKS的多密钥选项。 |
| |
#加密boot分区(GRUB)
展示了如何在使用GRUB时加密boot分区。 |
|
|
#Root on ZFS |
虽然上述方案对外界威胁提供了比文件系统层面的加密好得多的保护,它们也有一个共同的缺点:任何拥有加密密码的用户都能解密整个设备,进而访问其它用户的数据。若要防范上述风险,可以结合使用块设备加密和文件系统加密。要提前规划相关配置,见静态数据加密。
对于本文中所有方案的分区策略的概述,参见dm-crypt/准备磁盘#分区。
此外,还需考虑是否设置加密的swap分区及设置方式。详见dm-crypt/Swap encryption。
加密整个系统可在设备失窃时保护数据,若还要针对logical tampering提供额外的保护,在完成选择的方案后,还应参见dm-crypt/Specialties#Securing the unencrypted boot partition。
若使用固态硬盘,或许需要启用TRIM支持。然而,这可能影响安全性,详见dm-crypt/Specialties#Discard/TRIM support for solid state drives (SSD)。
- 无论选择何种方案,绝对不能直接在未解密的分区上运行文件系统修复软件(如fsck),否则解密数据所用的密钥将永久丢失!相关软件只能用于解密后的分区。
- 对于LUKS2格式:
- GRUB对LUKS2的支持并不完善,详见GRUB#加密的/boot。对于需要由GRUB解密的分区,建议使用LUKS1(
cryptsetup luksFormat --type luks1
)。 - LUKS2采用了占用内存较多的设计方式,默认情况下每个encrypted mapper将占用1GB内存。若设备的内存容量较少或要同时解密较多的LUKS2分区,可能导致系统启动失败。可通过
--pbkdf-memory
选择控制内存用量,详见[1]。
- GRUB对LUKS2的支持并不完善,详见GRUB#加密的/boot。对于需要由GRUB解密的分区,建议使用LUKS1(
在单一分区上配置LUKS[编辑 | 编辑源代码]
本例说明了通过“dm-crypt”与LUKS在单一分区上进行配置以加密整个系统。
+-----------------------+------------------------+-----------------------+ | boot分区 | LUKS2加密的根分区 | 可选的用于其它分区的 | | | partition | 空闲空间 | | | | | | /boot | / | | | | | | | | /dev/mapper/root | | | |------------------------| | | /dev/sda1 | /dev/sda2 | | +-----------------------+------------------------+-----------------------+
本例的前几步可在启动到Arch Linux安装镜像后直接进行。
准备磁盘[编辑 | 编辑源代码]
在创建任何分区前,应先查看dm-crypt/准备磁盘,明确安全地擦除整个磁盘的重要性及相关方法。
接下来创建本例中的相关分区(至少需要/
(/dev/sda2
)和/boot
(/dev/sda1
),详见分区。
准备非boot分区[编辑 | 编辑源代码]
下列命令创建并挂载加密的根分区。dm-crypt/加密非root文件系统#分区中对相关命令有详细解释(只要正确配置了mkinitcpio和引导加载程序,其中的内容同样适用于根分区。)
若要使用自定义的加密选项(如cipher、key length),在执行第一条命令前参看加密选项。若要改变默认扇区大小,参见dm-crypt/设备加密#扇区大小。
# cryptsetup -y -v luksFormat /dev/sda2 # cryptsetup open /dev/sda2 root # mkfs.ext4 /dev/mapper/root # mount /dev/mapper/root /mnt
确保设备mapper已正确设置:
# umount /mnt # cryptsetup close root # cryptsetup open /dev/sda2 root # mount /dev/mapper/root /mnt
若为不同目录(如/home
)创建了单独的分区,则需要对除了boot分区的所有要加密的分区重复上述步骤。为了在启动时挂载上述分区,见dm-crypt/加密非root文件系统#自动解锁并挂载。
注意对于每个加密分区都需要单独设置密码,并且导致启动时需要依次输入每个分区密码的不便。可通过在根分区存储一个keyfile并配置crypttab
来自动解密其它分区,详见dm-crypt/设备加密#使用LUKS来格式化带有密钥文件的分区。
准备boot分区[编辑 | 编辑源代码]
对于加密根分区的方案,需创建一个未加密的boot分区。对于传统BIOS启动的系统上,执行:
# mkfs.ext4 /dev/sda1
或者,在UEFI启动的系统上可将EFI 系统分区用作/boot
:
# mkfs.fat -F32 /dev/sda1
之后创建挂载用的目录并挂载boot分区:
# mount --mkdir /dev/sda1 /mnt/boot
挂载分区[编辑 | 编辑源代码]
在执行安装指南#挂载分区中的步骤时,注意需要挂载解密后创建的设备mapper而非硬盘上的实际分区。当然,未加密的boot分区需要直接挂载。
配置mkinitcpio[编辑 | 编辑源代码]
在mkinitcpio.conf中添加keyboard
和encrypt
钩子。若要使用非标准的键盘布局,还需添加keymap
钩子;若要修改控制台字体,还需添加consolefont
钩子:
HOOKS=(base udev autodetect modconf kms keyboard keymap consolefont block encrypt filesystems fsck)
若使用基于sysyemd的initramfs,需添加的应为keyboard
和sd-encrypt
钩子。若要使用非标准键盘布局或修改控制台字体,还需添加sd-vconsole
钩子:
HOOKS=(base systemd autodetect modconf kms keyboard sd-vconsole block sd-encrypt filesystems fsck)
在保存上述修改后,重新生成initramfs。对于可能需要的其它钩子,详见dm-crypt/System configuration#mkinitcpio。
配置引导加载程序[编辑 | 编辑源代码]
要在启动时解密已加密的根分区,需配置引导加载程序设置如下内核参数:
cryptdevice=UUID=device-UUID:root root=/dev/mapper/root
若使用的是[[sd-encrypt],则应该添加如下内核参数:
rd.luks.name=device-UUID=root root=/dev/mapper/root
device-UUID
指的是LUKS超级块的UUID,在本例中为/dev/sda2
。详见块设备持久化命名。
另见dm-crypt/System configuration#Kernel parameters。
{{提示|若根分区与boot分区在同一硬盘上,并且使用的引导加载程序支持GPT分区自动挂载,则可以修改根分区的partition type GUID为“Root partitio”并通过systemd-gpt-auto-generator(8)设置,而不必再设置上述内核参数。
在LUKS上配置LVM[编辑 | 编辑源代码]
在加密的单个分区上配置LVM是一种较为直接的方案。该方案将LVM设置在一个大的加密的块设备中。因此,在该块设备解密、卷结构被扫描、分区在启动时被挂载前,整个LVM都是不可见的。
以下为示例磁盘结构:
+-----------------------------------------------------------------------+ +----------------+ | 逻辑卷 1 | 逻辑卷 2 | 逻辑卷 3 | | boot分区 | | | | | | | | [SWAP] | / | /home | | /boot | | | | | | | | /dev/MyVolGroup/swap | /dev/MyVolGroup/root | /dev/MyVolGroup/home | | | |_ _ _ _ _ _ _ _ _ _ _ _|_ _ _ _ _ _ _ _ _ _ _ _|_ _ _ _ _ _ _ _ _ _ _ _| | (可能在其他 | | | | 设备上) | | 使用LUKS2加密的分区 | | | | /dev/sda1 | | /dev/sdb1 | +-----------------------------------------------------------------------+ +----------------+
encrypt
钩子,则不能将逻辑卷分配到多个磁盘上。可使用sd-encrypt钩子或参见dm-crypt/Specialties#Modifying the encrypt hook for multiple partitions。- dm-crypt/Specialties#Encrypted system using a detached LUKS header介绍了如何将LUKS头单独存放于USB设备上以实现两步验证。
- dm-crypt/Specialties#Encrypted /boot and a detached LUKS header on USB介绍的方案将LUKS头、加密的boot分区、加密的keyfile都存放在USB设备上。
准备磁盘[编辑 | 编辑源代码]
在创建任何分区前,应先查看dm-crypt/Drive preparation,明确安全地擦除整个磁盘的重要性及相关方法。
创建一个大于200MiB的分区用于/boot
。
/boot
。创建一个用于存储整个加密的LVM的分区。
在上述创建的分区上配置LUKS作为LVM的容器:
# cryptsetup luksFormat /dev/sda1
若要自定义cryptsetup的选项,在执行上述命令参考LUKS加密选项。
打开创建的容器:
# cryptsetup open /dev/sda1 cryptlvm
现在可通过/dev/mapper/cryptlvm
访问解密的容器。
准备逻辑卷[编辑 | 编辑源代码]
在解密的LUKS容器上创建物理卷:
# pvcreate /dev/mapper/cryptlvm
创建卷组(本例中使用MyVolGroup
,也可使用其它名称。),并将先前创建的物理卷加入其中:
# vgcreate MyVolGroup /dev/mapper/cryptlvm
在卷组上创建所有逻辑卷:
{{提示|若要将一个或多个逻辑卷格式化为ext4格式,应在卷组中预留至少256MiB的空闲空间以便能正常使用e2scrub(8)。要达到这一点,在使用-l 100%FREE
创建占满所有剩余空间最后一个逻辑卷后,使用类似lvreduce -L -256M MyVolGroup/home
的命令减少该逻辑卷的大小。}
# lvcreate -L 8G MyVolGroup -n swap # lvcreate -L 32G MyVolGroup -n root # lvcreate -l 100%FREE MyVolGroup -n home
格式化逻辑卷:
# mkfs.ext4 /dev/MyVolGroup/root # mkfs.ext4 /dev/MyVolGroup/home # mkswap /dev/MyVolGroup/swap
挂载文件系统:
# mount /dev/MyVolGroup/root /mnt # mount --mkdir /dev/MyVolGroup/home /mnt/home # swapon /dev/MyVolGroup/swap
准备boot分区[编辑 | 编辑源代码]
引导加载程序从/boot
目录加载内核、initramfs和自身的配置文件。可使用任何可被引导加载程序读取的文件系统。
在用于/boot
的分区上创建文件系统:
# mkfs.ext4 /dev/sdb1
将上述分区挂载到/mnt/boot
:
# mount --mkdir /dev/sdb1 /mnt/boot
继续执行安装指南#开始安装系统中的步骤。之后回到本页,执行修改后的关于 initramfs、安装引导程序步骤。
配置mkinitcpio[编辑 | 编辑源代码]
确保安装了lvm2包,之后在mkinitcpio.conf中添加keyboard
、encrypt
、lvm2
钩子。若要使用非标准的键盘布局,还需添加keymap
钩子;若要修改控制台字体,还需添加consolefont
钩子:
HOOKS=(base udev autodetect modconf kms keyboard keymap consolefont block encrypt lvm2 filesystems fsck)
若使用基于sysyemd的initramfs,需添加的应为keyboard
、lvm2
和sd-encrypt
钩子。若要使用非标准键盘布局或修改控制台字体,还需添加sd-vconsole
钩子:
HOOKS=(base systemd autodetect modconf kms keyboard sd-vconsole block sd-encrypt lvm2 filesystems fsck)
在保存上述修改后,重新生成initramfs。对于可能需要的其它钩子,详见dm-crypt/System configuration#mkinitcpio。
配置引导加载程序[编辑 | 编辑源代码]
要在启动时解密已加密的根分区,需配置引导加载程序设置如下内核参数:
cryptdevice=UUID=device-UUID:cryptlvm root=/dev/MyVolGroup/root
若使用的是[[sd-encrypt],则应该添加如下内核参数:
rd.luks.name=device-UUID=cryptlvm root=/dev/MyVolGroup/root
device-UUID
指的是LUKS超级块的UUID,在本例中为/dev/sda1
。详见块设备持久化命名。
若使用的是dracut,将需要更复杂的一系列参数:
kernel_cmdline="rd.luks.uuid=luks-deviceUUID rd.lvm.lv=MyVolGroup/root rd.lvm.lv=MyVolGroup/swap root=/dev/mapper/MyVolGroup-root rootfstype=ext4 rootflags=rw,relatime"
详见dm-crypt/System configuration#Kernel parameters。
在LVM上配置LUKS[编辑 | 编辑源代码]
为了在LVM上层设置加密,首先要设置LVM卷,将它们用作加密分区的基础。在这种配置方案下,可同时创建加密及未加密的分区。
下例展示了在LVM上配置LUKS的方法,并且对不同逻辑卷采用了不同的加密方式:/home卷采用keyfile,/tmp
、/swap
则采用临时随机密钥。采用临时随机密钥是基于安全性考虑,每次重启后加密都将重新配置,使得临时数据不会在重启后保留。若你熟悉LVM,可根据自己的方案修改下述案例。
若要创建横跨多个磁盘的逻辑卷,或要将已有逻辑卷扩展到其它已配置好的磁盘上,详见dm-crypt/Specialties#Expanding LVM on multiple disks。注意调整逻辑卷的大小后,LUKS加密容器的大小也需要调整。
准备磁盘[编辑 | 编辑源代码]
分区方案如下:
+----------------+-------------------------------------------------------------------------------------------------+ | boot分区 | dm-crypt plain加密的分区 | LUKS2加密的分区 | LUKS2加密的分区 | | | | | | | /boot | [SWAP] | / | /home | | | | | | | | /dev/mapper/swap | /dev/mapper/root | /dev/mapper/home | | |_ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _|_ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _|_ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _| | | 逻辑卷 1 | 逻辑卷 2 | 逻辑卷 3 | | | /dev/MyVolGroup/cryptswap | /dev/MyVolGroup/cryptroot | /dev/MyVolGroup/crypthome | | |_ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _|_ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _|_ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _| | | | | /dev/sda1 | /dev/sda2 | +----------------+-------------------------------------------------------------------------------------------------+
按照[dm-crypt/Drive preparation#dm-crypt wipe on an empty disk or partition]]随机擦除/dev/sda2
。
准备逻辑卷[编辑 | 编辑源代码]
# pvcreate /dev/sda2 # vgcreate MyVolGroup /dev/sda2 # lvcreate -L 32G -n cryptroot MyVolGroup # lvcreate -L 500M -n cryptswap MyVolGroup # lvcreate -L 500M -n crypttmp MyVolGroup # lvcreate -l 100%FREE -n crypthome MyVolGroup
# cryptsetup luksFormat /dev/MyVolGroup/cryptroot # cryptsetup open /dev/MyVolGroup/cryptroot root # mkfs.ext4 /dev/mapper/root # mount /dev/mapper/root /mnt
关于加密选项的更多信息,详见dm-crypt/Device encryption#Encryption options for LUKS mode。
注意/home
会在#加密/home逻辑卷步骤中进行。
open
操作。准备boot分区[编辑 | 编辑源代码]
# dd if=/dev/zero of=/dev/sda1 bs=1M status=progress # mkfs.ext4 /dev/sda1 # mount --mkdir /dev/sda1 /mnt/boot
配置mkinitcpio[编辑 | 编辑源代码]
确保安装了lvm2包,之后在mkinitcpio.conf中添加keyboard
、encrypt
、lvm2
钩子。若要使用非标准的键盘布局,还需添加keymap
钩子;若要修改控制台字体,还需添加consolefont
钩子:
HOOKS=(base udev autodetect modconf kms keyboard keymap consolefont block lvm2 encrypt filesystems fsck)
若使用基于sysyemd的initramfs,需添加的应为keyboard
和sd-encrypt
钩子。若要使用非标准键盘布局或修改控制台字体,还需添加sd-vconsole
钩子:
HOOKS=(base systemd autodetect modconf kms keyboard sd-vconsole block sd-encrypt lvm2 filesystems fsck)
在保存上述修改后,重新生成initramfs。对于可能需要的其它钩子,详见dm-crypt/System configuration#mkinitcpio。
配置引导加载程序[编辑 | 编辑源代码]
要在启动时解密已加密的根分区,需配置引导加载程序设置如下内核参数:
cryptdevice=UUID=device-UUID:root root=/dev/mapper/root
若使用的是[[sd-encrypt],则应该添加如下内核参数:
rd.luks.name=device-UUID=root root=/dev/mapper/root
device-UUID
指的是LUKS超级块的UUID,在本例中为/dev/MyVolGroup/cryptroot
。详见块设备持久化命名。
若使用的是dracut,将需要更复杂的一系列参数:
kernel_cmdline="rd.luks.uuid=luks-deviceUUID rd.lvm.lv=MyVolGroup/root rd.lvm.lv=MyVolGroup/swap root=/dev/mapper/MyVolGroup-root rootfstype=ext4 rootflags=rw,relatime"
详见dm-crypt/System configuration#Kernel parameters。
配置fstab和crypttab[编辑 | 编辑源代码]
解密设备和挂载其上的文件系统分别需要编辑crypttab、fstab。下列配置将在每次重启时重新加密临时文件系统:
/etc/crypttab
swap /dev/MyVolGroup/cryptswap /dev/urandom swap,cipher=aes-xts-plain64,size=256 tmp /dev/MyVolGroup/crypttmp /dev/urandom tmp,cipher=aes-xts-plain64,size=256
/etc/fstab
/dev/mapper/root / ext4 defaults 0 1 /dev/sda1 /boot ext4 defaults 0 2 /dev/mapper/tmp /tmp tmpfs defaults 0 0 /dev/mapper/swap none swap sw 0 0
加密/home逻辑卷[编辑 | 编辑源代码]
由于本方案使用LVM作为第一层,em-crypt作为第二层,每个加密的逻辑卷都需要单独设置加密。此外,和临时文件系统采用的临时随机密钥加密不同,/home
对应的逻辑卷显然应该采用持久不变的密钥。下列命令假定你已经重启进入了安装好的系统。若没有,需要调整相应路径。
为了在系统启动时无需输入两次密码,创建了keyfile:
# mkdir -m 700 /etc/luks-keys # dd if=/dev/random of=/etc/luks-keys/home bs=1 count=256 status=progress
该逻辑卷将采用上述keyfile加密:
# cryptsetup luksFormat -v /dev/MyVolGroup/crypthome /etc/luks-keys/home # cryptsetup -d /etc/luks-keys/home open /dev/MyVolGroup/crypthome home # mkfs.ext4 /dev/mapper/home # mount /dev/mapper/home /home
/etc/crypttab
home /dev/MyVolGroup/crypthome /etc/luks-keys/home
/etc/fstab
/dev/mapper/home /home ext4 defaults 0 2
在软RAID上配置LUKS[编辑 | 编辑源代码]
下例基于真实的工作站笔记本设置,该工作站有两块大小相同的SSD及一块大容量机械硬盘。该配置对所有磁盘采用LUKS1加密(包括/boot
),并将两块SSD组成了RAID0阵列。系统启动时,在GRUB中输入正确的密码后,将用解密出的keyfile解密所有分区。
该方案采用的分区配置非常简单,组成的RAID阵列挂载在/
(没有单独的/boot
分区),解密后的机械硬盘挂载在/data
。
注意对于该方案,定期的备份非常重要。若两块SSD中的任何一块损坏,整个RAID阵列中的数据将在实践上不可能被恢复。若对硬件错误的容忍能力对你来说非常重要,应选择其它的RAID级别。
The encryption is not deniable in this setup.
在之后的命令中,采用了如下块设备设置:
/dev/sda = 第一块SSD /dev/sdb = 第二块SSD /dev/sdc = 机械硬盘
+---------------------+---------------------------+---------------------------+ +---------------------+---------------------------+---------------------------+ +---------------------------+ | BIOS boot分区 | EFI系统分区 | LUKS1加密的分区 | | BIOS 启动分区 | EFI系统分区 | LUKS1加密的分区 | | LUKS2加密的分区 | | | | | | | | | | | | | /efi | / | | | /efi | / | | /data | | | | | | | | | | | | | | /dev/mapper/root | | | | /dev/mapper/root | | | | +---------------------------+---------------------------+ | +---------------------------+---------------------------+ | | | | RAID1 array (part 1 of 2) | RAID0 array (part 1 of 2) | | | RAID1 array (part 2 of 2) | RAID0 array (part 2 of 2) | | | | | | | | | | | | | | | /dev/md/ESP | /dev/md/root | | | /dev/md/ESP | /dev/md/root | | /dev/mapper/data | | +---------------------------+---------------------------+ | +---------------------------+---------------------------+ +---------------------------+ | /dev/sda1 | /dev/sda2 | /dev/sda3 | | /dev/sdb1 | /dev/sdb2 | /dev/sdb3 | | /dev/sdc1 | +---------------------+---------------------------+---------------------------+ +---------------------+---------------------------+---------------------------+ +---------------------------+
在之后的操作中,确保将上述对应的设备替换为你的配置。
准备磁盘[编辑 | 编辑源代码]
在创建任何分区前,应先查看dm-crypt/Drive preparation,明确安全地擦除整个磁盘的重要性及相关方法。
对于具有GPT分区表的BIOS启动的系统,需创建一个大小为1MiB的BIOS启动分区,GRUB将在此存储第二阶段的BIOS引导加载程序。不要挂载创建的该分区。
对于UEFI启动的系统,需创建大小适当的EFI 系统分区,之后该分区将被挂载到/efi
。
对于设备上的剩余空间,创建一个用于"Linux RAID"的分区(本例中为/dev/sda3
)。对于MBR分区表,选择fd
分区type ID;对于GPT分区表,选择A19D880F-05FC-4D3B-A006-743F0F84911E
分区type GUID。
一旦在/dev/sda
上创建好了分区,可使用下列命令将它们的配置克隆到/dev/sdb
上:
# sfdisk -d /dev/sda > sda.dump # sfdisk /dev/sdb < sda.dump
对于机械硬盘,创建单个占据整个磁盘的Linux分区(/dev/sdc1
)。
构建RAID阵列[编辑 | 编辑源代码]
创建用于两块SSD的RAID阵列:
- 对于两块SSD,EFI系统分区必须能在每块SSD上单独访问,因此ESP只能被存放在RAID1上。
- RAID superblock必须通过
--metadata=1.0
选项放置在EFI系统分区的后部,否则固件将不能访问EFI系统分区。
# mdadm --create --verbose --level=1 --metadata=1.0 --raid-devices=2 /dev/md/ESP /dev/sda2 /dev/sdb2
对于根分区,本例采用了RAID0。可根据实际偏好与需要选择其它RAID级别。
# mdadm --create --verbose --level=0 --metadata=1.2 --raid-devices=2 /dev/md/root /dev/sda3 /dev/sdb3
准备块设备[编辑 | 编辑源代码]
dm-crypt/Drive preparation中介绍了如何使用/dev/zero
与随机密钥加密来用随机数据擦除设备。此外,也可使用dd
与/dev/random
或/dev/urandom
,不过比前者慢得多。
# cryptsetup open --type plain /dev/md/root container --key-file /dev/random # dd if=/dev/zero of=/dev/mapper/container bs=1M status=progress # cryptsetup close container
对本例中的机械硬盘(/dev/sdc1
)也需重复上述操作。
为/dev/md/root
设置加密:
cryptsetup luksFormat --type luks1
)。# cryptsetup -y -v luksFormat --type luks1 /dev/md/root # cryptsetup open /dev/md/root root # mkfs.ext4 /dev/mapper/root # mount /dev/mapper/root /mnt
对机械硬盘,相似的:
# cryptsetup -y -v luksFormat /dev/sdc1 # cryptsetup open /dev/sdc1 data # mkfs.ext4 /dev/mapper/data # mount --mkdir /dev/mapper/data /mnt/data
对于UEFI启动的系统,需设置EFI系统分区:
# mkfs.fat -F32 /dev/md/ESP # mount --mkdir /dev/md/ESP /mnt/efi
配置GRUB[编辑 | 编辑源代码]
对于本例中通过LUKS1加密的系统,编辑/etc/default/grub
以配置GRUB:
GRUB_CMDLINE_LINUX="cryptdevice=/dev/md/root:root" GRUB_ENABLE_CRYPTODISK=y
若要在较新的设备上使用USB键盘在GRUB中输入密码,需启用固件中的”legacy USB support“或在/etc/default/grub
中添加:
GRUB_TERMINAL_INPUT="usb_keyboard" GRUB_PRELOAD_MODULES="usb usb_keyboard ohci uhci ehci"
否则在提示输入LUKS密码时可能无法使用键盘。详见dm-crypt/System configuration#Kernel parameters、GRUB#Encrypted /boot。
将GRUB安装到两块SSD上(事实上,仅安装到/dev/sda
也是可行的):
# grub-install --target=i386-pc /dev/sda # grub-install --target=i386-pc /dev/sdb # grub-install --target=x86_64-efi --efi-directory=/efi --bootloader-id=GRUB # grub-mkconfig -o /boot/grub/grub.cfg
配置keyfile[编辑 | 编辑源代码]
下面进行的步骤可避免在启动时需输入两次密码的情况(第一次发生在GRUB解密LUKS1设备,第二次发生在initramfs接管系统时)。首先要创建一个keyfile,然后将该keyfile加入到initramfs镜像中以便encrypt钩子解密根分区。详见dm-crypt/Device encryption#With a keyfile embedded in the initramfs。
- 创建keyfile并向其中添加
/dev/md/root
的密码。 - 创建另一个keyfile以便于在启动时解密机械硬盘(
/dev/sdc1
)。为了便于以后可能的需要恢复的情况,不要移动/删除上述keyfile。接下来编辑/etc/crypttab
以在启动时解密机械硬盘。详见dm-crypt/System configuration#Unlocking with a keyfile。
配置系统[编辑 | 编辑源代码]
配置fstab以挂载根分区、data分区、EFI系统分区:
/dev/mapper/root / ext4 rw,noatime 0 1 /dev/mapper/data /data ext4 defaults 0 2 /dev/md/ESP /efi vfat rw,relatime,codepage=437,iocharset=iso8859-1,shortname=mixed,utf8,tz=UTC,errors=remount-ro 0 2
保存RAID配置:
# mdadm --detail --scan >> /etc/mdadm.conf
编辑mkinitcpio.conf以包含keyfile,并添加所需的钩子:
FILES=(/crypto_keyfile.bin) HOOKS=(base udev autodetect modconf kms keyboard keymap consolefont block mdadm_udev encrypt filesystems fsck)
详见dm-crypt/System configuration#mkinitcpio。
Plain dm-crypt[编辑 | 编辑源代码]
与LUKS不同,dm-crypt的plain模式无需在加密设备前附加LUKS头,本方案利用这个特点在未分区的硬盘上设置加密的系统。对于他人,该硬盘将无法与用随机数据填充的硬盘区分,即所谓可抵赖加密。详见wikipedia:Disk encryption#Full disk encryption。
注意若不需要全盘(可抵赖)加密,无论对于加密整个系统还是加密某个分区,之前介绍的使用LUKS的方法都更好。在plain 模式下将无法使用多密码/keyfile的密钥管理、就地重新加密一个设备等LUKS的特色功能。
因为无需依赖单个加密的master-key,可避免因master-key损坏造成单点故障,plain dm-crypt加密对数据错误的容忍性较LUKS强。然而,使用plain模式也需要更多的手动配置以达到与LUKS相同的加密强度(详见Data-at-rest encryption#Cryptographic metadata)。此外,若担心dm-crypt/Specialties#Discard/TRIM support for solid state drives (SSD)中提到的问题,也可使用plain模式。
- 通过cryptsetup的
--header
选项使用分离的LUKS头。注意若使用该方案则不能使用标准的"encrypt"钩子,不过可对该钩子进行修改。 - tcplay 在headerless加密模式下提供了PBKDF2函数。
本方案使用了两个U盘:
- 一个用于引导设备,同时存储用于解密plain加密设备的配置选项;
- 一个用于keyfile,将存储为raw bits,故对于不不知道配置方案的攻击者而言,只能看到一串仿佛随机的数据而非可见的普通文件。详见Wikipedia:Security through obscurity。要准备keyfile,见dm-crypt/Device encryption#Keyfiles。
磁盘布局如下:
+----------------------+----------------------+----------------------+ +----------------+ +----------------+ | 逻辑卷 1 | 逻辑卷 2 | 逻辑卷 3 | | 启动设备 | | 密钥 | | | | | | | | 存储设备 | | / | [SWAP] | /home | | /boot | | (在本例中 | | | | | | | | 未分区) | | /dev/MyVolGroup/root | /dev/MyVolGroup/swap | /dev/MyVolGroup/home | | /dev/sdb1 | | /dev/sdc | |----------------------+----------------------+----------------------| |----------------| |----------------| | /dev/sda 使用plain模式加密并配置LVM | | U盘 1 | | U盘 2 | +--------------------------------------------------------------------+ +----------------+ +----------------+
- 也可以使用单个U盘:
- 可以将keyfile放到U盘(/dev/sdb)的另一个分区(/dev/sdb2)上。
- 或直接将keyfile复制到initramfs上例如,对于keyfile
/etc/keyfile
,可在/etc/mkinitcpio.conf
中设置FILES=(/etc/keyfile)
。要使encrypt
钩子读取initramfs镜像中的keyfile,在文件名前指定rootfs:
前缀,如cryptkey=rootfs:/etc/keyfile
。
- 另一种配置方法是结合密码与良好的熵源。
准备磁盘[编辑 | 编辑源代码]
向解密后的映射(mapped)设备中填充随机数据是非常重要的。尤其是在本例中。
详见dm-crypt/准备磁盘、[dm-crypt/准备磁盘#dm-crypt 专用方案]]
准备分区(除boot)[编辑 | 编辑源代码]
在本例中,使用/dev/sda
,并使用aes-xts cipher,512位的key size,使用keyfile加密:
# cryptsetup --cipher=aes-xts-plain64 --offset=0 --key-file=/dev/sdc --key-size=512 open --type plain /dev/sda cryptlvm
与使用LUKS不同,每次要解密设备时必须使用上述“完整”的命令,故一定要牢记cipher、key file相关信息。
检查有关/dev/mapper/cryptlvm
的映射条目已建立:
# fdisk -l
接下来,在解密后产生的映射设备上设置LVM逻辑卷,详见Install Arch Linux on LVM:
# pvcreate /dev/mapper/cryptlvm # vgcreate MyVolGroup /dev/mapper/cryptlvm # lvcreate -L 32G MyVolGroup -n root # lvcreate -L 10G MyVolGroup -n swap # lvcreate -l 100%FREE MyVolGroup -n home
格式化并挂载上述分区并激活swap,详见文件系统#创建文件系统。
# mkfs.ext4 /dev/MyVolGroup/root # mkfs.ext4 /dev/MyVolGroup/home # mount /dev/MyVolGroup/root /mnt # mount --mkdir /dev/MyVolGroup/home /mnt/home # mkswap /dev/MyVolGroup/swap # swapon /dev/MyVolGroup/swap
准备boot分区[编辑 | 编辑源代码]
可将U盘默认的vfat分区用作/boot
分区。但若要手动分区,仅需配置一个200MiB的分区用于/boot
。通过分区工具进行分区。
为/boot
创建文件系统:
# mkfs.fat -F32 /dev/sdb1 # mount --mkdir /dev/sdb1 /mnt/boot
配置mkinitcpio[编辑 | 编辑源代码]
确保安装了lvm2包,之后在mkinitcpio.conf中添加keyboard
、encrypt
、lvm2
钩子。若要使用非标准的键盘布局,还需添加keymap
钩子;若要修改控制台字体,还需添加consolefont
钩子:
HOOKS=(base udev autodetect modconf kms keyboard keymap consolefont block encrypt lvm2 filesystems fsck)
在保存上述修改后,重新生成initramfs。对于可能需要的其它钩子,详见dm-crypt/System configuration#mkinitcpio。
配置引导加载程序[编辑 | 编辑源代码]
要在加密的根分区上启动,需通过引导加载程序设置如下内核参数(注意512位对应64字节):
cryptdevice=/dev/disk/by-id/disk-ID-of-sda:cryptlvm cryptkey=/dev/disk/by-id/disk-ID-of-sdc:0:64 crypto=:aes-xts-plain64:512:0:
disk-ID-of-disk
表示相应磁盘的id,详见[[
块设备持久化命名]]。
对于其它可能需要的内核参数,详见dm-crypt/System configuration#Kernel parameters。
/boot
分区所在的U盘上。
对于BIOS启动的系统:
# grub-install --target=i386-pc --recheck /dev/sdb
对于UEFI启动的系统:
# grub-install --target=x86_64-efi --efi-directory=/boot --removable
安装完成后的操作[编辑 | 编辑源代码]
或许希望在系统启动后移除启动用的U盘。由于/boot
分区并不常用,可在/etc/fstab
的相应行中添加noauto
选项:
/etc/fstab
# /dev/sdb1 /dev/sdb1 /boot vfat noauto,rw,noatime 0 2
然而,若需要更新initramfs、内核、引导加载程序使用的相关文件时,必须挂载/boot
分区。由于fstab
中存在相关条目,只需:
# mount /boot
加密的boot分区(GRUB)[编辑 | 编辑源代码]
本配置使用与#在LUKS上配置LVM相同的分区结构和配置,除了使用了GRUB引导加载程序,其能够从LVM逻辑卷和LUKS1加密的/boot
上启动系统。另见GRUB#加密的/boot
本例中的磁盘结构如下:
+---------------------+----------------------+----------------------+----------------------+----------------------+ | BIOS boot分区 | EFI系统分区 | 逻辑卷 1 | 逻辑卷 2 | 逻辑卷 3 | | | | | | | | | /efi | / | [SWAP] | /home | | | | | | | | | | /dev/MyVolGroup/root | /dev/MyVolGroup/swap | /dev/MyVolGroup/home | | /dev/sda1 | /dev/sda2 |----------------------+----------------------+----------------------+ | 未加密 | 未加密 | /dev/sda3 使用LUKS1加密并配置LVM | +---------------------+----------------------+--------------------------------------------------------------------+
- 所有案例仅用作示例。当然也可以与其它案例结合使用。此外,可参见#在LUKS上配置LVM中介绍的两种变体。
- 可以使用cryptbootAUR中的
cryptboot
脚本简化加密boot分区的管理(包括挂载、卸载、软件包更新)。此外,该脚本与UEFI安全启动配合还可抵御Evil Maid攻击。更多信息及缺陷见cryptboot project。
准备磁盘[编辑 | 编辑源代码]
在创建任何分区前,应先查看dm-crypt/Drive preparation,明确安全地擦除整个磁盘的重要性及相关方法。
对于BIOS启动,GUID分区表的系统,需创建大小1MiB的BIOS启动,GUID分区表的系统|BIOS boot分区用于GRUB存储第二阶段引导加载程序。不要挂载该分区。BIOS启动,MBR分区表的系统无须上述操作。
对于UEFI启动的系统,创建大小合适的[EFI 系统分区]],其会被挂载到/efi
。
创建一个类型为8309
的分区用于加密的LVM。
创建LUKS加密容器:
cryptsetup luksFormat --type luks1
)。# cryptsetup luksFormat --type luks1 /dev/sda3
关于加密选项的更多信息,在执行上述命令前参见dm-crypt/设备加密#LUKS 模式的加密选项。
分区结构应与下述相似:
# gdisk -l /dev/sda
... Number Start (sector) End (sector) Size Code Name 1 2048 4095 1024.0 KiB EF02 BIOS boot partition 2 4096 1130495 550.0 MiB EF00 EFI System 3 1130496 68239360 32.0 GiB 8309 Linux LUKS
打开LUKS加密容器:
# cryptsetup open /dev/sda3 cryptlvm
解密后的容器将在/dev/mapper/cryptlvm
。
准备逻辑卷[编辑 | 编辑源代码]
本例中逻辑卷的结构与#在LUKS上配置LVM中的相同。因此,只需执行#准备逻辑卷中的步骤并按需调整。
若要使用UEFI启动,为EFI 系统分区创建/efi
挂载点以与grub-install
兼容。挂载该分区:
# mount --mkdir /dev/sda2 /mnt/efi
现在,在/mnt
下应挂载有如下分区与逻辑卷:
$ lsblk
NAME MAJ:MIN RM SIZE RO TYPE MOUNTPOINT sda 8:0 0 200G 0 disk ├─sda1 8:1 0 1M 0 part ├─sda2 8:2 0 550M 0 part /mnt/efi └─sda3 8:3 0 100G 0 part └─cryptlvm 254:0 0 100G 0 crypt ├─MyVolGroup-swap 254:1 0 8G 0 lvm [SWAP] ├─MyVolGroup-root 254:2 0 32G 0 lvm /mnt └─MyVolGroup-home 254:3 0 60G 0 lvm /mnt/home
配置mkinitcpio[编辑 | 编辑源代码]
确保安装了lvm2包,之后在mkinitcpio.conf中添加keyboard
、encrypt
、lvm2
钩子。若要使用非标准的键盘布局,还需添加keymap
钩子;若要修改控制台字体,还需添加consolefont
钩子:
HOOKS=(base udev autodetect modconf kms keyboard keymap consolefont block encrypt lvm2 filesystems fsck)
若使用基于sysyemd的initramfs,需添加的应为keyboard
和sd-encrypt
钩子。若要使用非标准键盘布局或修改控制台字体,还需添加sd-vconsole
钩子:
HOOKS=(base systemd autodetect modconf kms keyboard sd-vconsole block sd-encrypt lvm2 filesystems fsck)
在保存上述修改后,重新生成initramfs。对于可能需要的其它钩子,详见dm-crypt/System configuration#mkinitcpio。
配置GRUB[编辑 | 编辑源代码]
配置GRUB以从LUKS1加密的/boot
启动:
/etc/default/grub
GRUB_ENABLE_CRYPTODISK=y
设置内核参数以便initramfs解密加密的根分区。
若使用encrypt
钩子:
/etc/default/grub
GRUB_CMDLINE_LINUX="... cryptdevice=UUID=device-UUID:cryptlvm ..."
若使用sd-encrypt钩子:
/etc/default/grub
GRUB_CMDLINE_LINUX="... rd.luks.name=device-UUID=cryptlvm ..."
详见dm-crypt/System configuration#Kernel parameters、GRUB#加密的/boot。device-UUID
应为LUKS superblock的UUID(在本例中为/dev/sda3
,该分区包含了含有根分区的LVM)。详见块设备持久化命名。
对于UEFI启动的系统,安装GRUB 到挂载的EFI系统分区:
# grub-install --target=x86_64-efi --efi-directory=/efi --bootloader-id=GRUB --recheck
对于BIOS启动的系统,安装GRUB:
# grub-install --target=i386-pc --recheck /dev/sda
生成GRUB配置文件:
# grub-mkconfig -o /boot/grub/grub.cfg
若上述所有命令都成功执行,在下次重启后GRUB应提示输入密码解锁/dev/sda3
。
避免需输入两次密码[编辑 | 编辑源代码]
尽管在上述操作后,GRUB会要求输入密码解密LUKS1加密的分区,该信息并未传递到initramfs。因此,需要在启动时输入两次密码:一次用于GRUB,一次用于initramfs。
本节介绍了配置启动过程仅需输入一次密码(在GRUB中)的额外操作。可通过在initramfs中嵌入密钥文件达成。
首先,创建一个密钥文件,将其添加到LUKS key中:
# dd bs=512 count=4 if=/dev/random of=/root/cryptlvm.keyfile iflag=fullblock # chmod 000 /root/cryptlvm.keyfile # cryptsetup -v luksAddKey /dev/sda3 /root/cryptlvm.keyfile
将keyfile添加到initramfs镜像:
/etc/mkinitcpio.conf
FILES=(/root/cryptlvm.keyfile)
重新生成initramfs镜像。 之后,对keyfile进行安全保护:
# chmod 600 /boot/initramfs-linux*
设置如下内核参数以通过keyfile解密LUKS分区。
若使用encrypt
钩子:
GRUB_CMDLINE_LINUX="... cryptkey=rootfs:/root/cryptlvm.keyfile"
若使用sd-encrypt钩子:
GRUB_CMDLINE_LINUX="... rd.luks.key=device-UUID=/root/cryptlvm.keyfile"
若由于某些原因keyfile解密boot分区失败,systemd会要求手动输入密码。若密码输入正确,系统启动将正常进行。
Btrfs子卷,带swap[编辑 | 编辑源代码]
译注:原文采用的部分方法已过时,从内核版本5.0开始,btrfs已支持swapfile。
本例通过在LUKS1上使用Btrfs来加密整个系统,并使用Btrfs子卷来模拟分区。
对于UEFI启动的设备,需要EFI 系统分区(ESP),不能加密该分区。/boot
则可以与/
为同一个子卷并配置加密。在本例中,/dev/sda1
作为ESP并被挂载到/efi
。/boot
与/
在同一个分区(/dev/sda2
)上。
由于/boot
位于LUKS1加密的/
上,引导加载程序必须使用GRUB,因为只有GRUB可加载相应的模块(如crypto.mod,cryptodisk.mod和luks.mod)以解密/boot
。
此外, 本例还展示了一个可选的,以plain模式加密的swap分区:
+----------------------+----------------------+----------------------+ | EFI系统分区 | 根分区 | Swap分区 | | 未加密 | LUKS1模式加密 | plain模式加密 | | | | | | /efi | / | [SWAP] | | /dev/sda1 | /dev/sda2 | /dev/sda3 | +----------------------+----------------------+----------------------+
准备磁盘[编辑 | 编辑源代码]
在创建任何分区前,应先查看dm-crypt/Drive preparation,明确安全地擦除整个磁盘的重要性及相关方法。若使用UEFI,需创建大小合适的EFI 系统分区。若要创建加密的swap分区,可以先建立分区结构,但因为将使用plain模式加密该分区,不要将其直接标记为swap分区。
创建需要的分区,至少要创建根分区(本例中为/dev/sda2
)。详见分区。
准备根分区[编辑 | 编辑源代码]
创建LUKS容器[编辑 | 编辑源代码]
cryptsetup luksFormat --type luks1
)。按照dm-crypt/设备加密#使用 LUKS 模式加密设备中的步骤对/dev/sda2
设置LUKS。在设置之前,可以在dm-crypt/设备加密#LUKS 模式的加密选项查看可用的加密选择。
解锁LUKS容器[编辑 | 编辑源代码]
按照dm-crypt/设备加密#使用设备映射器解锁/映射 LUKS分区中的步骤解密并映射LUKS容器。
格式化映射后的设备[编辑 | 编辑源代码]
按照Btrfs#单一设备上的文件系统中的步骤格式化映射后的设备。在格式化时,注意/dev/分区名
应为映射后的设备(在本例中,即/dev/mapper/root
),而非/dev/sda2
。
挂载映射后的设备[编辑 | 编辑源代码]
最后,挂载已格式化的映射后的设备(在本例中,即/dev/mapper/root
)到/mnt
。
创建btrfs子卷[编辑 | 编辑源代码]
布局[编辑 | 编辑源代码]
子卷将用于模拟分区,但除此之外还将创建嵌套子卷。如下是本例配置的部分展示:
subvolid=5 | ├── @ -| | 包含如下目录: | ├── /usr | ├── /bin | ├── /.snapshots | ├── ... | ├── @home ├── @snapshots ├── @var_log └── @...
本例遵循了Snapper#Suggested filesystem layout的配置,有助于与Snapper配合使用。有关子卷布局,还可参考Btrfs Wiki SysadminGuide#Layout。
为第一次挂载准备子卷[编辑 | 编辑源代码]
依据惯例,对于用作挂载点的子卷,本例在子卷名称前加上@
。@
对应于挂载到/
的子卷。
按照Btrfs#创建子卷,在/mnt/@
、/mnt/@snapshots
和/mnt/@home
创建子卷。
此外,根据你的配置,现在还应创建用作挂载点的其它子卷。
创建例外子卷[编辑 | 编辑源代码]
对于在为/
制作快照时欲排除的目录,可创建相应子卷。例如,可能需要排除/var/cache/pacman/pkg
目录。这些子卷会嵌套在@
子卷下,但根据不同的设置,也可能已在上一步创建,并与@
同级。
常见的需创建例外子卷的有:/var/abs
、/var/tmp
、/srv
。
挂载顶层子卷[编辑 | 编辑源代码]
卸载已挂载到{{ic|/mnt}的根文件系统。
之后,通过subvol=
挂载选项将创建的用于/
的@
子卷挂载到/mnt
。假设解密产生的映射设备为root
,执行的命令与与如下的相似:
# mount -o compress=zstd,subvol=@ /dev/mapper/root /mnt
详见Btrfs#挂载子卷。
此外,将其它子卷挂载到相应的位置:@home
挂载到/mnt/home
,@snapshots
挂载到/mnt/.snapshots
。
挂载ESP[编辑 | 编辑源代码]
若之前准备了EFI系统分区,创建相应挂载点并挂载。
/efi
不是btrfs文件系统的一部分,btrfs快照将忽略它。在执行pacstrap安装步骤时,除base包元软件包外,还必须安装btrfs-progs包。
配置mkinitcpio[编辑 | 编辑源代码]
创建keyfile[编辑 | 编辑源代码]
为了允许GRUB解密LUKS分区,同时无需输入两次密码,本例将keyfile嵌入initramfs中。执行dm-crypt/设备加密#在 initramfs 中嵌入密钥文件中的步骤。在luksAddKey步骤时,确保目标为/dev/sda2
。
编辑mkinitcpio.conf[编辑 | 编辑源代码]
按上述步骤创建、添加并嵌入keyfile后,将encrypt
连同其它需要的钩子一同加入mkinitcpio.conf。
在保存上述修改后,重新生成initramfs。对于可能需要的其它钩子,详见dm-crypt/System configuration#mkinitcpio。
配置引导加载程序[编辑 | 编辑源代码]
安装GRUB,并按GRUB#额外的参数,GRUB#加密的/boot和dm-crypt/System configuration#Using encrypt hook中的步骤编辑/etc/default/grub
,注意要遵循有关加密的根分区、加密的boot分区两方面的内容。最后,生成GRUB配置文件。注意还需按照Btrfs#挂载子卷为根挂载点中的要求设定内核参数。
配置swap[编辑 | 编辑源代码]
若创建了用于加密的swap的分区,按照dm-crypt/Swap encryption中的步骤进行配置。
Root on ZFS[编辑 | 编辑源代码]
要将dm-crypt与ZFS一同使用,参见ZFS#Encryption in ZFS using dm-crypt。
此外,ZFS具有原生加密支持,可用于加密根分区(不包含引导加载程序和文件系统元数据)。详见:
对于UEFI启动的系统,在安装后可通过安全启动验证引导加载程序。
使用TPM2和安全启动的简单根分区加密[编辑 | 编辑源代码]
本例介绍了通过dm-crypt + LUKS以及TPM2、安全启动配置的全系统加密。
仅有包含统一内核镜像和systemd-boot的ESP是未加密的。可将统一内核镜像和systemd-boot签名并与安全启动一同使用,TPM会检测安全启动状态,所以若安全启动被关闭或其密钥数据库被修改,加密的分区将不会被自动解密。这提供了合理的安全启动链(与Windows的BitLocker,macOS的FileVault相似)。
本例中,分区按照systemd#GPT分区自动挂载的要求创建,故无需配置fstab或crypttab。
+-----------------------+------------------------+-----------------------+ | ESP | 根分区 | | 未加密 | 加密 | | | | | /efi | / | | | | | | /dev/mapper/root | | |------------------------------------------------| | /dev/sda1 | /dev/sda2 | +-----------------------+------------------------+-----------------------+
对硬盘分区[编辑 | 编辑源代码]
在创建任何分区前,应先查看dm-crypt/Drive preparation,明确安全地擦除整个磁盘的重要性及相关方法。
此外,确保硬盘报告的扇区大小是正确的,详见sector size。
确保分区表类型为GPT。之后创建相关分区(如/dev/sda2
用于/
,/dev/sda1
用于/efi
)。详见分区。
- 要创建ESP并设置正确的GUID,详见EFI 系统分区#GPT 分区磁盘。
- 对于LUKS加密的根分区,若使用gdisk可设置分区类型为
8304
;若使用fdisk可设置分区类型为23 Linux root (x86-64)
(若架构为x86-64)。详见}gPT fdisk#Partition type。
格式化分区[编辑 | 编辑源代码]
根分区[编辑 | 编辑源代码]
下列命令创建并挂载加密的根分区,并通过TPM存储密钥。详见dm-crypt/加密非root文件系统(只要正确配置了mkinitcpio、引导加载程序其中的内容也可用于加密根分区)、Trusted Platform Module#systemd-cryptenroll。
若要使用非默认的加密选项(如cipher、key length),或要求不使用基于TPM的解密,在执行下列命令前参见encryption options。
创建所需的LUKS卷(创建时可使用空密码,之后该密码将被擦除):
# cryptsetup luksFormat /dev/sda2 # cryptsetup open /dev/sda2 root # mkfs.ext4 /dev/mapper/root
ESP[编辑 | 编辑源代码]
挂载文件系统[编辑 | 编辑源代码]
在安装指南#挂载分区步骤时,注意应挂载解密后产生的映射设备,而不是对应的物理分区,即:
# mount /dev/mapper/root /mnt # mount --mkdir /dev/sda1 /mnt/efi
继续按照指南进行安装,直到安装指南#关于 initramfs。可以跳过安装指南#生成 fstab 文件步骤。
Initramfs[编辑 | 编辑源代码]
要构建可用的基于systemd的initramfs,修改mkinitcpio.conf中的HOOKS=
行:
HOOKS=(systemd autodetect modconf kms keyboard sd-vconsole block sd-encrypt filesystems fsck)
接下来配置需统一内核镜像,详见统一内核镜像#mkinitcpio。
由于/efi/EFI/Linux
文件夹需在引导加载程序的安装过程中创建,此时不要重新生成initramfs。
引导加载程序[编辑 | 编辑源代码]
通过以下命令安装systemd-boot:
# bootctl install
由mkinitcpio创建的统一内核镜像会被自动识别,故无需手动配置/efi/loader/entries
。
详见systemd-boot#更新 EFI 启动管理器、systemd-boot#启动选单配置。
完成安装[编辑 | 编辑源代码]
首先,[[mkinitcpio#创建和启用镜像|重新生成initramfs],确保镜像成功生成。
安全启动[编辑 | 编辑源代码]
若要启用安全启动,应对引导加载程序和EFI二进制文件进行签名。Unified Extensible Firmware Interface/Secure Boot#Assisted process with sbctl提供了一种简单快捷的方法。
使用TPM[编辑 | 编辑源代码]
在对引导加载程序进行签名并启用安全启动后,可使用TPM来解密LUKS卷。下列命令将移除在格式化LUKS卷时使用的空密码,并创建绑定到TPM PCR 7(默认,UEFI/安全启动#安全启动状态)的密钥。此外,还将创建恢复密钥以在需要时使用。只要启动链未被非法修改,TPM将自动释放密钥。详见Trusted Platform Module#systemd-cryptenroll、systemd-cryptenroll(1)。
# systemd-cryptenroll /dev/sda2 --recovery-key # systemd-cryptenroll /dev/sda2 --wipe-slot=empty --tpm2-device=auto