2023-06-27 15:15:44 +08:00
![Pine64 Star64 64-bit RISC-V SBC ](https://lupyuen.github.io/images/star64-title.jpg )
2023-06-26 15:38:26 +08:00
# Apache NuttX RTOS for Pine64 Star64 64-bit RISC-V SBC (StarFive JH7110)
2023-06-26 17:04:21 +08:00
Read the articles...
2023-06-26 15:38:26 +08:00
2023-07-25 15:27:14 +08:00
- ["Star64 JH7110 + NuttX RTOS: RISC-V Semihosting and Initial RAM Disk" ](https://lupyuen.github.io/articles/semihost )
2023-07-17 18:07:03 +08:00
- ["Star64 JH7110 + NuttX RTOS: RISC-V Privilege Levels and UART Registers" ](https://lupyuen.github.io/articles/privilege )
2023-07-18 11:14:16 +08:00
- ["Apache NuttX RTOS on RISC-V: Star64 JH7110 SBC" ](https://lupyuen.github.io/articles/nuttx2 )
2023-07-12 16:31:16 +08:00
- ["Star64 JH7110 RISC-V SBC: Boot from Network with U-Boot and TFTP" ](https://lupyuen.github.io/articles/tftp )
2023-07-18 11:14:16 +08:00
Earlier articles...
2023-07-03 10:08:51 +08:00
- ["Booting RISC-V Linux on Star64 JH7110 SBC" ](https://lupyuen.github.io/articles/linux )
2023-07-02 18:14:12 +08:00
2023-07-03 10:08:51 +08:00
- ["Inspecting the RISC-V Linux Images for Star64 JH7110 SBC" ](https://lupyuen.github.io/articles/star64 )
2023-06-28 21:20:24 +08:00
2023-06-26 15:38:26 +08:00
- ["64-bit RISC-V with Apache NuttX Real-Time Operating System" ](https://lupyuen.github.io/articles/riscv )
2023-06-26 17:04:21 +08:00
- ["Rolling to RISC-V" ](https://lupyuen.github.io/articles/pinephone2#rolling-to-risc-v )
2023-06-26 17:13:54 +08:00
Let's port Apache NuttX RTOS to [Pine64 Star64 ](https://wiki.pine64.org/wiki/STAR64 ) 64-bit RISC-V SBC!
(Based on [StarFive JH7110 SoC ](https://doc-en.rvspace.org/Doc_Center/jh7110.html ))
2023-06-26 17:06:24 +08:00
Hopefully NuttX will run on Pine64 PineTab-V, which is also based on StarFive JH7110 SoC.
2023-06-26 16:10:23 +08:00
# Linux Images for Star64
2023-06-26 17:02:37 +08:00
Let's examine the Linux Images for Star64 SBC, to see how U-Boot Bootloader is configured. (We'll boot NuttX later with U-Boot)
2023-06-26 16:10:23 +08:00
According to [Software Releases for Star64 ](https://wiki.pine64.org/wiki/STAR64#Software_releases ), we have...
2023-06-26 17:02:37 +08:00
- [Yocto Images ](https://github.com/Fishwaldo/meta-pine64 ) at [pine64.my-ho.st ](https://pine64.my-ho.st:8443/ )
2023-06-26 16:10:23 +08:00
2023-06-26 16:47:40 +08:00
Let's inspect [star64-image-minimal ](https://pine64.my-ho.st:8443/star64-image-minimal-star64-1.2.wic.bz2 )
2023-06-26 16:10:23 +08:00
2023-06-26 17:02:37 +08:00
- [Armbian Images ](https://www.armbian.com/star64/ )
2023-06-26 16:10:23 +08:00
Let's inspect [Armbian 23.8 Lunar (Minimal) ](https://github.com/armbianro/os/releases/download/23.8.0-trunk.56/Armbian_23.8.0-trunk.56_Star64_lunar_edge_5.15.0_minimal.img.xz )
2023-06-28 10:03:55 +08:00
Current state of RISC-V Linux: [Linux on RISC-V (2022) ](https://docs.google.com/presentation/d/1A0A6DnGyXR_MPpeg7QunQbv_yePPqid_uRswQe8Sj8M/edit#slide=id.p )
2023-06-26 16:10:23 +08:00
# Armbian Image for Star64
2023-06-26 16:59:17 +08:00
Let's inspect the Armbian Image for Star64: [Armbian 23.8 Lunar (Minimal) ](https://github.com/armbianro/os/releases/download/23.8.0-trunk.56/Armbian_23.8.0-trunk.56_Star64_lunar_edge_5.15.0_minimal.img.xz )
Uncompress the .xz, mount the .img file on Linux / macOS / Windows as an ISO Volume.
2023-06-28 09:51:38 +08:00
The image contains 1 used partition: `armbi_root` (612 MB) that contains the Linux Root Filesystem.
2023-06-26 16:59:17 +08:00
2023-06-28 22:35:12 +08:00
Plus one unused partition (4 MB) at the top. (Partition Table)
2023-06-28 07:30:00 +08:00
![Armbian Image for Star64 ](https://lupyuen.github.io/images/star64-armbian.png )
2023-07-01 18:10:38 +08:00
We see the U-Boot Bootloader Configuration at `armbi_root/boot/extlinux/extlinux.conf` ...
2023-06-26 16:59:17 +08:00
```text
2023-07-01 18:10:38 +08:00
label Armbian
kernel /boot/Image
initrd /boot/uInitrd
fdt /boot/dtb/starfive/jh7110-star64-pine64.dtb
append root=UUID=99f62df4-be35-475c-99ef-2ba3f74fe6b5 console=ttyS0,115200n8 console=tty0 earlycon=sbi rootflags=data=writeback stmmaceth=chain_mode:1 rw rw no_console_suspend consoleblank=0 fsck.fix=yes fsck.repair=yes net.ifnames=0 splash plymouth.ignore-serial-consoles
```
2023-06-26 16:59:17 +08:00
2023-07-01 18:10:38 +08:00
This says that U-Boot will load the Linux Kernel from `armbi_root/boot/Image`
2023-06-26 16:10:23 +08:00
2023-07-01 18:10:38 +08:00
Which is sym-linked to `armbi_root/boot/vmlinuz-5.15.0-starfive2`
2023-06-26 16:59:17 +08:00
2023-07-01 18:10:38 +08:00
_Where in RAM will the Kernel Image be loaded?_
2023-06-26 16:59:17 +08:00
2023-07-01 18:10:38 +08:00
According to [__kernel_addr_r__ ](https://u-boot.readthedocs.io/en/latest/develop/bootstd.html#environment-variables ) from the [__Default U-Boot Settings__ ](https://github.com/lupyuen/nuttx-star64#u-boot-settings-for-star64 ), the Linux Kernel will be loaded at RAM Address __ `0x4020` `0000` __...
2023-06-26 16:59:17 +08:00
2023-07-01 18:10:38 +08:00
```text
kernel_addr_r=0x40200000
2023-06-26 16:59:17 +08:00
```
2023-06-26 16:10:23 +08:00
2023-07-01 18:10:38 +08:00
[(Source) ](https://github.com/lupyuen/nuttx-star64#u-boot-settings-for-star64 )
2023-06-26 17:26:10 +08:00
2023-07-01 18:10:38 +08:00
_Everything looks hunky dory?_
Nope the [__Flattened Device Tree (FDT)__ ](https://u-boot.readthedocs.io/en/latest/develop/devicetree/index.html ) is missing!
2023-06-26 17:26:10 +08:00
2023-07-01 18:10:38 +08:00
But the Flattened Device Tree (FDT) is missing! `/boot/dtb/starfive/jh7110-star64-pine64.dtb`
2023-06-26 17:44:47 +08:00
2023-07-01 18:10:38 +08:00
```text
fdt /boot/dtb/starfive/jh7110-star64-pine64.dtb
```
2023-06-26 17:44:47 +08:00
2023-07-01 18:10:38 +08:00
Which means that Armbian will [__fail to boot__ ](https://github.com/lupyuen/nuttx-star64#boot-armbian-on-star64 ) on Star64!
2023-06-27 18:43:27 +08:00
```text
2023-07-01 18:10:38 +08:00
Retrieving file: /boot/uInitrd
10911538 bytes read in 466 ms (22.3 MiB/s)
Retrieving file: /boot/Image
22040576 bytes read in 936 ms (22.5 MiB/s)
Retrieving file: /boot/dtb/starfive/jh7110-star64-pine64.dtb
Failed to load '/boot/dtb/starfive/jh7110-star64-pine64.dtb'
2023-06-27 18:43:27 +08:00
```
2023-07-01 18:10:38 +08:00
[(Source) ](https://github.com/lupyuen/nuttx-star64#boot-armbian-on-star64 )
2023-06-26 18:39:13 +08:00
2023-07-01 18:10:38 +08:00
The missing Device Tree is noted in this [__Pine64 Forum Post__ ](https://forum.pine64.org/showthread.php?tid=18276&pid=117607#pid117607 ). So we might need to check back later for the Official Armbian Image, if it's fixed.
[(__balbes150__ suggests that we try this Armbian Image instead) ](https://forum.pine64.org/showthread.php?tid=18420&pid=118331#pid118331 )
2023-06-26 18:39:13 +08:00
2023-06-27 18:43:27 +08:00
2023-07-01 18:10:38 +08:00
For Reference: Here's the list of __Supported Device Trees__ ...
2023-06-28 09:51:38 +08:00
```text
→ ls /Volumes/armbi_root/boot/dtb-5.15.0-starfive2/starfive
evb-overlay jh7110-evb-usbdevice.dtb
jh7110-evb-can-pdm-pwmdac.dtb jh7110-evb.dtb
jh7110-evb-dvp-rgb2hdmi.dtb jh7110-fpga.dtb
jh7110-evb-i2s-ac108.dtb jh7110-visionfive-v2-A10.dtb
jh7110-evb-pcie-i2s-sd.dtb jh7110-visionfive-v2-A11.dtb
jh7110-evb-spi-uart2.dtb jh7110-visionfive-v2-ac108.dtb
jh7110-evb-uart1-rgb2hdmi.dtb jh7110-visionfive-v2-wm8960.dtb
jh7110-evb-uart4-emmc-spdif.dtb jh7110-visionfive-v2.dtb
jh7110-evb-uart5-pwm-i2c-tdm.dtb vf2-overlay
```
2023-06-27 18:43:27 +08:00
2023-07-01 18:10:38 +08:00
And here are the other files in __ /boot__...
2023-06-26 17:12:28 +08:00
```text
→ ls -l /Volumes/armbi_root/boot
total 94416
2023-07-01 18:10:38 +08:00
lrwxrwxrwx 24 Image -> vmlinuz-5.15.0-starfive2
-rw-r--r-- 4276712 System.map-5.15.0-starfive2
-rw-r--r-- 1536 armbian_first_run.txt.template
-rw-r--r-- 38518 boot.bmp
-rw-r--r-- 144938 config-5.15.0-starfive2
lrwxrwxrwx 20 dtb -> dtb-5.15.0-starfive2
drwxr-xr-x 0 dtb-5.15.0-starfive2
drwxrwxr-x 0 extlinux
lrwxrwxrwx 27 initrd.img -> initrd.img-5.15.0-starfive2
-rw-r--r-- 10911474 initrd.img-5.15.0-starfive2
lrwxrwxrwx 27 initrd.img.old -> initrd.img-5.15.0-starfive2
-rw-rw-r-- 341 uEnv.txt
lrwxrwxrwx 24 uInitrd -> uInitrd-5.15.0-starfive2
-rw-r--r-- 10911538 uInitrd-5.15.0-starfive2
lrwxrwxrwx 24 vmlinuz -> vmlinuz-5.15.0-starfive2
-rw-r--r-- 22040576 vmlinuz-5.15.0-starfive2
lrwxrwxrwx 24 vmlinuz.old -> vmlinuz-5.15.0-starfive2
2023-06-26 17:12:28 +08:00
```
2023-06-28 09:51:38 +08:00
TODO: Explain `boot/uInitrd` RAM Disk
2023-06-26 16:10:23 +08:00
# Yocto Image for Star64
2023-06-26 16:47:40 +08:00
Let's inspect the Yocto Image for Star64: [star64-image-minimal ](https://pine64.my-ho.st:8443/star64-image-minimal-star64-1.2.wic.bz2 )
2023-06-26 16:10:23 +08:00
Uncompress the .bz2, rename as .img. Balena Etcher won't work with .bz2 files!
2023-06-26 16:47:40 +08:00
Write the .img to a microSD Card with Balena Etcher. Insert the microSD Card into a Linux Machine. (Like Pinebook Pro)
2023-06-26 17:26:10 +08:00
We see 4 used partitions...
2023-06-26 16:47:40 +08:00
2023-06-28 22:35:12 +08:00
- spl (2 MB): [Secondary Program Loader ](https://u-boot.readthedocs.io/en/latest/board/starfive/visionfive2.html#flashing )
2023-06-26 16:47:40 +08:00
2023-06-28 22:35:12 +08:00
- uboot (4 MB): [U-Boot Bootloader ](https://u-boot.readthedocs.io/en/latest/board/starfive/visionfive2.html#flashing )
2023-06-26 16:47:40 +08:00
2023-06-27 08:54:44 +08:00
- boot (380 MB): U-Boot Configuration and Linux Kernel Image
2023-06-26 16:47:40 +08:00
2023-06-26 16:59:17 +08:00
- root (686 MB): Linux Root Filesystem
2023-06-26 16:47:40 +08:00
2023-06-28 22:35:12 +08:00
Plus one unused partition (2 MB) at the top. (Partition Table)
2023-06-26 16:47:40 +08:00
![Yocto Image for Star64 ](https://lupyuen.github.io/images/star64-yocto.png )
2023-06-26 16:59:17 +08:00
`boot` partition has 2 files...
```text
2023-07-22 08:45:49 +08:00
$ ls -l /run/media/$USER/boot
2023-06-26 16:59:17 +08:00
total 14808
2023-07-22 08:45:49 +08:00
-rw-r--r-- 1 15151064 Apr 6 2011 fitImage
-rw-r--r-- 1 1562 Apr 6 2011 vf2_uEnv.txt
2023-06-26 16:59:17 +08:00
```
2023-06-26 17:02:37 +08:00
`boot/vf2_uEnv.txt` contains the U-Boot Bootloader Configuration...
2023-06-26 16:59:17 +08:00
```text
# This is the sample jh7110_uEnv.txt file for starfive visionfive U-boot
# The current convention (SUBJECT TO CHANGE) is that this file
# will be loaded from the third partition on the
# MMC card.
#devnum=1
partnum=3
# The FIT file to boot from
fitfile=fitImage
# for debugging boot
bootargs_ext=if test ${devnum} = 0; then setenv bootargs "earlyprintk console=tty1 console=ttyS0,115200 rootwait earlycon=sbi root=/dev/mmcblk0p4"; else setenv bootargs "earlyprintk console=tty1 console=ttyS0,115200 rootwait earlycon=sbi root=/dev/mmcblk1p4"; fi;
#bootargs=earlyprintk console=ttyS0,115200 debug rootwait earlycon=sbi root=/dev/mmcblk1p4
# for addr info
fileaddr=0xa0000000
fdtaddr=0x46000000
# boot Linux flat or compressed 'Image' stored at 'kernel_addr_r'
kernel_addr_r=0x40200000
irdaddr=46100000
irdsize=5f00000
# Use the FDT in the FIT image..
setupfdt1=fdt addr ${fdtaddr}; fdt resize;
setupird=setexpr irdend ${irdaddr} + ${irdsize}; fdt set /chosen linux,initrd-start < 0x0 0x $ { irdaddr } > ; fdt set /chosen linux,initrd-end < 0x0 0x $ { irdend } >
setupfdt2=fdt set /chosen bootargs "${bootargs}";
bootwait=setenv _delay ${bootdelay}; echo ${_delay}; while test ${_delay} > 0; do sleep 1; setexpr _delay ${_delay} - 1; echo ${_delay}; done
boot2=run bootargs_ext; mmc dev ${devnum}; fatload mmc ${devnum}:${partnum} ${fileaddr} ${fitfile}; bootm start ${fileaddr}; run setupfdt1;run setupird;run setupfdt2; bootm loados ${fileaddr}; run chipa_set_linux; run cpu_vol_set; echo "Booting kernel in"; booti ${kernel_addr_r} ${irdaddr}:${filesize} ${fdtaddr}
```
2023-06-26 19:12:42 +08:00
[`kernel_addr_r` ](https://u-boot.readthedocs.io/en/latest/develop/bootstd.html#environment-variables ) says that Linux Kernel will be loaded at `0x4020` `0000` ...
2023-06-26 17:26:10 +08:00
```text
# boot Linux flat or compressed 'Image' stored at 'kernel_addr_r'
kernel_addr_r=0x40200000
```
2023-06-27 08:54:44 +08:00
Yocto boots from the [Flat Image Tree (FIT) ](https://u-boot.readthedocs.io/en/latest/usage/fit/index.html# ): `boot/fitImage`
2023-06-26 18:39:13 +08:00
2023-06-26 17:12:28 +08:00
Yocto's `root/boot` looks different from Armbian...
```text
2023-07-22 08:45:49 +08:00
$ ls -l /run/media/$USER/root/boot
2023-06-26 17:12:28 +08:00
total 24376
2023-07-22 08:45:49 +08:00
lrwxrwxrwx 1 17 Mar 9 2018 fitImage -> fitImage-5.15.107
-rw-r--r-- 1 9807808 Mar 9 2018 fitImage-5.15.107
-rw-r--r-- 1 15151064 Mar 9 2018 fitImage-initramfs-5.15.107
2023-06-26 17:12:28 +08:00
```
2023-06-26 17:26:10 +08:00
# Boot NuttX with U-Boot Bootloader
2023-07-01 18:13:05 +08:00
_Will we boot NuttX with Armbian or Yocto settings?_
2023-06-26 18:39:13 +08:00
Armbian looks simpler, since it uses a plain Linux Kernel Image File `Image` . (Instead of Yocto's complicated Flat Image Tree)
2023-06-27 18:43:27 +08:00
Hence we'll overwrite Armbian's `armbi_root/boot/Image` by the NuttX Kernel Image.
2023-07-01 18:13:05 +08:00
We'll compile NuttX Kernel to boot at `0x4020` `0000` .
2023-06-27 18:43:27 +08:00
NuttX Kernel will begin with a RISC-V Linux Header. (See next section)
We'll use a Temporary File for the Flattened Device Tree (FDT) since it's missing from Armbian.
2023-06-26 18:39:13 +08:00
2023-06-26 19:13:24 +08:00
# Inside the Armbian Kernel Image
2023-06-26 18:39:13 +08:00
2023-06-26 20:24:18 +08:00
_What's inside the Armbian Linux Kernel Image?_
2023-06-27 08:57:04 +08:00
Let's look inside `armbi_root/boot/vmlinuz-5.15.0-starfive2` ...
2023-06-26 20:24:18 +08:00
![Armbian Kernel Image ](https://lupyuen.github.io/images/star64-kernel.png )
See the "RISCV" at `0x30` ? That's the Magic Number for the RISC-V Linux Image Header!
- ["Boot image header in RISC-V Linux" ](https://www.kernel.org/doc/html/latest/riscv/boot-image-header.html )
```text
u32 code0; /* Executable code */
u32 code1; /* Executable code */
u64 text_offset; /* Image load offset, little endian */
u64 image_size; /* Effective Image size, little endian */
u64 flags; /* kernel flags, little endian */
u32 version; /* Version of this header */
u32 res1 = 0; /* Reserved */
u64 res2 = 0; /* Reserved */
u64 magic = 0x5643534952; /* Magic number, little endian, "RISCV" */
u32 magic2 = 0x05435352; /* Magic number 2, little endian, "RSC\x05" */
u32 res3; /* Reserved for PE COFF offset */
```
2023-06-30 11:28:21 +08:00
This is how we decode the RISC-V Linux Header...
- [__"Decode the RISC-V Linux Header"__ ](https://lupyuen.github.io/articles/star64#appendix-decode-the-risc-v-linux-header )
2023-06-27 08:57:04 +08:00
Let's decompile the Kernel Image...
2023-06-27 07:53:07 +08:00
2023-06-28 09:51:38 +08:00
TODO: Explain MZ and the funny RISC-V instruction at the top
2023-06-27 08:57:04 +08:00
# Decompile Armbian Kernel Image with Ghidra
We decompile the Armbian Linux Kernel Image with [Ghidra ](https://github.com/NationalSecurityAgency/ghidra ).
2023-06-27 07:53:07 +08:00
2023-06-27 08:27:19 +08:00
In Ghidra, create a New Project. Click File > Import File.
Select `armbi_root/boot/vmlinuz-5.15.0-starfive2` and enter these Import Options...
2023-06-26 20:24:18 +08:00
2023-06-26 20:36:36 +08:00
- Format: Raw Binary
2023-06-26 20:24:18 +08:00
- Language: RISCV > RV64GC (RISCV:LE:64:RV64GC:gcc)
[(StarFive JH7110 has 4 × RV64GC U74 Application Cores) ](https://doc-en.rvspace.org/JH7110/Datasheet/JH7110_DS/c_u74_quad_core.html )
2023-06-26 20:36:36 +08:00
- Options > Base Address: 0x44000000
2023-06-27 08:27:19 +08:00
(Based on the U-Boot Configuration from above)
2023-06-26 20:24:18 +08:00
![Load the Armbian Linux Kernel Image into Ghidra ](https://lupyuen.github.io/images/star64-ghidra.png )
2023-06-26 20:36:36 +08:00
![Load the Armbian Linux Kernel Image into Ghidra ](https://lupyuen.github.io/images/star64-ghidra2.png )
2023-06-27 07:53:07 +08:00
Double-click `vmlinuz-5.15.0-starfive2` , analyse the file with the Default Options.
2023-06-27 08:57:04 +08:00
Ghidra displays the Decompiled Linux Kernel...
2023-06-27 08:27:19 +08:00
![Disassembled Linux Kernel in Ghidra ](https://lupyuen.github.io/images/star64-ghidra3.png )
2023-06-28 16:14:41 +08:00
At Address `0x4400` `0002` we see a Jump to `FUN_440010c8` ...
2023-06-27 08:30:16 +08:00
2023-06-28 16:14:41 +08:00
```text
// Load -13 into Register S4
li s4,-0xd
2023-06-28 17:27:17 +08:00
2023-06-28 16:14:41 +08:00
// Jump to Actual Boot Code
j FUN_440010c8
```
Double-click `FUN_440010c8` to see the Linux Boot Code...
2023-06-27 08:27:19 +08:00
![Linux Boot Code in Ghidra ](https://lupyuen.github.io/images/star64-ghidra4.png )
2023-06-28 09:51:38 +08:00
TODO: Explain MZ and the funny RISC-V instruction at the top
2023-06-27 15:35:59 +08:00
TODO: Where is the source file?
2023-06-27 14:42:22 +08:00
TODO: Any interesting CSR Instructions?
2023-06-27 14:15:27 +08:00
# Serial Console on Star64
2023-06-27 15:35:59 +08:00
To access the Serial Console, we connect a [USB Serial Adapter ](https://pine64.com/product/serial-console-woodpecker-edition/ ) to Star64...
2023-06-27 14:15:27 +08:00
2023-07-03 10:07:36 +08:00
![Star64 JH7110 RISC-V SBC with Woodpecker USB Serial Adapter ](https://lupyuen.github.io/images/linux-title.jpg )
2023-06-27 15:11:01 +08:00
According to [Star64 Schematic ](https://files.pine64.org/doc/star64/Star64_Schematic_V1.1_20230504.pdf ), UART0 TX and RX (GPIO 5 and 6) are connected to the Pi GPIO Header (Pins 8 and 10).
2023-06-27 14:15:27 +08:00
2023-06-27 15:35:59 +08:00
Thus we connect these pins...
2023-06-27 14:15:27 +08:00
2023-06-27 15:11:01 +08:00
| Star64 GPIO Header | [USB Serial Adapter ](https://pine64.com/product/serial-console-woodpecker-edition/ ) | Wire Colour |
|:----:|:----:|:----|
| Pin 6 (GND) | GND | Brown
| Pin 8 (TX) | RX | Red
| Pin 10 (RX) | TX | Orange
2023-06-27 14:15:27 +08:00
2023-07-01 15:58:56 +08:00
Set the Voltage Jumper to 3V3. (Instead of 5V)
2023-07-03 10:07:36 +08:00
![Pine64 Woodpecker Serial Adapter ](https://lupyuen.github.io/images/star64-uart3.jpg )
2023-06-27 15:11:01 +08:00
On our computer, connect to the USB Serial Port at 115.2 kbps...
```bash
screen /dev/ttyUSB0 115200
```
2023-06-27 15:35:59 +08:00
Power up Star64. The DIP Switches for GPIO 0 and 1 default to Low and Low, so Star64 should boot from Flash Memory, which has the U-Boot Bootloader inside.
2023-06-29 08:27:07 +08:00
[(DIP Switch Labels are inverted: __"ON"__ actually means __"Low"__) ](https://wiki.pine64.org/wiki/STAR64#Prototype_Bringup_Notes )
2023-07-03 10:07:36 +08:00
![DIP Switches for GPIO 0 and 1 are set to Low and Low ](https://lupyuen.github.io/images/star64-uart2.jpg )
2023-06-27 15:35:59 +08:00
We'll see this U-Boot Bootloader Log...
2023-06-28 10:03:55 +08:00
TODO: Explain [OpenSBI ](https://www.thegoodpenguin.co.uk/blog/an-overview-of-opensbi/ )
2023-06-27 14:42:22 +08:00
# Star64 U-Boot Bootloader Log
2023-06-27 15:35:59 +08:00
Here's the log for U-Boot Bootloader on Star64 (without microSD Card)...
2023-06-27 14:44:51 +08:00
2023-06-28 17:52:49 +08:00
![U-Boot Bootloader Log ](https://lupyuen.github.io/images/star64-opensbi.jpg )
2023-06-27 14:42:22 +08:00
```text
U-Boot SPL 2021.10 (Jan 19 2023 - 04:09:41 +0800)
DDR version: dc2e84f0.
Trying to boot from SPI
OpenSBI v1.2
____ _____ ____ _____
/ __ \ / ____ | _ \_ _|
| | | |_ __ ___ _ __ | (___ | |_) || |
| | | | '_ \ / _ \ '_ \ \___ \| _ < | |
| |__| | |_) | __ / | | |____) | |_) || |_
\____/| .__/ \___|_| |_|_____/|____/_____|
| |
|_|
Platform Name : StarFive VisionFive V2
Platform Features : medeleg
Platform HART Count : 5
Platform IPI Device : aclint-mswi
Platform Timer Device : aclint-mtimer @ 4000000Hz
Platform Console Device : uart8250
Platform HSM Device : jh7110-hsm
Platform PMU Device : ---
Platform Reboot Device : pm-reset
Platform Shutdown Device : pm-reset
Firmware Base : 0x40000000
Firmware Size : 288 KB
Runtime SBI Version : 1.0
Domain0 Name : root
Domain0 Boot HART : 1
Domain0 HARTs : 0*,1*,2*,3*,4*
Domain0 Region00 : 0x0000000002000000-0x000000000200ffff (I)
Domain0 Region01 : 0x0000000040000000-0x000000004007ffff ()
Domain0 Region02 : 0x0000000000000000-0xffffffffffffffff (R,W,X)
Domain0 Next Address : 0x0000000040200000
Domain0 Next Arg1 : 0x0000000042200000
Domain0 Next Mode : S-mode
Domain0 SysReset : yes
Boot HART ID : 1
Boot HART Domain : root
Boot HART Priv Version : v1.11
Boot HART Base ISA : rv64imafdcbx
Boot HART ISA Extensions : none
Boot HART PMP Count : 8
Boot HART PMP Granularity : 4096
Boot HART PMP Address Bits: 34
Boot HART MHPM Count : 2
Boot HART MIDELEG : 0x0000000000000222
Boot HART MEDELEG : 0x000000000000b109
U-Boot 2021.10 (Jan 19 2023 - 04:09:41 +0800), Build: jenkins-github_visionfive2-6
CPU: rv64imacu
Model: StarFive VisionFive V2
DRAM: 8 GiB
MMC: sdio0@16010000: 0, sdio1@16020000: 1
Loading Environment from SPIFlash... SF: Detected gd25lq128 with page size 256 Bytes, erase size 4 KiB, total 16 MiB
*** Warning - bad CRC, using default environment
StarFive EEPROM format v2
--------EEPROM INFO--------
Vendor : PINE64
Product full SN: STAR64V1-2310-D008E000-00000003
data version: 0x2
PCB revision: 0xc1
BOM revision: A
Ethernet MAC0 address: 6c:cf:39:00:75:5d
Ethernet MAC1 address: 6c:cf:39:00:75:5e
--------EEPROM INFO--------
In: serial@10000000
Out: serial@10000000
Err: serial@10000000
Model: StarFive VisionFive V2
Net: eth0: ethernet@16030000, eth1: ethernet@16040000
Card did not respond to voltage select! : -110
Card did not respond to voltage select! : -110
bootmode flash device 0
Card did not respond to voltage select! : -110
Hit any key to stop autoboot: 2 1 0
Card did not respond to voltage select! : -110
Couldn't find partition mmc 0:3
Can't set block device
Importing environment from mmc0 ...
## Warning: Input data exceeds 1048576 bytes - truncated
## Info: input data size = 1048578 = 0x100002
Card did not respond to voltage select! : -110
Couldn't find partition mmc 1:2
Can't set block device
## Warning: defaulting to text format
## Error: "boot2" not defined
Card did not respond to voltage select! : -110
ethernet@16030000 Waiting for PHY auto negotiation to complete......... TIMEOUT !
phy_startup() failed: -110FAILED: -110ethernet@16040000 Waiting for PHY auto negotiation to complete......... TIMEOUT !
phy_startup() failed: -110FAILED: -110ethernet@16030000 Waiting for PHY auto negotiation to complete......... TIMEOUT !
phy_startup() failed: -110FAILED: -110ethernet@16040000 Waiting for PHY auto negotiation to complete......... TIMEOUT !
phy_startup() failed: -110FAILED: -110StarFive #
StarFive #
2023-06-28 09:51:38 +08:00
```
Which is OK because we haven't inserted a microSD Card.
2023-06-28 21:20:24 +08:00
## U-Boot Commands for Star64
2023-06-28 09:51:38 +08:00
Here are the U-Boot Commands...
```text
2023-06-27 14:42:22 +08:00
StarFive # help
? - alias for 'help'
base - print or set address offset
bdinfo - print Board Info structure
blkcache - block cache diagnostics and control
boot - boot default, i.e., run 'bootcmd'
bootd - boot default, i.e., run 'bootcmd'
bootefi - Boots an EFI payload from memory
bootelf - Boot from an ELF image in memory
booti - boot Linux kernel 'Image' format from memory
bootm - boot application image from memory
bootp - boot image via network using BOOTP/TFTP protocol
bootvx - Boot vxWorks from an ELF image
cmp - memory compare
config - print .config
coninfo - print console devices and information
cp - memory copy
cpu - display information about CPUs
crc32 - checksum calculation
dhcp - boot image via network using DHCP/TFTP protocol
dm - Driver model low level access
echo - echo args to console
editenv - edit environment variable
eeprom - EEPROM sub-system
efidebug - Configure UEFI environment
env - environment handling commands
erase - erase FLASH memory
eraseenv - erase environment variables from persistent storage
exit - exit script
ext2load - load binary file from a Ext2 filesystem
ext2ls - list files in a directory (default /)
ext4load - load binary file from a Ext4 filesystem
ext4ls - list files in a directory (default /)
ext4size - determine a file's size
ext4write - create a file in the root directory
false - do nothing, unsuccessfully
fatinfo - print information about filesystem
fatload - load binary file from a dos filesystem
fatls - list files in a directory (default /)
fatmkdir - create a directory
fatrm - delete a file
fatsize - determine a file's size
fatwrite - write file into a dos filesystem
fdt - flattened device tree utility commands
flinfo - print FLASH memory information
fstype - Look up a filesystem type
fstypes - List supported filesystem types
fsuuid - Look up a filesystem UUID
go - start application at address 'addr'
gpio - query and control gpio pins
gpt - GUID Partition Table
gzwrite - unzip and write memory to block device
help - print command description/usage
i2c - I2C sub-system
iminfo - print header information for application image
imxtract - extract a part of a multi-image
itest - return true/false on integer compare
ln - Create a symbolic link
load - load binary file from a filesystem
loadb - load binary file over serial line (kermit mode)
loads - load S-Record file over serial line
loadx - load binary file over serial line (xmodem mode)
loady - load binary file over serial line (ymodem mode)
log - log system
loop - infinite loop on address range
ls - list files in a directory (default /)
lzmadec - lzma uncompress a memory region
mac - display and program the system ID and MAC addresses in EEPROM
md - memory display
misc - Access miscellaneous devices with MISC uclass driver APIs
mm - memory modify (auto-incrementing address)
mmc - MMC sub system
mmcinfo - display MMC info
mw - memory write (fill)
net - NET sub-system
nfs - boot image via network using NFS protocol
nm - memory modify (constant address)
panic - Panic with optional message
part - disk partition related commands
ping - send ICMP ECHO_REQUEST to network host
pinmux - show pin-controller muxing
printenv - print environment variables
protect - enable or disable FLASH write protection
random - fill memory with random pattern
reset - Perform RESET of the CPU
run - run commands in an environment variable
save - save file to a filesystem
saveenv - save environment variables to persistent storage
setenv - set environment variables
setexpr - set environment variable as the result of eval expression
sf - SPI flash sub-system
showvar - print local hushshell variables
size - determine a file's size
sleep - delay execution for some time
source - run script from memory
sysboot - command to get and boot from syslinux files
test - minimal test like /bin/sh
tftpboot - boot image via network using TFTP protocol
tftpput - TFTP put command, for uploading files to a server
true - do nothing, successfully
unlz4 - lz4 uncompress a memory region
unzip - unzip a memory region
version - print monitor, compiler and linker version
2023-06-28 09:51:38 +08:00
```
2023-06-28 21:20:24 +08:00
## U-Boot Settings for Star64
2023-06-28 09:51:38 +08:00
Here are the U-Boot Settings...
2023-06-27 14:42:22 +08:00
2023-06-28 09:51:38 +08:00
```text
2023-06-27 14:42:22 +08:00
StarFive # printenv
baudrate=115200
boot_a_script=load ${devtype} ${devnum}:${distro_bootpart} ${scriptaddr} ${prefix}${script}; source ${scriptaddr}
boot_efi_binary=load ${devtype} ${devnum}:${distro_bootpart} ${kernel_addr_r} efi/boot/bootriscv64.efi; if fdt addr ${fdt_addr_r}; then bootefi ${kernel_addr_r} ${fdt_addr_r};else bootefi ${kernel_addr_r} ${fdtcontroladdr};fi
boot_efi_bootmgr=if fdt addr ${fdt_addr_r}; then bootefi bootmgr ${fdt_addr_r};else bootefi bootmgr;fi
boot_extlinux=sysboot ${devtype} ${devnum}:${distro_bootpart} any ${scriptaddr} ${prefix}${boot_syslinux_conf}
boot_prefixes=/ /boot/
boot_script_dhcp=boot.scr.uimg
boot_scripts=boot.scr.uimg boot.scr
boot_syslinux_conf=extlinux/extlinux.conf
boot_targets=mmc0 dhcp
bootargs=console=ttyS0,115200 debug rootwait earlycon=sbi
bootcmd=run load_vf2_env;run importbootenv;run load_distro_uenv;run boot2;run distro_bootcmd
bootcmd_dhcp=devtype=dhcp; if dhcp ${scriptaddr} ${boot_script_dhcp}; then source ${scriptaddr}; fi;setenv efi_fdtfile ${fdtfile}; setenv efi_old_vci ${bootp_vci};setenv efi_old_arch ${bootp_arch};setenv bootp_vci PXEClient:Arch:00027:UNDI:003000;setenv bootp_arch 0x1b;if dhcp ${kernel_addr_r}; then tftpboot ${fdt_addr_r} dtb/${efi_fdtfile};if fdt addr ${fdt_addr_r}; then bootefi ${kernel_addr_r} ${fdt_addr_r}; else bootefi ${kernel_addr_r} ${fdtcontroladdr};fi;fi;setenv bootp_vci ${efi_old_vci};setenv bootp_arch ${efi_old_arch};setenv efi_fdtfile;setenv efi_old_arch;setenv efi_old_vci;
bootcmd_distro=run fdt_loaddtb; run fdt_sizecheck; run set_fdt_distro; sysboot mmc ${fatbootpart} fat c0000000 ${bootdir}/${boot_syslinux_conf};
bootcmd_mmc0=devnum=0; run mmc_boot
bootdelay=2
bootdir=/boot
bootenv=uEnv.txt
bootmode=flash
bootpart=0:3
chip_vision=UNKOWN
chipa_gmac_set=fdt set /soc/ethernet@16030000/ethernet-phy@0 tx_inverted_10 < 0x0 > ;fdt set /soc/ethernet@16030000/ethernet-phy@0 tx_inverted_100 < 0x0 > ;fdt set /soc/ethernet@16030000/ethernet-phy@0 tx_inverted_1000 < 0x0 > ;fdt set /soc/ethernet@16030000/ethernet-phy@0 tx_delay_sel < 0x9 > ;fdt set /soc/ethernet@16040000/ethernet-phy@1 tx_inverted_10 < 0x0 > ;fdt set /soc/ethernet@16040000/ethernet-phy@1 tx_inverted_100 < 0x0 > ;fdt set /soc/ethernet@16040000/ethernet-phy@1 tx_inverted_1000 < 0x0 > ;fdt set /soc/ethernet@16040000/ethernet-phy@1 tx_delay_sel < 0x9 >
chipa_set=if test ${chip_vision} = A; then run chipa_gmac_set;fi;
chipa_set_linux=fdt addr ${fdt_addr_r};run visionfive2_mem_set;run chipa_set;
chipa_set_linux_force=fdt addr ${fdt_addr_r};run visionfive2_mem_set;run chipa_gmac_set;
chipa_set_uboot=fdt addr ${uboot_fdt_addr};run chipa_set;
chipa_set_uboot_force=fdt addr ${uboot_fdt_addr};run chipa_gmac_set;
devnum=0
distro_bootcmd=for target in ${boot_targets}; do run bootcmd_${target}; done
distroloadaddr=0xb0000000
efi_dtb_prefixes=/ /dtb/ /dtb/current/
eth0addr=6c:cf:39:00:75:5d
eth1addr=6c:cf:39:00:75:5e
ethact=ethernet@16030000
ethaddr=6c:cf:39:00:75:5d
ext4bootenv=ext4load mmc ${bootpart} ${loadaddr} ${bootdir}/${bootenv}
fatbootpart=1:2
fdt_addr_r=0x46000000
fdt_high=0xffffffffffffffff
fdt_loaddtb=fatload mmc ${fatbootpart} ${fdt_addr_r} ${bootdir}/dtbs/${fdtfile}; fdt addr ${fdt_addr_r};
fdt_sizecheck=fatsize mmc ${fatbootpart} ${bootdir}/dtbs/${fdtfile};
fdtaddr=fffc6aa0
fdtcontroladdr=fffc6aa0
fdtfile=starfive/starfive_visionfive2.dtb
importbootenv=echo Importing environment from mmc${devnum} ...; env import -t ${loadaddr} ${filesize}
initrd_high=0xffffffffffffffff
ipaddr=192.168.120.230
kernel_addr_r=0x40200000
load_distro_uenv=fatload mmc ${fatbootpart} ${distroloadaddr} ${bootdir}/${bootenv}; env import ${distroloadaddr} 17c;
load_efi_dtb=load ${devtype} ${devnum}:${distro_bootpart} ${fdt_addr_r} ${prefix}${efi_fdtfile}
load_vf2_env=fatload mmc ${bootpart} ${loadaddr} ${testenv}
loadaddr=0xa0000000
loadbootenv=fatload mmc ${bootpart} ${loadaddr} ${bootenv}
memory_addr=40000000
memory_size=200000000
mmc_boot=if mmc dev ${devnum}; then devtype=mmc; run scan_dev_for_boot_part; fi
mmcbootenv=run scan_mmc_dev; setenv bootpart ${devnum}:${mmcpart}; if mmc rescan; then run loadbootenv & & run importbootenv; run ext4bootenv & & run importbootenv; if test -n $uenvcmd; then echo Running uenvcmd ...; run uenvcmd; fi; fi
mmcpart=3
netmask=255.255.255.0
partitions=name=loader1,start=17K,size=1M,type=${type_guid_gpt_loader1};name=loader2,size=4MB,type=${type_guid_gpt_loader2};name=system,size=-,bootable,type=${type_guid_gpt_system};
preboot=run chipa_set_uboot;run mmcbootenv
pxefile_addr_r=0x45900000
ramdisk_addr_r=0x46100000
scan_dev_for_boot=echo Scanning ${devtype} ${devnum}:${distro_bootpart}...; for prefix in ${boot_prefixes}; do run scan_dev_for_extlinux; run scan_dev_for_scripts; done;run scan_dev_for_efi;
scan_dev_for_boot_part=part list ${devtype} ${devnum} -bootable devplist; env exists devplist || setenv devplist 1; for distro_bootpart in ${devplist}; do if fstype ${devtype} ${devnum}:${distro_bootpart} bootfstype; then run scan_dev_for_boot; fi; done; setenv devplist
scan_dev_for_efi=setenv efi_fdtfile ${fdtfile}; for prefix in ${efi_dtb_prefixes}; do if test -e ${devtype} ${devnum}:${distro_bootpart} ${prefix}${efi_fdtfile}; then run load_efi_dtb; fi;done;run boot_efi_bootmgr;if test -e ${devtype} ${devnum}:${distro_bootpart} efi/boot/bootriscv64.efi; then echo Found EFI removable media binary efi/boot/bootriscv64.efi; run boot_efi_binary; echo EFI LOAD FAILED: continuing...; fi; setenv efi_fdtfile
scan_dev_for_extlinux=if test -e ${devtype} ${devnum}:${distro_bootpart} ${prefix}${boot_syslinux_conf}; then echo Found ${prefix}${boot_syslinux_conf}; run boot_extlinux; echo SCRIPT FAILED: continuing...; fi
scan_dev_for_scripts=for script in ${boot_scripts}; do if test -e ${devtype} ${devnum}:${distro_bootpart} ${prefix}${script}; then echo Found U-Boot script ${prefix}${script}; run boot_a_script; echo SCRIPT FAILED: continuing...; fi; done
scan_mmc_dev=if test ${bootmode} = flash; then if mmc dev ${devnum}; then echo found device ${devnum};else setenv devnum 0;mmc dev 0;fi; fi; echo bootmode ${bootmode} device ${devnum};
scan_sf_for_scripts=${devtype} read ${scriptaddr} ${script_offset_f} ${script_size_f}; source ${scriptaddr}; echo SCRIPT FAILED: continuing...
script_offset_f=0x1fff000
script_size_f=0x1000
scriptaddr=0x43900000
serial#=STAR64V1-2310-D008E000-00000003
set_fdt_distro=if test ${chip_vision} = A; then if test ${memory_size} = 200000000; then run chipa_gmac_set;run visionfive2_mem_set;fatwrite mmc ${fatbootpart} ${fdt_addr_r} ${bootdir}/dtbs/${fdtfile} ${filesize};else run chipa_gmac_set;run visionfive2_mem_set;fatwrite mmc ${fatbootpart} ${fdt_addr_r} ${bootdir}/dtbs/${fdtfile} ${filesize};fi;else run visionfive2_mem_set;fatwrite mmc ${fatbootpart} ${fdt_addr_r} ${bootdir}/dtbs/${fdtfile} ${filesize};fi;
sf_boot=if sf probe ${busnum}; then devtype=sf; run scan_sf_for_scripts; fi
stderr=serial@10000000
stdin=serial@10000000
stdout=serial@10000000
testenv=vf2_uEnv.txt
type_guid_gpt_loader1=5B193300-FC78-40CD-8002-E86C45580B47
type_guid_gpt_loader2=2E54B353-1271-4842-806F-E436D6AF6985
type_guid_gpt_system=0FC63DAF-8483-4772-8E79-3D69D8477DE4
uboot_fdt_addr=0xfffc6aa0
ver=U-Boot 2021.10 (Jan 19 2023 - 04:09:41 +0800)
visionfive2_mem_set=fdt memory ${memory_addr} ${memory_size};
Environment size: 7246/65532 bytes
StarFive #
```
2023-06-27 18:03:46 +08:00
2023-06-27 18:22:53 +08:00
# Boot Armbian on Star64
2023-06-28 09:51:38 +08:00
Let's boot Armbian on Star64!
We download the Armbian Image for Star64: [Armbian 23.8 Lunar (Minimal) ](https://github.com/armbianro/os/releases/download/23.8.0-trunk.56/Armbian_23.8.0-trunk.56_Star64_lunar_edge_5.15.0_minimal.img.xz )
Uncompress the .xz, write the .img to a microSD Card with Balena Etcher.
Here's what happens when we boot the microSD Card on Star64...
2023-06-27 18:22:53 +08:00
2023-06-28 09:51:38 +08:00
- [Armbian Boot Log ](https://gist.github.com/lupyuen/d73ace627318375fe20e90e4950f9c50 )
2023-06-27 18:22:53 +08:00
2023-06-28 09:51:38 +08:00
Armbian fails to boot...
2023-06-27 18:43:27 +08:00
```text
Found /boot/extlinux/extlinux.conf
Retrieving file: /boot/extlinux/extlinux.conf
383 bytes read in 7 ms (52.7 KiB/s)
1: [6CArmbian
Retrieving file: /boot/uInitrd
10911538 bytes read in 466 ms (22.3 MiB/s)
Retrieving file: /boot/Image
22040576 bytes read in 936 ms (22.5 MiB/s)
append: root=UUID=99f62df4-be35-475c-99ef-2ba3f74fe6b5 console=ttyS0,115200n8 console=tty0 earlycon=sbi rootflags=data=writeback stmmaceth=chain_mode:1 rw rw no_console_suspend consoleblank=0 fsck.fix=yes fsck.repair=yes net.ifnames=0 splash plymouth.ignore-serial-consoles
Retrieving file: /boot/dtb/starfive/jh7110-star64-pine64.dtb
Failed to load '/boot/dtb/starfive/jh7110-star64-pine64.dtb'
Skipping Armbian for failure retrieving FDT
```
The Flattened Device Tree (FDT) is missing! `/boot/dtb/starfive/jh7110-star64-pine64.dtb`
```text
→ ls /Volumes/armbi_root/boot/dtb-5.15.0-starfive2/starfive
evb-overlay jh7110-evb-usbdevice.dtb
jh7110-evb-can-pdm-pwmdac.dtb jh7110-evb.dtb
jh7110-evb-dvp-rgb2hdmi.dtb jh7110-fpga.dtb
jh7110-evb-i2s-ac108.dtb jh7110-visionfive-v2-A10.dtb
jh7110-evb-pcie-i2s-sd.dtb jh7110-visionfive-v2-A11.dtb
jh7110-evb-spi-uart2.dtb jh7110-visionfive-v2-ac108.dtb
jh7110-evb-uart1-rgb2hdmi.dtb jh7110-visionfive-v2-wm8960.dtb
jh7110-evb-uart4-emmc-spdif.dtb jh7110-visionfive-v2.dtb
jh7110-evb-uart5-pwm-i2c-tdm.dtb vf2-overlay
```
2023-06-29 15:26:04 +08:00
The missing Device Tree is noted in this [__Pine64 Forum Post__ ](https://forum.pine64.org/showthread.php?tid=18276&pid=117607#pid117607 ). So we might need to check back later for the Official Armbian Image, if it's fixed.
[(__balbes150__ suggests that we try this Armbian Image instead) ](https://forum.pine64.org/showthread.php?tid=18420&pid=118331#pid118331 )
2023-06-28 09:51:38 +08:00
2023-06-27 18:03:46 +08:00
# Boot Yocto on Star64
2023-06-28 09:51:38 +08:00
Now we boot Yocto on Star64.
We download the Yocto Minimal Image for Star64: [star64-image-minimal ](https://pine64.my-ho.st:8443/star64-image-minimal-star64-1.2.wic.bz2 )
Uncompress the .bz2, rename as .img. Balena Etcher won't work with .bz2 files!
Write the .img to a microSD Card with Balena Etcher.
2023-06-27 18:03:46 +08:00
2023-06-28 09:51:38 +08:00
Here's what happens when we boot the microSD Card on Star64...
2023-06-27 18:03:46 +08:00
2023-06-28 09:51:38 +08:00
- [Yocto Boot Log ](https://gist.github.com/lupyuen/b23edf50cecbee13e5aab3c0bae6c528 )
Usernames and Passwords are...
- root / pine64
- pine64 / pine64
2023-06-27 18:03:46 +08:00
[(Source) ](https://github.com/Fishwaldo/meta-pine64#usernames )
2023-06-27 19:20:27 +08:00
2023-06-28 09:51:38 +08:00
Yep the Yocto Minimal Image boots OK on Star64!
2023-06-27 19:20:27 +08:00
# Boot Yocto Plasma on Star64
2023-06-28 09:51:38 +08:00
Finally we boot Yocto Plasma on Star64.
We download the Yocto Plasma Image for Star64: [star64-image-plasma ](https://pine64.my-ho.st:8443/star64-image-plasma-star64-1.2.wic.bz2 )
Uncompress the .bz2, rename as .img. Balena Etcher won't work with .bz2 files!
Write the .img to a microSD Card with Balena Etcher.
2023-06-27 19:20:27 +08:00
2023-06-28 09:51:38 +08:00
When we boot the microSD Card on Star64, the Plasma Desktop Environment runs OK on a HDMI Display! (Pic below)
Usernames and Passwords are...
- root / pine64
- pine64 / pine64
[(Source) ](https://github.com/Fishwaldo/meta-pine64#usernames )
2023-06-28 07:28:43 +08:00
![Yocto Plasma on Star64 ](https://lupyuen.github.io/images/star64-plasma.jpg )
2023-06-28 09:51:38 +08:00
2023-06-30 16:37:59 +08:00
# NuttX prints to QEMU Console
2023-06-28 09:51:38 +08:00
2023-07-01 15:18:16 +08:00
Our NuttX Kernel will print to Star64 Serial Console for debugging. Before that, let's write some RISC-V Assembly Code to print to the QEMU Console!
2023-06-29 17:09:41 +08:00
2023-06-30 16:37:59 +08:00
Earlier we ran NuttX on QEMU Emulator for 64-bit RISC-V...
2023-06-29 17:09:41 +08:00
2023-06-30 16:37:59 +08:00
- ["64-bit RISC-V with Apache NuttX Real-Time Operating System" ](https://lupyuen.github.io/articles/riscv )
2023-06-29 17:09:41 +08:00
2023-06-30 16:37:59 +08:00
QEMU emulates a 16550 UART Port. (Similar to Star64 / JH7110)
2023-06-29 17:09:41 +08:00
2023-07-01 15:18:16 +08:00
_What's the Base Address of QEMU's UART Port?_
According to the NuttX Configuration for QEMU: [nsh64/defconfig ](https://github.com/lupyuen2/wip-pinephone-nuttx/blob/star64/boards/risc-v/qemu-rv/rv-virt/configs/nsh64/defconfig#L10-L16 )
2023-06-29 17:09:41 +08:00
2023-06-30 16:37:59 +08:00
```text
CONFIG_16550_ADDRWIDTH=0
CONFIG_16550_UART0=y
CONFIG_16550_UART0_BASE=0x10000000
CONFIG_16550_UART0_CLOCK=3686400
CONFIG_16550_UART0_IRQ=37
CONFIG_16550_UART0_SERIAL_CONSOLE=y
CONFIG_16550_UART=y
```
Base Address of QEMU's UART Port is `0x1000` `0000` . (Same as Star64 / JH7110 yay!)
_How to print to the 16550 UART Port?_
2023-06-29 17:09:41 +08:00
2023-06-30 17:13:28 +08:00
Let's check the 16550 UART Driver in NuttX. From [uart_16550.c ](https://github.com/apache/nuttx/blob/master/drivers/serial/uart_16550.c#L1539-L1553 ):
2023-06-29 17:09:41 +08:00
```c
/****************************************************************************
* Name: u16550_send
*
* Description:
* This method will send one byte on the UART
*
** **************************************************************************/
static void u16550_send(struct uart_dev_s *dev, int ch)
{
FAR struct u16550_s *priv = (FAR struct u16550_s * )dev->priv;
u16550_serialout(priv, UART_THR_OFFSET, (uart_datawidth_t)ch);
}
```
2023-06-30 16:37:59 +08:00
[(u16550_serialout is defined here) ](https://github.com/apache/nuttx/blob/master/drivers/serial/uart_16550.c#L610-L624 )
To print a character, the driver writes to the UART Base Address (`0x1000` `0000` ) at Offset UART_THR_OFFSET.
2023-06-29 17:09:41 +08:00
2023-07-01 15:18:16 +08:00
And we discover that [UART_THR_OFFSET ](https://github.com/apache/nuttx/blob/dc69b108b8e0547ecf6990207526c27aceaf1e2e/include/nuttx/serial/uart_16550.h#L172-L200 ) is 0:
2023-06-29 17:09:41 +08:00
```c
#define UART_THR_INCR 0 /* (DLAB =0) Transmit Holding Register */
#define UART_THR_OFFSET (CONFIG_16550_REGINCR*UART_THR_INCR)
```
2023-06-30 16:37:59 +08:00
So we can transmit to UART Port by simply writing to `0x1000` `0000` . How convenient!
2023-06-29 17:10:15 +08:00
2023-06-30 16:37:59 +08:00
_How to print to the QEMU Console?_
2023-06-29 17:10:15 +08:00
2023-06-30 16:37:59 +08:00
Let's do the printing in RISC-V Assembly Code, so that we can debug the NuttX Boot Code.
2023-06-29 19:09:27 +08:00
2023-06-30 17:13:28 +08:00
From [qemu_rv_head.S ](https://github.com/lupyuen2/wip-pinephone-nuttx/blob/star64/arch/risc-v/src/qemu-rv/qemu_rv_head.S#L71-L93 ):
2023-06-29 20:44:49 +08:00
```text
/* Load UART Base Address to Register t0 */
li t0, 0x10000000
/* Load `1` to Register t1 */
li t1, 0x31
/* Store byte from Register t1 to UART Base Address, Offset 0 */
sb t1, 0(t0)
/* Load `2` to Register t1 */
li t1, 0x32
/* Store byte from Register t1 to UART Base Address, Offset 0 */
sb t1, 0(t0)
/* Load `3` to Register t1 */
li t1, 0x33
/* Store byte from Register t1 to UART Base Address, Offset 0 */
sb t1, 0(t0)
```
2023-06-30 16:37:59 +08:00
This prints "123" to the QEMU Console. Here's the output:
2023-06-29 20:44:49 +08:00
```text
2023-07-01 15:19:18 +08:00
+ qemu-system-riscv64 \
-semihosting \
-M virt,aclint=on \
-cpu rv64 \
-smp 8 \
-bios none \
-kernel nuttx \
-nographic
2023-06-29 20:44:49 +08:00
123123123123123123112323
NuttShell (NSH) NuttX-12.0.3
nsh>
```
2023-06-30 16:37:59 +08:00
Which is correct because QEMU is running with 8 CPUs. Yay!
2023-06-29 21:12:29 +08:00
2023-07-01 13:51:26 +08:00
![NuttX prints to QEMU Console ](https://lupyuen.github.io/images/riscv-print.png )
2023-06-30 16:20:29 +08:00
[Cody AI Assistant ](https://about.sourcegraph.com/cody ) explains our RISC-V Assembly Code...
![Cody AI Assistant explains our RISC-V Assembly Code ](https://lupyuen.github.io/images/riscv-cody1.png )
2023-06-30 17:10:21 +08:00
And offers to optimise our RISC-V Assembly Code...
2023-06-30 16:20:29 +08:00
![Cody AI Assistant optimises our RISC-V Assembly Code ](https://lupyuen.github.io/images/riscv-cody2.png )
2023-06-30 17:12:19 +08:00
But the output is incorrect ;-)
```text
+ qemu-system-riscv64 -semihosting -M virt,aclint=on -cpu rv64 -smp 8 -bios none -kernel nuttx -nographic
11111111
NuttShell (NSH) NuttX-12.0.3
nsh>
```
2023-06-30 17:27:16 +08:00
The correct output is `123123123123123123112323` . (Because of the 8 CPUs)
2023-07-01 15:53:57 +08:00
# UART Base Address for Star64
2023-06-30 16:37:59 +08:00
2023-07-01 20:04:44 +08:00
We'll take the UART Assembly Code from the previous section and run on Star64 / JH7110. (So we can troubleshoot the NuttX Boot Code)
2023-06-30 16:37:59 +08:00
2023-07-02 13:28:36 +08:00
_Does Star64 / JH7110 use a 16550 UART Controller like QEMU?_
2023-06-30 16:37:59 +08:00
2023-07-02 13:28:36 +08:00
According to the [JH7110 UART Developing Guide ](https://doc-en.rvspace.org/VisionFive2/DG_UART/JH7110_SDK/function_layer.html ), Star64 / JH7110 uses the 8250 UART Controller...
Which is [compatible with QEMU's 16550 UART Controller ](https://en.wikipedia.org/wiki/16550_UART ). So our UART Assembly Code for QEMU will run on Star64!
_What's the UART Base Address for Star64 / JH7110?_
Based on [JH7110 System Memory Map ](https://doc-en.rvspace.org/JH7110/TRM/JH7110_TRM/system_memory_map.html ), UART0 is at `0x1000` `0000` .
Also from the [JH7110 UART Device Tree ](https://doc-en.rvspace.org/VisionFive2/DG_UART/JH7110_SDK/general_uart_controller.html ): UART Register Base Address is `0x1000` `0000` with range `0x10000` .
2023-06-30 16:37:59 +08:00
2023-07-01 20:04:44 +08:00
[(JH7110 UART Datasheet) ](https://doc-en.rvspace.org/JH7110/Datasheet/JH7110_DS/uart.html )
2023-06-30 16:37:59 +08:00
2023-07-02 13:28:36 +08:00
_Isn't that the same UART Base Address as QEMU?_
Let's check the UART Base Address in NuttX for QEMU. From [nsh64/defconfig ](https://github.com/lupyuen2/wip-pinephone-nuttx/blob/star64/boards/risc-v/qemu-rv/rv-virt/configs/nsh64/defconfig#L10-L16 ):
2023-07-01 15:53:57 +08:00
```text
CONFIG_16550_ADDRWIDTH=0
CONFIG_16550_UART0=y
CONFIG_16550_UART0_BASE=0x10000000
CONFIG_16550_UART0_CLOCK=3686400
CONFIG_16550_UART0_IRQ=37
CONFIG_16550_UART0_SERIAL_CONSOLE=y
CONFIG_16550_UART=y
```
2023-07-02 13:28:36 +08:00
NuttX UART Base Address is `0x1000` `0000` . The exact same UART Base Address for QEMU AND Star64!
So no changes needed, our UART Assembly Code will run on QEMU AND Star64 yay!
2023-07-01 15:53:57 +08:00
2023-06-30 17:10:21 +08:00
# RISC-V Linux Kernel Header
2023-06-30 16:37:59 +08:00
2023-07-01 20:04:44 +08:00
For U-Boot Bootloader to boot NuttX, we need to embed the RISC-V Linux Kernel Header...
- [__"Inside the Kernel Image"__ ](https://lupyuen.github.io/articles/star64#inside-the-kernel-image )
2023-06-29 19:09:27 +08:00
2023-06-30 11:28:21 +08:00
This is how we decode the RISC-V Linux Header...
- [__"Decode the RISC-V Linux Header"__ ](https://lupyuen.github.io/articles/star64#appendix-decode-the-risc-v-linux-header )
2023-07-01 20:04:44 +08:00
We copy the Arm64 Linux Header from [arm64_head.S ](https://github.com/lupyuen2/wip-pinephone-nuttx/blob/star64/arch/arm64/src/common/arm64_head.S#L79-L118 )...
2023-06-30 13:37:48 +08:00
2023-07-01 20:04:44 +08:00
And tweak for RISC-V Linux Header, like this: [qemu_rv_head.S ](https://github.com/lupyuen2/wip-pinephone-nuttx/blob/star64/arch/risc-v/src/qemu-rv/qemu_rv_head.S#L42-L75 ):
2023-06-30 14:05:08 +08:00
```text
__start:
/* Begin Test */
/* DO NOT MODIFY. Image Header expected by Linux bootloaders.
*
* This `li` instruction has no meaningful effect except that
* its opcode forms the magic "MZ" signature of a PE/COFF file
* that is required for UEFI applications.
*
* Some bootloaders check the magic "MZ" to see if the image is a valid
* Linux image. But modifying the bootLoader is unnecessary unless we
* need to do a customized secure boot. So we just put "MZ" in the
* header to make the bootloader happy.
*/
2023-07-01 19:29:34 +08:00
c.li s4, -13 /* Magic Signature "MZ" (2 bytes) */
2023-06-30 14:05:08 +08:00
j real_start /* Jump to Kernel Start (2 bytes) */
.long 0 /* Executable Code padded to 8 bytes */
2023-07-01 17:09:39 +08:00
.quad 0x200000 /* Image load offset from start of RAM */
2023-06-30 14:05:08 +08:00
/* TODO: _e_initstack - __start */
.quad 171644 /* Effective size of kernel image, little-endian */
.quad 0x0 /* Kernel flags, little-endian */
.long 0x2 /* Version of this header */
.long 0 /* Reserved */
.quad 0 /* Reserved */
.ascii "RISCV\x00\x00\x00" /* Magic number, "RISCV" (8 bytes) */
.ascii "RSC\x05" /* Magic number 2, "RSC\x05" (4 bytes) */
.long 0 /* Reserved for PE COFF offset */
real_start:
/* Load UART Base Address to Register t0 */
li t0, 0x10000000
```
2023-07-01 17:09:39 +08:00
Note that Image Load Offset must be `0x20` `0000` !
2023-06-30 14:09:34 +08:00
```text
2023-07-01 17:09:39 +08:00
.quad 0x200000 /* Image load offset from start of RAM */
```
That's because our kernel starts at `0x4020` `0000`
Here's the assembled output...
```text
0000000040200000 < __start > :
li s4, -0xd /* Magic Signature "MZ" (2 bytes) */
40200000: 5a4d li s4,-13
j real_start /* Jump to Kernel Start (2 bytes) */
40200002: a83d j 40200040 < real_start >
40200004: 0000 unimp
40200006: 0000 unimp
40200008: 0000 unimp
4020000a: 0020 addi s0,sp,8
4020000c: 0000 unimp
4020000e: 0000 unimp
40200010: 9e7c 0x9e7c
40200012: 0002 c.slli64 zero
2023-06-30 14:09:34 +08:00
...
2023-07-01 17:09:39 +08:00
40200020: 0002 c.slli64 zero
2023-06-30 14:09:34 +08:00
...
2023-07-01 17:09:39 +08:00
4020002e: 0000 unimp
40200030: 4952 lw s2,20(sp)
40200032: 00564353 fadd.s ft6,fa2,ft5,rmm
40200036: 0000 unimp
40200038: 5352 lw t1,52(sp)
4020003a: 00000543 fmadd.s fa0,ft0,ft0,ft0,rne
2023-06-30 14:09:34 +08:00
...
2023-07-01 17:09:39 +08:00
0000000040200040 < real_start > :
2023-06-30 14:09:34 +08:00
```
2023-07-01 17:09:39 +08:00
Check that the lengths and offsets match the RISC-V Linux Header Format...
- [__"Decode the RISC-V Linux Header"__ ](https://lupyuen.github.io/articles/star64#appendix-decode-the-risc-v-linux-header )
2023-06-30 14:09:34 +08:00
2023-07-01 20:04:44 +08:00
And our RISC-V Boot Code tested OK with QEMU.
2023-06-30 14:50:22 +08:00
2023-07-01 13:51:26 +08:00
# Set Start Address of NuttX Kernel
2023-06-30 17:10:21 +08:00
2023-07-01 20:04:44 +08:00
Earlier we saw that Star64's U-Boot Bootloader will load Linux Kernels into RAM at Address `0x4020` `0000` ...
- ["Armbian Image for Star64" ](https://lupyuen.github.io/articles/star64#armbian-image-for-star64 )
- ["Yocto Image for Star64" ](https://lupyuen.github.io/articles/star64#yocto-image-for-star64 )
To boot NuttX on Star64, let's set the Start Address of the NuttX Kernel to `0x4020` `0000` .
2023-06-29 19:09:27 +08:00
2023-06-29 19:46:18 +08:00
From [nsh64/defconfig ](https://github.com/lupyuen2/wip-pinephone-nuttx/blob/star64/boards/risc-v/qemu-rv/rv-virt/configs/nsh64/defconfig#L56-L57 ):
```text
CONFIG_RAM_SIZE=33554432
CONFIG_RAM_START=0x80000000
```
2023-07-01 20:04:44 +08:00
We changed the above NuttX Build Config to `0x40200000`
2023-06-30 14:19:18 +08:00
2023-07-01 20:04:44 +08:00
We also updated the Linker Script: [ld.script ](https://github.com/lupyuen2/wip-pinephone-nuttx/blob/star64/boards/risc-v/qemu-rv/rv-virt/scripts/ld.script#L21-L26 )
2023-06-30 14:42:19 +08:00
```text
SECTIONS
{
/* Previously 0x80000000 */
2023-07-01 17:09:39 +08:00
. = 0x40200000;
2023-06-30 14:42:19 +08:00
.text :
```
2023-07-01 20:04:44 +08:00
Remember to change this if building for NuttX Kernel Mode: [ld-kernel64.script ](https://github.com/lupyuen2/wip-pinephone-nuttx/blob/star64/boards/risc-v/qemu-rv/rv-virt/scripts/ld-kernel64.script#L21-L51 ):
2023-06-30 14:42:19 +08:00
```text
MEMORY
{
/* Previously 0x80000000 */
2023-07-01 17:09:39 +08:00
kflash (rx) : ORIGIN = 0x40200000, LENGTH = 2048K /* w/ cache */
2023-06-30 14:42:19 +08:00
/* Previously 0x80200000 */
2023-07-01 17:09:39 +08:00
ksram (rwx) : ORIGIN = 0x40400000, LENGTH = 2048K /* w/ cache */
2023-06-30 14:42:19 +08:00
/* Previously 0x80400000 */
2023-07-01 17:09:39 +08:00
pgram (rwx) : ORIGIN = 0x40600000, LENGTH = 4096K /* w/ cache */
2023-06-30 14:42:19 +08:00
}
...
SECTIONS
{
/* Previously 0x80000000 */
2023-07-01 17:09:39 +08:00
. = 0x40200000;
2023-06-30 14:42:19 +08:00
.text :
```
2023-07-05 10:53:58 +08:00
Which should match [knsh64/defconfig ](https://github.com/lupyuen2/wip-pinephone-nuttx/blob/star64/boards/risc-v/qemu-rv/rv-virt/configs/knsh64/defconfig ):
```text
CONFIG_ARCH_PGPOOL_PBASE=0x40600000
CONFIG_ARCH_PGPOOL_VBASE=0x40600000
// TODO: Fix CONFIG_RAM_SIZE
CONFIG_RAM_SIZE=1048576
CONFIG_RAM_START=0x40200000
```
2023-07-01 13:51:26 +08:00
RISC-V Disassembly of NuttX Kernel shows that the Start Address is correct...
```text
2023-07-01 17:09:39 +08:00
0000000040200000 < __start > :
2023-07-01 13:51:26 +08:00
li s4, -0xd /* Magic Signature "MZ" (2 bytes) */
2023-07-01 17:09:39 +08:00
40200000: 5a4d li s4,-13
2023-07-01 13:51:26 +08:00
j real_start /* Jump to Kernel Start (2 bytes) */
2023-07-01 17:09:39 +08:00
40200002: a83d j 40200040 < real_start >
2023-07-01 13:51:26 +08:00
```
2023-07-01 20:04:44 +08:00
We're ready to boot NuttX on Star64!
2023-07-01 15:53:57 +08:00
# Boot NuttX on Star64
2023-06-30 17:10:21 +08:00
2023-07-01 21:55:32 +08:00
Let's boot NuttX on Star64! We compile [NuttX for 64-bit RISC-V QEMU ](https://lupyuen.github.io/articles/riscv#appendix-build-apache-nuttx-rtos-for-64-bit-risc-v-qemu ) with these tweaks...
- ["NuttX prints to QEMU Console" ](https://github.com/lupyuen/nuttx-star64#nuttx-prints-to-qemu-console )
- ["UART Base Address for Star64" ](https://github.com/lupyuen/nuttx-star64#uart-base-address-for-star64 )
- ["RISC-V Linux Kernel Header" ](https://github.com/lupyuen/nuttx-star64#risc-v-linux-kernel-header )
- ["Set Start Address of NuttX Kernel" ](https://github.com/lupyuen/nuttx-star64#set-start-address-of-nuttx-kernel )
For the microSD Image, we pick this [__Armbian Image for Star64__ ](https://www.armbian.com/star64/ )...
2023-07-01 18:29:50 +08:00
- [__Armbian 23.8 Lunar for Star64 (Minimal)__ ](https://github.com/armbianro/os/releases/download/23.8.0-trunk.56/Armbian_23.8.0-trunk.56_Star64_lunar_edge_5.15.0_minimal.img.xz )
Write the Armbian Image to a microSD Card with Balena Etcher.
2023-07-01 17:09:39 +08:00
2023-07-01 21:55:32 +08:00
We fix the [Missing Device Tree ](https://lupyuen.github.io/articles/star64#armbian-image-for-star64 )...
2023-06-29 19:44:48 +08:00
2023-07-01 16:29:04 +08:00
```bash
2023-07-04 06:56:20 +08:00
## Fix the Missing Device Tree
sudo chmod go+w /run/media/$USER/armbi_root/boot
sudo chmod go+w /run/media/$USER/armbi_root/boot/dtb/starfive
2023-07-01 16:29:04 +08:00
cp \
2023-07-04 06:56:20 +08:00
/run/media/$USER/armbi_root/boot/dtb/starfive/jh7110-visionfive-v2.dtb \
/run/media/$USER/armbi_root/boot/dtb/starfive/jh7110-star64-pine64.dtb
2023-06-29 19:44:48 +08:00
```
2023-07-01 18:29:50 +08:00
Then we delete the sym-link `/boot/Image` and copy the NuttX Binary Image `nuttx.bin` to `/boot/Image` ...
2023-07-01 16:29:04 +08:00
2023-07-01 17:09:39 +08:00
```bash
2023-07-04 06:56:20 +08:00
## We assume that `nuttx` contains the NuttX ELF Image.
## Export the NuttX Binary Image to `nuttx.bin`
2023-07-01 18:00:07 +08:00
riscv64-unknown-elf-objcopy \
-O binary \
nuttx \
nuttx.bin
2023-07-04 06:56:20 +08:00
## Delete Armbian Kernel `/boot/Image`
rm /run/media/$USER/armbi_root/boot/Image
2023-07-01 17:09:39 +08:00
2023-07-04 06:56:20 +08:00
## Copy `nuttx.bin` to Armbian Kernel `/boot/Image`
cp nuttx.bin /run/media/$USER/armbi_root/boot/Image
2023-07-01 17:09:39 +08:00
```
Insert the microSD Card into Star64 and power up.
2023-07-01 21:55:32 +08:00
NuttX boots with `123` yay! [(Which is printed by our Boot Code) ](https://github.com/lupyuen/nuttx-star64#nuttx-prints-to-qemu-console )
2023-07-01 18:00:07 +08:00
```text
Starting kernel ...
clk u5_dw_i2c_clk_core already disabled
clk u5_dw_i2c_clk_apb already disabled
123
Unhandled exception: Illegal instruction
```
2023-07-01 15:53:57 +08:00
2023-07-01 17:15:51 +08:00
![Boot NuttX on Star64 ](https://lupyuen.github.io/images/star64-nuttx.png )
2023-07-01 18:00:07 +08:00
Here's the complete log...
2023-07-01 15:53:57 +08:00
```text
2023-07-01 16:29:04 +08:00
Retrieving file: /boot/extlinux/extlinux.conf
383 bytes read in 7 ms (52.7 KiB/s)
2023-07-01 17:09:39 +08:00
1: [6CArmbian
2023-07-01 16:29:04 +08:00
Retrieving file: /boot/uInitrd
2023-07-01 17:09:39 +08:00
10911538 bytes read in 466 ms (22.3 MiB/s)
2023-07-01 16:29:04 +08:00
Retrieving file: /boot/Image
2023-07-01 17:09:39 +08:00
163201 bytes read in 14 ms (11.1 MiB/s)
2023-07-01 16:29:04 +08:00
append: root=UUID=99f62df4-be35-475c-99ef-2ba3f74fe6b5 console=ttyS0,115200n8 console=tty0 earlycon=sbi rootflags=data=writeback stmmaceth=chain_mode:1 rw rw no_console_suspend consoleblank=0 fsck.fix=yes fsck.repair=yes net.ifnames=0 splash plymouth.ignore-serial-consoles
Retrieving file: /boot/dtb/starfive/jh7110-star64-pine64.dtb
2023-07-01 17:09:39 +08:00
50235 bytes read in 14 ms (3.4 MiB/s)
## Loading init Ramdisk from Legacy Image at 46100000 ...
Image Name: uInitrd
Image Type: RISC-V Linux RAMDisk Image (gzip compressed)
Data Size: 10911474 Bytes = 10.4 MiB
Load Address: 00000000
Entry Point: 00000000
Verifying Checksum ... OK
## Flattened Device Tree blob at 46000000
Booting using the fdt blob at 0x46000000
Using Device Tree in place at 0000000046000000, end 000000004600f43a
Starting kernel ...
clk u5_dw_i2c_clk_core already disabled
clk u5_dw_i2c_clk_apb already disabled
123Unhandled exception: Illegal instruction
EPC: 000000004020005c RA: 00000000fff471c6 TVAL: 00000000f1402573
EPC: ffffffff804ba05c RA: 00000000402011c6 reloc adjusted
SP: 00000000ff733630 GP: 00000000ff735e00 TP: 0000000000000001
T0: 0000000010000000 T1: 0000000000000033 T2: 7869662e6b637366
S0: 0000000000000400 S1: 00000000ffff1428 A0: 0000000000000001
A1: 0000000046000000 A2: 0000000000000600 A3: 0000000000004000
A4: 0000000000000000 A5: 0000000040200000 A6: 00000000fffd5708
A7: 0000000000000000 S2: 00000000fff47194 S3: 0000000000000003
S4: fffffffffffffff3 S5: 00000000fffdbb50 S6: 0000000000000000
S7: 0000000000000000 S8: 00000000fff47194 S9: 0000000000000002
S10: 0000000000000000 S11: 0000000000000000 T3: 0000000000000023
T4: 000000004600b5cc T5: 000000000000ff00 T6: 000000004600b5cc
Code: 0313 0320 8023 0062 0313 0330 8023 0062 (2573 f140)
2023-07-01 16:29:04 +08:00
resetting ...
reset not supported yet
### ERROR ### Please RESET the board ###
2023-07-01 15:53:57 +08:00
```
2023-07-01 19:29:34 +08:00
Why does NuttX crash at `4020005c` ? See the next section...
2023-06-29 19:44:48 +08:00
2023-07-05 18:54:12 +08:00
![Cody AI Assistant tries to explain our RISC-V Exception ](https://lupyuen.github.io/images/star64-exception.jpg )
2023-07-02 18:14:12 +08:00
2023-07-05 18:54:12 +08:00
_[Cody AI Assistant](https://about.sourcegraph.com/cody) tries to explain our RISC-V Exception_
2023-07-02 18:14:12 +08:00
2023-07-01 21:33:03 +08:00
# NuttX Fails To Get Hart ID
2023-07-01 19:29:34 +08:00
Earlier we saw NuttX crashing when booting on Star64...
```text
Starting kernel ...
clk u5_dw_i2c_clk_core already disabled
clk u5_dw_i2c_clk_apb already disabled
123
Unhandled exception: Illegal instruction
EPC: 000000004020005c RA: 00000000fff471c6 TVAL: 00000000f1402573
```
2023-07-01 21:33:03 +08:00
_Why did NuttX crash at `4020005c` ?_
Here's our RISC-V Boot Code...
2023-07-01 19:29:34 +08:00
2023-07-04 09:52:46 +08:00
From [qemu_rv_head.S ](https://github.com/lupyuen2/wip-pinephone-nuttx/blob/ed09c34532ee7c51ac2da816cd6cf0adcce336e6/arch/risc-v/src/qemu-rv/qemu_rv_head.S#L92-L103 ):
2023-07-01 19:29:34 +08:00
```text
nuttx/arch/risc-v/src/chip/qemu_rv_head.S:95
/* Load mhartid (cpuid) */
csrr a0, mhartid
2023-07-01 21:33:03 +08:00
4020005c: f1402573 csrr a0, mhartid
2023-07-01 19:29:34 +08:00
```
2023-07-01 21:33:03 +08:00
NuttX tries loads the CPU ID or Hardware Thread "Hart" ID from the RISC-V Control and Status Register (CSR). [(Explained here) ](https://lupyuen.github.io/articles/riscv#get-cpu-id )
2023-07-06 09:45:17 +08:00
But it fails! Because we don't have sufficient privilege to access the Hart ID...
# RISC-V Privilege Levels
2023-07-01 19:29:34 +08:00
2023-07-01 21:33:03 +08:00
RISC-V runs at 3 Privilege Levels...
2023-07-01 19:29:34 +08:00
2023-07-07 14:50:31 +08:00
- M: Machine Mode (Most powerful)
2023-07-01 19:29:34 +08:00
2023-07-07 14:50:31 +08:00
- S: Supervisor Mode (Less powerful)
2023-07-01 19:29:34 +08:00
2023-07-07 14:50:31 +08:00
- U: User Mode (Least powerful)
2023-07-01 19:29:34 +08:00
2023-07-07 14:50:31 +08:00
NuttX runs at Supervisor Mode, which [doesn't allow access to Machine-Mode CSR Registers ](https://five-embeddev.com/riscv-isa-manual/latest/machine.html ). (Including [Hart ID ](https://five-embeddev.com/riscv-isa-manual/latest/machine.html#hart-id-register-mhartid ))
2023-07-01 21:44:02 +08:00
2023-07-07 14:50:31 +08:00
(The `m` in `mhartid` signifies that it's a Machine-Mode Register)
2023-07-01 21:33:03 +08:00
2023-07-07 14:43:34 +08:00
![RISC-V Privilege Levels ](https://lupyuen.github.io/images/nuttx2-privilege.jpg )
2023-07-07 14:50:31 +08:00
_What runs in Machine Mode?_
2023-07-01 21:33:03 +08:00
2023-07-07 14:50:31 +08:00
[OpenSBI ](https://www.thegoodpenguin.co.uk/blog/an-overview-of-opensbi/ ) (Supervisor Binary Interface) is the first thing that boots on Star64. It runs at Machine Mode and starts the U-Boot Bootloader.
2023-07-01 21:44:02 +08:00
[(See the RISC-V SBI Spec) ](https://github.com/riscv-non-isa/riscv-sbi-doc/blob/master/riscv-sbi.pdf )
2023-07-01 21:33:03 +08:00
2023-07-01 21:44:02 +08:00
_What about U-Boot Bootloader?_
2023-07-01 21:33:03 +08:00
2023-07-07 14:50:31 +08:00
U-Boot Bootloader runs in Supervisor Mode. And starts NuttX, also in Supervisor Mode.
2023-07-01 19:29:34 +08:00
2023-07-07 14:50:31 +08:00
So OpenSBI is the only thing that runs in Machine Mode. And can access the Machine-Level Registers.
2023-07-01 21:44:02 +08:00
2023-07-07 14:43:34 +08:00
_QEMU doesn't have this problem?_
Because QEMU runs everything in (super-powerful) __Machine Mode__ !
![NuttX QEMU runs in Machine Mode ](https://lupyuen.github.io/images/nuttx2-privilege2.jpg )
NuttX needs to fetch the Hart ID with a different recipe...
2023-07-06 09:45:17 +08:00
# Downgrade NuttX to Supervisor Mode
2023-07-01 21:33:03 +08:00
_How to get the Hart ID from OpenSBI?_
2023-07-01 21:44:02 +08:00
Let's refer to the Linux Boot Code: [linux/arch/riscv/kernel/head.S ](https://github.com/torvalds/linux/blob/master/arch/riscv/kernel/head.S )
2023-07-01 21:33:03 +08:00
2023-07-01 21:44:02 +08:00
(Tip: `CONFIG_RISCV_M_MODE` is False and `CONFIG_EFI` is True)
2023-07-01 21:33:03 +08:00
From [linux/blob/master/arch/riscv/kernel/head.S ](https://github.com/torvalds/linux/blob/master/arch/riscv/kernel/head.S#L292-L295 ):
```c
/* Save hart ID and DTB physical address */
mv s0, a0
mv s1, a1
```
2023-07-01 19:53:51 +08:00
2023-07-01 21:44:02 +08:00
Here we see that U-Boot [(or OpenSBI) ](https://github.com/riscv-non-isa/riscv-sbi-doc/blob/master/riscv-sbi.adoc#function-hart-start-fid-0 ) will pass 2 arguments when it starts our kernel...
2023-07-01 19:53:51 +08:00
2023-07-01 21:33:03 +08:00
- Register A0: Hart ID
2023-07-01 19:29:34 +08:00
2023-07-01 21:33:03 +08:00
- Register A1: RAM Address of Device Tree
2023-07-01 19:29:34 +08:00
2023-07-01 21:33:03 +08:00
So we'll simply read the Hart ID from Register A0. (And ignore A1)
2023-07-04 15:16:43 +08:00
We'll remove `csrr a0, mhartid` .
2023-07-01 21:33:03 +08:00
2023-07-02 13:35:13 +08:00
_What are the actual values of Registers A0 and A1?_
Thanks to our [earlier Crash Dump ](https://github.com/lupyuen/nuttx-star64#boot-nuttx-on-star64 ), we know the actual values of A0 and A1!
```text
SP: 00000000ff733630 GP: 00000000ff735e00 TP: 0000000000000001
T0: 0000000010000000 T1: 0000000000000033 T2: 7869662e6b637366
S0: 0000000000000400 S1: 00000000ffff1428 A0: 0000000000000001
A1: 0000000046000000 A2: 0000000000000600 A3: 0000000000004000
```
This says that...
- Hart ID is 1 (Register A0)
- RAM Address of Device Tree is `0x4600` `0000` (Register A1)
2023-07-04 15:16:43 +08:00
Yep looks correct! But we'll subtract 1 from Register A0 because NuttX expects Hart ID to start with 0.
2023-07-02 13:35:33 +08:00
2023-07-01 21:33:03 +08:00
_What about other CSR Instructions in our NuttX Boot Code?_
2023-07-01 22:20:45 +08:00
We change the Machine-Level `m` Registers to Supervisor-Level `s` Registers.
2023-07-04 15:16:43 +08:00
To Disable Interrupts: Change `mie` to [`sie` ](https://five-embeddev.com/riscv-isa-manual/latest/supervisor.html#supervisor-interrupt-registers-sip-and-sie )
2023-07-01 21:33:03 +08:00
```text
/* Disable all interrupts (i.e. timer, external) in mie */
csrw mie, zero
```
[(Source) ](https://lupyuen.github.io/articles/riscv#disable-interrupts )
2023-07-04 15:16:43 +08:00
To Load Interrupt Vector Table: Change `mtvec` to [`stvec` ](https://five-embeddev.com/riscv-isa-manual/latest/supervisor.html#supervisor-trap-vector-base-address-register-stvec )
2023-07-01 21:33:03 +08:00
```text
/* Load address of Interrupt Vector Table */
csrw mtvec, t0
```
2023-07-01 20:05:25 +08:00
2023-07-01 21:33:03 +08:00
[(Source) ](https://lupyuen.github.io/articles/riscv#load-interrupt-vector )
2023-07-01 19:29:34 +08:00
2023-07-01 22:20:45 +08:00
_The Linux Boot Code looks confusing. What are CSR_IE and CSR_IP?_
```text
/* Mask all interrupts */
csrw CSR_IE, zero
csrw CSR_IP, zero
```
[(Source) ](https://github.com/torvalds/linux/blob/master/arch/riscv/kernel/head.S#L195-L200 )
That's because the Linux Boot Code will work for Machine Level AND Supervisor Level! Here's how `CSR_IE` and `CSR_IP` are mapped to the `m` and `s` CSR Registers...
(Remember: `CONFIG_RISCV_M_MODE` is false for NuttX)
```text
#ifdef CONFIG_RISCV_M_MODE
/* Use Machine-Level CSR Registers */
# define CSR_IE CSR_MIE
# define CSR_IP CSR_MIP
...
#else
/* Use Supervisor-Level CSR Registers */
# define CSR_IE CSR_SIE
# define CSR_IP CSR_SIP
...
#endif /* !CONFIG_RISCV_M_MODE */
```
[(Source) ](https://github.com/torvalds/linux/blob/master/arch/riscv/include/asm/csr.h#L391-L444 )
2023-07-04 15:16:43 +08:00
Let's fix the Boot Code...
# Fix the NuttX Boot Code
From the previous section, we identified these fixes for the NuttX Boot Code...
1. Remove `csrr a0, mhartid` because OpenSBI will pass Hart ID in Register A0. Subtract 1 from Register A0 because NuttX expects Hart ID to start with 0.
1. To Disable Interrupts: Change `mie` to [`sie` ](https://five-embeddev.com/riscv-isa-manual/latest/supervisor.html#supervisor-interrupt-registers-sip-and-sie )
1. To Load Interrupt Vector Table: Change `mtvec` to [`stvec` ](https://five-embeddev.com/riscv-isa-manual/latest/supervisor.html#supervisor-trap-vector-base-address-register-stvec )
Here's the updated Boot Code, and our analysis: [qemu_rv_head.S ](https://github.com/lupyuen2/wip-pinephone-nuttx/blob/star64/arch/risc-v/src/qemu-rv/qemu_rv_head.S )
```text
real_start:
...
/* Load mhartid (cpuid) */
/* Previously: csrr a0, mhartid */
/* We assume that OpenSBI has passed Hart ID (value 1) in Register a0. */
/* But NuttX expects Hart ID to start at 0, so we subtract 1. */
addi a0, a0, -1
/* Print the Hart ID */
addi t1, a0, 0x30
/* Store byte from Register t1 to UART Base Address, Offset 0 */
sb t1, 0(t0)
```
__If Hart ID is 0:__
- Set Stack Pointer to the Idle Thread Stack
```text
/* Set stack pointer to the idle thread stack */
bnez a0, 1f
la sp, QEMU_RV_IDLESTACK_TOP
j 2f
```
__If Hart ID is 1, 2, 3, ...__
- Validate the Hart ID (Must be less than number of CPUs)
- Compute the Stack Base Address based on `g_cpu_basestack` and Hart ID
- Set the Stack Pointer to the computed Stack Base Address
```text
1:
/* Load the number of CPUs that the kernel supports */
#ifdef CONFIG_SMP
li t1, CONFIG_SMP_NCPUS
#else
li t1, 1
#endif
/* If a0 (mhartid) >= t1 (the number of CPUs), stop here */
blt a0, t1, 3f
csrw sie, zero
/* Previously: csrw mie, zero */
wfi
3:
/* To get g_cpu_basestack[mhartid], must get g_cpu_basestack first */
la t0, g_cpu_basestack
/* Offset = pointer width * hart id * /
#ifdef CONFIG_ARCH_RV32
slli t1, a0, 2
#else
slli t1, a0, 3
#endif
add t0, t0, t1
/* Load idle stack base to sp */
REGLOAD sp, 0(t0)
/*
* sp (stack top) = sp + idle stack size - XCPTCONTEXT_SIZE
*
* Note: Reserve some space used by up_initial_state since we are already
* running and using the per CPU idle stack.
*/
li t0, STACK_ALIGN_UP(CONFIG_IDLETHREAD_STACKSIZE - XCPTCONTEXT_SIZE)
add sp, sp, t0
```
__For All Hart IDs:__
- Disable Interrupts
- Load the Interrupt Vector Table
- Jump to `qemu_rv_start`
```
2:
/* Disable all interrupts (i.e. timer, external) in mie */
csrw sie, zero
/* Previously: csrw mie, zero */
2023-07-05 12:33:18 +08:00
/* Don't load the Interrupt Vector Table, use OpenSBI for crash logging */
/* la t0, __trap_vec */
/* csrw stvec, t0 */
2023-07-04 15:16:43 +08:00
/* Previously: csrw mtvec, t0 */
/* Jump to qemu_rv_start */
jal x1, qemu_rv_start
/* We shouldn't return from _start */
```
2023-07-05 12:33:18 +08:00
Note that we don't load the Interrupt Vector Table, because we'll use OpenSBI for crash logging. (Like when we hit M-Mode Instructions)
2023-07-04 15:45:44 +08:00
_What happens when we run this?_
Hart ID is now 0, which is correct...
```text
Starting kernel ...
clk u5_dw_i2c_clk_core already disabled
clk u5_dw_i2c_clk_apb already disabled
2023-07-04 20:15:33 +08:00
123067
2023-07-04 15:45:44 +08:00
```
But `qemu_rv_start` hangs. Why?
```text
/* Print `7` */
li t0, 0x10000000
li t1, 0x37
sb t1, 0(t0)
/* Jump to qemu_rv_start */
jal x1, qemu_rv_start
```
2023-07-12 09:47:14 +08:00
Let's trace `qemu_rv_start` ...
2023-07-04 15:16:43 +08:00
2023-07-12 09:47:14 +08:00
# Boot from Network with U-Boot and TFTP
2023-07-04 18:39:40 +08:00
2023-07-15 12:30:44 +08:00
Read the article...
- ["Star64 JH7110 RISC-V SBC: Boot from Network with U-Boot and TFTP" ](https://lupyuen.github.io/articles/tftp )
2023-07-12 09:47:14 +08:00
We really should configure U-Boot Bootloader to load the Kernel Image over the network via [TFTP over UDP ](https://en.wikipedia.org/wiki/Trivial_File_Transfer_Protocol ). Because testing NuttX by swapping microSD Card is getting so tiresome.
2023-07-04 18:39:40 +08:00
2023-07-12 13:05:47 +08:00
Here's how...
![Boot from Network with U-Boot and TFTP ](https://lupyuen.github.io/images/tftp-flow.jpg )
2023-07-12 09:47:14 +08:00
## Setup TFTP Server
2023-07-04 18:39:40 +08:00
2023-07-12 13:05:47 +08:00
First we set up a TFTP Server with [`tftpd` ](https://crates.io/crates/tftpd )...
2023-07-04 18:39:40 +08:00
2023-07-12 09:47:14 +08:00
```bash
cargo install tftpd
mkdir $HOME/tftproot
sudo tftpd -i 0.0.0.0 -p 69 -d "$HOME/tftproot"
## `sudo` because port 69 is a privileged low port
2023-07-04 19:13:57 +08:00
```
2023-07-12 09:47:14 +08:00
([`tftp_server`](https://crates.io/crates/tftp_server) won't work, it only supports localhost)
We should see...
2023-07-04 18:39:40 +08:00
2023-07-04 19:13:57 +08:00
```text
2023-07-12 09:47:14 +08:00
Running TFTP Server on 0.0.0.0:69 in $HOME/tftproot
Sending a.txt to 127.0.0.1:57125
Sent a.txt to 127.0.0.1:57125
Sending a.txt to 192.168.x.x:33499
Sent a.txt to 192.168.x.x:33499
2023-07-04 19:13:57 +08:00
```
2023-07-04 18:39:40 +08:00
2023-07-12 09:47:14 +08:00
Let's test the TFTP Server...
2023-07-04 19:13:57 +08:00
2023-07-12 09:47:14 +08:00
```bash
echo Test123 >$HOME/tftproot/a.txt
curl -v tftp://127.0.0.1/a.txt
curl -v tftp://192.168.x.x/a.txt
```
2023-07-04 19:13:57 +08:00
2023-07-12 09:47:14 +08:00
(`localhost` won't work because of IPv6, I think)
2023-07-04 19:13:57 +08:00
2023-07-12 09:47:14 +08:00
We should see...
2023-07-04 19:13:57 +08:00
2023-07-12 09:47:14 +08:00
```text
$ curl -v tftp://192.168.x.x/a.txt
* Trying 192.168.x.x:69...
* getpeername() failed with errno 107: Transport endpoint is not connected
* Connected to 192.168.x.x () port 69 (#0)
* getpeername() failed with errno 107: Transport endpoint is not connected
* set timeouts for state 0; Total 300000, retry 6 maxtry 50
* got option=(tsize) value=(8)
* tsize parsed from OACK (8)
* got option=(blksize) value=(512)
* blksize parsed from OACK (512) requested (512)
* got option=(timeout) value=(6)
* Connected for receive
* set timeouts for state 1; Total 0, retry 72 maxtry 50
Test123
* Closing connection 0
2023-07-04 18:39:40 +08:00
```
2023-07-12 09:47:14 +08:00
If it fails...
2023-07-04 19:13:57 +08:00
2023-07-12 09:47:14 +08:00
```text
$ curl -v tftp://192.168.x.x/a.txt
* Trying 192.168.x.x:69...
* getpeername() failed with errno 107: Transport endpoint is not connected
* Connected to 192.168.x.x () port 69 (#0)
* getpeername() failed with errno 107: Transport endpoint is not connected
* set timeouts for state 0; Total 300000, retry 6 maxtry 50
2023-07-04 19:13:57 +08:00
```
2023-07-12 09:47:14 +08:00
In the olden days we would actually do this...
2023-07-04 19:46:16 +08:00
2023-07-12 09:47:14 +08:00
```text
$ tftp 127.0.0.1
tftp> get a.txt
Received 8 bytes in 0.0 seconds
tftp> quit
2023-07-04 19:46:16 +08:00
```
2023-07-12 09:47:14 +08:00
Just like FTP!
2023-07-04 20:15:05 +08:00
2023-07-12 09:47:14 +08:00
## Copy NuttX Image to TFTP Server
2023-07-04 20:15:05 +08:00
2023-07-12 09:47:14 +08:00
Next we copy the NuttX Image and Device Tree to the TFTP Folder...
2023-07-04 20:15:05 +08:00
2023-07-12 09:47:14 +08:00
```bash
## Copy the Device Tree from Armbian microSD
cp \
/run/media/$USER/armbi_root/boot/dtb/starfive/jh7110-visionfive-v2.dtb \
jh7110-star64-pine64.dtb
2023-07-04 20:15:05 +08:00
2023-07-12 09:47:14 +08:00
## Copy NuttX Binary Image and Device Tree to TFTP Folder
## `nuttx.bin` comes from here:
## https://github.com/lupyuen2/wip-pinephone-nuttx/releases/tag/star64-0.0.1
cp nuttx.bin $HOME/tftproot/Image
cp jh7110-star64-pine64.dtb $HOME/tftproot
2023-07-04 20:15:05 +08:00
2023-07-12 18:47:00 +08:00
## Test NuttX Binary Image and Device Tree over TFTP
2023-07-12 09:47:14 +08:00
curl -v tftp://192.168.x.x/Image
curl -v tftp://192.168.x.x/jh7110-star64-pine64.dtb
2023-07-04 20:15:05 +08:00
2023-07-12 18:47:00 +08:00
## We should see...
2023-07-12 09:47:14 +08:00
## Warning: Binary output can mess up your terminal. Use "--output -" to tell
## Warning: curl to output it to your terminal anyway, or consider "--output
## Warning: <FILE>" to save to a file.
```
2023-07-04 20:15:05 +08:00
2023-07-12 09:47:14 +08:00
## Test U-Boot with TFTP
2023-07-04 20:15:05 +08:00
2023-07-12 09:47:14 +08:00
Now we boot Star64 JH7110 SBC and test the TFTP Commands.
2023-07-04 20:15:05 +08:00
2023-07-12 09:47:14 +08:00
Connect Star64 SBC to the Ethernet wired network and power up.
2023-07-04 20:15:05 +08:00
2023-07-12 09:47:14 +08:00
Star64 fails to boot over the network (because we don't have a [BOOTP Server ](https://en.wikipedia.org/wiki/Bootstrap_Protocol ) or DHCP+TFTP Combo Server), but that's OK...
2023-07-04 19:46:16 +08:00
2023-07-12 09:47:14 +08:00
```text
ethernet@16030000 Waiting for PHY auto negotiation to complete....... done
BOOTP broadcast 1
*** Unhandled DHCP Option in OFFER/ACK: 43
*** Unhandled DHCP Option in OFFER/ACK: 43
DHCP client bound to address 192.168.x.x (351 ms)
Using ethernet@16030000 device
TFTP from server 192.168.x.x; our IP address is 192.168.x.x
Filename 'boot.scr.uimg'.
Load address: 0x43900000
Loading: *
TFTP server died; starting again
BOOTP broadcast 1
*** Unhandled DHCP Option in OFFER/ACK: 43
*** Unhandled DHCP Option in OFFER/ACK: 43
DHCP client bound to address 192.168.x.x (576 ms)
Using ethernet@16030000 device
TFTP from server 192.168.x.x; our IP address is 192.168.x.x
Filename 'boot.scr.uimg'.
Load address: 0x40200000
Loading: *
TFTP server died; starting again
StarFive #
```
2023-07-04 20:15:05 +08:00
2023-07-12 09:47:14 +08:00
[(Source) ](https://github.com/lupyuen/nuttx-star64#u-boot-bootloader-log-for-tftp )
2023-07-04 20:15:05 +08:00
2023-07-12 09:47:14 +08:00
Run these commands...
2023-07-04 20:15:05 +08:00
2023-07-12 09:47:14 +08:00
```bash
## Set the TFTP Server IP
setenv tftp_server 192.168.x.x
2023-07-04 20:15:05 +08:00
2023-07-12 09:47:14 +08:00
## Load the NuttX Image from TFTP Server
## kernel_addr_r=0x40200000
## tftp_server=192.168.x.x
tftpboot ${kernel_addr_r} ${tftp_server}:Image
2023-07-04 20:15:05 +08:00
2023-07-12 09:47:14 +08:00
## Load the Device Tree from TFTP Server
## fdt_addr_r=0x46000000
## tftp_server=192.168.x.x
tftpboot ${fdt_addr_r} ${tftp_server}:jh7110-star64-pine64.dtb
2023-07-04 20:15:05 +08:00
2023-07-12 09:47:14 +08:00
## Set the RAM Address of Device Tree
## fdt_addr_r=0x46000000
fdt addr ${fdt_addr_r}
2023-07-04 20:15:05 +08:00
2023-07-12 09:47:14 +08:00
## Boot the NuttX Image with the Device Tree
## kernel_addr_r=0x40200000
## fdt_addr_r=0x46000000
booti ${kernel_addr_r} - ${fdt_addr_r}
2023-07-04 20:15:05 +08:00
```
2023-07-12 09:47:14 +08:00
[(Inspired by this article) ](https://community.arm.com/oss-platforms/w/docs/495/tftp-remote-network-kernel-using-u-boot )
2023-07-04 20:15:05 +08:00
2023-07-12 09:47:14 +08:00
We should see...
2023-07-04 20:15:05 +08:00
```text
2023-07-12 09:47:14 +08:00
StarFive # setenv tftp_server 192.168.x.x
2023-07-04 20:15:05 +08:00
2023-07-12 09:47:14 +08:00
StarFive # tftpboot ${kernel_addr_r} ${tftp_server}:Image
Using ethernet@16030000 device
TFTP from server 192.168.x.x; our IP address is 192.168.x.x
Filename 'Image'.
Load address: 0x40200000
Loading: #############################################################T ####
#################################################################
#############
221.7 KiB/s
done
Bytes transferred = 2097832 (2002a8 hex)
StarFive # tftpboot ${fdt_addr_r} ${tftp_server}:jh7110-star64-pine64.dtb
Using ethernet@16030000 device
TFTP from server 192.168.x.x; our IP address is 192.168.x.x
Filename 'jh7110-star64-pine64.dtb'.
Load address: 0x46000000
Loading: ####
374 KiB/s
done
Bytes transferred = 50235 (c43b hex)
StarFive # fdt addr ${fdt_addr_r}
StarFive # booti ${kernel_addr_r} - ${fdt_addr_r}
## Flattened Device Tree blob at 46000000
Booting using the fdt blob at 0x46000000
Using Device Tree in place at 0000000046000000, end 000000004600f43a
2023-07-11 14:35:07 +08:00
Starting kernel ...
2023-07-12 09:47:14 +08:00
2023-07-11 14:35:07 +08:00
clk u5_dw_i2c_clk_core already disabled
clk u5_dw_i2c_clk_apb already disabled
2023-07-12 09:47:14 +08:00
123067DFAGHBC
2023-07-11 14:35:07 +08:00
```
2023-07-12 09:47:14 +08:00
[(Source) ](https://github.com/lupyuen/nuttx-star64#u-boot-bootloader-log-for-tftp )
2023-07-11 16:30:39 +08:00
2023-07-12 09:47:14 +08:00
## Configure U-Boot for TFTP
2023-07-11 16:30:39 +08:00
2023-07-12 09:47:14 +08:00
Let's configure U-Boot so that it will boot from TFTP every time we power up!
2023-07-04 20:15:05 +08:00
2023-07-12 09:47:14 +08:00
```bash
## Remember the TFTP Server IP
setenv tftp_server 192.168.x.x
## Check that it's correct
printenv tftp_server
## Save it for future reboots
saveenv
2023-07-05 14:34:14 +08:00
2023-07-12 09:47:14 +08:00
## Add the Boot Command for TFTP
setenv bootcmd_tftp 'if tftpboot ${kernel_addr_r} ${tftp_server}:Image ; then if tftpboot ${fdt_addr_r} ${tftp_server}:jh7110-star64-pine64.dtb ; then if fdt addr ${fdt_addr_r} ; then booti ${kernel_addr_r} - ${fdt_addr_r} ; fi ; fi ; fi'
## Check that it's correct
printenv bootcmd_tftp
## Save it for future reboots
saveenv
2023-07-05 14:34:14 +08:00
2023-07-12 09:47:14 +08:00
## Test the Boot Command for TFTP
run bootcmd_tftp
2023-07-05 14:34:14 +08:00
2023-07-12 09:47:14 +08:00
## Remember the Original Boot Targets
setenv orig_boot_targets "$boot_targets"
## Should show `mmc0 dhcp`
printenv boot_targets
## Save it for future reboots
saveenv
2023-07-05 12:49:58 +08:00
2023-07-12 09:47:14 +08:00
## Add TFTP to the Boot Targets
setenv boot_targets "$boot_targets tftp"
## Should show `mmc0 dhcp tftp`
printenv boot_targets
## Save it for future reboots
saveenv
```
2023-07-05 12:49:58 +08:00
2023-07-12 09:47:14 +08:00
`bootcmd_tftp` expands to...
2023-07-05 12:49:58 +08:00
2023-07-12 09:47:14 +08:00
```bash
## Load the NuttX Image from TFTP Server
## kernel_addr_r=0x40200000
## tftp_server=192.168.x.x
if tftpboot ${kernel_addr_r} ${tftp_server}:Image ;
then
2023-07-05 12:49:58 +08:00
2023-07-12 09:47:14 +08:00
## Load the Device Tree from TFTP Server
## fdt_addr_r=0x46000000
if tftpboot ${fdt_addr_r} ${tftp_server}:jh7110-star64-pine64.dtb ;
then
2023-07-05 12:49:58 +08:00
2023-07-12 09:47:14 +08:00
## Set the RAM Address of Device Tree
## fdt_addr_r=0x46000000
if fdt addr ${fdt_addr_r} ;
then
2023-07-05 12:49:58 +08:00
2023-07-12 09:47:14 +08:00
## Boot the NuttX Image with the Device Tree
## kernel_addr_r=0x40200000
## fdt_addr_r=0x46000000
booti ${kernel_addr_r} - ${fdt_addr_r} ;
fi ;
fi ;
fi
```
2023-07-05 12:49:58 +08:00
2023-07-12 09:47:14 +08:00
[(From here) ](https://community.arm.com/oss-platforms/w/docs/495/tftp-remote-network-kernel-using-u-boot ) This is a persistent change, i.e. the device will boot via TFTP on every power up. To revert back to the default boot behaviour:
2023-07-05 12:49:58 +08:00
2023-07-12 09:47:14 +08:00
```bash
## Restore the Boot Targets
setenv boot_targets "$orig_boot_targets"
## Should show `mmc0 dhcp`
printenv boot_targets
## Save it for future reboots
saveenv
```
2023-07-05 12:49:58 +08:00
2023-07-15 12:30:44 +08:00
With Network Boot running, we're now ready for __Automated Testing of Apache NuttX RTOS__ on Star64 SBC!
Though we might need a __Smart Power Plug__ to power-cycle our SBC: [__IKEA TRÅDFRI__ ](https://www.ikea.com/sg/en/p/tradfri-control-outlet-kit-smart-10364797/ ) and [__DIRIGERA__ ](https://www.ikea.com/sg/en/p/dirigera-hub-for-smart-products-white-smart-50503409/ ) via [__Home Assistant API__ ](https://gist.github.com/lupyuen/01cff0d4ca225984ca8fd0d999d7c76d ) (pic below)
![Home Assistant controls Google Home (and potentially Smart Plugs) ](https://lupyuen.github.io/images/tftp-home.png )
2023-07-12 09:47:14 +08:00
## U-Boot Commands for Network Boot
2023-07-05 12:49:58 +08:00
2023-07-12 09:47:14 +08:00
_How does it work?_
2023-07-05 12:49:58 +08:00
2023-07-12 09:47:14 +08:00
`bootcmd` is now...
2023-07-05 12:49:58 +08:00
2023-07-12 09:47:14 +08:00
```text
bootcmd=run load_vf2_env;run importbootenv;run load_distro_uenv;run boot2;run distro_bootcmd
2023-07-04 20:15:05 +08:00
2023-07-12 09:47:14 +08:00
load_vf2_env=fatload mmc ${bootpart} ${loadaddr} ${testenv}
2023-07-04 18:39:40 +08:00
2023-07-12 09:47:14 +08:00
importbootenv=echo Importing environment from mmc${devnum} ...; env import -t ${loadaddr} ${filesize}
2023-07-04 18:39:40 +08:00
2023-07-12 09:47:14 +08:00
load_distro_uenv=fatload mmc ${fatbootpart} ${distroloadaddr} ${bootdir}/${bootenv}; env import ${distroloadaddr} 17c;
2023-07-04 18:39:40 +08:00
2023-07-12 09:47:14 +08:00
boot2 not defined, comes from `boot/vf2_uEnv.txt`
2023-07-04 18:39:40 +08:00
```
2023-07-04 16:05:57 +08:00
2023-07-12 09:47:14 +08:00
`bootcmd` calls `distro_bootcmd` , which runs `bootcmd_mmc0` and `bootcmd_dhcp` ...
2023-07-04 15:27:49 +08:00
2023-07-12 09:47:14 +08:00
```text
distro_bootcmd=for target in ${boot_targets}; do run bootcmd_${target}; done
2023-07-10 09:26:44 +08:00
2023-07-12 09:47:14 +08:00
boot_targets=mmc0 dhcp
2023-07-11 16:36:26 +08:00
2023-07-12 09:47:14 +08:00
bootcmd_mmc0=devnum=0; run mmc_boot
2023-07-10 10:32:07 +08:00
2023-07-12 09:47:14 +08:00
bootcmd_distro=run fdt_loaddtb; run fdt_sizecheck; run set_fdt_distro; sysboot mmc ${fatbootpart} fat c0000000 ${bootdir}/${boot_syslinux_conf};
2023-07-10 10:50:55 +08:00
```
2023-07-12 09:47:14 +08:00
`bootcmd_dhcp` is...
2023-07-10 10:50:55 +08:00
```text
2023-07-12 09:47:14 +08:00
bootcmd_dhcp=devtype=dhcp; if dhcp ${scriptaddr} ${boot_script_dhcp}; then source ${scriptaddr}; fi;setenv efi_fdtfile ${fdtfile}; setenv efi_old_vci ${bootp_vci};setenv efi_old_arch ${bootp_arch};setenv bootp_vci PXEClient:Arch:00027:UNDI:003000;setenv bootp_arch 0x1b;if dhcp ${kernel_addr_r}; then tftpboot ${fdt_addr_r} dtb/${efi_fdtfile};if fdt addr ${fdt_addr_r}; then bootefi ${kernel_addr_r} ${fdt_addr_r}; else bootefi ${kernel_addr_r} ${fdtcontroladdr};fi;fi;setenv bootp_vci ${efi_old_vci};setenv bootp_arch ${efi_old_arch};setenv efi_fdtfile;setenv efi_old_arch;setenv efi_old_vci;
2023-07-10 10:32:07 +08:00
```
2023-07-12 09:47:14 +08:00
Which expands to...
2023-07-10 10:22:07 +08:00
```bash
2023-07-12 09:47:14 +08:00
devtype=dhcp
2023-07-10 10:22:07 +08:00
2023-07-12 09:47:14 +08:00
## Load the Boot Script from DHCP+TFTP Server
## scriptaddr=0x43900000
## boot_script_dhcp=boot.scr.uimg
if dhcp ${scriptaddr} ${boot_script_dhcp}
then
source ${scriptaddr}
fi
2023-07-11 18:23:38 +08:00
2023-07-12 09:47:14 +08:00
## Set the EFI Variables
## fdtfile=starfive/starfive_visionfive2.dtb
setenv efi_fdtfile ${fdtfile}
setenv efi_old_vci ${bootp_vci}
setenv efi_old_arch ${bootp_arch}
setenv bootp_vci PXEClient:Arch:00027:UNDI:003000
setenv bootp_arch 0x1b
2023-07-10 10:22:07 +08:00
2023-07-12 09:47:14 +08:00
## Load the Kernel Image from DHCP+TFTP Server...
## kernel_addr_r=0x40200000
if dhcp ${kernel_addr_r}
then
2023-07-10 09:57:33 +08:00
2023-07-12 09:47:14 +08:00
## Load the Device Tree from the DHCP+TFTP Server
## fdt_addr_r=0x46000000
## efi_fdtfile=starfive/starfive_visionfive2.dtb
tftpboot ${fdt_addr_r} dtb/${efi_fdtfile}
2023-07-10 10:32:07 +08:00
2023-07-12 09:47:14 +08:00
## Set the RAM Address of Device Tree
## fdt_addr_r=0x46000000
if fdt addr ${fdt_addr_r}
then
2023-07-10 12:09:16 +08:00
2023-07-12 09:47:14 +08:00
## Boot the EFI Kernel Image
## fdt_addr_r=0x46000000
bootefi ${kernel_addr_r} ${fdt_addr_r}
else
2023-07-11 18:23:38 +08:00
2023-07-12 09:47:14 +08:00
## Boot the EFI Kernel Image
## fdtcontroladdr=fffc6aa0
bootefi ${kernel_addr_r} ${fdtcontroladdr}
fi
fi
## Unset the EFI Variables
setenv bootp_vci ${efi_old_vci}
setenv bootp_arch ${efi_old_arch}
setenv efi_fdtfile
setenv efi_old_arch
setenv efi_old_vci
2023-07-11 18:23:38 +08:00
```
2023-07-12 09:47:14 +08:00
`dhcp` command is...
2023-07-11 18:23:38 +08:00
2023-07-12 09:47:14 +08:00
```text
dhcp - boot image via network using DHCP/TFTP protocol
2023-07-11 16:36:26 +08:00
2023-07-12 09:47:14 +08:00
Usage:
dhcp [loadAddress] [[hostIPaddr:]bootfilename]
```
2023-07-10 12:09:16 +08:00
2023-07-12 09:47:14 +08:00
(Assume [DHCP/TFTP ](https://www.emcraft.com/som/using-dhcp ) is not used)
2023-07-10 12:09:16 +08:00
2023-07-12 09:47:14 +08:00
`tftpboot` command is...
2023-07-10 12:51:14 +08:00
2023-07-12 09:47:14 +08:00
```text
tftpboot - boot image via network using TFTP protocol
2023-07-10 12:51:14 +08:00
2023-07-12 09:47:14 +08:00
Usage:
tftpboot [loadAddress] [[hostIPaddr:]bootfilename]
2023-07-10 12:09:16 +08:00
```
2023-07-10 10:32:07 +08:00
2023-07-12 09:47:14 +08:00
`fdt` command is...
2023-07-11 16:36:26 +08:00
2023-07-12 09:47:14 +08:00
```text
fdt - flattened device tree utility commands
2023-07-11 18:09:45 +08:00
2023-07-12 09:47:14 +08:00
Usage:
fdt addr [-c] < addr > [< length > ] - Set the [control] fdt location to < addr >
fdt apply < addr > - Apply overlay to the DT
fdt move < fdt > < newaddr > < length > - Copy the fdt to < addr > and make it active
fdt resize [< extrasize > ] - Resize fdt to size + padding to 4k addr + some optional < extrasize > if needed
fdt print < path > [< prop > ] - Recursive print starting at < path >
fdt list < path > [< prop > ] - Print one level starting at < path >
fdt get value < var > < path > < prop > - Get < property > and store in < var >
fdt get name < var > < path > < index > - Get name of node < index > and store in < var >
fdt get addr < var > < path > < prop > - Get start address of < property > and store in < var >
fdt get size < var > < path > [< prop > ] - Get size of [< property > ] or num nodes and store in < var >
fdt set < path > < prop > [< val > ] - Set < property > [to < val > ]
fdt mknode < path > < node > - Create a new node after < path >
fdt rm < path > [< prop > ] - Delete the node or < property >
fdt header [get < var > < member > ] - Display header info
get - get header member < member > and store it in < var >
fdt bootcpu < id > - Set boot cpuid
fdt memory < addr > < size > - Add/Update memory node
fdt rsvmem print - Show current mem reserves
fdt rsvmem add < addr > < size > - Add a mem reserve
fdt rsvmem delete < index > - Delete a mem reserves
fdt chosen [< start > < end > ] - Add/update the /chosen branch in the tree
< start > /< end > - initrd start/end addr
NOTE: Dereference aliases by omitting the leading '/', e.g. fdt print ethernet0.
```
2023-07-11 18:09:45 +08:00
2023-07-12 09:47:14 +08:00
`booti` command is...
2023-07-11 18:09:45 +08:00
```text
2023-07-12 09:47:14 +08:00
booti - boot Linux kernel 'Image' format from memory
2023-07-11 18:09:45 +08:00
2023-07-12 09:47:14 +08:00
Usage:
booti [addr [initrd[:size]] [fdt]]
- boot Linux flat or compressed 'Image' stored at 'addr'
The argument 'initrd' is optional and specifies the address
of an initrd in memory. The optional parameter ':size' allows
specifying the size of a RAW initrd.
Currently only booting from gz, bz2, lzma and lz4 compression
types are supported. In order to boot from any of these compressed
images, user have to set kernel_comp_addr_r and kernel_comp_size environment
variables beforehand.
Since booting a Linux kernel requires a flat device-tree, a
third argument providing the address of the device-tree blob
is required. To boot a kernel with a device-tree blob but
without an initrd image, use a '-' for the initrd argument.
```
2023-07-11 18:09:45 +08:00
2023-07-12 09:47:14 +08:00
`bootefi` command is...
2023-07-10 11:55:54 +08:00
2023-07-12 09:47:14 +08:00
```text
bootefi - Boots an EFI payload from memory
2023-07-11 17:38:41 +08:00
2023-07-12 09:47:14 +08:00
Usage:
bootefi < image address > [fdt address]
- boot EFI payload stored at address < image address > .
If specified, the device tree located at < fdt address > gets
exposed as EFI configuration table.
bootefi bootmgr [fdt address]
- load and boot EFI payload based on BootOrder/BootXXXX variables.
2023-07-11 17:38:41 +08:00
2023-07-12 09:47:14 +08:00
If specified, the device tree located at < fdt address > gets
exposed as EFI configuration table.
```
2023-07-11 17:38:41 +08:00
2023-07-12 09:47:14 +08:00
Doesn't work for NuttX...
2023-07-11 17:38:41 +08:00
2023-07-12 09:47:14 +08:00
```text
StarFive # bootefi ${kernel_addr_r} ${fdt_addr_r}
Card did not respond to voltage select! : -110
Card did not respond to voltage select! : -110
No EFI system partition
No UEFI binary known at 0x40200000
2023-07-10 11:55:54 +08:00
```
2023-07-12 09:47:14 +08:00
[`autoload` ](https://u-boot.readthedocs.io/en/latest/usage/environment.html ) setting is...
2023-07-11 17:38:41 +08:00
```text
2023-07-12 09:47:14 +08:00
autoload:
if set to “no” (any string beginning with ‘ n’ ), “bootp” and “dhcp” will just load perform a lookup of the configuration from the BOOTP server, but not try to load any image.
```
2023-07-11 17:40:16 +08:00
2023-07-12 09:47:14 +08:00
# Hang in Enter Critical Section
2023-07-11 17:40:16 +08:00
2023-07-18 08:00:04 +08:00
Read the article...
- ["Star64 JH7110 + NuttX RTOS: RISC-V Privilege Levels and UART Registers" ](https://lupyuen.github.io/articles/privilege )
2023-07-14 14:14:06 +08:00
NuttX on Star64 JH7110 hangs when entering Critical Section...
2023-07-11 17:40:16 +08:00
2023-07-12 09:47:14 +08:00
From [uart_16550.c ](https://github.com/lupyuen2/wip-pinephone-nuttx/blob/star64/drivers/serial/uart_16550.c#L1713-L1737 ):
2023-07-11 17:40:16 +08:00
2023-07-12 09:47:14 +08:00
```c
int up_putc(int ch)
{
FAR struct u16550_s *priv = (FAR struct u16550_s * )CONSOLE_DEV.priv;
irqstate_t flags;
2023-07-11 17:40:16 +08:00
2023-07-12 09:47:14 +08:00
/* All interrupts must be disabled to prevent re-entrancy and to prevent
* interrupts from firing in the serial driver code.
*/
2023-07-11 17:40:16 +08:00
2023-07-12 09:47:14 +08:00
//// This will hang!
flags = enter_critical_section();
...
u16550_putc(priv, ch);
leave_critical_section(flags);
return ch;
}
2023-07-11 17:38:41 +08:00
```
2023-07-12 09:47:14 +08:00
Which assembles to...
2023-07-11 17:38:41 +08:00
2023-07-12 09:47:14 +08:00
```text
int up_putc(int ch)
{
...
up_irq_save():
2023-07-22 08:45:49 +08:00
nuttx/include/arch/irq.h:675
2023-07-12 09:47:14 +08:00
__asm__ __volatile__
40204598: 47a1 li a5,8
4020459a: 3007b7f3 csrrc a5,mstatus,a5
up_putc():
2023-07-22 08:45:49 +08:00
nuttx/drivers/serial/uart_16550.c:1726
2023-07-12 09:47:14 +08:00
flags = enter_critical_section();
```
2023-07-11 16:36:26 +08:00
2023-07-12 09:47:14 +08:00
But `mstatus` is not accessible at Supervisor Level! Let's trace this.
2023-07-10 11:55:54 +08:00
2023-07-12 09:47:14 +08:00
[`enter_critical_section` ](https://github.com/lupyuen2/wip-pinephone-nuttx/blob/star64/include/nuttx/irq.h#L156-L191 ) calls [`up_irq_save` ](https://github.com/lupyuen2/wip-pinephone-nuttx/blob/star64/arch/risc-v/include/irq.h#L660-L689 )...
2023-07-10 11:55:54 +08:00
2023-07-12 09:47:14 +08:00
```c
// Disable interrupts and return the previous value of the mstatus register
static inline irqstate_t up_irq_save(void)
{
irqstate_t flags;
2023-07-11 14:30:03 +08:00
2023-07-12 09:47:14 +08:00
/* Read mstatus & clear machine interrupt enable (MIE) in mstatus */
2023-07-11 11:13:58 +08:00
2023-07-12 09:47:14 +08:00
__asm__ __volatile__
(
"csrrc %0, " __XSTR(CSR_STATUS) ", %1\n"
: "=r" (flags)
: "r"(STATUS_IE)
: "memory"
);
2023-07-11 11:13:58 +08:00
2023-07-12 09:47:14 +08:00
/* Return the previous mstatus value so that it can be restored with
* up_irq_restore().
*/
return flags;
}
2023-07-10 11:55:54 +08:00
```
2023-07-12 09:47:14 +08:00
`CSR_STATUS` is defined in [mode.h ](https://github.com/lupyuen2/wip-pinephone-nuttx/blob/star64/arch/risc-v/include/mode.h#L35-L103 ):
2023-07-11 14:30:03 +08:00
2023-07-12 09:47:14 +08:00
```c
#ifdef CONFIG_ARCH_USE_S_MODE
# define CSR_STATUS sstatus /* Global status register */
#else
# define CSR_STATUS mstatus /* Global status register */
#endif
```
2023-07-11 18:09:45 +08:00
2023-07-12 09:47:14 +08:00
So we need to set [CONFIG_ARCH_USE_S_MODE ](https://github.com/lupyuen2/wip-pinephone-nuttx/blob/star64/arch/risc-v/Kconfig#L278-L296 ).
2023-07-11 18:09:45 +08:00
2023-07-12 09:47:14 +08:00
Which is defined in Kernel Mode: [`rv-virt:knsh64` ](https://github.com/lupyuen2/wip-pinephone-nuttx/blob/star64/boards/risc-v/qemu-rv/rv-virt/configs/knsh64/defconfig ). So we change Build Config to...
2023-07-11 18:09:45 +08:00
2023-07-12 09:47:14 +08:00
```bash
tools/configure.sh rv-virt:knsh64
2023-07-11 14:30:03 +08:00
```
2023-07-15 20:38:26 +08:00
And we bypassed Machine Mode Initialisation during startup...
2023-07-10 11:55:54 +08:00
2023-07-15 20:24:18 +08:00
From [qemu_rv_start.c ](https://github.com/lupyuen2/wip-pinephone-nuttx/blob/star64a/arch/risc-v/src/qemu-rv/qemu_rv_start.c#L166-L231 )
2023-07-10 11:55:54 +08:00
2023-07-12 09:47:14 +08:00
```c
void qemu_rv_start(int mhartid)
{
2023-07-15 20:24:18 +08:00
// Clear BSS
DEBUGASSERT(mhartid == 0);
if (0 == mhartid) { qemu_rv_clear_bss(); }
2023-07-15 12:40:55 +08:00
2023-07-15 20:24:18 +08:00
// Bypass to S-Mode Init
2023-07-15 12:40:55 +08:00
qemu_rv_start_s(mhartid);
2023-07-11 17:40:16 +08:00
2023-07-15 20:24:18 +08:00
// Skip M-Mode Init
// TODO: What about `satp` , `stvec` , `pmpaddr0` , `pmpcfg0` ?
2023-07-14 14:14:06 +08:00
...
2023-07-12 09:47:14 +08:00
}
```
2023-07-10 10:43:32 +08:00
2023-07-12 09:47:14 +08:00
grep for `csr` in `nuttx.S` shows that no more M-Mode Registers are used.
2023-07-10 10:43:32 +08:00
2023-07-14 14:14:06 +08:00
Now Critical Section is OK yay!
2023-07-10 10:43:32 +08:00
2023-07-12 09:47:14 +08:00
```text
Starting kernel ...
clk u5_dw_i2c_clk_core already disabled
clk u5_dw_i2c_clk_apb already disabled
2023-07-14 14:14:06 +08:00
123067DFAGHBCIcd
2023-07-12 09:47:14 +08:00
```
2023-07-13 15:33:34 +08:00
2023-07-14 14:14:06 +08:00
- [See the __Build Steps__ ](https://github.com/lupyuen2/wip-pinephone-nuttx/releases/tag/star64-0.0.1 )
2023-07-13 15:33:34 +08:00
2023-07-14 14:14:06 +08:00
- [See the __Modified Files__ ](https://github.com/lupyuen2/wip-pinephone-nuttx/pull/31/files )
2023-07-11 18:09:45 +08:00
2023-07-14 14:14:06 +08:00
- [See the __Build Outputs__ ](https://github.com/lupyuen2/wip-pinephone-nuttx/releases/tag/star64-0.0.1 )
2023-07-15 20:38:26 +08:00
_What about `satp` , `stvec` , `pmpaddr0` , `pmpcfg0` ?_
We'll handle them in a while.
2023-07-14 14:14:06 +08:00
Sometimes we see this...
2023-07-12 07:29:52 +08:00
2023-07-12 09:47:14 +08:00
```text
Starting kernel ...
clk u5_dw_i2c_clk_core already disabled
clk u5_dw_i2c_clk_apb already disabled
123067DFAGHBCUnhandled exception: Store/AMO access fault
EPC: 0000000040200628 RA: 00000000402004ba TVAL: ffffff8000008000
EPC: ffffffff804ba628 RA: ffffffff804ba4ba reloc adjusted
2023-07-12 07:29:52 +08:00
2023-07-12 09:47:14 +08:00
SP: 0000000040406a30 GP: 00000000ff735e00 TP: 0000000000000001
T0: 0000000010000000 T1: 0000000000000037 T2: ffffffffffffffff
S0: 0000000040400000 S1: 0000000000000200 A0: 0000000000000003
A1: 0000080000008000 A2: 0000000010100000 A3: 0000000040400000
A4: 0000000000000026 A5: 0000000000000000 A6: 00000000101000e7
A7: 0000000000000000 S2: 0000080000008000 S3: 0000000040600000
S4: 0000000040400000 S5: 0000000000000000 S6: 0000000000000026
S7: 00fffffffffff000 S8: 0000000040404000 S9: 0000000000001000
S10: 0000000040400ab0 S11: 0000000000200000 T3: 0000000000000023
T4: 000000004600f43a T5: 000000004600d000 T6: 000000004600cfff
2023-07-10 10:43:32 +08:00
2023-07-12 09:47:14 +08:00
Code: 879b 0277 d7b3 00f6 f793 1ff7 078e 95be (b023 0105)
2023-07-10 11:43:26 +08:00
```
2023-07-12 09:47:14 +08:00
Which fails at...
2023-07-10 11:43:26 +08:00
2023-07-10 12:51:14 +08:00
```text
2023-07-12 09:47:14 +08:00
nuttx/arch/risc-v/src/common/riscv_mmu.c:101
lntable[index] = (paddr | mmuflags);
40200620: 1ff7f793 andi a5,a5,511
40200624: 078e slli a5,a5,0x3
40200626: 95be add a1,a1,a5
40200628: 0105b023 sd a6,0(a1) /* Fails Here */
mmu_invalidate_tlb_by_vaddr():
nuttx/arch/risc-v/src/common/riscv_mmu.h:237
__asm__ __volatile__
4020062c: 12d00073 sfence.vma zero,a3
40200630: 8082 ret
2023-07-10 12:51:14 +08:00
```
2023-07-14 17:47:34 +08:00
TODO: Trace this Store/AMO Access Fault
2023-07-10 12:51:14 +08:00
2023-07-14 12:31:17 +08:00
# Hang in UART Transmit
2023-07-18 08:00:04 +08:00
Read the article...
- ["Star64 JH7110 + NuttX RTOS: RISC-V Privilege Levels and UART Registers" ](https://lupyuen.github.io/articles/privilege )
2023-07-14 17:47:34 +08:00
When printing to UART Port, the UART Transmit hangs while waiting for UART Transmit Ready...
2023-07-14 12:31:17 +08:00
From [uart_16550.c ](https://github.com/lupyuen2/wip-pinephone-nuttx/blob/star64/drivers/serial/uart_16550.c#L1638-L1642 )
```c
static void u16550_putc(FAR struct u16550_s *priv, int ch)
{
//// This will hang!
2023-07-14 13:50:52 +08:00
while ((u16550_serialin(priv, UART_LSR_OFFSET) & UART_LSR_THRE) == 0);
2023-07-14 12:31:17 +08:00
u16550_serialout(priv, UART_THR_OFFSET, (uart_datawidth_t)ch);
}
```
2023-07-14 14:14:06 +08:00
Where [`u16550_serialin` ](https://github.com/lupyuen2/wip-pinephone-nuttx/blob/star64a/drivers/serial/uart_16550.c#L596-L611 ) is defined as...
2023-07-14 13:50:52 +08:00
```c
*((FAR volatile uart_datawidth_t *)priv->uartbase + offset);
```
2023-07-14 14:14:06 +08:00
And [`UART_THR_OFFSET` ](https://github.com/lupyuen2/wip-pinephone-nuttx/blob/star64a/include/nuttx/serial/uart_16550.h#L197 ) is...
2023-07-14 13:50:52 +08:00
```c
#define UART_LSR_OFFSET (CONFIG_16550_REGINCR*UART_LSR_INCR)
```
2023-07-14 14:14:06 +08:00
`CONFIG_16550_REGINCR` is 1...
2023-07-14 12:31:17 +08:00
```text
2023-07-14 13:50:52 +08:00
→ grep 16550 .config
2023-07-14 12:31:17 +08:00
CONFIG_16550_REGINCR=1
CONFIG_16550_REGWIDTH=8
CONFIG_16550_ADDRWIDTH=0
```
2023-07-14 14:14:06 +08:00
As defined according to the Default Config for 16550: [Kconfig-16550 ](https://github.com/lupyuen2/wip-pinephone-nuttx/blob/star64a/drivers/serial/Kconfig-16550#L501-L520 ):
2023-07-14 12:31:17 +08:00
```text
config 16550_REGINCR
int "Address increment between 16550 registers"
default 1
---help---
The address increment between 16550 registers. Options are 1, 2, or 4.
Default: 1
config 16550_REGWIDTH
int "Bit width of 16550 registers"
default 8
---help---
The bit width of registers. Options are 8, 16, or 32. Default: 8
config 16550_ADDRWIDTH
int "Address width of 16550 registers"
default 8
---help---
The bit width of registers. Options are 0, 8, 16, or 32.
Default: 8
Note: 0 means auto detect address size (uintptr_t)
```
2023-07-14 17:47:34 +08:00
_But is CONFIG_16550_REGINCR correct for Star64 JH7110?_
Let's check the official Linux Driver. According to [JH7110 Linux Device Tree ](https://doc-en.rvspace.org/VisionFive2/DG_UART/JH7110_SDK/general_uart_controller.html )...
2023-07-14 12:31:17 +08:00
```text
reg = < 0x0 0x10000000 0x0 0xl0000 > ;
reg-io-width = < 4 > ;
reg-shift = < 2 > ;
```
2023-07-14 14:14:06 +08:00
`reg-shift` is 2.
And from the Linux 8250 Driver: [8250_dw.c ](https://github.com/torvalds/linux/blob/master/drivers/tty/serial/8250/8250_dw.c#L159-L169 )
2023-07-14 13:50:52 +08:00
```text
static void dw8250_serial_out(struct uart_port *p, int offset, int value)
{
struct dw8250_data *d = to_dw8250_data(p->private_data);
writeb(value, p->membase + (offset < < p- > regshift));
if (offset == UART_LCR & & !d->uart_16550_compatible)
dw8250_check_lcr(p, value);
}
```
2023-07-14 17:47:34 +08:00
We see that the UART Offset is shifted by 2 (`regshift`). Which means we multiply the UART Offset by 4.
2023-07-14 13:50:52 +08:00
2023-07-14 14:14:06 +08:00
Thus `CONFIG_16550_REGINCR` should be 4, not 1!
2023-07-14 13:50:52 +08:00
2023-07-14 17:47:34 +08:00
_How to fix CONFIG_16550_REGINCR?_
2023-07-14 14:14:06 +08:00
We fix the NuttX Configuration: Device Drivers > Serial Driver Support > 16550 UART Chip support > Address increment between 16550 registers
2023-07-14 13:50:52 +08:00
2023-07-14 14:14:06 +08:00
And change it from 1 to 4: [knsh64/defconfig ](https://github.com/lupyuen2/wip-pinephone-nuttx/blob/star64a/boards/risc-v/qemu-rv/rv-virt/configs/knsh64/defconfig#L11 )
2023-07-14 14:00:24 +08:00
```text
CONFIG_16550_REGINCR=4
```
2023-07-14 14:14:06 +08:00
Now UART Transmit doesn't hang yay!
2023-07-14 14:00:24 +08:00
```text
Starting kernel ...
clk u5_dw_i2c_clk_core already disabled
clk u5_dw_i2c_clk_apb already disabled
123067DFm45DTpAqGaclbHm45DTpBqm45DTpCqI
```
2023-07-14 13:50:52 +08:00
2023-07-14 17:47:34 +08:00
NuttX now hangs somewhere in `nx_start`
2023-07-15 20:24:18 +08:00
Let's log the NuttX Scheduler...
2023-07-14 14:14:06 +08:00
2023-07-15 15:21:31 +08:00
# Enable Scheduler Logging
2023-07-15 16:56:07 +08:00
Scheduler Logging in NuttX seems to have changed recently. To enable Scheduler Logging...
2023-07-15 15:21:31 +08:00
- `make menuconfig`
- Disable this setting: Device Drivers > System Logging > Prepend timestamp to syslog message
- Enable these settings: Build Setup > Debug Options > Scheduler Debug Features > Scheduler Error, Warnings and Info Output
2023-07-15 15:55:17 +08:00
- Also enable: Build Setup > Debug Options > Binary Loader Debug Features > Binary Loader Error, Warnings and Info Output
After enabling Scheduler Logging and Binary Loader Logging, we see...
2023-07-11 14:41:53 +08:00
2023-07-15 15:19:25 +08:00
```text
2023-07-15 15:55:17 +08:00
123067DFAGaclbHBCqemu_rv_kernel_mappings: map I/O regions
qemu_rv_kernel_mappings: map kernel text
qemu_rv_kernel_mappings: map kernel data
qemu_rv_kernel_mappings: connect the L1 and L2 page tables
qemu_rv_kernel_mappings: map the page pool
qemu_rv_mm_init: mmu_enable: satp=1077956608
Inx_start: Entry
elf_initialize: Registering ELF
2023-07-15 15:19:25 +08:00
uart_register: Registering /dev/console
uart_register: Registering /dev/ttyS0
work_start_lowpri: Starting low-priority kernel worker thread(s)
nx_start_application: Starting init task: /system/bin/init
2023-07-15 15:55:17 +08:00
load_absmodule: Loading /system/bin/init
elf_loadbinary: Loading file: /system/bin/init
elf_init: filename: /system/bin/init loadinfo: 0x404069e8
2023-07-15 15:19:25 +08:00
```
2023-07-16 13:30:34 +08:00
_What is `/system/bin/init` ?_
We'll find out in a while...
2023-07-15 15:44:46 +08:00
2023-07-18 20:03:09 +08:00
[Compare with QEMU Kernel Mode Run Log ](https://gist.github.com/lupyuen/6888376da6561bdc060c2459dffdef01 )
2023-07-15 17:35:35 +08:00
2023-07-15 20:18:59 +08:00
[See the QEMU Kernel Mode Build Log ](https://gist.github.com/lupyuen/dce0cdbbf4a4bdf9c79e617b3fe1b679 )
2023-07-15 17:35:35 +08:00
2023-07-15 20:38:26 +08:00
# Initialise RISC-V Supervisor Mode
2023-07-18 08:00:04 +08:00
Read the article...
- ["Star64 JH7110 + NuttX RTOS: RISC-V Privilege Levels and UART Registers" ](https://lupyuen.github.io/articles/privilege )
2023-07-16 13:30:34 +08:00
Earlier we bypassed the Machine Mode and Supervisor Mode Initialisation during NuttX startup...
2023-07-15 20:38:26 +08:00
From [qemu_rv_start.c ](https://github.com/lupyuen2/wip-pinephone-nuttx/blob/star64a/arch/risc-v/src/qemu-rv/qemu_rv_start.c#L166-L231 )
```c
void qemu_rv_start(int mhartid)
{
// Clear BSS
DEBUGASSERT(mhartid == 0);
if (0 == mhartid) { qemu_rv_clear_bss(); }
// Bypass to S-Mode Init
qemu_rv_start_s(mhartid);
// Skip M-Mode Init
// TODO: What about `satp` , `stvec` , `pmpaddr0` , `pmpcfg0` ?
...
}
```
2023-07-16 14:45:57 +08:00
Now we restore the Supervisor Mode Initialisation, commenting out the Machine Mode Initialisation...
2023-07-16 13:30:34 +08:00
From [qemu_rv_start.c ](https://github.com/lupyuen2/wip-pinephone-nuttx/blob/star64a/arch/risc-v/src/qemu-rv/qemu_rv_start.c#L165-L233 ):
```c
void qemu_rv_start(int mhartid)
{
2023-07-16 14:45:57 +08:00
DEBUGASSERT(mhartid == 0); //
2023-07-16 13:30:34 +08:00
/* NOTE: still in M-mode */
if (0 == mhartid)
{
qemu_rv_clear_bss();
/* Initialize the per CPU areas */
riscv_percpu_add_hart(mhartid);
}
/* Disable MMU and enable PMP */
WRITE_CSR(satp, 0x0);
2023-07-16 14:45:57 +08:00
//WRITE_CSR(pmpaddr0, 0x3fffffffffffffull);
//WRITE_CSR(pmpcfg0, 0xf);
2023-07-16 13:30:34 +08:00
/* Set exception and interrupt delegation for S-mode */
2023-07-16 14:45:57 +08:00
//WRITE_CSR(medeleg, 0xffff);
//WRITE_CSR(mideleg, 0xffff);
2023-07-16 13:30:34 +08:00
/* Allow to write satp from S-mode */
2023-07-16 14:45:57 +08:00
//CLEAR_CSR(mstatus, MSTATUS_TVM);
2023-07-16 13:30:34 +08:00
/* Set mstatus to S-mode and enable SUM */
2023-07-16 14:45:57 +08:00
//CLEAR_CSR(mstatus, ~MSTATUS_MPP_MASK);
//SET_CSR(mstatus, MSTATUS_MPPS | SSTATUS_SUM);
2023-07-16 13:30:34 +08:00
/* Set the trap vector for S-mode */
WRITE_CSR(stvec, (uintptr_t)__trap_vec);
/* Set the trap vector for M-mode */
2023-07-16 14:45:57 +08:00
//WRITE_CSR(mtvec, (uintptr_t)__trap_vec_m);
2023-07-16 13:30:34 +08:00
if (0 == mhartid)
{
/* Only the primary CPU needs to initialize mtimer
* before entering to S-mode
*/
2023-07-16 14:45:57 +08:00
// TODO
//up_mtimer_initialize();
2023-07-16 13:30:34 +08:00
}
/* Set mepc to the entry */
2023-07-16 14:45:57 +08:00
//WRITE_CSR(mepc, (uintptr_t)qemu_rv_start_s);
2023-07-16 13:30:34 +08:00
/* Set a0 to mhartid explicitly and enter to S-mode */
2023-07-16 14:45:57 +08:00
//asm volatile (
// "mv a0, %0 \n"
// "mret \n"
// :: "r" (mhartid)
//);
2023-07-16 13:30:34 +08:00
2023-07-16 14:45:57 +08:00
// Jump to S-Mode Init ourselves
qemu_rv_start_s(mhartid); //
2023-07-16 13:30:34 +08:00
}
```
2023-07-20 09:45:47 +08:00
TODO: Port [__up_mtimer_initialize__ ](https://github.com/lupyuen2/wip-pinephone-nuttx/blob/star64a/arch/risc-v/src/qemu-rv/qemu_rv_timerisr.c#L151-L210 ) to Star64
2023-07-16 13:30:34 +08:00
Now NuttX boots further!
2023-07-15 20:38:26 +08:00
2023-07-15 20:42:12 +08:00
```text
123067DFHBCqemu_rv_kernel_mappings: map I/O regions
qemu_rv_kernel_mappings: map kernel text
qemu_rv_kernel_mappings: map kernel data
qemu_rv_kernel_mappings: connect the L1 and L2 page tables
qemu_rv_kernel_mappings: map the page pool
qemu_rv_mm_init: mmu_enable: satp=1077956608
Inx_start: Entry
elf_initialize: Registering ELF
uart_register: Registering /dev/console
uart_register: Registering /dev/ttyS0
work_start_lowpri: Starting low-priority kernel worker thread(s)
nx_start_application: Starting init task: /system/bin/init
load_absmodule: Loading /system/bin/init
elf_loadbinary: Loading file: /system/bin/init
elf_init: filename: /system/bin/init loadinfo: 0x404069e8
riscv_exception: EXCEPTION: Breakpoint. MCAUSE: 0000000000000003, EPC: 0000000040200434, MTVAL: 0000000000000000
riscv_exception: PANIC!!! Exception = 0000000000000003
_assert: Current Version: NuttX 12.0.3 2261b80-dirty Jul 15 2023 20:38:57 risc-v
_assert: Assertion failed panic: at file: common/riscv_exception.c:85 task: Idle Task 0x40200ce6
up_dump_register: EPC: 0000000040200434
up_dump_register: A0: 0000000000000001 A1: 0000000040406778 A2: 0000000000000000 A3: 0000000000000001
up_dump_register: A4: 0000000000000000 A5: 00000000404067e0 A6: 0000000000000074 A7: fffffffffffffff8
up_dump_register: T0: 0000000000000030 T1: 0000000000000007 T2: 0000000000000020 T3: 0000000040406aa0
up_dump_register: T4: 0000000040406a98 T5: 00000000000001ff T6: 000000000000002d
up_dump_register: S0: 0000000000000000 S1: 0000000040406968 S2: 0000000040408720 S3: 0000000000000000
up_dump_register: S4: 0000000000000000 S5: 0000000000000000 S6: 0000000000000000 S7: 0000000000000000
up_dump_register: S8: 00000000fff47194 S9: 0000000000000000 S10: 0000000000000000 S11: 0000000000000000
up_dump_register: SP: 0000000040406760 FP: 0000000000000000 TP: 0000000000000001 RA: 0000000040213e24
dump_stack: User Stack:
dump_stack: base: 0x40406030
dump_stack: size: 00003024
dump_stack: sp: 0x40406760
stack_dump: 0x40406760: 00000000 00000000 40213e6a 00000000 fff47194 00000000 404067d0 00000000
stack_dump: 0x40406780: 00000001 00000000 00000010 00000000 00000000 00000000 40213ffc 00000000
stack_dump: 0x404067a0: 40408720 00000000 40406968 00000000 00000000 00000000 4020c7ec 00000000
stack_dump: 0x404067c0: 00000800 00000000 40219f30 00000000 612f2e2e 2f737070 2f6e6962 74696e69
stack_dump: 0x404067e0: 00000a00 00000000 00000000 00000000 00000000 00000000 00000000 00000000
stack_dump: 0x40406800: fff47194 00000000 00000000 00000000 00000000 00000000 00000000 00000000
stack_dump: 0x40406820: 00000000 00000000 00000000 00000000 40219f28 00000000 404069e8 00000000
stack_dump: 0x40406840: 40219f28 00000000 40212bde 00000000 40227776 00000000 40406870 00000000
stack_dump: 0x40406860: 00000000 00000000 fffffffc ffffffff 40219f28 00000000 404069e8 00000000
stack_dump: 0x40406880: 40400170 00000000 40204fea 00000000 0000006c 00000000 404069e8 00000000
stack_dump: 0x404068a0: 40400170 00000000 402050ae 00000000 40406908 00000000 40208f66 00000000
stack_dump: 0x404068c0: 40406908 00000000 4020c8c6 00000000 40219f28 00000000 404086d0 00000000
stack_dump: 0x404068e0: ffffffda ffffffff 40215be6 00000000 40406968 00000000 00000001 00000000
stack_dump: 0x40406900: 40400b28 00000000 40219f30 00000000 404086d0 00000000 40407e30 00000000
stack_dump: 0x40406920: 40407370 00000000 40219f30 00000000 00000000 00000000 40219f01 00000000
stack_dump: 0x40406940: 404069e8 00000000 404069e8 00000000 40219f28 00000000 4020dfdc 00000000
stack_dump: 0x40406960: 40219f28 00000000 40205ede 00000000 fff47194 00000000 404069b0 00000000
stack_dump: 0x40406980: 00000000 00000000 40205efe 00000000 00000000 00000000 404069b0 00000000
stack_dump: 0x404069a0: 40408830 00000000 4020d88c 00000000 40226bc0 00000000 40219f28 00000000
stack_dump: 0x404069c0: 40408830 00000000 00000000 00000000 40219f28 00000000 4020d894 00000000
stack_dump: 0x404069e0: 40406a18 00000000 00000000 00000000 00000000 00000000 00000000 00000000
stack_dump: 0x40406a00: 00000000 00000000 00000000 00000000 00000000 00000000 00000000 00000000
stack_dump: 0x40406a20: 00000000 00000000 00000000 00000000 00000000 00000000 00000000 00000000
stack_dump: 0x40406a40: 00000000 00000000 00000000 00000000 00000000 00000000 00000000 00000000
stack_dump: 0x40406a60: 00000000 00000000 00000000 00000000 00000000 00000000 00000000 00000000
stack_dump: 0x40406a80: 00000000 00000000 00000000 00000000 00000000 00000000 00000000 00000000
stack_dump: 0x40406aa0: 402277d0 00000000 40219f28 00000000 40408830 00000000 404001f8 00000000
stack_dump: 0x40406ac0: fffffffe ffffffff 4020eb36 00000000 00000000 00000000 00000000 00000000
stack_dump: 0x40406ae0: 40406b60 00000000 40406b68 00000000 40219f28 00000000 40408830 00000000
stack_dump: 0x40406b00: 00000c00 00000000 4020d38a 00000000 00000000 00000000 00000000 00000000
stack_dump: 0x40406b20: 00000000 00000000 fffda848 00000000 fffffff3 ffffffff 40400b18 00000000
stack_dump: 0x40406b40: 4040177c 00000000 00000064 00000000 00000c00 00000000 40200ff4 00000000
stack_dump: 0x40406b60: 00000000 00000000 40016400 00000000 00000000 00000000 00000c00 00000000
stack_dump: 0x40406b80: 4040177c 00000000 40401780 00000000 40400b28 00000000 40200ee6 00000000
stack_dump: 0x40406ba0: 40600000 00000000 00400000 00000000 00000026 00000000 00000003 00000000
stack_dump: 0x40406bc0: fff47194 00000000 ffff1428 00000000 10000000 00000000 40200514 00000000
stack_dump: 0x40406be0: 00000400 00000000 40200552 00000000 40000000 00000000 402000de 00000000
dump_tasks: PID GROUP PRI POLICY TYPE NPX STATE EVENT SIGMASK STACKBASE STACKSIZE USED FILLED COMMAND
dump_tasks: ---- --- --- -------- ------- --- ------- ---------- -------- 0x404002b0 2048 1160 56.6% irq
dump_task: 0 0 0 FIFO Kthread N-- Running 0000000000000000 0x40406030 3024 1448 47.8% Idle Task
dump_task: 1 1 100 RR Kthread --- Waiting Unlock 0000000000000000 0x4040a060 1952 264 13.5% lpwork 0x404013e0
```
2023-07-16 13:30:34 +08:00
But NuttX crashes. Let's find out why...
# QEMU Semihosting in NuttX
2023-07-27 18:32:43 +08:00
Read the article...
- ["Star64 JH7110 + NuttX RTOS: RISC-V Semihosting and Initial RAM Disk" ](https://lupyuen.github.io/articles/semihost )
2023-07-21 12:09:49 +08:00
NuttX crashes while booting on Star64 JH7110 SBC. From the Crash Dump above, [`mcause` ](https://five-embeddev.com/riscv-isa-manual/latest/machine.html#sec:mcause ) is 3: "Machine Software Interrupt".
2023-07-15 21:00:42 +08:00
Exception Program Counter `0x4020` `0434` is in RISC-V Semihosting `smh_call` ...
```text
0000000040200430 < smh_call > :
smh_call():
2023-07-22 08:45:49 +08:00
nuttx/arch/risc-v/src/common/riscv_semihost.S:37
2023-07-15 21:00:42 +08:00
.global smh_call
.type smh_call @function
smh_call:
slli zero, zero, 0x1f
40200430: 01f01013 slli zero,zero,0x1f
2023-07-22 08:45:49 +08:00
nuttx/arch/risc-v/src/common/riscv_semihost.S:38
2023-07-15 21:00:42 +08:00
ebreak
//// Crashes here (Trigger semihosting breakpoint)
40200434: 00100073 ebreak
2023-07-22 08:45:49 +08:00
nuttx/arch/risc-v/src/common/riscv_semihost.S:39
2023-07-15 21:00:42 +08:00
srai zero, zero, 0x7
40200438: 40705013 srai zero,zero,0x7
2023-07-22 08:45:49 +08:00
nuttx/arch/risc-v/src/common/riscv_semihost.S:40
2023-07-15 21:00:42 +08:00
ret
4020043c: 00008067 ret
40200440: 0000 unimp
```
2023-07-19 10:46:50 +08:00
When we log `smh_call` ...
2023-07-15 21:00:42 +08:00
```text
host_call: nbr=0x1, parm=0x40406778, size=24
```
[host_call ](https://github.com/lupyuen2/wip-pinephone-nuttx/blob/star64a/arch/risc-v/src/common/riscv_hostfs.c#L35-L73 ) says that the Semihosting Call is for HOST_OPEN. (Open a file)
2023-07-19 10:46:50 +08:00
So NuttX crashes on Star64 because it's trying to read `/system/bin/init` via Semihosting!
(See next section)
Let's disable Semihosting and replace by Initial RAM Disk and ROMFS.
2023-07-21 12:09:49 +08:00
(See https://github.com/apache/nuttx/issues/9501)
2023-07-19 10:46:50 +08:00
2023-07-25 11:35:04 +08:00
![QEMU reads the Apps Filesystem over Semihosting ](https://lupyuen.github.io/images/semihost-qemu.jpg )
2023-07-25 09:55:23 +08:00
2023-07-19 10:46:50 +08:00
Here's the Crash Dump after we disabled Semihosting...
2023-07-15 21:07:26 +08:00
```text
123067DFHBCqemu_rv_kernel_mappings: map I/O regions
qemu_rv_kernel_mappings: map kernel text
qemu_rv_kernel_mappings: map kernel data
qemu_rv_kernel_mappings: connect the L1 and L2 page tables
qemu_rv_kernel_mappings: map the page pool
qemu_rv_mm_init: mmu_enable: satp=1077956608
Inx_start: Entry
elf_initialize: Registering ELF
uart_register: Registering /dev/console
uart_register: Registering /dev/ttyS0
work_start_lowpri: Starting low-priority kernel worker thread(s)
nx_start_application: Starting init task: /system/bin/init
load_absmodule: Loading /system/bin/init
elf_loadbinary: Loading file: /system/bin/init
elf_init: filename: /system/bin/init loadinfo: 0x404069e8
host_call: nbr=0x1, parm=0x40406788, size=24
_assert: Current Version: NuttX 12.0.3 6ed2880-dirty Jul 15 2023 21:00:59 risc-v
_assert: Assertion failed panic: at file: common/riscv_hostfs.c:58 task: Idle Task 0x40200cd0
up_dump_register: EPC: 000000004020f590
up_dump_register: A0: 0000000040401630 A1: 000000000000003a A2: 0000000040219ee8 A3: 0000000000000000
up_dump_register: A4: 000000000000000a A5: 0000000000000000 A6: 0000000000000009 A7: 0000000000000068
up_dump_register: T0: 0000000000000030 T1: 0000000000000009 T2: 0000000000000020 T3: 000000000000002a
up_dump_register: T4: 000000000000002e T5: 00000000000001ff T6: 000000000000002d
up_dump_register: S0: 0000000000000000 S1: 0000000040400b28 S2: 0000000040401768 S3: 0000000040219ee8
up_dump_register: S4: 0000000040229b10 S5: 000000000000003a S6: 0000000000000000 S7: 0000000000000000
up_dump_register: S8: 00000000fff47194 S9: 0000000000000000 S10: 0000000000000000 S11: 0000000000000000
up_dump_register: SP: 0000000040406650 FP: 0000000000000000 TP: 0000000000000001 RA: 000000004020f590
dump_stack: User Stack:
dump_stack: base: 0x40406030
dump_stack: size: 00003024
dump_stack: sp: 0x40406650
stack_dump: 0x40406640: 40406650 00000000 4020f688 00000000 00000000 00000000 40212bc8 00000000
stack_dump: 0x40406660: deadbeef deadbeef 40406680 00000000 deadbeef deadbeef 7474754e 00000058
stack_dump: 0x40406680: 404066b8 00000000 00000001 00000000 40406788 00000000 40205cc0 00000000
stack_dump: 0x404066a0: 00000074 00000000 fffffff8 2e323100 00332e30 00000000 40229ae8 00000000
stack_dump: 0x404066c0: 65366708 38383264 69642d30 20797472 206c754a 32203531 20333230 303a3132
stack_dump: 0x404066e0: 39353a30 00000000 0000000a 00000000 00000000 73697200 00762d63 00000000
stack_dump: 0x40406700: ffff9fef ffffffff 40406740 00000000 fff47194 00000000 00000000 00000000
stack_dump: 0x40406720: 00000000 00000000 00000000 00000000 00000000 00000000 00000000 00000000
stack_dump: 0x40406740: 40408720 00000000 40406968 00000000 00000000 00000000 40204e80 00000000
stack_dump: 0x40406760: 00000074 00000000 40213e3c 00000000 fff47194 00000000 40213e64 00000000
stack_dump: 0x40406780: 00000000 00000000 404067d0 00000000 00000001 00000000 00000010 00000000
stack_dump: 0x404067a0: 40408720 00000000 40213f7e 00000000 00000000 00000000 4020c7d6 00000000
stack_dump: 0x404067c0: 00000800 00000000 40219e70 00000000 612f2e2e 2f737070 2f6e6962 74696e69
stack_dump: 0x404067e0: 00000a00 00000000 00000000 00000000 00000000 00000000 00000000 00000000
stack_dump: 0x40406800: fff47194 00000000 00000000 00000000 00000000 00000000 00000000 00000000
stack_dump: 0x40406820: 00000000 00000000 00000000 00000000 40219e68 00000000 404069e8 00000000
stack_dump: 0x40406840: 40219e68 00000000 40212bc8 00000000 402276b6 00000000 40406870 00000000
stack_dump: 0x40406860: 00000000 00000000 fffffffc ffffffff 40219e68 00000000 404069e8 00000000
stack_dump: 0x40406880: 40400170 00000000 40204fd4 00000000 0000006c 00000000 404069e8 00000000
stack_dump: 0x404068a0: 40400170 00000000 40205098 00000000 40406908 00000000 40208f50 00000000
stack_dump: 0x404068c0: 40406908 00000000 4020c8b0 00000000 40219e68 00000000 404086d0 00000000
stack_dump: 0x404068e0: ffffffda ffffffff 40215b2a 00000000 40406968 00000000 00000001 00000000
stack_dump: 0x40406900: 40400b28 00000000 40219e70 00000000 404086d0 00000000 40407e30 00000000
stack_dump: 0x40406920: 40407370 00000000 40219e70 00000000 00000000 00000000 40219e01 00000000
stack_dump: 0x40406940: 404069e8 00000000 404069e8 00000000 40219e68 00000000 4020dfc6 00000000
stack_dump: 0x40406960: 40219e68 00000000 40205ec8 00000000 fff47194 00000000 404069b0 00000000
stack_dump: 0x40406980: 00000000 00000000 40205ee8 00000000 00000000 00000000 404069b0 00000000
stack_dump: 0x404069a0: 40408830 00000000 4020d876 00000000 40226b00 00000000 40219e68 00000000
stack_dump: 0x404069c0: 40408830 00000000 00000000 00000000 40219e68 00000000 4020d87e 00000000
stack_dump: 0x404069e0: 40406a18 00000000 00000000 00000000 00000000 00000000 00000000 00000000
stack_dump: 0x40406a00: 00000000 00000000 00000000 00000000 00000000 00000000 00000000 00000000
stack_dump: 0x40406a20: 00000000 00000000 00000000 00000000 00000000 00000000 00000000 00000000
stack_dump: 0x40406a40: 00000000 00000000 00000000 00000000 00000000 00000000 00000000 00000000
stack_dump: 0x40406a60: 00000000 00000000 00000000 00000000 00000000 00000000 00000000 00000000
stack_dump: 0x40406a80: 00000000 00000000 00000000 00000000 00000000 00000000 00000000 00000000
stack_dump: 0x40406aa0: 40227710 00000000 40219e68 00000000 40408830 00000000 404001f8 00000000
stack_dump: 0x40406ac0: fffffffe ffffffff 4020eb20 00000000 00000000 00000000 00000000 00000000
stack_dump: 0x40406ae0: 40406b60 00000000 40406b68 00000000 40219e68 00000000 40408830 00000000
stack_dump: 0x40406b00: 00000c00 00000000 4020d374 00000000 00000000 00000000 00000000 00000000
stack_dump: 0x40406b20: 00000000 00000000 fffda848 00000000 fffffff3 ffffffff 40400b18 00000000
stack_dump: 0x40406b40: 4040177c 00000000 00000064 00000000 00000c00 00000000 40200fde 00000000
stack_dump: 0x40406b60: 00000000 00000000 40016400 00000000 00000000 00000000 00000c00 00000000
stack_dump: 0x40406b80: 4040177c 00000000 40401780 00000000 40400b28 00000000 40200ed0 00000000
stack_dump: 0x40406ba0: 40600000 00000000 00400000 00000000 00000026 00000000 00000003 00000000
stack_dump: 0x40406bc0: fff47194 00000000 ffff1428 00000000 10000000 00000000 402004fe 00000000
stack_dump: 0x40406be0: 00000400 00000000 4020053c 00000000 00000000 00000000 402000de 00000000
dump_tasks: PID GROUP PRI POLICY TYPE NPX STATE EVENT SIGMASK STACKBASE STACKSIZE USED FILLED COMMAND
dump_tasks: ---- --- --- -------- ------- --- ------- ---------- -------- 0x404002b0 2048 0 0.0% irq
dump_task: 0 0 0 FIFO Kthread N-- Running 0000000000000000 0x40406030 3024 2248 74.3% Idle Task
dump_task: 1 1 100 RR Kthread --- Waiting Unlock 0000000000000000 0x4040a060 1952 264 13.5% lpwork 0x404013e0
```
2023-07-15 20:45:30 +08:00
2023-07-20 09:45:47 +08:00
# NuttX Apps Filesystem
2023-07-15 21:55:03 +08:00
2023-07-27 18:32:43 +08:00
Read the article...
- ["Star64 JH7110 + NuttX RTOS: RISC-V Semihosting and Initial RAM Disk" ](https://lupyuen.github.io/articles/semihost )
2023-07-21 12:09:49 +08:00
_Where is `/system/bin/init` ? Why is it loaded by NuttX over Semihosting?_
2023-07-19 10:46:50 +08:00
2023-07-21 12:09:49 +08:00
`/system/bin/init` is needed for starting the NuttX Shell (and NuttX Apps) on Star64 JH7110 SBC.
We see it in the NuttX Build Configuration...
2023-07-15 21:55:03 +08:00
```text
→ grep INIT .config
CONFIG_INIT_FILE=y
CONFIG_INIT_ARGS=""
CONFIG_INIT_STACKSIZE=3072
CONFIG_INIT_PRIORITY=100
CONFIG_INIT_FILEPATH="/system/bin/init"
CONFIG_INIT_MOUNT=y
CONFIG_INIT_MOUNT_SOURCE=""
CONFIG_INIT_MOUNT_TARGET="/system"
CONFIG_INIT_MOUNT_FSTYPE="hostfs"
CONFIG_INIT_MOUNT_FLAGS=0x1
CONFIG_INIT_MOUNT_DATA="fs=../apps"
CONFIG_PATH_INITIAL="/system/bin"
CONFIG_NSH_ARCHINIT=y
```
2023-07-21 12:09:49 +08:00
Which says that `../apps` is mounted as `/system` , via Semihosting HostFS.
2023-07-15 21:55:03 +08:00
That's how `/system/bin/init` gets loaded over Semihosting...
```
→ ls ../apps/bin
getprime hello init sh
```
2023-07-25 11:35:04 +08:00
![QEMU reads the Apps Filesystem over Semihosting ](https://lupyuen.github.io/images/semihost-qemu.jpg )
2023-07-19 10:46:50 +08:00
We traced the Semihosting Calls in QEMU Kernel Mode, here's what we observed...
[QEMU Kernel Mode Run Log ](https://gist.github.com/lupyuen/6888376da6561bdc060c2459dffdef01 )
```text
nx_start_application: Starting init task: /system/bin/init
load_absmodule: Loading /system/bin/init
elf_loadbinary: Loading file: /system/bin/init
elf_init: filename: /system/bin/init loadinfo: 0x802069e8
hostfs_open: relpath=bin/init, oflags=0x1, mode=0x1b6
...
NuttShell (NSH) NuttX-12.2.1-RC0
nsh> nx_start: CPU0: Beginning Idle Loop
nsh>
nsh> uname -a
posix_spawn: pid=0xc0202978 path=uname file_actions=0xc0202980 attr=0xc0202988 argv=0xc0202a28
hostfs_stat: relpath=bin/uname
host_call: nbr=0x1, parm=0x80208fe0, size=24
exec_spawn: ERROR: Failed to load program 'uname': -2
nxposix_spawn_exec: ERROR: exec failed: 2
NuttX 12.2.1-RC0 cafbbb1 Jul 15 2023 16:55:00 risc-v rv-virt
nsh>
nsh> ls /
posix_spawn: pid=0xc0202978 path=ls file_actions=0xc0202980 attr=0xc0202988 argv=0xc0202a28
hostfs_stat: relpath=bin/ls
host_call: nbr=0x1, parm=0x80208fe0, size=24
exec_spawn: ERROR: Failed to load program 'ls': -2
nxposix_spawn_exec: ERROR: exec failed: 2
/:
dev/
proc/
system/
nsh>
nsh> ls /system
posix_spawn: pid=0xc0202978 path=ls file_actions=0xc0202980 attr=0xc0202988 argv=0xc0202a28
hostfs_stat: relpath=bin/ls
host_call: nbr=0x1, parm=0x80208fe0, size=24
exec_spawn: ERROR: Failed to load program 'ls': -2
nxposix_spawn_exec: ERROR: exec failed: 2
hostfs_stat: relpath=
host_call: nbr=0x1, parm=0x80209180, size=24
host_call: nbr=0xc, parm=0x80209180, size=8
host_call: nbr=0x2, parm=0x80209190, size=8
/system
nsh>
nsh> ls /system/bin
posix_spawn: pid=0xc0202978 path=ls file_actions=0xc0202980 attr=0xc0202988 argv=0xc0202a28
hostfs_stat: relpath=bin/ls
host_call: nbr=0x1, parm=0x80208fe0, size=24
exec_spawn: ERROR: Failed to load program 'ls': -2
nxposix_spawn_exec: ERROR: exec failed: 2
hostfs_stat: relpath=bin
host_call: nbr=0x1, parm=0x80209180, size=24
host_call: nbr=0xc, parm=0x80209180, size=8
host_call: nbr=0x2, parm=0x80209190, size=8
/system/bin
nsh>
nsh> ls /system/bin/init
posix_spawn: pid=0xc0202978 path=ls file_actions=0xc0202980 attr=0xc0202988 argv=0xc0202a28
hostfs_stat: relpath=bin/ls
host_call: nbr=0x1, parm=0x80208fe0, size=24
exec_spawn: ERROR: Failed to load program 'ls': -2
nxposix_spawn_exec: ERROR: exec failed: 2
hostfs_stat: relpath=bin/init
host_call: nbr=0x1, parm=0x80209180, size=24
host_call: nbr=0xc, parm=0x80209180, size=8
host_call: nbr=0x2, parm=0x80209190, size=8
/system/bin/init
```
2023-07-21 12:09:49 +08:00
Semihosting won't work on Star64 SBC. Let's replace this with Initial RAM Disk and ROMFS...
2023-07-19 10:46:50 +08:00
2023-07-21 12:09:49 +08:00
(See https://github.com/apache/nuttx/issues/9501)
2023-07-19 10:46:50 +08:00
2023-07-20 16:43:28 +08:00
# Initial RAM Disk for LiteX Arty-A7
2023-07-20 09:45:47 +08:00
2023-07-27 18:32:43 +08:00
Read the article...
- ["Star64 JH7110 + NuttX RTOS: RISC-V Semihosting and Initial RAM Disk" ](https://lupyuen.github.io/articles/semihost )
2023-07-21 10:07:47 +08:00
Let's modify NuttX for QEMU to mount the Apps Filesystem from an Initial RAM Disk (instead of Semihosting).
2023-07-21 12:09:49 +08:00
(So later we can replicate this on Star64 JH7110 SBC)
2023-07-20 09:45:47 +08:00
2023-07-20 16:43:28 +08:00
First we look at the Initial RAM Disk for LiteX Arty-A7...
2023-07-20 09:45:47 +08:00
2023-07-20 16:43:28 +08:00
[(About NuttX RAM Disks and ROM Disks) ](https://cwiki.apache.org/confluence/plugins/servlet/mobile?contentId=139629548#content/view/139629548 )
2023-07-20 09:45:47 +08:00
2023-07-21 12:09:49 +08:00
To generate the RAM Disk, we run this command: [VexRISCV_SMP Core ](https://nuttx.apache.org/docs/latest/platforms/risc-v/litex/cores/vexriscv_smp/index.html )
2023-07-20 09:45:47 +08:00
```bash
2023-07-20 16:43:28 +08:00
cd nuttx
genromfs -f romfs.img -d ../apps/bin -V "NuttXBootVol"
2023-07-20 09:45:47 +08:00
```
2023-07-20 09:52:08 +08:00
[(About `genromfs`) ](https://www.systutorials.com/docs/linux/man/8-genromfs/ )
2023-07-21 12:09:49 +08:00
LiteX Memory Map says where the RAM Disk is loaded...
2023-07-20 09:45:47 +08:00
```text
"romfs.img": "0x40C00000",
"nuttx.bin": "0x40000000",
"opensbi.bin": "0x40f00000"
```
2023-07-21 12:09:49 +08:00
This is the LiteX Build Configuration for mounting the RAM Disk: [knsh/defconfig ](https://github.com/lupyuen2/wip-pinephone-nuttx/blob/star64/boards/risc-v/litex/arty_a7/configs/knsh/defconfig#L34 )
2023-07-20 09:45:47 +08:00
```bash
CONFIG_BOARDCTL_ROMDISK=y
CONFIG_BOARD_LATE_INITIALIZE=y
CONFIG_BUILD_KERNEL=y
CONFIG_FS_ROMFS=y
CONFIG_INIT_FILEPATH="/system/bin/init"
CONFIG_INIT_MOUNT=y
CONFIG_INIT_MOUNT_FLAGS=0x1
CONFIG_INIT_MOUNT_TARGET="/system/bin"
CONFIG_LITEX_APPLICATION_RAMDISK=y
CONFIG_NSH_FILE_APPS=y
CONFIG_NSH_READLINE=y
CONFIG_PATH_INITIAL="/system/bin"
CONFIG_RAM_SIZE=4194304
CONFIG_RAM_START=0x40400000
CONFIG_RAW_BINARY=y
CONFIG_SYSTEM_NSH_PROGNAME="init"
CONFIG_TESTING_GETPRIME=y
```
2023-07-21 12:09:49 +08:00
According to [NSH Start-Up Script ](https://nuttx.apache.org/docs/latest/applications/nsh/nsh.html#nsh-start-up-script ):
2023-07-20 09:45:47 +08:00
```text
CONFIG_DISABLE_MOUNTPOINT not set
CONFIG_FS_ROMFS enabled
```
2023-07-21 12:09:49 +08:00
The RAM Disk is mounted at LiteX Startup: [litex_appinit.c ](https://github.com/apache/nuttx/blob/master/boards/risc-v/litex/arty_a7/src/litex_appinit.c#L76-L103 )
2023-07-20 09:45:47 +08:00
```c
void board_late_initialize(void)
{
#ifdef CONFIG_LITEX_APPLICATION_RAMDISK
litex_mount_ramdisk();
#endif
litex_bringup();
}
```
`litex_bringup` mounts the RAM Disk at startup: [litex_ramdisk.c ](https://github.com/apache/nuttx/blob/master/boards/risc-v/litex/arty_a7/src/litex_ramdisk.c#L41-L98 )
```c
#ifndef CONFIG_BUILD_KERNEL
#error "Ramdisk usage is intended to be used with kernel build only"
#endif
#define SECTORSIZE 512
#define NSECTORS(b) (((b) + SECTORSIZE - 1) / SECTORSIZE)
#define RAMDISK_DEVICE_MINOR 0
// Mount a ramdisk defined in the ld-kernel.script to /dev/ramX.
// The ramdisk is intended to contain a romfs with applications which can
// be spawned at runtime.
int litex_mount_ramdisk(void)
{
int ret;
struct boardioc_romdisk_s desc;
desc.minor = RAMDISK_DEVICE_MINOR;
desc.nsectors = NSECTORS((ssize_t)__ramdisk_size);
desc.sectsize = SECTORSIZE;
desc.image = __ramdisk_start;
ret = boardctl(BOARDIOC_ROMDISK, (uintptr_t)&desc);
if (ret < 0 )
{
syslog(LOG_ERR, "Ramdisk register failed: %s\n", strerror(errno));
syslog(LOG_ERR, "Ramdisk mountpoint /dev/ram%d\n",
RAMDISK_DEVICE_MINOR);
syslog(LOG_ERR, "Ramdisk length %u, origin %x\n",
(ssize_t)__ramdisk_size,
(uintptr_t)__ramdisk_start);
}
return ret;
}
```
2023-07-20 18:15:59 +08:00
`__ramdisk_start` is defined in [board_memorymap.h ](https://github.com/lupyuen2/wip-pinephone-nuttx/blob/star64b/boards/risc-v/litex/arty_a7/include/board_memorymap.h#L58-L91 ):
```c
/* RAMDisk */
#define RAMDISK_START (uintptr_t)__ramdisk_start
#define RAMDISK_SIZE (uintptr_t)__ramdisk_size
/* ramdisk (RW) */
extern uint8_t __ramdisk_start[];
extern uint8_t __ramdisk_size[];
```
And [ld-kernel.script ](https://github.com/lupyuen2/wip-pinephone-nuttx/blob/star64b/boards/risc-v/litex/arty_a7/scripts/ld-kernel.script#L20-L49 ):
```text
MEMORY
{
kflash (rx) : ORIGIN = 0x40000000, LENGTH = 4096K /* w/ cache */
ksram (rwx) : ORIGIN = 0x40400000, LENGTH = 4096K /* w/ cache */
pgram (rwx) : ORIGIN = 0x40800000, LENGTH = 4096K /* w/ cache */
ramdisk (rwx) : ORIGIN = 0x40C00000, LENGTH = 4096K /* w/ cache */
}
...
2023-07-20 20:32:47 +08:00
/* Page heap */
__pgheap_start = ORIGIN(pgram);
__pgheap_size = LENGTH(pgram) + LENGTH(ramdisk);
2023-07-20 18:15:59 +08:00
/* Application ramdisk */
__ramdisk_start = ORIGIN(ramdisk);
__ramdisk_size = LENGTH(ramdisk);
__ramdisk_end = ORIGIN(ramdisk) + LENGTH(ramdisk);
```
2023-07-20 20:32:47 +08:00
Note that `__pgheap_size` needs to include `ramdisk` .
2023-07-20 18:15:59 +08:00
Let's do the same to NuttX for QEMU...
2023-07-21 10:12:56 +08:00
# Modify NuttX QEMU to Load Initial RAM Disk
2023-07-20 16:43:28 +08:00
2023-07-27 18:32:43 +08:00
Read the article...
- ["Star64 JH7110 + NuttX RTOS: RISC-V Semihosting and Initial RAM Disk" ](https://lupyuen.github.io/articles/semihost )
2023-07-21 10:12:56 +08:00
Now we can modify NuttX for QEMU to mount the Apps Filesystem from an Initial RAM Disk instead of Semihosting.
2023-07-20 16:43:28 +08:00
2023-07-21 12:09:49 +08:00
(So later we can replicate this on Star64 JH7110 SBC)
2023-07-21 10:12:56 +08:00
2023-07-25 11:10:28 +08:00
![NuttX for QEMU will mount the Apps Filesystem from an Initial RAM Disk ](https://lupyuen.github.io/images/semihost-qemu2.jpg )
2023-07-21 12:09:49 +08:00
We follow the steps from LiteX Arty-A7 (from the previous section)...
2023-07-21 10:12:56 +08:00
We build NuttX QEMU in Kernel Mode: [Build Steps ](https://github.com/lupyuen2/wip-pinephone-nuttx/tree/master/boards/risc-v/qemu-rv/rv-virt )
2023-07-20 16:43:28 +08:00
2023-07-20 18:15:59 +08:00
```bash
## Build NuttX QEMU Kernel Mode
./tools/configure.sh rv-virt:knsh64
make V=1 -j7
## Build Apps Filesystem
make export V=1
pushd ../apps
./tools/mkimport.sh -z -x ../nuttx/nuttx-export-*.tar.gz
make import V=1
popd
```
2023-07-21 12:09:49 +08:00
We generate the Initial RAM Disk `initrd` ...
2023-07-20 18:23:16 +08:00
```bash
cd nuttx
genromfs -f initrd -d ../apps/bin -V "NuttXBootVol"
```
[(About `genromfs`) ](https://www.systutorials.com/docs/linux/man/8-genromfs/ )
2023-07-21 16:23:36 +08:00
Initial RAM Disk `initrd` is 7.9 MB...
```text
→ ls -l initrd
2023-07-22 08:45:49 +08:00
-rw-r--r-- 1 7902208 Jul 21 13:41 initrd
2023-07-21 16:23:36 +08:00
```
2023-07-21 12:09:49 +08:00
This is how we load the Initial RAM Disk on QEMU: [‘ virt’ Generic Virtual Platform (virt) ](https://www.qemu.org/docs/master/system/riscv/virt.html#running-linux-kernel )
2023-07-20 11:06:07 +08:00
```bash
qemu-system-riscv64 \
-semihosting \
-M virt,aclint=on \
-cpu rv64 \
-smp 8 \
-bios none \
-kernel nuttx \
-initrd initrd \
2023-07-20 16:43:28 +08:00
-nographic
2023-07-20 11:06:07 +08:00
```
2023-07-21 10:12:56 +08:00
_What is the RAM Address of the Initial RAM Disk in QEMU?_
Initial RAM Disk is loaded by QEMU at `0x8400` `0000` ...
- ["RAM Disk Address for RISC-V QEMU" ](https://github.com/lupyuen/nuttx-star64#ram-disk-address-for-risc-v-qemu )
Below are the files that we changed in NuttX for QEMU to load the Initial RAM Disk (instead of Semihosting)...
- [Modified Files for QEMU with Initial RAM Disk ](https://github.com/lupyuen2/wip-pinephone-nuttx/pull/33/files )
2023-07-20 20:32:47 +08:00
2023-07-21 09:58:58 +08:00
We configured QEMU to mount the RAM Disk as ROMFS (instead of Semihosting): [knsh64/defconfig ](https://github.com/lupyuen2/wip-pinephone-nuttx/blob/ramdisk/boards/risc-v/qemu-rv/rv-virt/configs/knsh64/defconfig )
2023-07-20 20:32:47 +08:00
2023-07-21 09:58:58 +08:00
```bash
CONFIG_BOARDCTL_ROMDISK=y
CONFIG_BOARD_LATE_INITIALIZE=y
CONFIG_FS_ROMFS=y
CONFIG_INIT_FILEPATH="/system/bin/init"
CONFIG_INIT_MOUNT=y
CONFIG_INIT_MOUNT_FLAGS=0x1
CONFIG_INIT_MOUNT_TARGET="/system/bin"
## We removed these...
## CONFIG_FS_HOSTFS=y
## CONFIG_RISCV_SEMIHOSTING_HOSTFS=y
```
We defined the RAM Disk Memory in the Linker Script: [ld-kernel64.script ](https://github.com/lupyuen2/wip-pinephone-nuttx/blob/ramdisk/boards/risc-v/qemu-rv/rv-virt/scripts/ld-kernel64.script#L20-L54 )
```text
MEMORY
{
...
/* Added RAM Disk */
2023-07-21 16:30:00 +08:00
ramdisk (rwx) : ORIGIN = 0x80800000, LENGTH = 16M /* w/ cache */
2023-07-21 09:58:58 +08:00
/* This won't work, crashes with a Memory Mgmt Fault...
2023-07-21 16:30:00 +08:00
ramdisk (rwx) : ORIGIN = 0x84000000, LENGTH = 16M */ /* w/ cache */
2023-07-21 09:58:58 +08:00
}
/* Added RAM Disk */
/* Page heap */
__pgheap_start = ORIGIN(pgram);
__pgheap_size = LENGTH(pgram) + LENGTH(ramdisk);
/* Previously: __pgheap_size = LENGTH(pgram); */
/* Added RAM Disk */
/* Application ramdisk */
__ramdisk_start = ORIGIN(ramdisk);
__ramdisk_size = LENGTH(ramdisk);
__ramdisk_end = ORIGIN(ramdisk) + LENGTH(ramdisk);
```
2023-07-21 16:30:00 +08:00
(We increased RAM Disk Memory from 4 MB to 16 MB because our RAM Disk is now bigger)
2023-07-21 09:58:58 +08:00
At Startup, we mount the RAM Disk: [qemu_rv_appinit.c ](https://github.com/lupyuen2/wip-pinephone-nuttx/blob/ramdisk/boards/risc-v/qemu-rv/rv-virt/src/qemu_rv_appinit.c#L83C1-L179C2 )
```c
// Called at NuttX Startup
void board_late_initialize(void) {
// Mount the RAM Disk
mount_ramdisk();
/* Perform board-specific initialization */
#ifdef CONFIG_NSH_ARCHINIT
mount(NULL, "/proc", "procfs", 0, NULL);
#endif
}
// Mount the RAM Disk
int mount_ramdisk(void) {
int ret;
struct boardioc_romdisk_s desc;
desc.minor = RAMDISK_DEVICE_MINOR;
desc.nsectors = NSECTORS((ssize_t)__ramdisk_size);
desc.sectsize = SECTORSIZE;
desc.image = __ramdisk_start;
ret = boardctl(BOARDIOC_ROMDISK, (uintptr_t)&desc);
if (ret < 0 )
{
syslog(LOG_ERR, "Ramdisk register failed: %s\n", strerror(errno));
syslog(LOG_ERR, "Ramdisk mountpoint /dev/ram%d\n",
RAMDISK_DEVICE_MINOR);
syslog(LOG_ERR, "Ramdisk length %lu, origin %lx\n",
(ssize_t)__ramdisk_size,
(uintptr_t)__ramdisk_start);
}
return ret;
}
```
We copied the RAM Disk from the QEMU Address (0x84000000) to the NuttX Address (__ramdisk_start): [qemu_rv_mm_init.c ](https://github.com/lupyuen2/wip-pinephone-nuttx/blob/ramdisk/arch/risc-v/src/qemu-rv/qemu_rv_mm_init.c#L271-L280 )
```c
void qemu_rv_kernel_mappings(void) {
...
// Copy 0x84000000 to __ramdisk_start (__ramdisk_size bytes)
// TODO: RAM Disk must not exceed __ramdisk_size bytes
memcpy((void *)__ramdisk_start, (void * )0x84000000, (size_t)__ramdisk_size);
```
2023-07-20 20:32:47 +08:00
2023-07-21 12:09:49 +08:00
[(Because somehow `map_region` crashes when we try to map 0x84000000) ](https://github.com/lupyuen2/wip-pinephone-nuttx/blob/ramdisk/arch/risc-v/src/qemu-rv/qemu_rv_mm_init.c#L280-L287 )
2023-07-21 16:41:48 +08:00
We check that the RAM Disk Memory is sufficient: [fs_romfsutil.c ](https://github.com/lupyuen2/wip-pinephone-nuttx/blob/ramdisk/fs/romfs/fs_romfsutil.c#L85-L105 )
```c
static uint32_t romfs_devread32(struct romfs_mountpt_s *rm, int ndx) {
//// Stop if RAM Disk Memory is too small
DEBUGASSERT(& rm->rm_buffer[ndx] < __ramdisk_start + ( size_t ) __ramdisk_size ) ; / / / /
```
2023-07-21 12:09:49 +08:00
Before making the above changes, here's the log for QEMU Kernel Mode with Semihosting...
2023-07-20 18:23:16 +08:00
```text
+ genromfs -f initrd -d ../apps/bin -V NuttXBootVol
+ riscv64-unknown-elf-size nuttx
text data bss dec hex filename
171581 673 21872 194126 2f64e nuttx
+ riscv64-unknown-elf-objcopy -O binary nuttx nuttx.bin
+ cp .config nuttx.config
+ riscv64-unknown-elf-objdump -t -S --demangle --line-numbers --wide nuttx
+ sleep 10
2023-07-20 18:24:04 +08:00
+ qemu-system-riscv64 -semihosting -M virt,aclint=on -cpu rv64 -smp 8 -bios none -kernel nuttx -initrd initrd -nographic
2023-07-20 18:23:16 +08:00
ABCnx_start: Entry
uart_register: Registering /dev/console
uart_register: Registering /dev/ttyS0
work_start_lowpri: Starting low-priority kernel worker thread(s)
nx_start_application: Starting init task: /system/bin/init
hostfs_stat: relpath=bin/init
hostfs_open: relpath=bin/init, oflags=0x1, mode=0x1b6
elf_symname: Symbol has no name
elf_symvalue: SHN_UNDEF: Failed to get symbol name: -3
elf_relocateadd: Section 2 reloc 2: Undefined symbol[0] has no name: -3
NuttShell (NSH) NuttX-12.0.3
nsh> nx_start: CPU0: Beginning Idle Loop
```
2023-07-21 12:09:49 +08:00
Now we run QEMU Kernel Mode with Initial RAM Disk, without Semihosting...
2023-07-21 09:46:13 +08:00
2023-07-22 17:50:18 +08:00
And it boots OK on QEMU yay!
2023-07-21 16:41:48 +08:00
2023-07-21 16:53:11 +08:00
[See the Run Log ](https://gist.github.com/lupyuen/8afee5b07b61bb7f9f202f7f8c5e3ab3 )
2023-07-21 16:41:48 +08:00
2023-07-22 17:50:18 +08:00
# Modify NuttX Star64 to Load Initial RAM Disk
2023-07-27 18:32:43 +08:00
Read the article...
- ["Star64 JH7110 + NuttX RTOS: RISC-V Semihosting and Initial RAM Disk" ](https://lupyuen.github.io/articles/semihost )
2023-07-25 15:24:11 +08:00
Finally we can modify NuttX for Star64 JH7110 RISC-V SBC to mount the Apps Filesystem from an Initial RAM Disk. (Instead of Semihosting)
2023-07-22 20:09:05 +08:00
2023-07-25 09:55:23 +08:00
![NuttX for Star64 JH7110 RISC-V SBC will mount the Apps Filesystem from an Initial RAM Disk ](https://lupyuen.github.io/images/semihost-star64.jpg )
2023-07-22 20:58:38 +08:00
We follow the steps from QEMU Kernel Mode's Initial RAM Disk. (See previous section)
We build NuttX Star64 in Kernel Mode: [Build Steps ](https://github.com/lupyuen2/wip-pinephone-nuttx/tree/master/boards/risc-v/qemu-rv/rv-virt )
```bash
## Build NuttX Star64 in Kernel Mode
tools/configure.sh rv-virt:knsh64
make V=1 -j7
## Build Apps Filesystem
make export V=1
pushd ../apps
./tools/mkimport.sh -z -x ../nuttx/nuttx-export-*.tar.gz
make import V=1
popd
```
We generate the Initial RAM Disk `initrd` and copy to TFTP Folder (for Network Booting)...
```bash
## Generate Initial RAM Disk
cd nuttx
genromfs -f initrd -d ../apps/bin -V "NuttXBootVol"
## Copy NuttX Binary Image, Device Tree and Initial RAM Disk to TFTP Folder
cp nuttx.bin $HOME/tftproot/Image
cp ../jh7110-star64-pine64.dtb $HOME/tftproot
cp initrd $HOME/tftproot
```
[(About `genromfs`) ](https://www.systutorials.com/docs/linux/man/8-genromfs/ )
Initial RAM Disk `initrd` is 7.9 MB...
```text
→ ls -l initrd
-rw-r--r-- 1 7930880 Jul 21 13:41 initrd
```
Below are the files that we changed in NuttX for Star64 to load the Initial RAM Disk (instead of Semihosting)...
- [Modified Files for Initial RAM Disk on Star64 ](https://github.com/lupyuen2/wip-pinephone-nuttx/pull/34/files )
These are the same changes that we made earlier for QEMU Kernel Mode's Initial RAM Disk.
2023-07-27 08:41:52 +08:00
(For a detailed explanation of the modified files, see the previous section)
2023-07-22 20:58:38 +08:00
Note that we copy the Initial RAM Disk from `0x4610` `0000` (instead of QEMU's `0x8400` `0000` ): [qemu_rv_mm_init.c ](https://github.com/lupyuen2/wip-pinephone-nuttx/blob/star64c/arch/risc-v/src/qemu-rv/qemu_rv_mm_init.c#L271-L280 )
```c
// Copy 0x46100000 to __ramdisk_start (__ramdisk_size bytes)
// TODO: RAM Disk must not exceed __ramdisk_size bytes
memcpy((void *)__ramdisk_start, (void * )0x46100000, (size_t)__ramdisk_size);
```
(Why `0x4610` `0000` ? See `ramdisk_addr_r` below)
This is how we updated the NuttX Build Configuration in `make menuconfig` ...
2023-07-22 19:51:34 +08:00
- Board Selection > Enable boardctl() interface > Enable application space creation of ROM disks
- RTOS Features > RTOS hooks > Custom board late initialization
- File Systems > ROMFS file system
- RTOS Features > Tasks and Scheduling > Auto-mount init file system
Set to `/system/bin`
- Build Setup > Debug Options > File System Debug Features > File System Error, Warnings and Info Output
- Disable: File Systems > Host File System
2023-07-22 20:58:38 +08:00
- Manually delete from [`knsh64/defconfig` ](https://github.com/lupyuen2/wip-pinephone-nuttx/blob/star64c/boards/risc-v/qemu-rv/rv-virt/configs/knsh64/defconfig )...
2023-07-22 19:51:34 +08:00
```text
CONFIG_HOST_MACOS=y
CONFIG_INIT_MOUNT_DATA="fs=../apps"
CONFIG_INIT_MOUNT_FSTYPE="hostfs"
CONFIG_INIT_MOUNT_SOURCE=""
```
2023-07-22 20:58:38 +08:00
Updated Build Configuration: [knsh64/defconfig ](https://github.com/lupyuen2/wip-pinephone-nuttx/blob/star64c/boards/risc-v/qemu-rv/rv-virt/configs/knsh64/defconfig )
2023-07-22 17:50:18 +08:00
_What is the RAM Address of the Initial RAM Disk in Star64?_
Initial RAM Disk is loaded by Star64's U-Boot Bootloader at `0x4610` `0000` ...
```bash
ramdisk_addr_r=0x46100000
```
[(Source) ](https://lupyuen.github.io/articles/linux#u-boot-settings-for-star64 )
2023-07-22 20:58:38 +08:00
Which means that we need to add these TFTP Commands to U-Boot Bootloader...
2023-07-22 18:11:28 +08:00
```bash
2023-07-22 20:09:05 +08:00
## Assume Initial RAM Disk is max 16 MB
setenv ramdisk_size 0x1000000
## Check that it's correct
printenv ramdisk_size
## Save it for future reboots
saveenv
2023-07-22 18:11:28 +08:00
## Load Kernel and Device Tree over TFTP
tftpboot ${kernel_addr_r} ${tftp_server}:Image
tftpboot ${fdt_addr_r} ${tftp_server}:jh7110-star64-pine64.dtb
fdt addr ${fdt_addr_r}
## Added this: Load Initial RAM Disk over TFTP
tftpboot ${ramdisk_addr_r} ${tftp_server}:initrd
2023-07-22 20:09:05 +08:00
## Changed this: Replaced `-` by `ramdisk_addr_r:ramdisk_size`
booti ${kernel_addr_r} ${ramdisk_addr_r}:${ramdisk_size} ${fdt_addr_r}
2023-07-22 18:11:28 +08:00
```
2023-07-22 18:31:01 +08:00
Which will change our U-Boot Boot Script to...
```bash
## Load the NuttX Image from TFTP Server
## kernel_addr_r=0x40200000
## tftp_server=192.168.x.x
if tftpboot ${kernel_addr_r} ${tftp_server}:Image;
then
## Load the Device Tree from TFTP Server
## fdt_addr_r=0x46000000
if tftpboot ${fdt_addr_r} ${tftp_server}:jh7110-star64-pine64.dtb;
then
## Set the RAM Address of Device Tree
## fdt_addr_r=0x46000000
if fdt addr ${fdt_addr_r};
then
## Load the Intial RAM Disk from TFTP Server
## ramdisk_addr_r=0x46100000
if tftpboot ${ramdisk_addr_r} ${tftp_server}:initrd;
then
## Boot the NuttX Image with the Initial RAM Disk and Device Tree
## kernel_addr_r=0x40200000
## ramdisk_addr_r=0x46100000
2023-07-22 20:09:05 +08:00
## ramdisk_size=0x1000000
2023-07-22 18:31:01 +08:00
## fdt_addr_r=0x46000000
2023-07-22 20:09:05 +08:00
booti ${kernel_addr_r} ${ramdisk_addr_r}:${ramdisk_size} ${fdt_addr_r};
2023-07-22 18:31:01 +08:00
fi;
fi;
fi;
fi
```
Which becomes...
```bash
2023-07-22 20:09:05 +08:00
## Assume Initial RAM Disk is max 16 MB
setenv ramdisk_size 0x1000000
## Check that it's correct
printenv ramdisk_size
## Save it for future reboots
saveenv
2023-07-22 18:31:01 +08:00
## Add the Boot Command for TFTP
2023-07-22 20:09:05 +08:00
setenv bootcmd_tftp 'if tftpboot ${kernel_addr_r} ${tftp_server}:Image ; then if tftpboot ${fdt_addr_r} ${tftp_server}:jh7110-star64-pine64.dtb ; then if fdt addr ${fdt_addr_r} ; then if tftpboot ${ramdisk_addr_r} ${tftp_server}:initrd ; then booti ${kernel_addr_r} ${ramdisk_addr_r}:${ramdisk_size} ${fdt_addr_r} ; fi ; fi ; fi ; fi'
2023-07-22 18:31:01 +08:00
## Check that it's correct
printenv bootcmd_tftp
## Save it for future reboots
saveenv
```
2023-07-22 20:09:05 +08:00
_What happens if we omit the RAM Disk Size?_
2023-07-22 19:35:03 +08:00
2023-07-22 19:51:34 +08:00
```text
$ booti ${kernel_addr_r} ${ramdisk_addr_r} ${fdt_addr_r}
Wrong Ramdisk Image Format
Ramdisk image is corrupt or invalid
2023-07-22 19:35:03 +08:00
2023-07-22 19:51:34 +08:00
## Assume max 16 MB
$ booti ${kernel_addr_r} ${ramdisk_addr_r}:0x1000000 ${fdt_addr_r}
2023-07-22 20:09:05 +08:00
## Boots OK
2023-07-22 19:51:34 +08:00
```
2023-07-22 19:35:03 +08:00
2023-07-22 20:58:38 +08:00
_Does the Initial RAM Disk work on Star64?_
2023-07-22 20:09:05 +08:00
Star64 JH7110 boots OK with the Initial RAM Disk yay!
2023-07-22 20:01:03 +08:00
```text
StarFive # booti ${kernel_addr_r} ${ramdisk_addr_r}:0x1000000 ${fdt_addr_r}
## Flattened Device Tree blob at 46000000
Booting using the fdt blob at 0x46000000
Using Device Tree in place at 0000000046000000, end 000000004600f43a
Starting kernel ...
clk u5_dw_i2c_clk_core already disabled
clk u5_dw_i2c_clk_apb already disabled
123067DFHBCInx_start: Entry
uart_register: Registering /dev/console
uart_register: Registering /dev/ttyS0
work_start_lowpri: Starting low-priority kernel worker thread(s)
board_late_initialize:
nx_start_application: Starting init task: /system/bin/init
elf_symname: Symbol has no name
elf_symvalue: SHN_UNDEF: Failed to get symbol name: -3
elf_relocateadd: Section 2 reloc 2: Undefined symbol[0] has no name: -3
nx_start_application: ret=3
up_exit: TCB=0x404088d0 exiting
nx_start: CPU0: Beginning Idle Loop
```
2023-07-22 20:09:05 +08:00
TODO: Why no shell?
2023-07-22 20:58:38 +08:00
TODO: Why `nx_start_application: ret=3` ?
TODO: Check User Address Space
TODO: Boot from MicroSD with Initial RAM Disk
2023-07-20 11:09:37 +08:00
# RAM Disk Address for RISC-V QEMU
2023-07-20 11:04:43 +08:00
2023-07-27 18:32:43 +08:00
Read the article...
- ["Star64 JH7110 + NuttX RTOS: RISC-V Semihosting and Initial RAM Disk" ](https://lupyuen.github.io/articles/semihost )
2023-07-20 11:04:43 +08:00
_Can we enable logging for RISC-V QEMU?_
Yep we use the `-trace "*"` option like this...
2023-07-20 09:52:08 +08:00
```bash
2023-07-20 10:29:46 +08:00
qemu-system-riscv64 \
-semihosting \
-M virt,aclint=on \
-cpu rv64 \
-smp 8 \
-bios none \
-kernel nuttx \
-initrd initrd \
-nographic \
-trace "*"
2023-07-20 09:52:08 +08:00
```
2023-07-20 11:04:43 +08:00
In the QEMU Command above we loaded the Initial RAM Disk `initrd` .
To discover the RAM Address of the Initial RAM Disk, we check the QEMU Trace Log:
2023-07-20 10:29:46 +08:00
```text
resettablloader_write_rom nuttx
ELF program header segment 0:
@0x80000000 size=0x2b374 ROM=0
loader_write_rom nuttx
ELF program header segment 1:
@0x80200000 size=0x2a1 ROM=0
loader_write_rom initrd:
@0x84000000 size=0x2fc3e8 ROM=0
loader_write_rom fdt:
@0x87000000 size=0x100000 ROM=0
```
So Initial RAM Disk is loaded at `0x8400` `0000`
2023-07-20 11:33:38 +08:00
(`__ramdisk_start` from the previous section)
2023-07-20 11:04:43 +08:00
Also we see that Kernel is loaded at `0x8000` `0000` , Device Tree at `0x8700` `0000` .
2023-07-20 10:29:46 +08:00
# Device Tree for RISC-V QEMU
2023-07-20 09:45:47 +08:00
2023-07-27 18:32:43 +08:00
Read the article...
- ["Star64 JH7110 + NuttX RTOS: RISC-V Semihosting and Initial RAM Disk" ](https://lupyuen.github.io/articles/semihost )
2023-07-20 12:28:29 +08:00
To dump the Device Tree for QEMU RISC-V, we specify `dumpdtb` ...
2023-07-20 10:02:59 +08:00
```bash
## Dump Device Tree for QEMU RISC-V
qemu-system-riscv64 \
-semihosting \
-M virt,aclint=on,dumpdtb=qemu-riscv64.dtb \
-cpu rv64 \
-smp 8 \
-bios none \
-kernel nuttx \
-nographic
## Convert Device Tree to text format
dtc \
-o qemu-riscv64.dts \
-O dts \
-I dtb \
qemu-riscv64.dtb
```
2023-07-20 12:28:29 +08:00
This produces the Device Tree for QEMU RISC-V...
- [qemu-riscv64.dts: Device Tree for QEMU RISC-V ](https://github.com/lupyuen/nuttx-star64/blob/main/qemu-riscv64.dts )
Which is helpful for browsing the Memory Addresses of I/O Peripherals.
2023-07-20 10:02:59 +08:00
2023-07-15 15:21:31 +08:00
# TODO
2023-07-15 12:40:55 +08:00
2023-07-27 18:50:11 +08:00
```text
Starting kernel ...
clk u5_dw_i2c_clk_core already disabled
clk u5_dw_i2c_clk_apb already disabled
123067DFHBCqemu_rv_kernel_mappings: map I/O regions
qemu_rv_kernel_mappings: map kernel text
qemu_rv_kernel_mappings: map kernel data
qemu_rv_kernel_mappings: connect the L1 and L2 page tables
qemu_rv_kernel_mappings: map the page pool
qemu_rv_mm_init: mmu_enable: satp=1077956608
Inx_start: Entry
elf_initialize: Registering ELF
uart_register: Registering /dev/console
uart_register: Registering /dev/ttyS0
work_start_lowpri: Starting low-priority kernel worker thread(s)
board_late_initialize:
nx_start_application: Starting init task: /system/bin/init
load_absmodule: Loading /system/bin/init
elf_loadbinary: Loading file: /system/bin/init
elf_init: filename: /system/bin/init loadinfo: 0x4040c638
elf_read: Read 64 bytes from offset 0
elf_dumploadinfo: LOAD_INFO:
elf_dumploadinfo: textalloc: 00000000
elf_dumploadinfo: dataalloc: 00000000
elf_dumploadinfo: textsize: 0
elf_dumploadinfo: datasize: 0
elf_dumploadinfo: textalign: 0
elf_dumploadinfo: dataalign: 0
elf_dumploadinfo: filelen: 3289528
elf_dumploadinfo: symtabidx: 0
elf_dumploadinfo: strtabidx: 0
elf_dumploadinfo: ELF Header:
elf_dumploadinfo: e_ident: 7f 45 4c 46
elf_dumploadinfo: e_type: 0001
elf_dumploadinfo: e_machine: 00f3
elf_dumploadinfo: e_version: 00000001
elf_dumploadinfo: e_entry: 0000004a
elf_dumploadinfo: e_phoff: 0
elf_dumploadinfo: e_shoff: 3286264
elf_dumploadinfo: e_flags: 00000001
elf_dumploadinfo: e_ehsize: 64
elf_dumploadinfo: e_phentsize: 0
elf_dumploadinfo: e_phnum: 0
elf_dumploadinfo: e_shentsize: 64
elf_dumploadinfo: e_shnum: 51
elf_dumploadinfo: e_shstrndx: 50
elf_load: loadinfo: 0x4040c638
elf_read: Read 3264 bytes from offset 3286264
elf_loadfile: Loaded sections:
elf_read: Read 39900 bytes from offset 64
elf_loadfile: 1. 00000000->c0000000
elf_read: Read 45804 bytes from offset 39968
elf_loadfile: 3. 00009be0->c0009be0
elf_read: Read 9 bytes from offset 85776
elf_loadfile: 5. 00000000->c0014ed0
elf_read: Read 3 bytes from offset 85792
elf_loadfile: 6. 00000000->c0014ee0
elf_read: Read 3 bytes from offset 85800
elf_loadfile: 7. 00000000->c0014ee8
elf_read: Read 2 bytes from offset 85808
elf_loadfile: 8. 00000000->c0014ef0
elf_read: Read 2 bytes from offset 85816
...
elf_read: Read 24 bytes from offset 1313160
elf_symvalue: Other: 00005c88+c0000000=c0005c88
elf_read: Read 24 bytes from offset 1313352
elf_symvalue: Other: 00005c82+c0000000=c0005c82
elf_read: Read 24 bytes from offset 1313376
elf_symvalue: Other: 00005c72+c0000000=c0005c72
elf_read: Read 24 bytes from offset 1313400
elf_symvalue: Other: 00005c60+c0000000=c0005c60
elf_read: Read 24 bytes from offset 1313424
elf_symvalue: Other: 00005c78+c0000000=c0005c78
elf_read: Read 24 bytes from offset 1313448
elf_symvalue: Other: 00005c66+c0000000=c0005c66
elf_read: Read 24 bytes from offset 1313472
elf_symvalue: Other: 00005c6c+c0000000=c0005c6c
elf_read: Read 24 bytes from offset 1313496
elf_symvalue: Other: 00005d70+c0000000=c0005d70
elf_read: Read 24 bytes from offset 1312896
elf_symvalue: Other: 000019d0+c0009be0=c000b5b0
elf_read: Read 24 bytes from offset 1313232
elf_symvalue: Other: 00005d5a+c0000000=c0005d5a
elf_read: Read 24 bytes from offset 1313520
elf_symvalue: Other: 00005cf8+c0000000=c0005cf8
elf_read: Read 24 bytes from offset 1313544
elf_symvalue: Other: 00005d76+c0000000=c0005d76
elf_read: Read 24 bytes from offset 1313568
elf_symvalue: Other: 00005d4c+c0000000=c0005d4c
elf_read: Read 24 bytes from offset 1313592
elf_symvalue: Other: 00005d52+c0000000=c0005d52
elf_read: Read 24 bytes from offset 1313616
elf_symvalue: Other: 00005d56+c0000000=c0005d56
elf_read: Read 24 bytes from offset 1961968
elf_read: Read 24 bytes from offset 1781688
elf_symvalue: Other: 00000000+c0101028=c0101028
up_relocateadd: RISCV_64 at c0101028 [00000000] to sym=0x4040a850 st_value=c0101028
load_absmodule: Successfully loaded module /system/bin/init
binfmt_dumpmodule: Module:
binfmt_dumpmodule: entrypt: 0xc000004a
binfmt_dumpmodule: mapped: 0 size=0
binfmt_dumpmodule: alloc: 0 0 0
binfmt_dumpmodule: addrenv: 0x40409f60
binfmt_dumpmodule: stacksize: 2048
binfmt_dumpmodule: unload: 0
exec_module: Executing /system/bin/init
binfmt_copyargv: args=0 argsize=0
binfmt_copyargv: args=2 argsize=23
exec_module: Initialize the user heap (heapsize=528384)
nx_start_application: ret=3
up_exit: TCB=0x404088d0 exiting
nx_start: CPU0: Beginning Idle Loop
```
2023-07-20 09:45:47 +08:00
TODO: Port [__up_mtimer_initialize__ ](https://github.com/lupyuen2/wip-pinephone-nuttx/blob/star64a/arch/risc-v/src/qemu-rv/qemu_rv_timerisr.c#L151-L210 ) to Star64
2023-07-16 13:30:34 +08:00
2023-07-21 09:46:13 +08:00
TODO: Check [board_memorymap.h ](https://github.com/lupyuen2/wip-pinephone-nuttx/blob/ramdisk/boards/risc-v/qemu-rv/rv-virt/include/board_memorymap.h#L34-L37 )
2023-07-11 14:41:53 +08:00
2023-07-21 09:46:13 +08:00
```c
/* DDR start address */
#define QEMURV_DDR_BASE (0x80000000)
#define QEMURV_DDR_SIZE (0x40000000)
```
2023-07-11 14:41:53 +08:00
2023-07-21 09:46:13 +08:00
TODO: Check [qemu_rv_mm_init.c ](https://github.com/lupyuen2/wip-pinephone-nuttx/blob/ramdisk/arch/risc-v/src/qemu-rv/qemu_rv_mm_init.c#L43-L46 )
2023-07-11 14:41:53 +08:00
2023-07-21 09:46:13 +08:00
```c
/* Map the whole I/O memory with vaddr = paddr mappings */
#define MMU_IO_BASE (0x00000000)
#define MMU_IO_SIZE (0x80000000)
```
2023-07-11 14:41:53 +08:00
TODO: RISC-V Exceptions [riscv_exception_common.S ](https://github.com/lupyuen2/wip-pinephone-nuttx/blob/star64/arch/risc-v/src/common/riscv_exception_common.S#L77 )
TODO: Set CLINT and PLIC Addresses
From [U74 Memory Map ](https://doc-en.rvspace.org/JH7110/TRM/JH7110_TRM/u74_memory_map.html ):
```text
0x00_0200_0000 0x00_0200_FFFF RW A CLINT
0x00_0C00_0000 0x00_0FFF_FFFF RW A PLIC
```
TODO: We update [qemu_rv_memorymap.h ](https://github.com/lupyuen2/wip-pinephone-nuttx/blob/star64/arch/risc-v/src/qemu-rv/hardware/qemu_rv_memorymap.h#L27-L33 ):
```c
#define QEMU_RV_CLINT_BASE 0x02000000
#define QEMU_RV_ACLINT_BASE 0x02f00000
#define QEMU_RV_PLIC_BASE 0x0c000000
```
2023-07-10 12:51:14 +08:00
# U-Boot Bootloader Log for TFTP
2023-07-10 09:26:44 +08:00
2023-07-10 09:57:33 +08:00
```text
2023-07-10 12:51:14 +08:00
U-Boot SPL 2021.10 (Jan 19 2023 - 04:09:41 +0800)
DDR version: dc2e84f0.
Trying to boot from SPI
U-Boot SPL 2021.10 (Jan 19 2023 - 04:09:41 +0800)
DDR version: dc2e84f0.
Trying to boot from SPI
2023-07-10 09:57:33 +08:00
2023-07-10 12:51:14 +08:00
OpenSBI v1.2
____ _____ ____ _____
/ __ \ / ____ | _ \_ _|
| | | |_ __ ___ _ __ | (___ | |_) || |
| | | | '_ \ / _ \ '_ \ \___ \| _ < | |
| |__| | |_) | __ / | | |____) | |_) || |_
\____/| .__/ \___|_| |_|_____/|____/_____|
| |
|_|
Platform Name : StarFive VisionFive V2
Platform Features : medeleg
Platform HART Count : 5
Platform IPI Device : aclint-mswi
Platform Timer Device : aclint-mtimer @ 4000000Hz
Platform Console Device : uart8250
Platform HSM Device : jh7110-hsm
Platform PMU Device : ---
Platform Reboot Device : pm-reset
Platform Shutdown Device : pm-reset
Firmware Base : 0x40000000
Firmware Size : 288 KB
Runtime SBI Version : 1.0
2023-07-10 09:26:44 +08:00
2023-07-10 12:51:14 +08:00
Domain0 Name : root
Domain0 Boot HART : 1
Domain0 HARTs : 0*,1*,2*,3*,4*
Domain0 Region00 : 0x0000000002000000-0x000000000200ffff (I)
Domain0 Region01 : 0x0000000040000000-0x000000004007ffff ()
Domain0 Region02 : 0x0000000000000000-0xffffffffffffffff (R,W,X)
Domain0 Next Address : 0x0000000040200000
Domain0 Next Arg1 : 0x0000000042200000
Domain0 Next Mode : S-mode
Domain0 SysReset : yes
Boot HART ID : 1
Boot HART Domain : root
Boot HART Priv Version : v1.11
Boot HART Base ISA : rv64imafdcbx
Boot HART ISA Extensions : none
Boot HART PMP Count : 8
Boot HART PMP Granularity : 4096
Boot HART PMP Address Bits: 34
Boot HART MHPM Count : 2
Boot HART MIDELEG : 0x0000000000000222
Boot HART MEDELEG : 0x000000000000b109
U-Boot 2021.10 (Jan 19 2023 - 04:09:41 +0800), Build: jenkins-github_visionfive2-6
CPU: rv64imacu
Model: StarFive VisionFive V2
DRAM: 8 GiB
MMC: sdio0@16010000: 0, sdio1@16020000: 1
Loading Environment from SPIFlash... SF: Detected gd25lq128 with page size 256 Bytes, erase size 4 KiB, total 16 MiB
*** Warning - bad CRC, using default environment
StarFive EEPROM format v2
--------EEPROM INFO--------
Vendor : PINE64
Product full SN: STAR64V1-2310-D008E000-00000003
data version: 0x2
PCB revision: 0xc1
BOM revision: A
Ethernet MAC0 address: 6c:cf:39:00:75:5d
Ethernet MAC1 address: 6c:cf:39:00:75:5e
--------EEPROM INFO--------
In: serial@10000000
Out: serial@10000000
Err: serial@10000000
Model: StarFive VisionFive V2
Net: eth0: ethernet@16030000, eth1: ethernet@16040000
Card did not respond to voltage select! : -110
Card did not respond to voltage select! : -110
bootmode flash device 0
Card did not respond to voltage select! : -110
Hit any key to stop autoboot: 2 1 0
Card did not respond to voltage select! : -110
Couldn't find partition mmc 0:3
Can't set block device
Importing environment from mmc0 ...
## Warning: Input data exceeds 1048576 bytes - truncated
## Info: input data size = 1048578 = 0x100002
Card did not respond to voltage select! : -110
Couldn't find partition mmc 1:2
Can't set block device
## Warning: defaulting to text format
## Error: "boot2" not defined
Card did not respond to voltage select! : -110
ethernet@16030000 Waiting for PHY auto negotiation to complete....... done
BOOTP broadcast 1
*** Unhandled DHCP Option in OFFER/ACK: 43
*** Unhandled DHCP Option in OFFER/ACK: 43
DHCP client bound to address 192.168.x.x (351 ms)
Using ethernet@16030000 device
TFTP from server 192.168.x.x; our IP address is 192.168.x.x
Filename 'boot.scr.uimg'.
Load address: 0x43900000
Loading: *
TFTP server died; starting again
BOOTP broadcast 1
*** Unhandled DHCP Option in OFFER/ACK: 43
*** Unhandled DHCP Option in OFFER/ACK: 43
DHCP client bound to address 192.168.x.x (576 ms)
Using ethernet@16030000 device
TFTP from server 192.168.x.x; our IP address is 192.168.x.x
Filename 'boot.scr.uimg'.
Load address: 0x40200000
Loading: *
TFTP server died; starting again
2023-07-11 17:38:41 +08:00
2023-07-11 14:35:07 +08:00
StarFive # setenv tftp_server 192.168.x.x
2023-07-11 17:38:41 +08:00
2023-07-11 14:35:07 +08:00
StarFive # tftpboot ${kernel_addr_r} ${tftp_server}:Image
2023-07-10 12:51:14 +08:00
Using ethernet@16030000 device
TFTP from server 192.168.x.x; our IP address is 192.168.x.x
Filename 'Image'.
Load address: 0x40200000
Loading: * #############################################################T ####
[8C #################################################################
[8C #############
[8C 221.7 KiB/s
done
Bytes transferred = 2097832 (2002a8 hex)
2023-07-11 17:38:41 +08:00
2023-07-11 14:35:07 +08:00
StarFive # tftpboot ${fdt_addr_r} ${tftp_server}:jh7110-star64-pine64.dtb
2023-07-10 12:51:14 +08:00
Using ethernet@16030000 device
TFTP from server 192.168.x.x; our IP address is 192.168.x.x
Filename 'jh7110-star64-pine64.dtb'.
Load address: 0x46000000
Loading: * ####
[8C 374 KiB/s
done
Bytes transferred = 50235 (c43b hex)
2023-07-11 17:38:41 +08:00
StarFive # fdt addr ${fdt_addr_r}
2023-07-10 12:51:14 +08:00
StarFive # booti ${kernel_addr_r} - ${fdt_addr_r}
## Flattened Device Tree blob at 46000000
Booting using the fdt blob at 0x46000000
Using Device Tree in place at 0000000046000000, end 000000004600f43a
Starting kernel ...
clk u5_dw_i2c_clk_core already disabled
clk u5_dw_i2c_clk_apb already disabled
123067DFAGHBC
```
2023-07-11 11:13:58 +08:00
2023-07-11 14:41:53 +08:00
# U-Boot Bootloader Log for Auto Network Boot
2023-07-11 11:13:58 +08:00
2023-07-11 14:41:53 +08:00
```text
U-Boot SPL 2021.10 (Jan 19 2023 - 04:09:41 +0800)
DDR version: dc2e84f0.
Trying to boot from SPI
2023-07-11 11:13:58 +08:00
2023-07-11 14:41:53 +08:00
OpenSBI v1.2
____ _____ ____ _____
/ __ \ / ____ | _ \_ _|
| | | |_ __ ___ _ __ | (___ | |_) || |
| | | | '_ \ / _ \ '_ \ \___ \| _ < | |
| |__| | |_) | __ / | | |____) | |_) || |_
\____/| .__/ \___|_| |_|_____/|____/_____|
| |
|_|
2023-07-11 11:13:58 +08:00
2023-07-11 14:41:53 +08:00
Platform Name : StarFive VisionFive V2
Platform Features : medeleg
Platform HART Count : 5
Platform IPI Device : aclint-mswi
Platform Timer Device : aclint-mtimer @ 4000000Hz
Platform Console Device : uart8250
Platform HSM Device : jh7110-hsm
Platform PMU Device : ---
Platform Reboot Device : pm-reset
Platform Shutdown Device : pm-reset
Firmware Base : 0x40000000
Firmware Size : 288 KB
Runtime SBI Version : 1.0
2023-07-11 11:13:58 +08:00
2023-07-11 14:41:53 +08:00
Domain0 Name : root
Domain0 Boot HART : 1
Domain0 HARTs : 0*,1*,2*,3*,4*
Domain0 Region00 : 0x0000000002000000-0x000000000200ffff (I)
Domain0 Region01 : 0x0000000040000000-0x000000004007ffff ()
Domain0 Region02 : 0x0000000000000000-0xffffffffffffffff (R,W,X)
Domain0 Next Address : 0x0000000040200000
Domain0 Next Arg1 : 0x0000000042200000
Domain0 Next Mode : S-mode
Domain0 SysReset : yes
2023-07-11 11:13:58 +08:00
2023-07-11 14:41:53 +08:00
Boot HART ID : 1
Boot HART Domain : root
Boot HART Priv Version : v1.11
Boot HART Base ISA : rv64imafdcbx
Boot HART ISA Extensions : none
Boot HART PMP Count : 8
Boot HART PMP Granularity : 4096
Boot HART PMP Address Bits: 34
Boot HART MHPM Count : 2
Boot HART MIDELEG : 0x0000000000000222
Boot HART MEDELEG : 0x000000000000b109
2023-07-11 11:13:58 +08:00
2023-07-11 14:41:53 +08:00
U-Boot 2021.10 (Jan 19 2023 - 04:09:41 +0800), Build: jenkins-github_visionfive2-6
2023-07-11 11:13:58 +08:00
2023-07-11 14:41:53 +08:00
CPU: rv64imacu
Model: StarFive VisionFive V2
DRAM: 8 GiB
MMC: sdio0@16010000: 0, sdio1@16020000: 1
Loading Environment from SPIFlash... SF: Detected gd25lq128 with page size 256 Bytes, erase size 4 KiB, total 16 MiB
OK
StarFive EEPROM format v2
2023-07-11 11:13:58 +08:00
2023-07-11 14:41:53 +08:00
--------EEPROM INFO--------
Vendor : PINE64
Product full SN: STAR64V1-2310-D008E000-00000003
data version: 0x2
PCB revision: 0xc1
BOM revision: A
Ethernet MAC0 address: 6c:cf:39:00:75:5d
Ethernet MAC1 address: 6c:cf:39:00:75:5e
--------EEPROM INFO--------
2023-07-11 11:13:58 +08:00
2023-07-11 14:41:53 +08:00
In: serial@10000000
Out: serial@10000000
Err: serial@10000000
Model: StarFive VisionFive V2
Net: eth0: ethernet@16030000, eth1: ethernet@16040000
Card did not respond to voltage select! : -110
Card did not respond to voltage select! : -110
bootmode flash device 0
Card did not respond to voltage select! : -110
Hit any key to stop autoboot: 2 1 0
Card did not respond to voltage select! : -110
Couldn't find partition mmc 0:3
Can't set block device
Importing environment from mmc0 ...
Card did not respond to voltage select! : -110
Couldn't find partition mmc 1:2
Can't set block device
## Warning: defaulting to text format
## Error: "boot2" not defined
Card did not respond to voltage select! : -110
ethernet@16030000 Waiting for PHY auto negotiation to complete....... done
BOOTP broadcast 1
*** Unhandled DHCP Option in OFFER/ACK: 43
*** Unhandled DHCP Option in OFFER/ACK: 43
DHCP client bound to address 192.168.x.x (550 ms)
Using ethernet@16030000 device
TFTP from server 192.168.x.x; our IP address is 192.168.x.x
Filename 'boot.scr.uimg'.
Load address: 0x43900000
Loading: *
TFTP server died; starting again
BOOTP broadcast 1
*** Unhandled DHCP Option in OFFER/ACK: 43
*** Unhandled DHCP Option in OFFER/ACK: 43
DHCP client bound to address 192.168.x.x (547 ms)
Using ethernet@16030000 device
TFTP from server 192.168.x.x; our IP address is 192.168.x.x
Filename 'boot.scr.uimg'.
Load address: 0x40200000
Loading: *
TFTP server died; starting again
Using ethernet@16030000 device
TFTP from server 192.168.x.x; our IP address is 192.168.x.x
Filename 'Image'.
Load address: 0x40200000
Loading: * #################################################################
[8C ###########################################################T ######T
[8C #############
[8C 147.5 KiB/s
done
Bytes transferred = 2097832 (2002a8 hex)
Using ethernet@16030000 device
TFTP from server 192.168.x.x; our IP address is 192.168.x.x
Filename 'jh7110-star64-pine64.dtb'.
Load address: 0x46000000
Loading: * #T ###
[8C 8.8 KiB/s
done
Bytes transferred = 50235 (c43b hex)
## Flattened Device Tree blob at 46000000
Booting using the fdt blob at 0x46000000
Using Device Tree in place at 0000000046000000, end 000000004600f43a
2023-07-11 11:13:58 +08:00
2023-07-11 14:41:53 +08:00
Starting kernel ...
2023-07-11 11:13:58 +08:00
2023-07-11 14:41:53 +08:00
clk u5_dw_i2c_clk_core already disabled
clk u5_dw_i2c_clk_apb already disabled
123067DFAGHBCUnhandled exception: Store/AMO access fault
EPC: 0000000040200628 RA: 00000000402004ba TVAL: ffffff8000008000
EPC: ffffffff804ba628 RA: ffffffff804ba4ba reloc adjusted
2023-07-11 11:13:58 +08:00
2023-07-11 14:41:53 +08:00
SP: 0000000040406a30 GP: 00000000ff735e00 TP: 0000000000000001
T0: 0000000010000000 T1: 0000000000000037 T2: ffffffffffffffff
S0: 0000000040400000 S1: 0000000000000200 A0: 0000000000000003
A1: 0000080000008000 A2: 0000000010100000 A3: 0000000040400000
A4: 0000000000000026 A5: 0000000000000000 A6: 00000000101000e7
A7: 0000000000000000 S2: 0000080000008000 S3: 0000000040600000
S4: 0000000040400000 S5: 0000000000000000 S6: 0000000000000026
S7: 00fffffffffff000 S8: 0000000040404000 S9: 0000000000001000
S10: 0000000040400ab0 S11: 0000000000200000 T3: 0000000000000023
T4: 000000004600f43a T5: 000000004600d000 T6: 000000004600cfff
2023-07-11 11:13:58 +08:00
2023-07-11 14:41:53 +08:00
Code: 879b 0277 d7b3 00f6 f793 1ff7 078e 95be (b023 0105)
2023-07-11 11:13:58 +08:00
2023-07-11 14:41:53 +08:00
resetting ...
reset not supported yet
### ERROR ### Please RESET the board ###
2023-07-11 11:13:58 +08:00
```