On some recent PCs it can be necessary, or desirable, to load
        firmware to make them work at their best. The kernel or kernel
        drivers look for firmware images in /lib/firmware. But in LFS /lib is a symlink to usr/lib, so the directory containing the firmware
        files is actually /usr/lib/firmware.
      
        Currently, most firmware can be found at a git repository which can be viewed
        in the browser with the URL 
        https://git.kernel.org/pub/scm/linux/kernel/git/firmware/linux-firmware.git/plain.
        For convenience, the LFS Project has created a mirror, updated daily,
        where these firmware files can be accessed via wget or a web browser at https://anduin.linuxfromscratch.org/BLFS/linux-firmware/.
      
        To get the firmware, point a browser to one of the above repositories
        and then download the item(s) which you need. If you want all these
        firmware files (for example you are distributing the system onto
        multiple hardware systems), either install git-2.50.1 and clone
        
        https://git.kernel.org/pub/scm/linux/kernel/git/firmware/linux-firmware.git,
        or open this URL in a browser and download the latest snapshot listed
        in the Tag table.
      
![[Important]](../images/important.png) 
        
          
          Note that some required firmware files (and directories) are
          actually symlinks to other files (or directories). Those symlinks
          are not stored in the Git repository, instead they are documented
          in the WHENCE file. Each line
          beginning with Link: describes a
          symlink. If you are downloading an item you need and it's such a
          symlink (or in a directory which is such a symlink), you need to
          create the symlink manually and download the target. On the other
          hand, if you want all the firmware files, use the ./copy-firmware.sh
          /usr/lib/firmware command which will copy the
          firmware files and create all the symlinks in the WHENCE file, instead of a plain copy command like
          cp -r *
          /usr/lib/firmware/.
        
For some other firmware, particularly for Intel microcode and certain wifi devices, the needed firmware is not available in the above repository. Some of this will be addressed below, but a search of the Internet for needed firmware is sometimes necessary.
Firmware files are conventionally referred to as blobs because you cannot determine what they will do. Note that firmware is distributed under various different licenses which do not permit disassembly or reverse-engineering.
Firmware for PCs falls into four categories:
Updates to the CPU to work around errata, usually referred to as microcode.
Firmware for video controllers. On x86 machines this is required for ATI devices (Radeon and AMDGPU chips) and may be useful for Intel (Skylake and later) and NVIDIA (Kepler and later) GPUs.
ATI Radeon and AMDGPU devices all require firmware to be able to use KMS (kernel modesetting - the preferred option) as well as for Xorg.
Intel integrated GPUs from Skylake onwards can use firmware for GuC (the Graphics microcontroller), and also for the HuC (HEVC/H265 microcontroller which offloads to the GPU) and the DMC (Display Microcontroller) to provide additional low-power states. The GuC and HuC have had a chequered history in the kernel and updated firmware may be disabled by default, depending on your kernel version. Further details may be found at 01.org and Arch linux.
NVIDIA GPUs from Kepler onwards require signed firmware, otherwise the nouveau driver is unable to provide hardware acceleration. NVIDIA has now released firmware up to Ada Lovelace (GeForce 40 series) to linux-firmware.
Firmware updates for wired network ports. Most of them work even without the updates, but they will probably work better with the updated firmware. For some modern laptops, firmware for both wired ethernet (e.g. rtl_nic) and also for bluetooth devices (e.g. qca) is required before the wired network can be used.
Firmware for other devices, such as wireless NICs. These devices are not required for the PC to boot, but need the firmware before these devices can be used.
![[Note]](../images/note.png) 
        Although not needed to load a firmware blob, the following tools may be useful for determining, obtaining, or preparing the needed firmware in order to load it into the system: cpio-2.15, git-2.50.1, pciutils-3.14.0, and Wget-1.25.0
In general, microcode can be loaded by the BIOS or UEFI, and it might be updated by upgrading to a newer version of those. On linux, you can also load the microcode from the kernel if you are using an AMD family 10h or later processor (first introduced late 2007), or an Intel processor from 1998 and later (Pentium4, Core, etc), if updated microcode has been released. These updates only last until the machine is powered off, so they need to be applied on every boot.
![[Note]](../images/note.png) 
          In some rare cases a microcode update from the kernel can be ineffective. For example, despite intel-microcode-20241029 contains the Raptor Lake microcode revision 0x12b which is intended to fix a notorious excessive voltage issue causing stability issues and even permanent CPU damages, when the kernel boots it's already too late for the microcode to fix the issue. The only way to fix this issue is updating the BIOS.
