Dev/Embedded/FromScratch
Introduction
First of all, building your own system image for your board is a good idea.
This may feel a little bit confusing or difficult at first, but you'll find most information here and across this Wiki. It may not be exhaustive or up-to-date, but you'll have all the basics.
Building your own images is really important.
It means that :
- You have access to all the sources
- You do not depend on others to get what you want, update, or fix your system
- You will learn a lot
External requirements
In order to build your own system image you will need a few things :
- A system running a Linux based distribution. It will be your "host" or "development" system. (Well, the "Linux based" is not mandatory, but it's the one I use). I'm using a Devuan GNU/Linux system running on a x86_64 computer, but there are alternatives out there, for the distribution as well as for the hardware. It's best if it has some powerfull processors and plenty of available RAM and storage, but even the smallest system will do the job, it will only take more time.
- A (cross-)compiler toolchain. You'll find (most of) the information related to this part on my page dedicated to cross-compilation.
- A way to fetch the sources of course.
- A target system to run the binaries you'll build. Well you certainly have one, or you'll not be here, though it is also possible to use an emulator on your development system.
- A way to connect to your target's serial console. It may be possible to do it differently, but it's way easier to have a serial link to connect to the system console (UART), and I'll consider you are using this solution.
- Access to the sources and documentations for your hardware system and it's components. Though you won't need the documentation of all the parts, or even none of them if your system is already fully supported, it's always better if they are not black-boxes.
- Some time, curiosity and will. It's not very difficult, but you may enconter some problem as the systems you'll use are in constant evolution and may not behave the same as when I wrote these lines.
System content
All system images can be split in three parts :
- The bootloader
- The Linux kernel
- The userspace
These three parts may be split further, but it's the basics.
Bootloader
The bootloader is a small piece of software which is the first thing to run when the system is powered up.
The role of the Bootloader is to setup the hardware so that it can execute a bigger system : the Linux Kernel in our case.
Depending on your target hardware you may need more than one boatloader before you can get your Linux Kernel running. This is usually due to technical reasons, such as available memory or hardware security protections.
In most cases, the very first boatloader is a part you have no control over, and is usually called a "ROM bootloader" (or RBL), and is integrated in the ROM of the processor. This one will load your first stage bootloader from a given source, depending (most of the time) on the state of some configuration pins (boot selection or boot configuration) or internal fuses (e-fuses, which may be one-time programmable or not).
Depending on the hardware you'll use, this first stage bootloader can be :
- The only bootloader you'll need
- A smaller version of your bootloader : some processor will load the bootloader in the instruction cache of the processor, which may be too small for a full featured bootloader (tough if you manage to get everything you need in the available space, then you can get rid of the second stage bootloader and use only this one).
- A specific piece of firmware required for the processor, provided as source or binary : many recent ARM targets require an "ARM trusted firmware", now called "Trusted Firmware-A (TF-A)". Other may need other hadware specific bootloader to start.
When the first stage bootloader (sometimes called "Secondary Program Loader" (SPL)) is not the only required one, it will load your second stage bootloader (sometimes called "next stage" bootloader). Most of the time the second stage bootloader is from the same project as the first stage one (both u-boot, or both barebox for example), and when your board is supported the compilation is made in one go, both binaries being bundled in a single image.
In order to know what can and cannot be done with a specific board you will need the board's documentation or schematics (and the processor's documentation).
ARM trusted firmware (or Trusted Firmware-A)
If you are building U-Boot for an arm64/aarch64 device, you need to include the "ARM trusted firmware" (ATF) or "Trusted Firmware-A" (TF-A).
- Get the sources :
You can get this either from the trustedfirmware.org git repository :
git clone https://git.trustedfirmware.org/TF-A/trusted-firmware-a.git
These sources are also available from the arm-trusted-firmware git repository on github : https://github.com/ARM-software/arm-trusted-firmware.git
Note that both repository seems to have the same code, only some of the tags appear to be different.
- Configure :
There's no configuration stage for this part, though you'll need to know the "platform" name and the "target" name you want to build.
You can run "make help" to get the list of supported platform and the target names.
You can refer to the documentation for your SoC (processor) under "docs/plat" directory.
I'll indicate the right plaform and target on the pages dedicated to each board when I get my hands on them :)
- Compile :
To build this firmware part get to the root of the git tree and run:
platform=
target=
compiler_prefix=
make CROSS_COMPILE=${compiler_prefix} PLAT=${platform} DEBUG=1 ${target}
Set "platform" and "target" according to your needs as identified in the "Configure" section above
U-Boot
"Das U-Boot" (https://www.denx.de/project/u-boot/) is one of the most used bootloader for embedded systems running Linux.
It supports a wide range of hardware and many boot mechanisms (serial, ethernet, flash, USB, SD card, emmc, ...) to load the necessary images required to boot the system.
Details will be found in u-boot documentation.
- Get the sources :
Most of the time the official u-boot sources (from denx.de) are fine :
git clone git://git.denx.de/u-boot.git
If your board is not supported by the official u-boot sources, refer to your board documentation to get the bootloader sources. Depending on your hardware, you may need to use a specific release of u-boot. Use git checkout to get the right one.
- Configure :
If your hardware is supported by u-boot you should find a "<board_name>_defconfig" file in the "configs" directory at the root of the git tree.
If you do not know which one to use, you can refer to your board's documentation, or to u-boot documentation under "doc/board".
Then the configuration is made in two steps :
board_name=
compiler_prefix=
make CROSS_COMPILE=${compiler_prefix} ${board_name}_defconfig
make CROSS_COMPILE=${compiler_prefix} menuconfig
If the default configuration fits your needs the second step (menuconfig) is not required.
- Compile :
Once you're done with the configuration step, the compilation is made by running make from the source tree root with the "CROSS_COMPILE" variable set.
Note that for boards which require an ARM trusted firmware image it may be required to export a variable with it's path first :
platform=
target=
path_to_atf=
export BL31=${path_to_atf}/build/${platform}/debug/${target}.bin
And compile :
compiler_prefix=
make CROSS_COMPILE=${compiler_prefix}
The result is at the root of the source tree. The image name to use depends on your target.
- U-Boot environment :
Part of u-boot configuration includes a default environment, which holds various variables used to help loading and starting the Linux kernel.
Some of these variables have predefined meaning and other are user defined to help configure u-boot behavior.
Depending on your hardware, the default variables can be overridden from the content of a saved environment from a flash location, or from the content of a script. This helps save board specific boot configuration and select boot mechanism used to load and start the required binaries.
Whatever the mechanism used to get them, it usually boils down to loading the Linux kernel image, an optional initramfs or initrd image (which may also be included in the Linux kernel image or not required at all), and a device tree blob (dtb) if it is not included within the Linux kernel image.
The bootloader also has to provide a command line to the Linux kernel, unless the Linux kernel image has a default command line compiled in. This is done by setting the "bootargs" environment variable.
The effective boot is done then by running one of the boot commands supported by the u-boot image.
Refer to each board description for a list of board specific configurations and for example boot scripts when supported.
Other bootloaders
Some boards make use of barebox (https://www.barebox.org/) as bootloader (which is an u-boot derivative, and for some boards you may have the choice between different solutions.
Currently I always sticked to the bootloader advertised by the board vendor, and used barebox only on industrial boards not sold to the public, so I won't add any more information here.