This document describes my preferred way to install Arch Linux.
-
LUKS allows full disk encryption.
-
LVM (Logical Volume Management) is a more flexible way to set up a hard drive, as it allows partitions to be dynamically resized.
-
UEFI is the modern replacement for Legacy BIOS.
If you are curious as to why I use Arch you can read Why Arch Linux.
Before you begin you must prepare your installation media and make sure your BIOS is configured correctly:
Download the Arch Linux ISO and create a bootable USB drive. The simplest way to create bootable media on Linux is using the dd command:
$ sudo dd bs=4M if=/path_to_arch_.iso of=/dev/sd* && sync
On Mac use Etcher or UNetBootin. On Windows use Rufus.
Hold F12 (or whatever key is used on your system) during startup to access bios. Then...
-
Make sure UEFI is ON. Most modern systems use UEFI, so it's generally on by default.
-
Disable Secure Boot. If secure boot is enabled it must be turned off since Linux boot loaders don't typically have digital signatures. Note that if you intend on running a dual-boot system with Windows and Linux you won't be able to use disk encryption on the partition containing Windows, as it requires secure boot.
-
Disable Fast Startup Mode. If you are dual booting with Windows turn off Fast Startup. This feature puts Windows into hibernation when you power off. Because some systems are still active during hibernation, booting into Linux can cause various nasty problems.
All references to disk nodes in this document are shown as:
/dev/sd*
You will need to change sd*
to the actual node name you want to use on your drive. To get this info use either of these commands:
$ fdisk -l
$ lsblk
Drive nodes might be called "sda", "sdb", etc., or they might be something completely different. On my Dell XPS, for example, they are called "nvme0n1" to indicate they are interfaced via PCIe.
Here we go...
Hold F12 (or whatever key is used on your system) during startup to access startup menu. Select the USB drive and boot into Arch.
The most reliable way is to use a wired connection, as Arch is setup by default to connect to DHCP. However, you can usually get WiFi working by running:
$ wifi-menu
To test your connection:
$ ping -c 3 www.google.com
If the terminal font is too small, which can happen if you have a high res display, then install terminus fonts.
First, update the pacman databases:
$ pacman -Sy
Then install the fonts:
$ pacman -S terminus-font
Update font cache
$ fc-cache -fv
Set the font to a large size:
$ setfont ter-v32b
If you are installing Arch on a previously used hard drive you can remove partitions using fdisk
First, get the drive node name containing the partition(s) you want to remove and run:
$ fdisk /dev/sd*
Then enter "p" for the partition list:
$ p
Then enter "d" to delete:
$ d
You'll be prompted to enter the number corresponding to the partition you want to remove.
To commit the changes enter:
$ w
Optional step if you are using a hard drive with existing data. Here's how to do it using dd:
$ dd if=/dev/urandom of=/dev/sd* status=progress
Or if you're paranoid (and have a couple days to wait) you can use a multi-pass tool like shred.
$ shred -vfz -n 3 /dev/sd*
NOTE: Since we're using LVM we only need two drive partitions: boot and root. The LVM will be created on root later, where we will use the device mapper to define root, home and swap partitions.
First, launch parted on your desired drive node:
$ parted /dev/sd*
Then run the following commands with your particular size values. Sizes can be specified in MiB
, GiB
, or as a percentage:
$ (parted) mklabel gpt
$ (parted) mkpart primary 1MiB 512MiB name 1 boot
$ (parted) set 1 boot on
$ (parted) mkpart primary 512MiB 100% name 2 root
$ (parted) quit
Before we setup our LVM we need to encrypt the root partition we just created. I use AES 256. Note that cryptsetup splits the supplied key in half, so to use AES-256 we set the key size to 512. For more information about LUKS you can visit the Arch Wiki.
$ cryptsetup luksFormat -v -s 512 -h sha512 /dev/sd*
Now let's decrypt it so we can use it.
Note: I'm labeling this partition as "lvm". We will use this label later when we create the LVM.
$ cryptsetup open --type luks /dev/sd* lvm
To verify our "lvm" label we can use:
$ ls /dev/mapper/lvm
$ pvcreate /dev/mapper/lvm
Note: I'm labelling my volume group as "vg". If you use something else, make sure to replace every instance of it, not only in this section, but in the bootloader config section much later.
$ vgcreate vg /dev/mapper/lvm
At minimum we need two volumes. One for swap, the other for root. We can additionally put home on its own volume.
Note: The sizes below can be specified in megabytes (100M) or gigs (10G).
Also the "L" arguments below are case sensitive. The capital L is used when you want to specify a fixed size volume, the lowercase l lets you specify percentages.
$ lvcreate -L 4G vg -n swap
$ lvcreate -L 80G vg -n root
$ lvcreate -l 100%FREE vg -n home
Note: The boot partition is on the non-LVM partition, so use the disk node you specified when you created that partition.
$ mkfs.vfat -F32 /dev/sd*
$ mkfs.ext4 /dev/mapper/vg-root
$ mkfs.ext4 /dev/mapper/vg-home
$ mkswap /dev/mapper/vg-swap
We need to create a couple directories while we're at it.
$ mount /dev/mapper/vg-root /mnt
$ mkdir /mnt/home
$ mount /dev/mapper/vg-home /mnt/home
$ mkdir /mnt/boot
$ mount /dev/sd* /mnt/boot
$ swapon -s /dev/mapper/vg-swap
Before we download the Arch packages we should rank the mirrorlist to ensure our download speeds are as good as possible, and that the server being used is from our locale. There are two ways to accomplish this. The first (rankmirrors, which is what I typically use) does not require any additional packages, the other (reflector) requires that we install a package. Both methods are described next. Pick one.
Open the mirrorlist file using:
$ nano /etc/pacman.d/mirrorlist
Then make sure that only servers in your country are un-commented. Alternately (what I do), you can shift + arrow down
to highlight servers you don't want and Ctrl + K
to cut them. Save the file. Then:
Make a backup copy of the mirrorlist file:
$ sudo cp /etc/pacman.d/mirrorlist /etc/pacman.d/mirrorlist.backup
Now run rankmirrors
. It will take the server data from the backup mirrorlist, rank them, then copy the data to the original mirrorlist file:
$ rankmirrors -n 6 /etc/pacman.d/mirrorlist.backup > /etc/pacman.d/mirrorlist
First, make sure the pacman databases are up-to-date:
$ pacman -Sy
Install Reflector:
$ pacman -S reflector rsync curl
Now generate the new mirrorlist. Note: If you are in a different country change "United States" to your country.
$ reflector --verbose --country 'United States' -l 5 --sort rate --save /etc/pacman.d/mirrorlist
If you only want a base Arch install with no additional packages, run:
$ pacstrap -i /mnt base base-devel
Typically I also install git
so I can clone my post-install setup and config scripts, along with dialog
and wpa_supplicant
so that wifi-menu
will work after booting into the new system. I also install intel-ucode
to allow the Linux kernel to update the processor microcode.
$ pacstrap -i /mnt base base-devel git dialog wpa_supplicant intel-ucode
We now need to update the filesystem table on the new installation. Fstab contains the association between filesystems and mountpoints.
$ genfstab -U -p /mnt >> /mnt/etc/fstab
You can verify fstab with:
$ cat /mnt/etc/fstab
Since we're still booted via USB, in order to configure our new system we need to change root. If we don't do that, every change we make will be applied to the USB installation.
$ arch-chroot /mnt
While there are various bootloaders that may be used, since the Linux kernel has a built-in EFI image, all we need is a way to execute it. For that we will install systemd-boot:
$ bootctl --path=/boot install
Using nano we can edit the config file:
$ nano /boot/loader/loader.conf
Make sure that only the following lines are in the file:
default arch
timeout 3
editor 0
Notes: The timeout setting is the number of seconds the menu is displayed before it automatically boots the default choice. For security, setting editor 0 disables kernel parameter editing via the terminal. These can always be edited in the mkinitcpio.conf
file.
In the next step we will update the boot loader config file. But first, we need to determine the UUID of our root partition. The root partition is where we installed our LVM on. If you don't recall the name of that node you can look it up using:
$ lsblk
Now use the node name you just looked up to get the UUID:
$ blkid /dev/sd*
You can either write down the UUID (which is painful given the length), or what I prefer to do is pipe the output of the above command into the config file that we will need that information in:
$ blkid /dev/sda2 > /boot/loader/entries/arch.conf
Then open the config file in nano:
$ nano /boot/loader/entries/arch.conf
Arrow over to the UUID and shift + arrow
to highlight it (only the ID, not the surrounding quotes). Use Ctl+K
to cut the line, putting into the clipboard.
Now delete everything in that file and add the following info. Make sure to replace YOUR_ID with the ID gathered previously, which you can paste from your clipboard using Ctrl+U
.
title Arch Linux
linux /vmlinuz-linux
initrd /initramfs-linux.img
options cryptdevice=UUID=YOUR_ID:vg root=/dev/mapper/vg-root quiet splash rw
Note: If you installed the intel-ucode
package earlier your arch.conf file will need an additional line:
title Arch Linux
linux /vmlinuz-linux
initrd /intel-ucode.img
initrd /initramfs-linux.img
options cryptdevice=UUID=YOUR_ID:vg root=/dev/mapper/vg-root quiet splash rw
$ bootctl update
Since we're using disk encryption we need to make sure that the LUKS module gets initialized by the kernel so we can decrypt our drive prior to booting. We also need to make sure that the keyboard is available for use prior to initializing the filesystem, otherwise we will have no input device to type in our password.
Edit the following config file:
$ nano /etc/mkinitcpio.conf
Scroll down to the HOOKS section. It should look similar to this:
HOOKS=("base udev autodetect modconf block filesystems keyboard fsck")
Change it to this:
HOOKS=("base udev autodetect modconf block keyboard keymap encrypt lvm2 filesystems fsck")
If you installed terminus fonts earlier in order to increase the font size of the temrinal, add consolefont
to the HOOKS as well. I put the hook first, so that it gets initiallzed early enough to increase the font size during boot. Note that the fonts will have to be installed once you've booted into the new system.
HOOKS=(consolefont base udev autodetect modconf block keyboard keymap encrypt lvm2 filesystems fsck)
Lastly, If your computer is running PCIe storage rather than SATA add nvme
to MODULES. NVMe is a specification for accessing SSDs attached through the PCI Express bus. The Linux kernel includes an NVMe driver, so we just need to tell the kernel to load it.
Scroll up to the MODULES section and change it to:
MODULES=(nvme)
Now update the initramfs image with our changes:
$ mkinitcpio -p linux
If you're curious what modules are available as intcpio hooks:
$ ls /usr/lib/initcpio/install
Open the locale.gen file and uncomment your preferred language (I'm using en_US.UTF-8):
$ nano /etc/locale.gen
Now save the file and generate the locale:
$ locale-gen
Copy your language choice to the locale.conf file:
$ echo LANG=en_US.UTF-8 > /etc/locale.conf
Export the language as an environmental shell variable:
$ export LANG=en_US.UTF-8
Run this command to find your timezone:
$ tzselect
Now use the timezone you just looked up to create a symbolic link to /etc/localtime. Note: Be sure to change America/Denver to your timezone.
$ ln -s /usr/share/zoneinfo/America/Denver /etc/localtime
Note: If you get an error that says failed to create symbolic link '/etc/localtime': File exists
, you must first delete the localtime file:
$ rm /etc/localtime
Then you should be able to create the above symbolic link.
Update the hardware clock. I use UTC:
$ hwclock --systohc --utc
This is the name of your computer. I name mine "Arch", but you can change it to whatever you want your host to be.
$ echo Arch > /etc/hostname
$ passwd
Make sure to replace <username> with your username.
$ useradd -m -G wheel,users -s /bin/bash <username>
And set the user password:
$ passwd <username>
Install sudo:
$ pacman -S sudo
Then run the following command, which will open the sudoers file:
$ EDITOR=nano visudo
Find this line and un-comment it:
$ %wheel ALL=(ALL) ALL
Open the pacman.conf file:
$ nano /etc/pacman.conf
If you want to be able to run 32bit software on a 64bit system then uncomment:
#[multilib]
#Include = /etc/pacman.d/mirrorlist
If you plan on downloading packages from the Arch User Repository, add this:
[archlinuxfr]
SigLevel = Never
Server = http://repo.archlinux.fr/$arch
In that same file add these (or uncomment if either are there). The color
directive gives you colored output when running pacman commands, and ILoveCandy
enables the little yellow pacman animation.
color
ILoveCandy
Then save the file.
The installation is basically done so we now update the databases and all installed packages:
$ pacman -Syu
You should now have a working Arch Linux installation. It doesn't have a desktop environment or any applications yet...for that you can check out my ArchMatic repo, but the base installation is done.
First, exit chroot:
$ exit
Now unmount and reboot
$ umount -R /mnt
$ reboot
Or, if you prefer you can shutdown:
$ poweroff
Instead of rebooting, as indicated above, shut down the virtual machine and do the following;
Remove the ISO you booted from at:
Settings > Storage > Controller:IDE
Make sure EFI is enabled:
Settings > System > Enable EFI
For WiFi connectivity, first get the name of your connection using:
$ ip link
It should be something like enp0s3
. Then run:
$ sudo ip link set dev enp0s3 up
$ sudo dhcpcd enp0s3