Intel provide updates of their microcode for Skylake and later processors as new vulnerabilities come to light, and have in the past provided updates for processors from SandyBridge onwards, although those are no-longer supported for new fixes. New versions of AMD firmware are rare and usually only apply to a few models, although motherboard manufacturers get AGESA (AMD Generic Encapsulated Software Architecture) updates to change BIOS values, e.g. to support more memory variants, new vulnerability fixes or newer CPUs.
There were two ways of loading the microcode, described as 'early' and 'late'. Early loading happens before userspace has been started, late loading happens after userspace has started. However, late loading is known to be problematic and not supported anymore (see the kernel commit x86/microcode: Taint and warn on late loading). Indeed, early loading is needed to work around one particular erratum in early Intel Haswell processors which had TSX enabled. (See Intel Disables TSX Instructions: Erratum Found in Haswell, Haswell-E/EP, Broadwell-Y.) Without this update glibc can do the wrong thing in uncommon situations.
In previous versions of this book, late loading of microcode to see if it gets applied was recommended, followed by using an initrd to force early loading. But now that the contents of the Intel microcode tarball is documented, and AMD microcode can be read by a Python script to determine which machines it covers, there is no real reason to use late loading.
It might be still possible to manually force late loading of microcode. But it may cause kernel malfunction and you should take the risk yourself. You will need to reconfigure your kernel for late loading, but early loading is always supported by Linux kernel version 6.6 or later on a x86 (no matter 32-bit or 64-bit) system. The instructions here will show you how to create an initrd for early loading. It is also possible to build the same microcode bin file into the kernel, which allows early loading but requires the kernel to be recompiled to update the microcode.
To confirm what processor(s) you have (if more than one, they will be identical) look in /proc/cpuinfo. Determine the decimal values of the cpu family, model and stepping by running the following command (it will also report the current microcode version):
head -n7 /proc/cpuinfo
Convert the cpu family, model and stepping to pairs of hexadecimal digits, and remember the value of the “microcode” field. You can now check if there is any microcode available.
If you are creating an initrd to update firmware for different machines, as a distro would do, go down to 'Early loading of microcode' and cat all the Intel blobs to GenuineIntel.bin or cat all the AMD blobs to AuthenticAMD.bin. This creates a larger initrd - for all Intel machines in the 20200609 update the size was 3.0 MB compared to typically 24 KB for one machine.
            The first step is to get the most recent version of the Intel
            microcode. This must be done by navigating to 
            https://github.com/intel/Intel-Linux-Processor-Microcode-Data-Files/releases/
            and downloading the latest file there. As of this writing the
            most secure version of the microcode is microcode-20250812.
            Extract this file in the normal way, the microcode is in the
            intel-ucode directory, containing
            various blobs with names in the form XX-YY-ZZ. There are also
            various other files, and a release note.
          
            In the past, intel did not provide any details of which blobs had
            changed versions, but now the release note details this. You can
            compare the microcode version in /proc/cpuinfo with the version for your CPU
            model in the releasenote to know if there is an update.
          
The recent firmware for older processors is provided to deal with vulnerabilities which have now been made public, and for some of these such as Microarchitectural Data Sampling (MDS) you might wish to increase the protection by disabling hyperthreading, or alternatively to disable the kernel's default mitigation because of its impact on compile times. Please read the online documentation at https://www.kernel.org/doc/html/latest/admin-guide/hw-vuln/index.html.
            For an Tigerlake mobile (described as Intel(R) Core(TM) i5-11300H
            CPU) the relevant values are cpu family 6, model 140, stepping 1
            so in this case the required identification is 06-8c-01. The
            releasenote says the latest microcode for it is versioned 0xb8.
            If the value of the “microcode” field in /proc/cpuinfo is 0xb8 or greater, it indicates
            the microcode update is already applied by the BIOS. Otherwise,
            proceed to the section called “Early
            loading of microcode”.
          
