e0299f71b3
Currently, the documentation outlines the process for building the S-mode U-Boot image using `make menuconfig` and manual actions within the menuconfig UI. However, this approach is fragile due to Kconfig options potentially changing across different releases. For example, CONFIG_OF_PRIOR_STAGE has been replaced by CONFIG_BOARD since v2022.01 release, and CONFIG_TEXT_BASE has been moved to the 'General setup' menu from the 'Boot options' menu in v2024.01 release. This update aims to make the S-mode U-Boot image build instructions future-proof. It leverages the 'config' script provided in the U-Boot source tree to edit the .config file, followed by a `make olddefconfig`. Validated with U-Boot v2024.01 release. Signed-off-by: Bin Meng <bmeng@tinylab.org> Reviewed-by: Alistair Francis <alistair.francis@wdc.com> Message-ID: <20240104071523.273702-1-bmeng@tinylab.org> Signed-off-by: Alistair Francis <alistair.francis@wdc.com>
367 lines
13 KiB
ReStructuredText
367 lines
13 KiB
ReStructuredText
SiFive HiFive Unleashed (``sifive_u``)
|
||
======================================
|
||
|
||
SiFive HiFive Unleashed Development Board is the ultimate RISC-V development
|
||
board featuring the Freedom U540 multi-core RISC-V processor.
|
||
|
||
Supported devices
|
||
-----------------
|
||
|
||
The ``sifive_u`` machine supports the following devices:
|
||
|
||
* 1 E51 / E31 core
|
||
* Up to 4 U54 / U34 cores
|
||
* Core Local Interruptor (CLINT)
|
||
* Platform-Level Interrupt Controller (PLIC)
|
||
* Power, Reset, Clock, Interrupt (PRCI)
|
||
* L2 Loosely Integrated Memory (L2-LIM)
|
||
* DDR memory controller
|
||
* 2 UARTs
|
||
* 1 GEM Ethernet controller
|
||
* 1 GPIO controller
|
||
* 1 One-Time Programmable (OTP) memory with stored serial number
|
||
* 1 DMA controller
|
||
* 2 QSPI controllers
|
||
* 1 ISSI 25WP256 flash
|
||
* 1 SD card in SPI mode
|
||
* PWM0 and PWM1
|
||
|
||
Please note the real world HiFive Unleashed board has a fixed configuration of
|
||
1 E51 core and 4 U54 core combination and the RISC-V core boots in 64-bit mode.
|
||
With QEMU, one can create a machine with 1 E51 core and up to 4 U54 cores. It
|
||
is also possible to create a 32-bit variant with the same peripherals except
|
||
that the RISC-V cores are replaced by the 32-bit ones (E31 and U34), to help
|
||
testing of 32-bit guest software.
|
||
|
||
Hardware configuration information
|
||
----------------------------------
|
||
|
||
The ``sifive_u`` machine automatically generates a device tree blob ("dtb")
|
||
which it passes to the guest, if there is no ``-dtb`` option. This provides
|
||
information about the addresses, interrupt lines and other configuration of
|
||
the various devices in the system. Guest software should discover the devices
|
||
that are present in the generated DTB instead of using a DTB for the real
|
||
hardware, as some of the devices are not modeled by QEMU and trying to access
|
||
these devices may cause unexpected behavior.
|
||
|
||
If users want to provide their own DTB, they can use the ``-dtb`` option.
|
||
These DTBs should have the following requirements:
|
||
|
||
* The /cpus node should contain at least one subnode for E51 and the number
|
||
of subnodes should match QEMU's ``-smp`` option
|
||
* The /memory reg size should match QEMU’s selected ram_size via ``-m``
|
||
* Should contain a node for the CLINT device with a compatible string
|
||
"riscv,clint0" if using with OpenSBI BIOS images
|
||
|
||
Boot options
|
||
------------
|
||
|
||
The ``sifive_u`` machine can start using the standard -kernel functionality
|
||
for loading a Linux kernel, a VxWorks kernel, a modified U-Boot bootloader
|
||
(S-mode) or ELF executable with the default OpenSBI firmware image as the
|
||
-bios. It also supports booting the unmodified U-Boot bootloader using the
|
||
standard -bios functionality.
|
||
|
||
Machine-specific options
|
||
------------------------
|
||
|
||
The following machine-specific options are supported:
|
||
|
||
- serial=nnn
|
||
|
||
The board serial number. When not given, the default serial number 1 is used.
|
||
|
||
SiFive reserves the first 1 KiB of the 16 KiB OTP memory for internal use.
|
||
The current usage is only used to store the serial number of the board at
|
||
offset 0xfc. U-Boot reads the serial number from the OTP memory, and uses
|
||
it to generate a unique MAC address to be programmed to the on-chip GEM
|
||
Ethernet controller. When multiple QEMU ``sifive_u`` machines are created
|
||
and connected to the same subnet, they all have the same MAC address hence
|
||
it creates an unusable network. In such scenario, user should give different
|
||
values to serial= when creating different ``sifive_u`` machines.
|
||
|
||
- start-in-flash
|
||
|
||
When given, QEMU's ROM codes jump to QSPI memory-mapped flash directly.
|
||
Otherwise QEMU will jump to DRAM or L2LIM depending on the msel= value.
|
||
When not given, it defaults to direct DRAM booting.
|
||
|
||
- msel=[6|11]
|
||
|
||
Mode Select (MSEL[3:0]) pins value, used to control where to boot from.
|
||
|
||
The FU540 SoC supports booting from several sources, which are controlled
|
||
using the Mode Select pins on the chip. Typically, the boot process runs
|
||
through several stages before it begins execution of user-provided programs.
|
||
These stages typically include the following:
|
||
|
||
1. Zeroth Stage Boot Loader (ZSBL), which is contained in an on-chip mask
|
||
ROM and provided by QEMU. Note QEMU implemented ROM codes are not the
|
||
same as what is programmed in the hardware. The QEMU one is a simplified
|
||
version, but it provides the same functionality as the hardware.
|
||
2. First Stage Boot Loader (FSBL), which brings up PLLs and DDR memory.
|
||
This is U-Boot SPL.
|
||
3. Second Stage Boot Loader (SSBL), which further initializes additional
|
||
peripherals as needed. This is U-Boot proper combined with an OpenSBI
|
||
fw_dynamic firmware image.
|
||
|
||
msel=6 means FSBL and SSBL are both on the QSPI flash. msel=11 means FSBL
|
||
and SSBL are both on the SD card.
|
||
|
||
Running Linux kernel
|
||
--------------------
|
||
|
||
Linux mainline v5.10 release is tested at the time of writing. To build a
|
||
Linux mainline kernel that can be booted by the ``sifive_u`` machine in
|
||
64-bit mode, simply configure the kernel using the defconfig configuration:
|
||
|
||
.. code-block:: bash
|
||
|
||
$ export ARCH=riscv
|
||
$ export CROSS_COMPILE=riscv64-linux-
|
||
$ make defconfig
|
||
$ make
|
||
|
||
To boot the newly built Linux kernel in QEMU with the ``sifive_u`` machine:
|
||
|
||
.. code-block:: bash
|
||
|
||
$ qemu-system-riscv64 -M sifive_u -smp 5 -m 2G \
|
||
-display none -serial stdio \
|
||
-kernel arch/riscv/boot/Image \
|
||
-initrd /path/to/rootfs.ext4 \
|
||
-append "root=/dev/ram"
|
||
|
||
Alternatively, we can use a custom DTB to boot the machine by inserting a CLINT
|
||
node in fu540-c000.dtsi in the Linux kernel,
|
||
|
||
.. code-block:: none
|
||
|
||
clint: clint@2000000 {
|
||
compatible = "riscv,clint0";
|
||
interrupts-extended = <&cpu0_intc 3 &cpu0_intc 7
|
||
&cpu1_intc 3 &cpu1_intc 7
|
||
&cpu2_intc 3 &cpu2_intc 7
|
||
&cpu3_intc 3 &cpu3_intc 7
|
||
&cpu4_intc 3 &cpu4_intc 7>;
|
||
reg = <0x00 0x2000000 0x00 0x10000>;
|
||
};
|
||
|
||
with the following command line options:
|
||
|
||
.. code-block:: bash
|
||
|
||
$ qemu-system-riscv64 -M sifive_u -smp 5 -m 8G \
|
||
-display none -serial stdio \
|
||
-kernel arch/riscv/boot/Image \
|
||
-dtb arch/riscv/boot/dts/sifive/hifive-unleashed-a00.dtb \
|
||
-initrd /path/to/rootfs.ext4 \
|
||
-append "root=/dev/ram"
|
||
|
||
To build a Linux mainline kernel that can be booted by the ``sifive_u`` machine
|
||
in 32-bit mode, use the rv32_defconfig configuration. A patch is required to
|
||
fix the 32-bit boot issue for Linux kernel v5.10.
|
||
|
||
.. code-block:: bash
|
||
|
||
$ export ARCH=riscv
|
||
$ export CROSS_COMPILE=riscv64-linux-
|
||
$ curl https://patchwork.kernel.org/project/linux-riscv/patch/20201219001356.2887782-1-atish.patra@wdc.com/mbox/ > riscv.patch
|
||
$ git am riscv.patch
|
||
$ make rv32_defconfig
|
||
$ make
|
||
|
||
Replace ``qemu-system-riscv64`` with ``qemu-system-riscv32`` in the command
|
||
line above to boot the 32-bit Linux kernel. A rootfs image containing 32-bit
|
||
applications shall be used in order for kernel to boot to user space.
|
||
|
||
Running VxWorks kernel
|
||
----------------------
|
||
|
||
VxWorks 7 SR0650 release is tested at the time of writing. To build a 64-bit
|
||
VxWorks mainline kernel that can be booted by the ``sifive_u`` machine, simply
|
||
create a VxWorks source build project based on the sifive_generic BSP, and a
|
||
VxWorks image project to generate the bootable VxWorks image, by following the
|
||
BSP documentation instructions.
|
||
|
||
A pre-built 64-bit VxWorks 7 image for HiFive Unleashed board is available as
|
||
part of the VxWorks SDK for testing as well. Instructions to download the SDK:
|
||
|
||
.. code-block:: bash
|
||
|
||
$ wget https://labs.windriver.com/downloads/wrsdk-vxworks7-sifive-hifive-1.01.tar.bz2
|
||
$ tar xvf wrsdk-vxworks7-sifive-hifive-1.01.tar.bz2
|
||
$ ls bsps/sifive_generic_1_0_0_0/uboot/uVxWorks
|
||
|
||
To boot the VxWorks kernel in QEMU with the ``sifive_u`` machine, use:
|
||
|
||
.. code-block:: bash
|
||
|
||
$ qemu-system-riscv64 -M sifive_u -smp 5 -m 2G \
|
||
-display none -serial stdio \
|
||
-nic tap,ifname=tap0,script=no,downscript=no \
|
||
-kernel /path/to/vxWorks \
|
||
-append "gem(0,0)host:vxWorks h=192.168.200.1 e=192.168.200.2:ffffff00 u=target pw=vxTarget f=0x01"
|
||
|
||
It is also possible to test 32-bit VxWorks on the ``sifive_u`` machine. Create
|
||
a 32-bit project to build the 32-bit VxWorks image, and use exact the same
|
||
command line options with ``qemu-system-riscv32``.
|
||
|
||
Running U-Boot
|
||
--------------
|
||
|
||
U-Boot mainline v2024.01 release is tested at the time of writing. To build a
|
||
U-Boot mainline bootloader that can be booted by the ``sifive_u`` machine, use
|
||
the sifive_unleashed_defconfig with similar commands as described above for
|
||
Linux:
|
||
|
||
.. code-block:: bash
|
||
|
||
$ export CROSS_COMPILE=riscv64-linux-
|
||
$ export OPENSBI=/path/to/opensbi-riscv64-generic-fw_dynamic.bin
|
||
$ make sifive_unleashed_defconfig
|
||
|
||
You will get spl/u-boot-spl.bin and u-boot.itb file in the build tree.
|
||
|
||
To start U-Boot using the ``sifive_u`` machine, prepare an SPI flash image, or
|
||
SD card image that is properly partitioned and populated with correct contents.
|
||
genimage_ can be used to generate these images.
|
||
|
||
A sample configuration file for a 128 MiB SD card image is:
|
||
|
||
.. code-block:: bash
|
||
|
||
$ cat genimage_sdcard.cfg
|
||
image sdcard.img {
|
||
size = 128M
|
||
|
||
hdimage {
|
||
gpt = true
|
||
}
|
||
|
||
partition u-boot-spl {
|
||
image = "u-boot-spl.bin"
|
||
offset = 17K
|
||
partition-type-uuid = 5B193300-FC78-40CD-8002-E86C45580B47
|
||
}
|
||
|
||
partition u-boot {
|
||
image = "u-boot.itb"
|
||
offset = 1041K
|
||
partition-type-uuid = 2E54B353-1271-4842-806F-E436D6AF6985
|
||
}
|
||
}
|
||
|
||
SPI flash image has slightly different partition offsets, and the size has to
|
||
be 32 MiB to match the ISSI 25WP256 flash on the real board:
|
||
|
||
.. code-block:: bash
|
||
|
||
$ cat genimage_spi-nor.cfg
|
||
image spi-nor.img {
|
||
size = 32M
|
||
|
||
hdimage {
|
||
gpt = true
|
||
}
|
||
|
||
partition u-boot-spl {
|
||
image = "u-boot-spl.bin"
|
||
offset = 20K
|
||
partition-type-uuid = 5B193300-FC78-40CD-8002-E86C45580B47
|
||
}
|
||
|
||
partition u-boot {
|
||
image = "u-boot.itb"
|
||
offset = 1044K
|
||
partition-type-uuid = 2E54B353-1271-4842-806F-E436D6AF6985
|
||
}
|
||
}
|
||
|
||
Assume U-Boot binaries are put in the same directory as the config file,
|
||
we can generate the image by:
|
||
|
||
.. code-block:: bash
|
||
|
||
$ genimage --config genimage_<boot_src>.cfg --inputpath .
|
||
|
||
Boot U-Boot from SD card, by specifying msel=11 and pass the SD card image
|
||
to QEMU ``sifive_u`` machine:
|
||
|
||
.. code-block:: bash
|
||
|
||
$ qemu-system-riscv64 -M sifive_u,msel=11 -smp 5 -m 8G \
|
||
-display none -serial stdio \
|
||
-bios /path/to/u-boot-spl.bin \
|
||
-drive file=/path/to/sdcard.img,if=sd
|
||
|
||
Changing msel= value to 6, allows booting U-Boot from the SPI flash:
|
||
|
||
.. code-block:: bash
|
||
|
||
$ qemu-system-riscv64 -M sifive_u,msel=6 -smp 5 -m 8G \
|
||
-display none -serial stdio \
|
||
-bios /path/to/u-boot-spl.bin \
|
||
-drive file=/path/to/spi-nor.img,if=mtd
|
||
|
||
Note when testing U-Boot, QEMU automatically generated device tree blob is
|
||
not used because U-Boot itself embeds device tree blobs for U-Boot SPL and
|
||
U-Boot proper. Hence the number of cores and size of memory have to match
|
||
the real hardware, ie: 5 cores (-smp 5) and 8 GiB memory (-m 8G).
|
||
|
||
Above use case is to run upstream U-Boot for the SiFive HiFive Unleashed
|
||
board on QEMU ``sifive_u`` machine out of the box. This allows users to
|
||
develop and test the recommended RISC-V boot flow with a real world use
|
||
case: ZSBL (in QEMU) loads U-Boot SPL from SD card or SPI flash to L2LIM,
|
||
then U-Boot SPL loads the combined payload image of OpenSBI fw_dynamic
|
||
firmware and U-Boot proper.
|
||
|
||
However sometimes we want to have a quick test of booting U-Boot on QEMU
|
||
without the needs of preparing the SPI flash or SD card images, an alternate
|
||
way can be used, which is to create a U-Boot S-mode image by modifying the
|
||
configuration of U-Boot:
|
||
|
||
.. code-block:: bash
|
||
|
||
$ export CROSS_COMPILE=riscv64-linux-
|
||
$ make sifive_unleashed_defconfig
|
||
$ ./scripts/config --enable OF_BOARD
|
||
$ ./scripts/config --disable BINMAN_FDT
|
||
$ ./scripts/config --disable SPL
|
||
$ make olddefconfig
|
||
|
||
This changes U-Boot to use the QEMU generated device tree blob, and bypass
|
||
running the U-Boot SPL stage.
|
||
|
||
Boot the 64-bit U-Boot S-mode image directly:
|
||
|
||
.. code-block:: bash
|
||
|
||
$ qemu-system-riscv64 -M sifive_u -smp 5 -m 2G \
|
||
-display none -serial stdio \
|
||
-kernel /path/to/u-boot.bin
|
||
|
||
It's possible to create a 32-bit U-Boot S-mode image as well.
|
||
|
||
.. code-block:: bash
|
||
|
||
$ export CROSS_COMPILE=riscv64-linux-
|
||
$ make sifive_unleashed_defconfig
|
||
$ ./scripts/config --disable ARCH_RV64I
|
||
$ ./scripts/config --enable ARCH_RV32I
|
||
$ ./scripts/config --set-val TEXT_BASE 0x80400000
|
||
$ ./scripts/config --enable OF_BOARD
|
||
$ ./scripts/config --disable BINMAN_FDT
|
||
$ ./scripts/config --disable SPL
|
||
$ make olddefconfig
|
||
|
||
Use the same command line options to boot the 32-bit U-Boot S-mode image:
|
||
|
||
.. code-block:: bash
|
||
|
||
$ qemu-system-riscv32 -M sifive_u -smp 5 -m 2G \
|
||
-display none -serial stdio \
|
||
-kernel /path/to/u-boot.bin
|
||
|
||
.. _genimage: https://github.com/pengutronix/genimage
|