Dev/Embedded/FromScratch: Difference between revisions

From Embeded Linux (and more) Wiki by Nathael
< Dev‎ | Embedded
Jump to navigation Jump to search
 
(17 intermediate revisions by the same user not shown)
Line 13: Line 13:
In order to build your own system image you will need a few things :
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 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 [[Dev/Embedded/CrossCompile|page dedicated to cross-compilation]].
* A (cross-)compiler toolchain. You'll find (most of) the information related to this part on my '''[[Dev/Embedded/CrossCompile|page dedicated to cross-compilation]]'''.
* A way to fetch the sources of course.
* 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 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.
Line 24: Line 24:
* The bootloader
* The bootloader
* The Linux kernel
* The Linux kernel
* The userspace
* The "UserLand"


These three parts may be split further, but it's the basics.
These three parts may be split further, but it's the basics.
Line 46: Line 46:
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).
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) ====
==== U-Boot (and Arm Trusted Firmware) ====
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).
Most boards make use of U-Boot as bootloader.


* '''Get the sources :'''
Detailed information will be found '''on [[Dev/Embedded/FromScratch/UBoot|the page dedicated to U-Boot]]'''.
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<br />
Note that both repository seems to have the same code, only some of the tags appear to be different.


* '''Configure :'''
==== Other bootloaders ====
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.
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.


You can run "make help" to get the list of supported platform and the target names.
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.


You can refer to the documentation for your SoC (processor) under "docs/plat" directory.
== Linux Kernel ==
On all my systems (embedded and PC) I'm using Linux based systems.


I'll indicate the right plaform and target on the pages dedicated to each board when I get my hands on them :)
By definition, these use Linux as the kernel for the system, whatever the bootloader or the user-space part.


* '''Compile :'''
The Linux kernel can run on almost any processor you can find which has an MMU and two separate execution modes, tough it has even been ported on MMU-less systems and micro-controllers with a single execution mode.
To build this firmware part get to the root of the git tree and run:
platform=
target=
make CROSS_COMPILE=aarch64-linux-gnu- PLAT=${platform} DEBUG=1 ${target}


Set "platform" and "target" according to your needs as identified in the "Configure" section above
It's purpose is to handle the hardware and provide an unified interface to the user-space programs. As such, it has to be taillored to the effective hardware present on your system.


==== U-Boot ====
Detailed information will be found on '''[[Dev/Embedded/FromScratch/LinuxKernel|the page dedicated to the Linux Kernel]]'''.
"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.<br />
Details will be found in u-boot documentation.


* '''Get the sources :'''
== UserLand ==
Most of the time the official u-boot sources (from denx.de) are fine :
The "last" part you'll need to get a running system is the "userland", which is simply all the binaries (executables), the libraries, the configuration files, the scripts, and all the other files and directories required by the system and your application.
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.
This "userland" is usually split in two parts : the "base system", and your application(s).<br />
Depending on your hardware, you may need to use a specific release of u-boot. Use git checkout to get the right one.
The "base system" is the part you get from an external source (or many), while the "application" part is pretty obvious : it's your own code and data.


* '''Configure :'''
Note that there are many confusions out there : some call the "base system" the "rootfs", others even add the application, though the "rootfs" can be either all (base + app) or only a subpart (not even whole base).<br />
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.<br />
The rootfs is the part which the kernel is looking for in order to mount it as the root ("/") of the filesystems stack/tree, and where it will need to find your init binary. It can be reduced to an initramfs file containing a single init binary file.
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".<br />
Then the configuration is made in two steps :
board_name=
make CROSS_COMPILE=arm-linux-gnueabihf- ${board_name}_defconfig
make CROSS_COMPILE=arm-linux-gnueabihf- menuconfig


If the default configuration fits your needs the second step (menuconfig) is not required.
Also note that though this "userland" holds everything that will run in user-space, it will also hold your modules, which will run in kernel-space. Some more information about this "kernel space" and "user space" separation can be found on this "[https://en.wikipedia.org/wiki/User_space_and_kernel_space User space and kernel space]]" wikipedia page.


* '''Compile :'''
=== Chose a solution for your base system ===
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.<br />
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 :
To build the base part of your userland you have a few options :
make CROSS_COMPILE=arm-linux-gnueabihf-
* Build everything from scratch (follow [https://www.linuxfromscratch.org/ Linux From Scratch ] instructions, I will not explain it on this Wiki).
* Use [https://busybox.net/ BusyBox] : recommended for space constrained systems. Get more information for this choice '''[[Dev/Embedded/Rootfs/BusyBox|on the BusyBox dedicated page]]'''.
* Use a GNU/Linux distribution : the easy way (I'm using [https://www.devuan.org/ Devuan GNU/Linux], but many other distribution may fit, just avoid those targeting only desktop or those making stupid choices). Also get more information for this choice '''[[Dev/Embedded/Rootfs/Devuan/FromScratch|on my "Devuan from scratch" dedicated page]]'''.
* Use an image provided from a third party (well, I'll never trust any of these, but it's up to you).
* Use an image builder tool (there's some unusable ones out there, [[Dev/Embedded/NotOEYocto|Yocto]] may be the most popular, but I strongly advise against these "solutions").


The result is at the root of the source tree.
=== Organize your data ===


==== Other bootloaders ====
=== Configuration ===
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.


== Linux Kernel ==
=== Auto-run your application ===


== Userspace ==
== Burn, Flash, Upload, Remote booting, ... ==

Latest revision as of 18:52, 3 January 2025

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 "UserLand"

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).

U-Boot (and Arm Trusted Firmware)

Most boards make use of U-Boot as bootloader.

Detailed information will be found on the page dedicated to U-Boot.

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.

Linux Kernel

On all my systems (embedded and PC) I'm using Linux based systems.

By definition, these use Linux as the kernel for the system, whatever the bootloader or the user-space part.

The Linux kernel can run on almost any processor you can find which has an MMU and two separate execution modes, tough it has even been ported on MMU-less systems and micro-controllers with a single execution mode.

It's purpose is to handle the hardware and provide an unified interface to the user-space programs. As such, it has to be taillored to the effective hardware present on your system.

Detailed information will be found on the page dedicated to the Linux Kernel.

UserLand

The "last" part you'll need to get a running system is the "userland", which is simply all the binaries (executables), the libraries, the configuration files, the scripts, and all the other files and directories required by the system and your application.

This "userland" is usually split in two parts : the "base system", and your application(s).
The "base system" is the part you get from an external source (or many), while the "application" part is pretty obvious : it's your own code and data.

Note that there are many confusions out there : some call the "base system" the "rootfs", others even add the application, though the "rootfs" can be either all (base + app) or only a subpart (not even whole base).
The rootfs is the part which the kernel is looking for in order to mount it as the root ("/") of the filesystems stack/tree, and where it will need to find your init binary. It can be reduced to an initramfs file containing a single init binary file.

Also note that though this "userland" holds everything that will run in user-space, it will also hold your modules, which will run in kernel-space. Some more information about this "kernel space" and "user space" separation can be found on this "User space and kernel space]" wikipedia page.

Chose a solution for your base system

To build the base part of your userland you have a few options :

  • Build everything from scratch (follow Linux From Scratch instructions, I will not explain it on this Wiki).
  • Use BusyBox : recommended for space constrained systems. Get more information for this choice on the BusyBox dedicated page.
  • Use a GNU/Linux distribution : the easy way (I'm using Devuan GNU/Linux, but many other distribution may fit, just avoid those targeting only desktop or those making stupid choices). Also get more information for this choice on my "Devuan from scratch" dedicated page.
  • Use an image provided from a third party (well, I'll never trust any of these, but it's up to you).
  • Use an image builder tool (there's some unusable ones out there, Yocto may be the most popular, but I strongly advise against these "solutions").

Organize your data

Configuration

Auto-run your application

Burn, Flash, Upload, Remote booting, ...