Begin by downloading a container of firmware for your CPU family from https://anduin.linuxfromscratch.org/BLFS/linux-firmware/amd-ucode/. The family is always specified in hex. Families 10h to 14h (16 to 20) are in microcode_amd.bin. Families 15h, 16h, 17h (Zen, Zen+, Zen2) and 19h (Zen3) have their own containers, but very few machines are likely to get updated microcode. Instead, AMD provide an updated AGESA to the motherboard makers, who may provide an updated BIOS using this. There is a Python3 script at https://github.com/AMDESE/amd_ucode_info/blob/master/amd_ucode_info.py. Download that script and run it against the bin file to check which processors have updates.
For the very old Athlon(tm) II X2 in these examples the values were cpu family 16, model 5, stepping 3 giving an identification of Family=0x10 Model=0x05 Stepping=0x03. One line of the amd_ucode_info.py script output describes the microcode version for it:
Family=0x10 Model=0x05 Stepping=0x03: Patch=0x010000c8 Length=960 bytes
            If the value of the “microcode” field in /proc/cpuinfo is 0x10000c8 or greater, it
            indicates the BIOS has already applied the microcode update.
            Otherwise, proceed to the section called “Early loading of
            microcode”.
          
If you have established that updated microcode is available for your system, it is time to prepare it for early loading. This requires an additional package, cpio-2.15 and the creation of an initrd which will need to be added to grub.cfg.
It does not matter where you prepare the initrd, and once it is working you can apply the same initrd to later LFS systems or newer kernels on this same machine, at least until any newer microcode is released. Use the following commands:
mkdir -p initrd/kernel/x86/microcode cd initrd
For an AMD machine, use the following command (replace <MYCONTAINER> with the name of the container for your CPU's family):
cp -v ../<MYCONTAINER> kernel/x86/microcode/AuthenticAMD.bin
Or for an Intel machine copy the appropriate blob using this command:
cp -v ../intel-ucode/<XX-YY-ZZ> kernel/x86/microcode/GenuineIntel.bin
Now prepare the initrd:
find . | cpio -o -H newc > /boot/microcode.img
You now need to add a new entry to /boot/grub/grub.cfg and here you should add a new line after the linux line within the stanza. If /boot is a separate mountpoint:
initrd /microcode.img
or this if it is not:
initrd /boot/microcode.img
            If you are already booting with an initrd (see the section called
            “About initramfs”), you should run mkinitramfs again after putting
            the appropriate blob or container into /usr/lib/firmware. More precisely, put an intel
            blob in a /usr/lib/firmware/intel-ucode directory or an
            AMD container in a /usr/lib/firmware/amd-ucode directory before
            running mkinitramfs. Alternatively, you
            can have both initrd on the same line, such as initrd /microcode.img
            /other-initrd.img (adapt that as above if /boot
            is not a separate mountpoint).
          
You can now reboot with the added initrd, and then use the following command to check that the early load worked:
dmesg | grep -e 'microcode' -e 'Linux version' -e 'Command line'
If you updated to address vulnerabilities, you can look at the output of the lscpu command to see what is now reported.
The places and times where early loading happens are very different in AMD and Intel machines. First, an example of an Intel (Tigerlake mobile) with early loading:
[    0.000000] Linux version 6.10.4 (xry111@stargazer) (gcc (GCC) 14.2.0, GNU ld (GNU Binutils) 2.43) #4 SMP PREEMPT_DYNAMIC Tue Aug 15 18:04:11 CST 2024
[    0.000000] Command line: BOOT_IMAGE=/boot/vmlinuz-6.10.0 root=PARTUUID=<CLASSIFIED> ro
[    0.585605] microcode: Current revision: 0x000000b8
[    0.585611] microcode: Updated early from: 0x00000086A historic AMD example:
[    0.000000] Linux version 4.15.3 (ken@testserver) (gcc version 7.3.0 (GCC))
               #2 SMP Sun Feb 18 02:32:03 GMT 2018
[    0.000000] Command line: BOOT_IMAGE=/vmlinuz-4.15.3-sda5 root=/dev/sda5 ro
[    0.307619] microcode: microcode updated early to new patch_level=0x010000c8
[    0.307678] microcode: CPU0: patch_level=0x010000c8
[    0.307723] microcode: CPU1: patch_level=0x010000c8
[    0.307795] microcode: Microcode Update Driver: v2.2.Early radeon devices only needed a single 2K blob of firmware. Recent devices need several different blobs, and some of them are much bigger. The total size of the radeon firmware directory is over 500K — on a large modern system you can probably spare the space, but it is still redundant to install all the unused files each time you build a system.
            A better approach is to install pciutils-3.14.0 and then use lspci to identify which VGA
            controller is installed.
          
With that information, check the RadeonFeature page of the Xorg wiki for Decoder ring for engineering vs marketing names to identify the family (you may need to know this for the Xorg driver in BLFS — Southern Islands and Sea Islands use the radeonsi driver) and the specific model.
Now that you know which controller you are using, consult the Radeon page of the Gentoo wiki which has a table listing the required firmware blobs for the various chipsets. Note that Southern Islands and Sea Islands chips use different firmware for kernel 3.17 and later compared to earlier kernels. Identify and download the required blobs then install them:
mkdir -pv /usr/lib/firmware/radeon cp -v <YOUR_BLOBS> /usr/lib/firmware/radeon
Building the kernel amdgpu driver as a module is recommended because the firmware files need to be accessible at the time it is loaded. If you are building it as a part of the kernel image for any reason, you need to either include the firmware files in the initramfs (read the section called “About initramfs” for details), or include them in the kernel image itself (read the section called “Include Firmware Blobs in the Kernel Image” for details).
All video controllers using the amdgpu kernel driver require firmware, whether you will be using the xorg amdgpu driver, the xserver's modesetting driver, or just kernel modesetting to get a console framebuffer larger than 80x25.
Install pciutils-3.14.0 and use that to check the model name (look for 'VGA compatible controller:'). If you have an APU (Accelerated Processing Unit, i.e. CPU and video on the same chip) that will probably tell you the name. If you have a separate amdgpu video card you will need to search to determine which name it uses (e.g. a card described as Advanced Micro Devices, Inc. [AMD/ATI] Baffin [Radeon RX 550 640SP / RX 560/560X] needs Polaris11 firmware. There is a table of "Family, Chipset name, Product name and Firmware" at the end of the Kernel sections in AMDGPU page of the Gentoo wiki.
Once you have identified the firmware name, install all the relevant files for it. For example, the Baffin card mentioned above has 21 different polaris11* files, APUs such as renoir and picasso have at least 12 files and might gain more in future updates (e.g. the raven APU now has a 13th file, raven_ta.bin).
mkdir -pv /usr/lib/firmware/amdgpu cp -v <YOUR_BLOBS> /usr/lib/firmware/amdgpu
If disk space is not a problem, you could install all the current amdgpu firmware files and not worry about exactly which chipset is installed.
Building the kernel amdgpu driver as a module is recommended because the firmware files need to be accessible at the time it is loaded. If you are building it as a part of the kernel image for any reason, you need to either include the firmware files in the initramfs (read the section called “About initramfs” for details), or include them in the kernel image itself (read the section called “Include Firmware Blobs in the Kernel Image” for details).
            Some NVIDIA GPUs from Kepler onwards (including all NVIDIA
            GeForce RTX GPUs and GTX 9xx series onwards) require firmware
            signed by NVIDIA to work. For the Nouveau driver, NVIDIA has
            released the signed firmware files in the nvidia/ directory of linux-firmware. Read
            the CodeNames
            page of the Nouveau wiki to figure out the code name of your
            NVIDIA GPU (for example, the code name of GeForce RTX 5080 is
            GB203). The directory for the firmware files you need is named as
            the result converting the code name to lower case (gb203 for the example). The directory (or some
            of its contents) may be actually a symlink to another directory
            (or some files in another directory) and you'll then need to
            create the symlink and
            download its target yourself (for example, gb207 is a symlink to gb202).
          
            If you can find the directory for your GPU in neither the
            linux-firmware tree nor the WHENCE
            file (as a symlink), it means either your GPU does not need any
            firmware, or it's newer than Blackwell (RTX 50xx series) and the
            firmware for it isn't released yet (as at mid 2025).
          
For the NVIDIA GPUs from Tesla to Kepler, the Nouveau driver requires some extra firmware to support video decoding acceleration. These firmware files are not released by NVIDIA but they can be extracted from the NVIDIA binary driver. For more exact information about which chips need extracted firmware, see the VideoAcceleration of the Nouveau wiki. To extract and install these firmware files, issue:
wget https://anduin.linuxfromscratch.org/BLFS/nvidia-firmware/extract_firmware.py wget https://us.download.nvidia.com/XFree86/Linux-x86/340.32/NVIDIA-Linux-x86-340.32.run sh NVIDIA-Linux-x86-340.32.run --extract-only python3 extract_firmware.py mkdir -p /usr/lib/firmware/nouveau cp -d nv* vuc-* /usr/lib/firmware/nouveau/
          The kernel likes to load firmware for some network drivers,
          particularly those from Realtek (the /usr/lib/linux-firmware/rtl_nic/ directory), but
          they generally appear to work without it. Therefore, you can boot
          the kernel, check dmesg for messages about this missing firmware,
          and if necessary download the firmware and put it in the specified
          directory in /usr/lib/firmware so
          that it will be found on subsequent boots. Note that with current
          kernels this works whether or not the driver is compiled in or
          built as a module, there is no need to build this firmware into the
          kernel. Here is an example where the R8169 driver has been compiled
          in but the firmware was not made available. Once the firmware had
          been provided, there was no mention of it on later boots.
        
dmesg | grep firmware | grep r8169
[    7.018028] r8169 0000:01:00.0: Direct firmware load for rtl_nic/rtl8168g-2.fw failed with error -2
[    7.018036] r8169 0000:01:00.0 eth0: unable to load firmware patch rtl_nic/rtl8168g-2.fw (-2)
          Different countries have different regulations on the radio
          spectrum usage of wireless devices. You can install a firmware to
          make the wireless devices obey local spectrum regulations, so you
          won't be inquired by local authority or find your wireless NIC
          jamming the frequencies of other devices (for example, remote
          controllers). The regulatory database firmware can be downloaded
          from https://kernel.org/pub/software/network/wireless-regdb/.
          To install it, simply extract regulatory.db and regulatory.db.p7s from the tarball into
          /usr/lib/firmware. Note that either
          the cfg80211 driver needs to be
          selected as a module for the regulatory.* files to be loaded, or those files
          need to be included as firmware into the kernel, as explained above
          in the section called “Firmware for Video
          Cards”.
        
          The access point (AP) would send a country code to your wireless
          NIC, and wpa_supplicant-2.11 would tell the
          kernel to load the regulation of this country from regulatory.db, and enforce it. Note that several
          AP don't send this country code, so you may be locked to a rather
          restricted usage (specially if you want to use your interface as an
          AP).
        
          Some systems (especially laptops with a recent Intel CPU) utilize a
          DSP shipped with the CPU for connection with the audio codec. The
          Sound Open Firmware must be loaded onto the DSP to make it
          functional. These firmware files can be downloaded from https://github.com/thesofproject/sof-bin/releases.
          Extract the tarball and changing into the extracted directory, then
          as the root user install the
          firmware:
        
install -vdm755 /usr/lib/firmware/intel && cp -rv sof* /usr/lib/firmware/intel/
alsa-lib-1.2.14 needs Use Case Manager configuration files for the systems using Sound Open Firmware as well. Read the alsa-lib-1.2.14 page for the instructions to install them. Once the firmware is loaded (you may need a reboot so the kernel will load them) and the UCM configuration files are installed, following the section called “Configuring ALSA Utilities” to set up your sound card for ALSA properly.
          Identifying the correct firmware will typically require you to
          install pciutils-3.14.0, and then use lspci to identify the device. You
          should then search online to check which module it uses, which
          firmware, and where to obtain the firmware — not all of it is in
          linux-firmware.
        
If possible, you should begin by using a wired connection when you first boot your LFS system. To use a wireless connection you will need to use a network tools such as iw-6.9, Wireless Tools-29, or wpa_supplicant-2.11.
Firmware may also be needed for other devices such as some SCSI controllers, bluetooth adaptors, or TV recorders. The same principles apply.
          Some drivers, notably the drivers for ATI or AMD GPU, requires the
          firmware files accessible at the time it is loaded. The easiest
          method to handle these drivers is building them as a kernel module.
          An alternative method is creating an initramfs (read the section
          called “About initramfs” for details) including the firmware
          files. If you don't want to use either methods, you may include the
          firmware files in the kernel image itself. Install the needed
          firmware files into /usr/lib/firmware
          first, then set the following kernel configuration and rebuild the
          kernel:
        
Device Drivers ---> Generic Driver Options ---> Firmware loader ---> <*> Firmware loading facility [FW_LOADER] (xx/aa.bin xx/bb.bin) Build named firmware blobs into the kernel binary ... [EXTRA_FIRMWARE] # This is the default, you can change it to "/usr/lib/firmware" # if you want: (/lib/firmware) Firmware blobs root directory ... [EXTRA_FIRMWARE_DIR]
          Replace xx/aa.bin
          xx/bb.bin with a whitespace-separated list of paths to
          the needed firmware files, relative to /usr/lib/firmware. A method easier than manually
          typing the list (it may be long) is running the following command:
        
echo CONFIG_EXTRA_FIRMWARE='"'$({ cd /usr/lib/firmware; echo amdgpu/* })'"' >> .config
make oldconfig
        
          Replace amdgpu/* with a
          shell pattern matching the needed firmware files.
        
![[Warning]](../images/warning.png) 
          Do not distribute a kernel image containing the firmware to others or you may violate the GPL.