![Pine64 Star64 64-bit RISC-V SBC](https://lupyuen.github.io/images/star64-title.jpg) # Apache NuttX RTOS for Pine64 Star64 64-bit RISC-V SBC (StarFive JH7110) [![Daily Build of NuttX for Star64](https://github.com/lupyuen/nuttx-star64/actions/workflows/star64.yml/badge.svg)](https://github.com/lupyuen/nuttx-star64/actions/workflows/star64.yml) Read the articles... - ["RISC-V Star64 JH7110: Power Up the Display Controller with U-Boot Bootloader"](https://lupyuen.github.io/articles/display3) - ["RISC-V Star64 JH7110: Inside the Display Controller"](https://lupyuen.github.io/articles/display2) - ["RTOS on a RISC-V SBC: Star64 JH7110 + Apache NuttX"](https://www.hackster.io/lupyuen/rtos-on-a-risc-v-sbc-star64-jh7110-apache-nuttx-2a7429) - ["Star64 JH7110 + NuttX RTOS: Creating the First Release for the RISC-V SBC"](https://lupyuen.github.io/articles/release) - ["Star64 JH7110 + NuttX RTOS: RISC-V PLIC Interrupts and Serial I/O"](https://lupyuen.github.io/articles/plic) - ["Star64 JH7110 + NuttX RTOS: RISC-V Semihosting and Initial RAM Disk"](https://lupyuen.github.io/articles/semihost) - ["Star64 JH7110 + NuttX RTOS: RISC-V Privilege Levels and UART Registers"](https://lupyuen.github.io/articles/privilege) - ["Apache NuttX RTOS on RISC-V: Star64 JH7110 SBC"](https://lupyuen.github.io/articles/nuttx2) - ["Star64 JH7110 RISC-V SBC: Boot from Network with U-Boot and TFTP"](https://lupyuen.github.io/articles/tftp) - ["Strange Workaround for TFTP Timeout in U-Boot Bootloader (Star64 JH7110 RISC-V SBC)"](https://lupyuen.github.io/articles/tftp2) Earlier articles... - ["Booting RISC-V Linux on Star64 JH7110 SBC"](https://lupyuen.github.io/articles/linux) - ["Inspecting the RISC-V Linux Images for Star64 JH7110 SBC"](https://lupyuen.github.io/articles/star64) - ["64-bit RISC-V with Apache NuttX Real-Time Operating System"](https://lupyuen.github.io/articles/riscv) - ["Rolling to RISC-V"](https://lupyuen.github.io/articles/pinephone2#rolling-to-risc-v) Many thanks to CNXSoft for the coverage... - ["Star64 RISC-V SBC can now boot Apache NuttX real-time operating system"](https://www.cnx-software.com/2023/08/17/star64-risc-v-sbc-apache-nuttx-real-time-operating-system/) 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)) [(NuttX now officially supports Star64 JH7110)](https://nuttx.apache.org/docs/latest/platforms/risc-v/jh7110/boards/star64/index.html) [(NuttX works OK on StarFive VisionFive 2 SBC too)](https://lupyuen.github.io/articles/release#whats-next) Hopefully NuttX will run on Pine64 PineTab-V, which is also based on StarFive JH7110 SoC. # NuttX Automated Daily Build for Star64 NuttX for Star64 is now built automatically every day via GitHub Actions. The Daily Releases are available here... - [nuttx-star64/releases](https://github.com/lupyuen/nuttx-star64/releases) [nuttx.hash](https://github.com/lupyuen/nuttx-star64/releases/download/nuttx-star64-2023-08-11/nuttx.hash) contains the Commit Hash of the NuttX Kernel and NuttX Apps repos... ```text NuttX Source: https://github.com/apache/nuttx/tree/fa676f264fa16253213ff665052ba4691e56bf05 NuttX Apps: https://github.com/apache/nuttx-apps/tree/6196e03337a3c677aba0c74e815af16e7075bc71 ``` The GitHub Actions Workflow is here... - [star64.yml](https://github.com/lupyuen/nuttx-star64/blob/main/.github/workflows/star64.yml) Maybe someday we'll do Daily Automated Testing... 1. Download the Daily Build to TFTP Server 1. Power on Star64 with an [IKEA Smart Power Plug via Home Assistant](https://lupyuen.github.io/articles/tftp#whats-next) 1. Star64 boots the Daily Build over TFTP 1. Capture the Automated Testing Log and write to the Release Notes [(Similar to BL602)](https://lupyuen.github.io/articles/auto) # Linux Images for Star64 Read the article... - ["Inspecting the RISC-V Linux Images for Star64 JH7110 SBC"](https://lupyuen.github.io/articles/star64) 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) According to [Software Releases for Star64](https://wiki.pine64.org/wiki/STAR64#Software_releases), we have... - [Yocto Images](https://github.com/Fishwaldo/meta-pine64) at [pine64.my-ho.st](https://pine64.my-ho.st:8443/) Let's inspect [star64-image-minimal](https://pine64.my-ho.st:8443/star64-image-minimal-star64-1.2.wic.bz2) - [Armbian Images](https://www.armbian.com/star64/) 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) 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) # Armbian Image for Star64 Read the article... - ["Inspecting the RISC-V Linux Images for Star64 JH7110 SBC"](https://lupyuen.github.io/articles/star64) 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. The image contains 1 used partition: `armbi_root` (612 MB) that contains the Linux Root Filesystem. Plus one unused partition (4 MB) at the top. (Partition Table) ![Armbian Image for Star64](https://lupyuen.github.io/images/star64-armbian.png) We see the U-Boot Bootloader Configuration at `armbi_root/boot/extlinux/extlinux.conf`... ```text 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 ``` This says that U-Boot will load the Linux Kernel from `armbi_root/boot/Image` Which is sym-linked to `armbi_root/boot/vmlinuz-5.15.0-starfive2` _Where in RAM will the Kernel Image be loaded?_ 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`__... ```text kernel_addr_r=0x40200000 ``` [(Source)](https://github.com/lupyuen/nuttx-star64#u-boot-settings-for-star64) _Everything looks hunky dory?_ Nope the [__Flattened Device Tree (FDT)__](https://u-boot.readthedocs.io/en/latest/develop/devicetree/index.html) is missing! But the Flattened Device Tree (FDT) is missing! `/boot/dtb/starfive/jh7110-star64-pine64.dtb` ```text fdt /boot/dtb/starfive/jh7110-star64-pine64.dtb ``` Which means that Armbian will [__fail to boot__](https://github.com/lupyuen/nuttx-star64#boot-armbian-on-star64) on Star64! ```text 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' ``` [(Source)](https://github.com/lupyuen/nuttx-star64#boot-armbian-on-star64) 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) For Reference: Here's the list of __Supported Device Trees__... ```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 ``` And here are the other files in __/boot__... ```text → ls -l /Volumes/armbi_root/boot total 94416 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 ``` TODO: Explain `boot/uInitrd` RAM Disk # Yocto Image for Star64 Read the article... - ["Inspecting the RISC-V Linux Images for Star64 JH7110 SBC"](https://lupyuen.github.io/articles/star64) 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) Uncompress the .bz2, rename as .img. Balena Etcher won't work with .bz2 files! Write the .img to a microSD Card with Balena Etcher. Insert the microSD Card into a Linux Machine. (Like Pinebook Pro) We see 4 used partitions... - spl (2 MB): [Secondary Program Loader](https://u-boot.readthedocs.io/en/latest/board/starfive/visionfive2.html#flashing) - uboot (4 MB): [U-Boot Bootloader](https://u-boot.readthedocs.io/en/latest/board/starfive/visionfive2.html#flashing) - boot (380 MB): U-Boot Configuration and Linux Kernel Image - root (686 MB): Linux Root Filesystem Plus one unused partition (2 MB) at the top. (Partition Table) ![Yocto Image for Star64](https://lupyuen.github.io/images/star64-yocto.png) `boot` partition has 2 files... ```text $ ls -l /run/media/$USER/boot total 14808 -rw-r--r-- 1 15151064 Apr 6 2011 fitImage -rw-r--r-- 1 1562 Apr 6 2011 vf2_uEnv.txt ``` `boot/vf2_uEnv.txt` contains the U-Boot Bootloader Configuration... ```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} ``` [`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`... ```text # boot Linux flat or compressed 'Image' stored at 'kernel_addr_r' kernel_addr_r=0x40200000 ``` Yocto boots from the [Flat Image Tree (FIT)](https://u-boot.readthedocs.io/en/latest/usage/fit/index.html#): `boot/fitImage` Yocto's `root/boot` looks different from Armbian... ```text $ ls -l /run/media/$USER/root/boot total 24376 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 ``` # Boot NuttX with U-Boot Bootloader Read the article... - ["Inspecting the RISC-V Linux Images for Star64 JH7110 SBC"](https://lupyuen.github.io/articles/star64) _Will we boot NuttX with Armbian or Yocto settings?_ Armbian looks simpler, since it uses a plain Linux Kernel Image File `Image`. (Instead of Yocto's complicated Flat Image Tree) Hence we'll overwrite Armbian's `armbi_root/boot/Image` by the NuttX Kernel Image. We'll compile NuttX Kernel to boot at `0x4020` `0000`. 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. # Inside the Armbian Kernel Image Read the article... - ["Inspecting the RISC-V Linux Images for Star64 JH7110 SBC"](https://lupyuen.github.io/articles/star64) _What's inside the Armbian Linux Kernel Image?_ Let's look inside `armbi_root/boot/vmlinuz-5.15.0-starfive2`... ![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 */ ``` 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) Let's decompile the Kernel Image... TODO: Explain MZ and the funny RISC-V instruction at the top # Decompile Armbian Kernel Image with Ghidra Read the article... - ["Inspecting the RISC-V Linux Images for Star64 JH7110 SBC"](https://lupyuen.github.io/articles/star64) We decompile the Armbian Linux Kernel Image with [Ghidra](https://github.com/NationalSecurityAgency/ghidra). In Ghidra, create a New Project. Click File > Import File. Select `armbi_root/boot/vmlinuz-5.15.0-starfive2` and enter these Import Options... - Format: Raw Binary - 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) - Options > Base Address: 0x44000000 (Based on the U-Boot Configuration from above) ![Load the Armbian Linux Kernel Image into Ghidra](https://lupyuen.github.io/images/star64-ghidra.png) ![Load the Armbian Linux Kernel Image into Ghidra](https://lupyuen.github.io/images/star64-ghidra2.png) Double-click `vmlinuz-5.15.0-starfive2`, analyse the file with the Default Options. Ghidra displays the Decompiled Linux Kernel... ![Disassembled Linux Kernel in Ghidra](https://lupyuen.github.io/images/star64-ghidra3.png) At Address `0x4400` `0002` we see a Jump to `FUN_440010c8`... ```text // Load -13 into Register S4 li s4,-0xd // Jump to Actual Boot Code j FUN_440010c8 ``` Double-click `FUN_440010c8` to see the Linux Boot Code... ![Linux Boot Code in Ghidra](https://lupyuen.github.io/images/star64-ghidra4.png) TODO: Explain MZ and the funny RISC-V instruction at the top TODO: Where is the source file? TODO: Any interesting CSR Instructions? # Serial Console on Star64 Read the article... - ["Booting RISC-V Linux on Star64 JH7110 SBC"](https://lupyuen.github.io/articles/linux) To access the Serial Console, we connect a [USB Serial Adapter](https://pine64.com/product/serial-console-woodpecker-edition/) to Star64... ![Star64 JH7110 RISC-V SBC with Woodpecker USB Serial Adapter](https://lupyuen.github.io/images/linux-title.jpg) 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). Thus we connect these pins... | 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 Set the Voltage Jumper to 3V3. (Instead of 5V) ![Pine64 Woodpecker Serial Adapter](https://lupyuen.github.io/images/star64-uart3.jpg) On our computer, connect to the USB Serial Port at 115.2 kbps... ```bash screen /dev/ttyUSB0 115200 ``` 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. [(DIP Switch Labels are inverted: __"ON"__ actually means __"Low"__)](https://wiki.pine64.org/wiki/STAR64#Prototype_Bringup_Notes) ![DIP Switches for GPIO 0 and 1 are set to Low and Low](https://lupyuen.github.io/images/star64-uart2.jpg) We'll see this U-Boot Bootloader Log... TODO: Explain [OpenSBI](https://www.thegoodpenguin.co.uk/blog/an-overview-of-opensbi/) # Star64 U-Boot Bootloader Log Read the article... - ["Booting RISC-V Linux on Star64 JH7110 SBC"](https://lupyuen.github.io/articles/linux) Here's the log for U-Boot Bootloader on Star64 (without microSD Card)... ![U-Boot Bootloader Log](https://lupyuen.github.io/images/star64-opensbi.jpg) ```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 # ``` Which is OK because we haven't inserted a microSD Card. ## U-Boot Commands for Star64 Here are the U-Boot Commands... ```text 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 ``` ## U-Boot Settings for Star64 Here are the U-Boot Settings... ```text 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 # ``` # Boot Armbian on Star64 Read the article... - ["Booting RISC-V Linux on Star64 JH7110 SBC"](https://lupyuen.github.io/articles/linux) 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... - [Armbian Boot Log](https://gist.github.com/lupyuen/d73ace627318375fe20e90e4950f9c50) Armbian fails to boot... ```text Found /boot/extlinux/extlinux.conf Retrieving file: /boot/extlinux/extlinux.conf 383 bytes read in 7 ms (52.7 KiB/s) 1:Armbian 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 ``` 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) # Boot Yocto on Star64 Read the article... - ["Booting RISC-V Linux on Star64 JH7110 SBC"](https://lupyuen.github.io/articles/linux) 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. Here's what happens when we boot the microSD Card on Star64... - [Yocto Boot Log](https://gist.github.com/lupyuen/b23edf50cecbee13e5aab3c0bae6c528) Usernames and Passwords are... - root / pine64 - pine64 / pine64 [(Source)](https://github.com/Fishwaldo/meta-pine64#usernames) Yep the Yocto Minimal Image boots OK on Star64! # Boot Yocto Plasma on Star64 Read the article... - ["Booting RISC-V Linux on Star64 JH7110 SBC"](https://lupyuen.github.io/articles/linux) 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. 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) ![Yocto Plasma on Star64](https://lupyuen.github.io/images/star64-plasma.jpg) # NuttX prints to QEMU Console Read the article... - ["Apache NuttX RTOS on RISC-V: Star64 JH7110 SBC"](https://lupyuen.github.io/articles/nuttx2) 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! Earlier we ran NuttX on QEMU Emulator for 64-bit RISC-V... - ["64-bit RISC-V with Apache NuttX Real-Time Operating System"](https://lupyuen.github.io/articles/riscv) QEMU emulates a 16550 UART Port. (Similar to Star64 / JH7110) _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) ```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?_ 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): ```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); } ``` [(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. And we discover that [UART_THR_OFFSET](https://github.com/apache/nuttx/blob/dc69b108b8e0547ecf6990207526c27aceaf1e2e/include/nuttx/serial/uart_16550.h#L172-L200) is 0: ```c #define UART_THR_INCR 0 /* (DLAB =0) Transmit Holding Register */ #define UART_THR_OFFSET (CONFIG_16550_REGINCR*UART_THR_INCR) ``` So we can transmit to UART Port by simply writing to `0x1000` `0000`. How convenient! _How to print to the QEMU Console?_ Let's do the printing in RISC-V Assembly Code, so that we can debug the NuttX Boot Code. 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): ```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) ``` This prints "123" to the QEMU Console. Here's the output: ```text + qemu-system-riscv64 \ -semihosting \ -M virt,aclint=on \ -cpu rv64 \ -smp 8 \ -bios none \ -kernel nuttx \ -nographic 123123123123123123112323 NuttShell (NSH) NuttX-12.0.3 nsh> ``` Which is correct because QEMU is running with 8 CPUs. Yay! ![NuttX prints to QEMU Console](https://lupyuen.github.io/images/riscv-print.png) [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) And offers to optimise our RISC-V Assembly Code... ![Cody AI Assistant optimises our RISC-V Assembly Code](https://lupyuen.github.io/images/riscv-cody2.png) 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> ``` The correct output is `123123123123123123112323`. (Because of the 8 CPUs) # UART Base Address for Star64 Read the article... - ["Apache NuttX RTOS on RISC-V: Star64 JH7110 SBC"](https://lupyuen.github.io/articles/nuttx2) We'll take the UART Assembly Code from the previous section and run on Star64 / JH7110. (So we can troubleshoot the NuttX Boot Code) _Does Star64 / JH7110 use a 16550 UART Controller like QEMU?_ 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`. [(JH7110 UART Datasheet)](https://doc-en.rvspace.org/JH7110/Datasheet/JH7110_DS/uart.html) _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): ```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 ``` 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! # RISC-V Linux Kernel Header Read the article... - ["Apache NuttX RTOS on RISC-V: Star64 JH7110 SBC"](https://lupyuen.github.io/articles/nuttx2) 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) 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) 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)... 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): ```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. */ c.li s4, -13 /* Magic Signature "MZ" (2 bytes) */ j real_start /* Jump to Kernel Start (2 bytes) */ .long 0 /* Executable Code padded to 8 bytes */ .quad 0x200000 /* Image load offset from start of RAM */ /* 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 ``` Note that Image Load Offset must be `0x20` `0000`! ```text .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 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 ... 40200020: 0002 c.slli64 zero ... 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 ... 0000000040200040 : ``` 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) And our RISC-V Boot Code tested OK with QEMU. # Set Start Address of NuttX Kernel Read the article... - ["Apache NuttX RTOS on RISC-V: Star64 JH7110 SBC"](https://lupyuen.github.io/articles/nuttx2) 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`. 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 ``` We changed the above NuttX Build Config to `0x40200000` 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) ```text SECTIONS { /* Previously 0x80000000 */ . = 0x40200000; .text : ``` 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): ```text MEMORY { /* Previously 0x80000000 */ kflash (rx) : ORIGIN = 0x40200000, LENGTH = 2048K /* w/ cache */ /* Previously 0x80200000 */ ksram (rwx) : ORIGIN = 0x40400000, LENGTH = 2048K /* w/ cache */ /* Previously 0x80400000 */ pgram (rwx) : ORIGIN = 0x40600000, LENGTH = 4096K /* w/ cache */ } ... SECTIONS { /* Previously 0x80000000 */ . = 0x40200000; .text : ``` 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 ``` RISC-V Disassembly of NuttX Kernel shows that the Start Address is correct... ```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 ``` We're ready to boot NuttX on Star64! # Boot NuttX on Star64 Read the article... - ["Apache NuttX RTOS on RISC-V: Star64 JH7110 SBC"](https://lupyuen.github.io/articles/nuttx2) 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/)... - [__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. We fix the [Missing Device Tree](https://lupyuen.github.io/articles/star64#armbian-image-for-star64)... ```bash ## 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 cp \ /run/media/$USER/armbi_root/boot/dtb/starfive/jh7110-visionfive-v2.dtb \ /run/media/$USER/armbi_root/boot/dtb/starfive/jh7110-star64-pine64.dtb ``` Then we delete the sym-link `/boot/Image` and copy the NuttX Binary Image `nuttx.bin` to `/boot/Image`... ```bash ## We assume that `nuttx` contains the NuttX ELF Image. ## Export the NuttX Binary Image to `nuttx.bin` riscv64-unknown-elf-objcopy \ -O binary \ nuttx \ nuttx.bin ## Delete Armbian Kernel `/boot/Image` rm /run/media/$USER/armbi_root/boot/Image ## Copy `nuttx.bin` to Armbian Kernel `/boot/Image` cp nuttx.bin /run/media/$USER/armbi_root/boot/Image ``` Insert the microSD Card into Star64 and power up. NuttX boots with `123` yay! [(Which is printed by our Boot Code)](https://github.com/lupyuen/nuttx-star64#nuttx-prints-to-qemu-console) ```text Starting kernel ... clk u5_dw_i2c_clk_core already disabled clk u5_dw_i2c_clk_apb already disabled 123 Unhandled exception: Illegal instruction ``` ![Boot NuttX on Star64](https://lupyuen.github.io/images/star64-nuttx.png) Here's the complete log... ```text Retrieving file: /boot/extlinux/extlinux.conf 383 bytes read in 7 ms (52.7 KiB/s) 1:Armbian Retrieving file: /boot/uInitrd 10911538 bytes read in 466 ms (22.3 MiB/s) Retrieving file: /boot/Image 163201 bytes read in 14 ms (11.1 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 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) resetting ... reset not supported yet ### ERROR ### Please RESET the board ### ``` Why does NuttX crash at `4020005c`? See the next section... ![Cody AI Assistant tries to explain our RISC-V Exception](https://lupyuen.github.io/images/star64-exception.jpg) _[Cody AI Assistant](https://about.sourcegraph.com/cody) tries to explain our RISC-V Exception_ # NuttX Fails To Get Hart ID Read the article... - ["Apache NuttX RTOS on RISC-V: Star64 JH7110 SBC"](https://lupyuen.github.io/articles/nuttx2) 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 ``` _Why did NuttX crash at `4020005c`?_ Here's our RISC-V Boot Code... 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): ```text nuttx/arch/risc-v/src/chip/qemu_rv_head.S:95 /* Load mhartid (cpuid) */ csrr a0, mhartid 4020005c: f1402573 csrr a0, mhartid ``` 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) But it fails! Because we don't have sufficient privilege to access the Hart ID... # RISC-V Privilege Levels Read the article... - ["Apache NuttX RTOS on RISC-V: Star64 JH7110 SBC"](https://lupyuen.github.io/articles/nuttx2) RISC-V runs at 3 Privilege Levels... - M: Machine Mode (Most powerful) - S: Supervisor Mode (Less powerful) - U: User Mode (Least powerful) 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)) (The `m` in `mhartid` signifies that it's a Machine-Mode Register) ![RISC-V Privilege Levels](https://lupyuen.github.io/images/nuttx2-privilege.jpg) _What runs in Machine Mode?_ [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. [(See the RISC-V SBI Spec)](https://github.com/riscv-non-isa/riscv-sbi-doc/blob/master/riscv-sbi.pdf) _What about U-Boot Bootloader?_ U-Boot Bootloader runs in Supervisor Mode. And starts NuttX, also in Supervisor Mode. So OpenSBI is the only thing that runs in Machine Mode. And can access the Machine-Level Registers. _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... # Downgrade NuttX to Supervisor Mode Read the article... - ["Apache NuttX RTOS on RISC-V: Star64 JH7110 SBC"](https://lupyuen.github.io/articles/nuttx2) _How to get the Hart ID from OpenSBI?_ 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) (Tip: `CONFIG_RISCV_M_MODE` is False and `CONFIG_EFI` is True) 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 ``` 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... - Register A0: Hart ID - Register A1: RAM Address of Device Tree So we'll simply read the Hart ID from Register A0. (And ignore A1) We'll remove `csrr a0, mhartid`. _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) Yep looks correct! But we'll subtract 1 from Register A0 because NuttX expects Hart ID to start with 0. _What about other CSR Instructions in our NuttX Boot Code?_ We change the Machine-Level `m` Registers to Supervisor-Level `s` Registers. To Disable Interrupts: Change `mie` to [`sie`](https://five-embeddev.com/riscv-isa-manual/latest/supervisor.html#supervisor-interrupt-registers-sip-and-sie) ```text /* Disable all interrupts (i.e. timer, external) in mie */ csrw mie, zero ``` [(Source)](https://lupyuen.github.io/articles/riscv#disable-interrupts) 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) ```text /* Load address of Interrupt Vector Table */ csrw mtvec, t0 ``` [(Source)](https://lupyuen.github.io/articles/riscv#load-interrupt-vector) _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) Let's fix the Boot Code... # Fix the NuttX Boot Code Read the article... - ["Apache NuttX RTOS on RISC-V: Star64 JH7110 SBC"](https://lupyuen.github.io/articles/nuttx2) 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 */ /* Don't load the Interrupt Vector Table, use OpenSBI for crash logging */ /* la t0, __trap_vec */ /* csrw stvec, t0 */ /* Previously: csrw mtvec, t0 */ /* Jump to qemu_rv_start */ jal x1, qemu_rv_start /* We shouldn't return from _start */ ``` 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) _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 123067 ``` 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 ``` Let's trace `qemu_rv_start`... # Boot from Network with U-Boot and TFTP Read the article... - ["Star64 JH7110 RISC-V SBC: Boot from Network with U-Boot and TFTP"](https://lupyuen.github.io/articles/tftp) 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. Here's how... ![Boot from Network with U-Boot and TFTP](https://lupyuen.github.io/images/tftp-flow.jpg) ## Setup TFTP Server First we set up a TFTP Server with [`tftpd`](https://crates.io/crates/tftpd)... ```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 ``` ([`tftp_server`](https://crates.io/crates/tftp_server) won't work, it only supports localhost) We should see... ```text 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 ``` Let's test the TFTP Server... ```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 ``` (`localhost` won't work because of IPv6, I think) We should see... ```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 ``` If it fails... ```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 ``` In the olden days we would actually do this... ```text $ tftp 127.0.0.1 tftp> get a.txt Received 8 bytes in 0.0 seconds tftp> quit ``` Just like FTP! ## Copy NuttX Image to TFTP Server Next we copy the NuttX Image and Device Tree to the TFTP Folder... ```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 ## 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 ## Test NuttX Binary Image and Device Tree over TFTP curl -v tftp://192.168.x.x/Image curl -v tftp://192.168.x.x/jh7110-star64-pine64.dtb ## We should see... ## 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: " to save to a file. ``` ## Test U-Boot with TFTP Now we boot Star64 JH7110 SBC and test the TFTP Commands. Connect Star64 SBC to the Ethernet wired network and power up. 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... ```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 # ``` [(Source)](https://github.com/lupyuen/nuttx-star64#u-boot-bootloader-log-for-tftp) Run these commands... ```bash ## Set the TFTP Server IP setenv tftp_server 192.168.x.x ## Load the NuttX Image from TFTP Server ## kernel_addr_r=0x40200000 ## tftp_server=192.168.x.x tftpboot ${kernel_addr_r} ${tftp_server}:Image ## 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 ## Set the RAM Address of Device Tree ## fdt_addr_r=0x46000000 fdt addr ${fdt_addr_r} ## Boot the NuttX Image with the Device Tree ## kernel_addr_r=0x40200000 ## fdt_addr_r=0x46000000 booti ${kernel_addr_r} - ${fdt_addr_r} ``` [(Inspired by this article)](https://community.arm.com/oss-platforms/w/docs/495/tftp-remote-network-kernel-using-u-boot) We should see... ```text StarFive # setenv tftp_server 192.168.x.x 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 Starting kernel ... clk u5_dw_i2c_clk_core already disabled clk u5_dw_i2c_clk_apb already disabled 123067DFAGHBC ``` [(Source)](https://github.com/lupyuen/nuttx-star64#u-boot-bootloader-log-for-tftp) ## Configure U-Boot for TFTP Let's configure U-Boot so that it will boot from TFTP every time we power up! ```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 ## 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 ## Test the Boot Command for TFTP run bootcmd_tftp ## Remember the Original Boot Targets setenv orig_boot_targets "$boot_targets" ## Should show `mmc0 dhcp` printenv boot_targets ## Save it for future reboots saveenv ## 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 ``` `bootcmd_tftp` expands 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 ## 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 ``` [(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: ```bash ## Restore the Boot Targets setenv boot_targets "$orig_boot_targets" ## Should show `mmc0 dhcp` printenv boot_targets ## Save it for future reboots saveenv ``` 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) ## U-Boot Commands for Network Boot _How does it work?_ `bootcmd` is now... ```text bootcmd=run load_vf2_env;run importbootenv;run load_distro_uenv;run boot2;run distro_bootcmd load_vf2_env=fatload mmc ${bootpart} ${loadaddr} ${testenv} importbootenv=echo Importing environment from mmc${devnum} ...; env import -t ${loadaddr} ${filesize} load_distro_uenv=fatload mmc ${fatbootpart} ${distroloadaddr} ${bootdir}/${bootenv}; env import ${distroloadaddr} 17c; boot2 not defined, comes from `boot/vf2_uEnv.txt` ``` `bootcmd` calls `distro_bootcmd`, which runs `bootcmd_mmc0` and `bootcmd_dhcp`... ```text distro_bootcmd=for target in ${boot_targets}; do run bootcmd_${target}; done boot_targets=mmc0 dhcp bootcmd_mmc0=devnum=0; run mmc_boot bootcmd_distro=run fdt_loaddtb; run fdt_sizecheck; run set_fdt_distro; sysboot mmc ${fatbootpart} fat c0000000 ${bootdir}/${boot_syslinux_conf}; ``` `bootcmd_dhcp` is... ```text 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; ``` Which expands to... ```bash devtype=dhcp ## 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 ## 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 ## Load the Kernel Image from DHCP+TFTP Server... ## kernel_addr_r=0x40200000 if dhcp ${kernel_addr_r} then ## 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} ## Set the RAM Address of Device Tree ## fdt_addr_r=0x46000000 if fdt addr ${fdt_addr_r} then ## Boot the EFI Kernel Image ## fdt_addr_r=0x46000000 bootefi ${kernel_addr_r} ${fdt_addr_r} else ## 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 ``` `dhcp` command is... ```text dhcp - boot image via network using DHCP/TFTP protocol Usage: dhcp [loadAddress] [[hostIPaddr:]bootfilename] ``` (Assume [DHCP/TFTP](https://www.emcraft.com/som/using-dhcp) is not used) `tftpboot` command is... ```text tftpboot - boot image via network using TFTP protocol Usage: tftpboot [loadAddress] [[hostIPaddr:]bootfilename] ``` `fdt` command is... ```text fdt - flattened device tree utility commands Usage: fdt addr [-c] [] - Set the [control] fdt location to fdt apply - Apply overlay to the DT fdt move - Copy the fdt to and make it active fdt resize [] - Resize fdt to size + padding to 4k addr + some optional if needed fdt print [] - Recursive print starting at fdt list [] - Print one level starting at fdt get value - Get and store in fdt get name - Get name of node and store in fdt get addr - Get start address of and store in fdt get size [] - Get size of [] or num nodes and store in fdt set [] - Set [to ] fdt mknode - Create a new node after fdt rm [] - Delete the node or fdt header [get ] - Display header info get - get header member and store it in fdt bootcpu - Set boot cpuid fdt memory - Add/Update memory node fdt rsvmem print - Show current mem reserves fdt rsvmem add - Add a mem reserve fdt rsvmem delete - Delete a mem reserves fdt chosen [ ] - Add/update the /chosen branch in the tree / - initrd start/end addr NOTE: Dereference aliases by omitting the leading '/', e.g. fdt print ethernet0. ``` `booti` command is... ```text booti - boot Linux kernel 'Image' format from memory 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. ``` `bootefi` command is... ```text bootefi - Boots an EFI payload from memory Usage: bootefi [fdt address] - boot EFI payload stored at address . If specified, the device tree located at gets exposed as EFI configuration table. bootefi bootmgr [fdt address] - load and boot EFI payload based on BootOrder/BootXXXX variables. If specified, the device tree located at gets exposed as EFI configuration table. ``` Doesn't work for NuttX... ```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 ``` [`autoload`](https://u-boot.readthedocs.io/en/latest/usage/environment.html) setting is... ```text 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. ``` # Strange Workaround for TFTP Timeout in U-Boot Bootloader Read the article... - ["Strange Workaround for TFTP Timeout in U-Boot Bootloader (Star64 JH7110 RISC-V SBC)"](https://lupyuen.github.io/articles/tftp2) Why does TFTP time out so often on our SBC? Is it because our TFTP Server sends packets too quickly to our SBC? Frequent TFTP Timeouts ("T" below) are affecting our NuttX Testing on Star64 JH7110 SBC. Effective transfer rate is only 430 kbps! ```text Loading: . ##############T ################################################### . ####################T #########T #################################### . 53.7 KiB/s ``` [(Source)](https://gist.github.com/lupyuen/9bdb1f5478318631d0480f03f6041d83#file-jh7110-nuttx-math-log-L140-L173) Let's try something: We send every TFTP Packet twice. From https://github.com/lupyuen/rs-tftpd-timeout/blob/main/src/worker.rs#L232-L255 ```rust fn send_window( socket: &T, window: &Window, mut block_num: u16, ) -> Result<(), Box> { // For Every Frame... for frame in window.get_elements() { // Send the TFTP Packet socket.send(&Packet::Data { block_num, data: frame.to_vec(), })?; // Wait 1 millisecond static mut DELAY_MS: u64 = 1; let millis = std::time::Duration::from_millis(DELAY_MS); std::thread::sleep(millis); // Send the same TFTP Packet again // Why does this work? socket.send(&Packet::Data { block_num, data: frame.to_vec(), })?; ``` Let's test this... __Before Fixing:__ TFTP Transfer Rate is 48.8 KiB/s (with 6 timeouts) [(See the log: xinetd + tftpd on Raspberry Pi 4 32-bit)](https://gist.github.com/lupyuen/b36278130fbd281d03fc20189de5485e) [(Watch the Demo on YouTube)](https://youtu.be/MPBc2Qec6jo) [(Based on this configuration)](https://community.arm.com/oss-platforms/w/docs/495/tftp-remote-network-kernel-using-u-boot) __After Fixing:__ TFTP Transfer Rate is 1.1 MiB/s (with NO timeouts) [(See the log: rs-tftpd-timeout on Raspberry Pi 4 32-bit)](https://gist.github.com/lupyuen/19ab2e16c0c2bb46175bcd8fba7116f2) [(Watch the Demo on YouTube)](https://youtu.be/ABpi2ABln5o) [(Based on rs-tftpd-timeout)](https://github.com/lupyuen/rs-tftpd-timeout/blob/main/src/worker.rs#L232-L255) Yep it works! No more TFTP Timeouts! (Tested on 32-bit Raspberry Pi 4 and on macOS x64) TODO: Why does it work? Dropped UDP Packets? We should check with Wireshark TODO: What if we throttle our TFTP Server to send packets slower? Nope doesn't help TODO: What if we we reduce the timeout? Nope doesn't work _Does this problem happen for devices other than Star64 JH7110?_ Nope this TFTP Timeout seems specific to Star64 JH7110. We downloaded a 9 MB file from Pi to macOS over TFTP on Wired Ethernet... ```text # Before Fixing TFTP Server: 19 Mbps → curl --output initrd tftp://192.168.31.10/initrd % Total % Received % Xferd Average Speed Time Time Time Current Dload Upload Total Spent Left Speed 100 9015k 0 9015k 0 0 2374k 0 --:--:-- 0:00:03 --:--:-- 2374k # After Fixing TFTP Server: 3.3 Mbps → curl --output initrd tftp://192.168.31.10/initrd % Total % Received % Xferd Average Speed Time Time Time Current Dload Upload Total Spent Left Speed 100 9015k 100 9015k 0 0 411k 0 0:00:21 0:00:21 --:--:-- 411k ``` The fixed TFTP Server is slower because of the 1 millisecond delay between packets. And we sent every packet twice. So maybe U-Boot Bootloader on Star64 JH7110 is too slow to catch all the TFTP Packets? # Hang in Enter Critical Section Read the article... - ["Star64 JH7110 + NuttX RTOS: RISC-V Privilege Levels and UART Registers"](https://lupyuen.github.io/articles/privilege) NuttX on Star64 JH7110 hangs when entering Critical Section... From [uart_16550.c](https://github.com/lupyuen2/wip-pinephone-nuttx/blob/star64/drivers/serial/uart_16550.c#L1713-L1737): ```c int up_putc(int ch) { FAR struct u16550_s *priv = (FAR struct u16550_s *)CONSOLE_DEV.priv; irqstate_t flags; /* All interrupts must be disabled to prevent re-entrancy and to prevent * interrupts from firing in the serial driver code. */ //// This will hang! flags = enter_critical_section(); ... u16550_putc(priv, ch); leave_critical_section(flags); return ch; } ``` Which assembles to... ```text int up_putc(int ch) { ... up_irq_save(): nuttx/include/arch/irq.h:675 __asm__ __volatile__ 40204598: 47a1 li a5,8 4020459a: 3007b7f3 csrrc a5,mstatus,a5 up_putc(): nuttx/drivers/serial/uart_16550.c:1726 flags = enter_critical_section(); ``` But `mstatus` is not accessible at Supervisor Level! Let's trace this. [`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)... ```c // Disable interrupts and return the previous value of the mstatus register static inline irqstate_t up_irq_save(void) { irqstate_t flags; /* Read mstatus & clear machine interrupt enable (MIE) in mstatus */ __asm__ __volatile__ ( "csrrc %0, " __XSTR(CSR_STATUS) ", %1\n" : "=r" (flags) : "r"(STATUS_IE) : "memory" ); /* Return the previous mstatus value so that it can be restored with * up_irq_restore(). */ return flags; } ``` `CSR_STATUS` is defined in [mode.h](https://github.com/lupyuen2/wip-pinephone-nuttx/blob/star64/arch/risc-v/include/mode.h#L35-L103): ```c #ifdef CONFIG_ARCH_USE_S_MODE # define CSR_STATUS sstatus /* Global status register */ #else # define CSR_STATUS mstatus /* Global status register */ #endif ``` 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). 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... ```bash tools/configure.sh rv-virt:knsh64 ``` And we bypassed Machine Mode Initialisation during startup... 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`? ... } ``` grep for `csr` in `nuttx.S` shows that no more M-Mode Registers are used. Now Critical Section is OK yay! ```text Starting kernel ... clk u5_dw_i2c_clk_core already disabled clk u5_dw_i2c_clk_apb already disabled 123067DFAGHBCIcd ``` - [See the __Build Steps__](https://github.com/lupyuen2/wip-pinephone-nuttx/releases/tag/star64-0.0.1) - [See the __Modified Files__](https://github.com/lupyuen2/wip-pinephone-nuttx/pull/31/files) - [See the __Build Outputs__](https://github.com/lupyuen2/wip-pinephone-nuttx/releases/tag/star64-0.0.1) _What about `satp`, `stvec`, `pmpaddr0`, `pmpcfg0`?_ We'll handle them in a while. Sometimes we see this... ```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 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 Code: 879b 0277 d7b3 00f6 f793 1ff7 078e 95be (b023 0105) ``` Which fails at... ```text 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 ``` TODO: Trace this Store/AMO Access Fault # Hang in UART Transmit Read the article... - ["Star64 JH7110 + NuttX RTOS: RISC-V Privilege Levels and UART Registers"](https://lupyuen.github.io/articles/privilege) When printing to UART Port, the UART Transmit hangs while waiting for UART Transmit Ready... 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! while ((u16550_serialin(priv, UART_LSR_OFFSET) & UART_LSR_THRE) == 0); u16550_serialout(priv, UART_THR_OFFSET, (uart_datawidth_t)ch); } ``` Where [`u16550_serialin`](https://github.com/lupyuen2/wip-pinephone-nuttx/blob/star64a/drivers/serial/uart_16550.c#L596-L611) is defined as... ```c *((FAR volatile uart_datawidth_t *)priv->uartbase + offset); ``` And [`UART_THR_OFFSET`](https://github.com/lupyuen2/wip-pinephone-nuttx/blob/star64a/include/nuttx/serial/uart_16550.h#L197) is... ```c #define UART_LSR_OFFSET (CONFIG_16550_REGINCR*UART_LSR_INCR) ``` `CONFIG_16550_REGINCR` is 1... ```text → grep 16550 .config CONFIG_16550_REGINCR=1 CONFIG_16550_REGWIDTH=8 CONFIG_16550_ADDRWIDTH=0 ``` 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): ```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) ``` _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)... ```text reg = <0x0 0x10000000 0x0 0xl0000>; reg-io-width = <4>; reg-shift = <2>; ``` `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) ```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); } ``` We see that the UART Offset is shifted by 2 (`regshift`). Which means we multiply the UART Offset by 4. Thus `CONFIG_16550_REGINCR` should be 4, not 1! _How to fix CONFIG_16550_REGINCR?_ We fix the NuttX Configuration: Device Drivers > Serial Driver Support > 16550 UART Chip support > Address increment between 16550 registers 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) ```text CONFIG_16550_REGINCR=4 ``` Now UART Transmit doesn't hang yay! ```text Starting kernel ... clk u5_dw_i2c_clk_core already disabled clk u5_dw_i2c_clk_apb already disabled 123067DFm45DTpAqGaclbHm45DTpBqm45DTpCqI ``` NuttX now hangs somewhere in `nx_start` Let's log the NuttX Scheduler... # Enable Scheduler Logging Scheduler Logging in NuttX seems to have changed recently. To enable Scheduler Logging... - `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 - 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... ```text 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 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 ``` _What is `/system/bin/init`?_ We'll find out in a while... [Compare with QEMU Kernel Mode Run Log](https://gist.github.com/lupyuen/6888376da6561bdc060c2459dffdef01) [See the QEMU Kernel Mode Build Log](https://gist.github.com/lupyuen/dce0cdbbf4a4bdf9c79e617b3fe1b679) # Initialise RISC-V Supervisor Mode Read the article... - ["Star64 JH7110 + NuttX RTOS: RISC-V Privilege Levels and UART Registers"](https://lupyuen.github.io/articles/privilege) Earlier we bypassed the Machine Mode and Supervisor Mode Initialisation during NuttX startup... 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`? ... } ``` Now we restore the Supervisor Mode Initialisation, commenting out the Machine Mode Initialisation... 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) { DEBUGASSERT(mhartid == 0); // /* 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); //WRITE_CSR(pmpaddr0, 0x3fffffffffffffull); //WRITE_CSR(pmpcfg0, 0xf); /* Set exception and interrupt delegation for S-mode */ //WRITE_CSR(medeleg, 0xffff); //WRITE_CSR(mideleg, 0xffff); /* Allow to write satp from S-mode */ //CLEAR_CSR(mstatus, MSTATUS_TVM); /* Set mstatus to S-mode and enable SUM */ //CLEAR_CSR(mstatus, ~MSTATUS_MPP_MASK); //SET_CSR(mstatus, MSTATUS_MPPS | SSTATUS_SUM); /* Set the trap vector for S-mode */ WRITE_CSR(stvec, (uintptr_t)__trap_vec); /* Set the trap vector for M-mode */ //WRITE_CSR(mtvec, (uintptr_t)__trap_vec_m); if (0 == mhartid) { /* Only the primary CPU needs to initialize mtimer * before entering to S-mode */ // TODO //up_mtimer_initialize(); } /* Set mepc to the entry */ //WRITE_CSR(mepc, (uintptr_t)qemu_rv_start_s); /* Set a0 to mhartid explicitly and enter to S-mode */ //asm volatile ( // "mv a0, %0 \n" // "mret \n" // :: "r" (mhartid) //); // Jump to S-Mode Init ourselves qemu_rv_start_s(mhartid); // } ``` 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 Now NuttX boots further! ```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 ``` But NuttX crashes. Let's find out why... # QEMU Semihosting in NuttX Read the article... - ["Star64 JH7110 + NuttX RTOS: RISC-V Semihosting and Initial RAM Disk"](https://lupyuen.github.io/articles/semihost) 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". Exception Program Counter `0x4020` `0434` is in RISC-V Semihosting `smh_call`... ```text 0000000040200430 : smh_call(): nuttx/arch/risc-v/src/common/riscv_semihost.S:37 .global smh_call .type smh_call @function smh_call: slli zero, zero, 0x1f 40200430: 01f01013 slli zero,zero,0x1f nuttx/arch/risc-v/src/common/riscv_semihost.S:38 ebreak //// Crashes here (Trigger semihosting breakpoint) 40200434: 00100073 ebreak nuttx/arch/risc-v/src/common/riscv_semihost.S:39 srai zero, zero, 0x7 40200438: 40705013 srai zero,zero,0x7 nuttx/arch/risc-v/src/common/riscv_semihost.S:40 ret 4020043c: 00008067 ret 40200440: 0000 unimp ``` When we log `smh_call`... ```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) 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. (See https://github.com/apache/nuttx/issues/9501) ![QEMU reads the Apps Filesystem over Semihosting](https://lupyuen.github.io/images/semihost-qemu.jpg) Here's the Crash Dump after we disabled Semihosting... ```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 ``` # NuttX Apps Filesystem Read the article... - ["Star64 JH7110 + NuttX RTOS: RISC-V Semihosting and Initial RAM Disk"](https://lupyuen.github.io/articles/semihost) _Where is `/system/bin/init`? Why is it loaded by NuttX over Semihosting?_ `/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... ```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 ``` Which says that `../apps` is mounted as `/system`, via Semihosting HostFS. That's how `/system/bin/init` gets loaded over Semihosting... ``` → ls ../apps/bin getprime hello init sh ``` ![QEMU reads the Apps Filesystem over Semihosting](https://lupyuen.github.io/images/semihost-qemu.jpg) 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 ``` Semihosting won't work on Star64 SBC. Let's replace this with Initial RAM Disk and ROMFS... (See https://github.com/apache/nuttx/issues/9501) # Initial RAM Disk for LiteX Arty-A7 Read the article... - ["Star64 JH7110 + NuttX RTOS: RISC-V Semihosting and Initial RAM Disk"](https://lupyuen.github.io/articles/semihost) Let's modify NuttX for QEMU to mount the Apps Filesystem from an Initial RAM Disk (instead of Semihosting). (So later we can replicate this on Star64 JH7110 SBC) First we look at the Initial RAM Disk for LiteX Arty-A7... [(About NuttX RAM Disks and ROM Disks)](https://cwiki.apache.org/confluence/plugins/servlet/mobile?contentId=139629548#content/view/139629548) 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) ```bash cd nuttx genromfs -f romfs.img -d ../apps/bin -V "NuttXBootVol" ``` [(About `genromfs`)](https://www.systutorials.com/docs/linux/man/8-genromfs/) LiteX Memory Map says where the RAM Disk is loaded... ```text "romfs.img": "0x40C00000", "nuttx.bin": "0x40000000", "opensbi.bin": "0x40f00000" ``` 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) ```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 ``` According to [NSH Start-Up Script](https://nuttx.apache.org/docs/latest/applications/nsh/nsh.html#nsh-start-up-script): ```text CONFIG_DISABLE_MOUNTPOINT not set CONFIG_FS_ROMFS enabled ``` 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) ```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; } ``` `__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 */ } ... /* Page heap */ __pgheap_start = ORIGIN(pgram); __pgheap_size = LENGTH(pgram) + LENGTH(ramdisk); /* Application ramdisk */ __ramdisk_start = ORIGIN(ramdisk); __ramdisk_size = LENGTH(ramdisk); __ramdisk_end = ORIGIN(ramdisk) + LENGTH(ramdisk); ``` Note that `__pgheap_size` needs to include `ramdisk`. Let's do the same to NuttX for QEMU... # Modify NuttX QEMU to Load Initial RAM Disk Read the article... - ["Star64 JH7110 + NuttX RTOS: RISC-V Semihosting and Initial RAM Disk"](https://lupyuen.github.io/articles/semihost) Now we can modify NuttX for QEMU to mount the Apps Filesystem from an Initial RAM Disk instead of Semihosting. (So later we can replicate this on Star64 JH7110 SBC) ![NuttX for QEMU will mount the Apps Filesystem from an Initial RAM Disk](https://lupyuen.github.io/images/semihost-qemu2.jpg) We follow the steps from LiteX Arty-A7 (from the previous section)... 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) ```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 ``` We generate the Initial RAM Disk `initrd`... ```bash cd nuttx genromfs -f initrd -d ../apps/bin -V "NuttXBootVol" ``` [(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 7902208 Jul 21 13:41 initrd ``` 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) ```bash qemu-system-riscv64 \ -semihosting \ -M virt,aclint=on \ -cpu rv64 \ -smp 8 \ -bios none \ -kernel nuttx \ -initrd initrd \ -nographic ``` _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) 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) ```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 */ ramdisk (rwx) : ORIGIN = 0x80800000, LENGTH = 16M /* w/ cache */ /* This won't work, crashes with a Memory Mgmt Fault... ramdisk (rwx) : ORIGIN = 0x84000000, LENGTH = 16M */ /* w/ cache */ } /* 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); ``` (We increased RAM Disk Memory from 4 MB to 16 MB because our RAM Disk is now bigger) 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); ``` [(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) 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); //// ``` Before making the above changes, here's the log for QEMU Kernel Mode with Semihosting... ```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 + qemu-system-riscv64 -semihosting -M virt,aclint=on -cpu rv64 -smp 8 -bios none -kernel nuttx -initrd initrd -nographic 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 ``` Now we run QEMU Kernel Mode with Initial RAM Disk, without Semihosting... And it boots OK on QEMU yay! [See the Run Log](https://gist.github.com/lupyuen/8afee5b07b61bb7f9f202f7f8c5e3ab3) # Modify NuttX Star64 to Load Initial RAM Disk Read the article... - ["Star64 JH7110 + NuttX RTOS: RISC-V Semihosting and Initial RAM Disk"](https://lupyuen.github.io/articles/semihost) Finally we can modify NuttX for Star64 JH7110 RISC-V SBC to mount the Apps Filesystem from an Initial RAM Disk. (Instead of Semihosting) ![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) 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. (For a detailed explanation of the modified files, see the previous section) 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`... - 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 - Manually delete from [`knsh64/defconfig`](https://github.com/lupyuen2/wip-pinephone-nuttx/blob/star64c/boards/risc-v/qemu-rv/rv-virt/configs/knsh64/defconfig)... ```text CONFIG_HOST_MACOS=y CONFIG_INIT_MOUNT_DATA="fs=../apps" CONFIG_INIT_MOUNT_FSTYPE="hostfs" CONFIG_INIT_MOUNT_SOURCE="" ``` Updated Build Configuration: [knsh64/defconfig](https://github.com/lupyuen2/wip-pinephone-nuttx/blob/star64c/boards/risc-v/qemu-rv/rv-virt/configs/knsh64/defconfig) _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) Which means that we need to add these TFTP Commands to U-Boot Bootloader... ```bash ## 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 ## 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 ## Changed this: Replaced `-` by `ramdisk_addr_r:ramdisk_size` booti ${kernel_addr_r} ${ramdisk_addr_r}:${ramdisk_size} ${fdt_addr_r} ``` 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 ## ramdisk_size=0x1000000 ## fdt_addr_r=0x46000000 booti ${kernel_addr_r} ${ramdisk_addr_r}:${ramdisk_size} ${fdt_addr_r}; fi; fi; fi; fi ``` Which becomes... ```bash ## 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 ## 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 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' ## Check that it's correct printenv bootcmd_tftp ## Save it for future reboots saveenv ``` _What happens if we omit the RAM Disk Size?_ ```text $ booti ${kernel_addr_r} ${ramdisk_addr_r} ${fdt_addr_r} Wrong Ramdisk Image Format Ramdisk image is corrupt or invalid ## Assume max 16 MB $ booti ${kernel_addr_r} ${ramdisk_addr_r}:0x1000000 ${fdt_addr_r} ## Boots OK ``` _Does the Initial RAM Disk work on Star64?_ Star64 JH7110 boots OK with the Initial RAM Disk yay! ```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 ``` TODO: Why no shell? TODO: Why `nx_start_application: ret=3`? # Increase RAM Disk Limit Note that the Initial RAM Disk (initrd) is limited to __16 MB__, because of the Linker Script: [ld.script](https://github.com/apache/nuttx/blob/master/boards/risc-v/jh7110/star64/scripts/ld.script) ```text MEMORY { kflash (rx) : ORIGIN = 0x40200000, LENGTH = 2048K /* w/ cache */ ksram (rwx) : ORIGIN = 0x40400000, LENGTH = 2048K /* w/ cache */ pgram (rwx) : ORIGIN = 0x40600000, LENGTH = 4096K /* w/ cache */ ramdisk (rwx) : ORIGIN = 0x40A00000, LENGTH = 16M /* w/ cache */ } ``` _If initrd exceeds 16 MB, how to increase the RAM Disk Limit?_ Edit the Linker Script above and change the RAM Disk Length accordingly. If we're booting NuttX over TFTP, remember to update the U-Boot Script... ```bash ## 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 ``` [(Source)](https://lupyuen.github.io/articles/tftp#configure-u-boot-for-tftp) `ramdisk_size` needs to be increased also. TODO: Can we fix the U-Boot Script so that it doesn't hardcode the RAM Disk Size? # Memory Map for RAM Disk Note that the RAM Disk is mapped into the Page Heap (`__pgheap_size`) in the Linker Script... From [ld.script](https://github.com/apache/nuttx/blob/master/boards/risc-v/jh7110/star64/scripts/ld.script): ```text MEMORY { kflash (rx) : ORIGIN = 0x40200000, LENGTH = 2048K /* w/ cache */ ksram (rwx) : ORIGIN = 0x40400000, LENGTH = 2048K /* w/ cache */ pgram (rwx) : ORIGIN = 0x40600000, LENGTH = 4096K /* w/ cache */ ramdisk (rwx) : ORIGIN = 0x40A00000, LENGTH = 16M /* w/ cache */ } /* Note: Page Heap includes RAM Disk */ __pgheap_start = ORIGIN(pgram); __pgheap_size = LENGTH(pgram) + LENGTH(ramdisk); /* Application ramdisk */ __ramdisk_start = ORIGIN(ramdisk); __ramdisk_size = LENGTH(ramdisk); __ramdisk_end = ORIGIN(ramdisk) + LENGTH(ramdisk); ``` _Won't the RAM Disk get incorrectly allocated to NuttX Apps?_ Nope, that won't happen because the Actual Page Heap Size (without RAM Disk) is defined by `CONFIG_ARCH_PGPOOL_SIZE` in the NuttX Config... From [nsh/defconfig](https://github.com/apache/nuttx/blob/master/boards/risc-v/jh7110/star64/configs/nsh/defconfig#L33-L34): ```bash CONFIG_ARCH_PGPOOL_SIZE=4194304 CONFIG_ARCH_PGPOOL_VBASE=0x40600000 ``` This says that the Actual Page Heap Size is 4 MB (excluding the RAM Disk). Let's increase the Page Heap Size... TODO: Folks might forget to update `CONFIG_ARCH_PGPOOL_SIZE`. Let's verify that... ```c DEBUGASSERT(__pgheap_size == CONFIG_ARCH_PGPOOL_SIZE + __ramdisk_size) ``` # Increase Page Heap Size _NuttX Apps will allocate Dynamic Memory (malloc) from the NuttX Page Heap..._ _Why does NuttX report that the Page Heap Size is 20 MB?_ ```text NuttShell (NSH) NuttX-12.0.3 nsh> free total used free largest nused nfree Kmem: 2065400 14600 2050800 2049440 50 3 Page: 20971520 643072 20328448 20328448 ``` [(Source)](https://gist.github.com/lupyuen/fe062fe61a646c465329b80b1fe5fcac#file-nuttx-heap-log-L161-L165) That's because the 20 MB (`__pgheap_size`) includes the 4 MB Page Heap + 16 MB RAM Disk, as defined in the Linker Script... From [ld.script](https://github.com/apache/nuttx/blob/master/boards/risc-v/jh7110/star64/scripts/ld.script): ```text MEMORY { kflash (rx) : ORIGIN = 0x40200000, LENGTH = 2048K /* w/ cache */ ksram (rwx) : ORIGIN = 0x40400000, LENGTH = 2048K /* w/ cache */ pgram (rwx) : ORIGIN = 0x40600000, LENGTH = 4096K /* w/ cache */ ramdisk (rwx) : ORIGIN = 0x40A00000, LENGTH = 16M /* w/ cache */ } /* Note: Page Heap includes RAM Disk */ __pgheap_start = ORIGIN(pgram); __pgheap_size = LENGTH(pgram) + LENGTH(ramdisk); ``` By default the Page Heap Size is actually 4 MB, as defined by `CONFIG_ARCH_PGPOOL_SIZE` in the NuttX Config... From [nsh/defconfig](https://github.com/apache/nuttx/blob/master/boards/risc-v/jh7110/star64/configs/nsh/defconfig#L33-L34): ```bash CONFIG_ARCH_PGPOOL_SIZE=4194304 CONFIG_ARCH_PGPOOL_VBASE=0x40600000 ``` _How to increase the Page Heap Size, so that NuttX Apps will have more Dynamic Memory (malloc)?_ This is how we increase the Page Heap Size from 4 MB to 16 MB... From [ld.script](https://github.com/lupyuen2/wip-pinephone-nuttx/blob/malloc/boards/risc-v/jh7110/star64/scripts/ld.script#L25-L26): ```text /* Previously: LENGTH = 4096K */ pgram (rwx) : ORIGIN = 0x40600000, LENGTH = 16M /* w/ cache */ /* Previously: ORIGIN = 0x40A00000 */ ramdisk (rwx) : ORIGIN = 0x41600000, LENGTH = 16M /* w/ cache */ ``` Note that the RAM Disk Origin shifts down to make room for the increased Page Heap size. (0x41600000 = 0x40600000 + 16 MB) Then we update `CONFIG_ARCH_PGPOOL_SIZE` in the NuttX Config... From [nsh/defconfig](https://github.com/lupyuen2/wip-pinephone-nuttx/blob/malloc/boards/risc-v/jh7110/star64/configs/nsh/defconfig#L33-L34): ```bash ## Previously 4194304 (4 MB) CONFIG_ARCH_PGPOOL_SIZE=16777216 CONFIG_ARCH_PGPOOL_VBASE=0x40600000 ``` We reconfigure NuttX and rebuild... ```bash make distclean tools/configure.sh star64:nsh make ## Remember to rebuild the Initial RAM Disk (initrd) ``` Now NuttX reports that the Free Memory is 32 MB (16 MB Page Heap + 16 MB RAM Disk)... ```text NuttShell (NSH) NuttX-12.0.3 nsh> free total used free largest nused nfree Kmem: 2065400 15000 2050400 2049040 50 4 Page: 33554432 643072 32911360 32911360 ``` [(Source)](https://gist.github.com/lupyuen/83980edd8d970d5530070fd78f6b0242#file-nuttx-heap-increase-ok-log-L162-L166) Let's test to see if it works... # Test the Page Heap _In the previous section we increased the NuttX Page Heap Size from 4 MB to 16 MB, so that NuttX Apps will have more Dynamic Memory (malloc)..._ _How do we test this?_ We run a loop in our `hello` app to allocate 8 KB blocks repeatedly... From [hello_main.c](https://github.com/lupyuen2/wip-pinephone-nuttx-apps/blob/malloc/examples/hello/hello_main.c#L41-L47): ```c for (i = 0; ; i++) { void *p = malloc(8192); if (p == NULL) { break; } if (i % 10 == 0) { printf("i=%d\n", i); } } printf("malloc failed at i=%d\n", i); ``` We run the Heap Test Code before and after increasing the Page Heap Size... __Before: Page Heap Size was 4 MB__ The `hello` app crashes after allocating 160-plus blocks of 8 KB... ```text nsh> hello ... i=160 i=riscv_exception: EXCEPTION: Store/AMO access fault. MCAUSE: 0000000000000007, EPC: 0000000040206652, MTVAL: 0000000000000000 riscv_exception: PANIC!!! Exception = 0000000000000007 ``` [(Source)](https://gist.github.com/lupyuen/fe062fe61a646c465329b80b1fe5fcac#file-nuttx-heap-log-L184-L188) Which is around 1.2 MB. __After: Page Heap Size is now 16 MB__ The `hello` app crashes after allocating 1,603-plus blocks of 8 KB... ```text nsh> hello ... i=1630 iriscv_exception: EXCEPTION: Store/AMO access fault. MCAUSE: 0000000000000007, EPC: 0000000040206652, MTVAL: 0000000000000000 ``` [(Source)](https://gist.github.com/lupyuen/83980edd8d970d5530070fd78f6b0242#file-nuttx-heap-increase-ok-log-L332-L333) Which is around 13 MB. So yep the increase in Page Heap Size works! _Why does it crash?_ Based on the [Crash Dump](https://gist.github.com/lupyuen/83980edd8d970d5530070fd78f6b0242#file-nuttx-heap-increase-ok-log-L332-L333 )... ```text riscv_exception: EXCEPTION: Store/AMO access fault MCAUSE: 7 EPC: 40206652 MTVAL: 0 ``` The Exception Program Counter (EPC) says 0x40206652, which is inside the code for `memset()`. (We looked up the NuttX Disassemly) MTVAL is 0, which means that `memset()` crashed while accessing a Null Pointer. Maybe `printf` tried to allocate a Dynamic Buffer and failed? TODO: NuttX QEMU doesn't crash when we run the same Heap Test, the app exits gracefully to the NuttX Shell. Might need more investigation. [(See this)](https://github.com/lupyuen2/wip-pinephone-nuttx-apps/blob/malloc2/examples/hello/hello_main.c#L53-L73) # Increase Stack Size The Default Stack Sizes for NuttX Kernel and NuttX Apps are rather small. This might crash our NuttX Apps when they run out of Stack Space... ```text STACKSIZE USED FILLED COMMAND 2048 2040 99.6%! irq 3056 1808 59.1% Idle_Task 1968 752 38.2% lpwork 0x802015f0 0x80201618 3008 744 24.7% /system/bin/init 2000 2000 100.0%! scheme ``` Let's increase the Interrupt and App Stack Sizes. _But we already set [CONFIG_YOURAPPNAME_STACKSIZE=8192](https://github.com/KenDickey/nuttx-umb-scheme/blob/main/Kconfig)!_ Apparently it doesn't work. Maybe because we're running in NuttX Kernel Mode? (Instead of NuttX Flat Mode) This is how we increase the App Stack Size to 8192 in `.config`... ```bash CONFIG_POSIX_SPAWN_DEFAULT_STACKSIZE=8192 ``` After increasing the App Stack Size, our App no longer crashes! - [Before Setting CONFIG_POSIX_SPAWN_DEFAULT_STACKSIZE](https://gist.github.com/lupyuen/5c225a3a30086cd35455463955f5ff64) - [After Setting CONFIG_POSIX_SPAWN_DEFAULT_STACKSIZE](https://gist.github.com/lupyuen/572e6018ed982fe42b2b5ed40ffae505) - [Scheme runs OK on Star64 yay!](https://gist.github.com/lupyuen/1e009a3343da70257d6f24400339053f) This is how we increase the Default Stack Size, Interrupt Stack Size and other Kernel Stack Sizes... From [nsh/defconfig](https://github.com/lupyuen2/wip-pinephone-nuttx/commit/5441201b82755f979958e8b33a955554bbe345b6) ```bash CONFIG_ARCH_INTERRUPTSTACK=8192 CONFIG_ARCH_KERNEL_STACKSIZE=8192 CONFIG_DEFAULT_TASK_STACKSIZE=8192 CONFIG_IDLETHREAD_STACKSIZE=8192 ## Use Default Values: ## CONFIG_INIT_STACKSIZE=8192 ## CONFIG_POSIX_SPAWN_DEFAULT_STACKSIZE=8192 ``` The log shows that the Interrupt and Kernel Stacks have been increased... ```text STACKSIZE USED FILLED COMMAND 8192 8184 99.9%! irq 8176 1824 22.3% Idle_Task 8112 720 8.8% lpwork 0x80202df0 0x80202e18 8128 848 10.4% /system/bin/init ``` TODO: Why is Interrupt Stack full again? _How do we search menuconfig for a setting like DEFAULT_TASK_STACKSIZE?_ Normally we don't edit `.config` directly. We do it in menuconfig like this... 1. Run `make menuconfig` 1. Press "`/`" to search 1. Enter `DEFAULT_TASK_STACKSIZE` (or a substring) We should see the location of the setting (Like "RTOS Features > Stack and Heap Information") 1. To update the NuttX Build Config: ```bash make savedefconfig \ && grep -v CONFIG_HOST defconfig \ >boards/risc-v/jh7110/star64/configs/nsh/defconfig ``` # No UART Output from NuttX Shell Read the article... - ["Star64 JH7110 + NuttX RTOS: RISC-V PLIC Interrupts and Serial I/O"](https://lupyuen.github.io/articles/plic) From the previous section, we found out that NuttX Shell didn't appear on Star64 JH7110 SBC. When we log [`uart_write`](https://github.com/lupyuen2/wip-pinephone-nuttx/blob/ramdisk2/drivers/serial/serial.c#L1172-L1341), we see that the NuttX Shell is actually started! ```text uart_write (0xc000a610): 0000 0a 4e 75 74 74 53 68 65 6c 6c 20 28 4e 53 48 29 .NuttShell (NSH) 0010 20 4e 75 74 74 58 2d 31 32 2e 30 2e 33 0a NuttX-12.0.3. uart_write (0xc0015338): 0000 6e 73 68 3e 20 nsh> uart_write (0xc0015310): 0000 1b 5b 4b .[K ``` Just that the NuttX Shell couldn't produce any UART Output. (This happens to all NuttX Apps, but not to NuttX Kernel) Let's find out why, by tracing the UART Output in NuttX QEMU... # UART Output in NuttX QEMU Read the article... - ["Star64 JH7110 + NuttX RTOS: RISC-V PLIC Interrupts and Serial I/O"](https://lupyuen.github.io/articles/plic) To understand how UART Output (`printf`) works in NuttX Apps (and NuttX Shell), we add logs to NuttX QEMU... ```text ABCnx_start: Entry up_irq_enable: up_enable_irq: irq=17 up_enable_irq: RISCV_IRQ_SOFT=17 uart_register: Registering /dev/console uart_register: Registering /dev/ttyS0 up_enable_irq: irq=35 up_enable_irq: extirq=10, RISCV_IRQ_EXT=25 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 up_exit: TCB=0x802088d0 exiting $%&riscv_doirq: irq=8 $%&riscv_doirq: irq=8 $%&riscv_doirq: irq=8 ... $%&riscv_doirq: irq=8 $%&riscv_doirq: irq=8 $%&riscv_doirq: irq=8 uart_write (0xc0200428): 0000 2a 2a 2a 6d 61 69 6e 0a ***main. FAAAAAAAADEF*F*F*FmFaFiFnF $%&riscv_doirq: irq=8 $%&riscv_doirq: irq=8 $%&riscv_doirq: irq=8 ... $%&riscv_doirq: irq=8 $%&riscv_doirq: irq=8 $%&riscv_doirq: irq=8 uart_write (0xc000a610): 0000 0a 4e 75 74 74 53 68 65 6c 6c 20 28 4e 53 48 29 .NuttShell (NSH) 0010 20 4e 75 74 74 58 2d 31 32 2e 30 2e 33 0a NuttX-12.0.3. FAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAADEF FNFuFtFtFSFhFeFlFlF F(FNFSFHF)F FNFuFtFtFXF-F1F2F.F0F.F3F $%&riscv_doirq: irq=8 uart_write (0xc0015340): 0000 6e 73 68 3e 20 nsh> AAAAADEFnFsFhF>F $%&riscv_doirq: irq=8 uart_write (0xc0015318): 0000 1b 5b 4b .[K AAADEF[FK$%&riscv_doirq: irq=8 nx_start: CPU0: Beginning Idle Loop $%^&riscv_doirq: irq=35 #*ADEFa$%&riscv_doirq: irq=8 $%^&riscv_doirq: irq=35 #*ADEFa$%&riscv_doirq: irq=8 $%^&riscv_doirq: irq=35 #*ADEFa$%&riscv_doirq: irq=8 ``` This says that NuttX Apps call [`uart_write`](https://github.com/lupyuen2/wip-pinephone-nuttx/blob/ramdisk2/drivers/serial/serial.c#L1172-L1341), which calls... - `A`: [`uart_putxmitchar`](https://github.com/lupyuen2/wip-pinephone-nuttx/blob/ramdisk2/drivers/serial/serial.c#L150-L286) which calls... - `D`: [`uart_xmitchars`](https://github.com/lupyuen2/wip-pinephone-nuttx/blob/ramdisk2/drivers/serial/serial_io.c#L42-L107) which calls... - `E`: [`uart_txready`](https://github.com/lupyuen2/wip-pinephone-nuttx/blob/ramdisk2/drivers/serial/serial_io.c#L63-L68) and... `F`: [`u16550_send`](https://github.com/lupyuen2/wip-pinephone-nuttx/blob/ramdisk2/drivers/serial/uart_16550.c#L1542-L1556) When we type something, the UART Input will trigger an Interrupt... (Also for NuttX Apps calling a System Function in NuttX Kernel) - `$`: [`exception_common`](https://github.com/lupyuen2/wip-pinephone-nuttx/blob/ramdisk2/arch/risc-v/src/common/riscv_exception_common.S#L63-L189) calls... - `%^&`: [`riscv_dispatch_irq`](https://github.com/lupyuen2/wip-pinephone-nuttx/blob/ramdisk2/arch/risc-v/src/qemu-rv/qemu_rv_irq_dispatch.c#L51-L92) which calls... - [`riscv_doirq`](https://github.com/lupyuen2/wip-pinephone-nuttx/blob/ramdisk2/arch/risc-v/src/common/riscv_doirq.c#L58-L131) which calls... - `#`: [`u16550_interrupt`](https://github.com/lupyuen2/wip-pinephone-nuttx/blob/ramdisk2/drivers/serial/uart_16550.c#L918-L1021) _What is `riscv_doirq: irq=35`?_ This is the Interrupt triggered by UART Input. QEMU UART is at [RISC-V IRQ 10](https://github.com/lupyuen/nuttx-star64/blob/main/qemu-riscv64.dts#L225-L226), which becomes NuttX IRQ 35 (10 + 25). [(RISCV_IRQ_EXT = RISCV_IRQ_SEXT = 16 + 9 = 25)](https://github.com/lupyuen2/wip-pinephone-nuttx/blob/ramdisk2/arch/risc-v/include/irq.h#L75-L86) _Why so many `riscv_doirq: irq=8`?_ NuttX IRQ 8 is [`RISCV_IRQ_ECALLU`](https://github.com/lupyuen2/wip-pinephone-nuttx/blob/ramdisk2/arch/risc-v/include/irq.h#L52-L74): ECALL from RISC-V User Mode to Supervisor Mode. This happens when the NuttX App (User Mode) calls a System Function in NuttX Kernel (Supervisor Mode). ![UART Output in NuttX QEMU](https://lupyuen.github.io/images/plic-qemu.png) Now we compare the above with Star64... # Compare UART Output: Star64 vs QEMU Read the article... - ["Star64 JH7110 + NuttX RTOS: RISC-V PLIC Interrupts and Serial I/O"](https://lupyuen.github.io/articles/plic) In the previous section we added logs to UART I/O in NuttX QEMU. We add the same logs to NuttX Star64 and compare... ```text 123067BCnx_start: Entry up_irq_enable: up_enable_irq: irq=17 up_enable_irq: RISCV_IRQ_SOFT=17 uart_register: Registering /dev/console uart_register: Registering /dev/ttyS0 up_enable_irq: irq=57 up_enable_irq: extirq=32, RISCV_IRQ_EXT=25 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 $%&riscv_doirq: irq=8 $%&riscv_doirq: irq=8 $%&riscv_doirq: irq=8 ... $%&riscv_doirq: irq=8 $%&riscv_doirq: irq=8 $%&riscv_doirq: irq=8 uart_write (0xc0200428): 0000 2a 2a 2a 6d 61 69 6e 0a ***main. AAAAAAAAAD$%&riscv_doirq: irq=8 $%&riscv_doirq: irq=8 $%&riscv_doirq: irq=8 $%&riscv_doirq: irq=8 ... $%&riscv_doirq: irq=8 $%&riscv_doirq: irq=8 $%&riscv_doirq: irq=8 uart_write (0xc000a610): 0000 0a 4e 75 74 74 53 68 65 6c 6c 20 28 4e 53 48 29 .NuttShell (NSH) 0010 20 4e 75 74 74 58 2d 31 32 2e 30 2e 33 0a NuttX-12.0.3. AAAAAAAAAAAAAAAriscv_doirq: irq=8 uart_write (0xc0015338): 0000 6e 73 68 3e 20 nsh> AAAAAD$%&riscv_doirq: irq=8 uart_write (0xc0015310): 0000 1b 5b 4b .[K AAAD$%&riscv_doirq: irq=8 nx_start: CPU0: Beginning Idle Loop ``` From the previous section, we know that [`uart_write`](https://github.com/lupyuen2/wip-pinephone-nuttx/blob/ramdisk2/drivers/serial/serial.c#L1172-L1341), should call... - `A`: [`uart_putxmitchar`](https://github.com/lupyuen2/wip-pinephone-nuttx/blob/ramdisk2/drivers/serial/serial.c#L150-L286) which calls... - `D`: [`uart_xmitchars`](https://github.com/lupyuen2/wip-pinephone-nuttx/blob/ramdisk2/drivers/serial/serial_io.c#L42-L107) which calls... - `E`: [`uart_txready`](https://github.com/lupyuen2/wip-pinephone-nuttx/blob/ramdisk2/drivers/serial/serial_io.c#L63-L68) and... `F`: [`u16550_send`](https://github.com/lupyuen2/wip-pinephone-nuttx/blob/ramdisk2/drivers/serial/uart_16550.c#L1542-L1556) BUT from the above Star64 Log, we see that [`uart_txready`](https://github.com/lupyuen2/wip-pinephone-nuttx/blob/ramdisk2/drivers/serial/serial_io.c#L63-L68) is NOT Ready. That's why NuttX Star64 doesn't call [`u16550_send`](https://github.com/lupyuen2/wip-pinephone-nuttx/blob/ramdisk2/drivers/serial/uart_16550.c#L1542-L1556) to print the output. _Is our [__Interrupt Controller__](https://github.com/lupyuen2/wip-pinephone-nuttx/blob/star64c/arch/risc-v/src/qemu-rv/hardware/qemu_rv_memorymap.h#L27-L33) OK?_ NuttX Star64 doesn't respond to UART Input. We'll check why in a while. [(See the __JH7110 U74 Memory Map__)](https://doc-en.rvspace.org/JH7110/TRM/JH7110_TRM/u74_memory_map.html) _Is the UART IRQ Number correct?_ Star64 UART is [RISC-V IRQ 32](https://doc-en.rvspace.org/VisionFive2/DG_UART/JH7110_SDK/general_uart_controller.html), which becomes NuttX IRQ 57 (32 + 25). [(RISCV_IRQ_EXT = RISCV_IRQ_SEXT = 16 + 9 = 25)](https://github.com/lupyuen2/wip-pinephone-nuttx/blob/ramdisk2/arch/risc-v/include/irq.h#L75-L86) ```bash CONFIG_16550_UART0_IRQ=57 ``` [(Source)](https://github.com/lupyuen2/wip-pinephone-nuttx/blob/star64d/boards/risc-v/qemu-rv/rv-virt/configs/knsh64/defconfig#L10-L17) Also from [JH7110 Interrupt Connections](https://doc-en.rvspace.org/JH7110/TRM/JH7110_TRM/interrupt_connections.html): `u0_uart` is at `global_interrupts[27]` Which is correct because [SiFive U74-MC Core Complex Manual](https://starfivetech.com/uploads/u74mc_core_complex_manual_21G1.pdf) (Page 198) says that `global_interrupts[0]` is PLIC Interrupt ID 5. Thus `u0_uart`(IRQ 32) is at `global_interrupts[27]`. _Is it the same UART IRQ as Linux?_ We check the Linux Device Tree... ```text dtc \ -o jh7110-visionfive-v2.dts \ -O dts \ -I dtb \ jh7110-visionfive-v2.dtb ``` Which produces [jh7110-visionfive-v2.dts](https://github.com/lupyuen/nuttx-star64/blob/main/jh7110-visionfive-v2.dts) UART0 is indeed RISC-V IRQ 32: [jh7110-visionfive-v2.dts](https://github.com/lupyuen/nuttx-star64/blob/main/jh7110-visionfive-v2.dts#L619-L631) ```text serial@10000000 { compatible = "snps,dw-apb-uart"; reg = <0x00 0x10000000 0x00 0x10000>; reg-io-width = <0x04>; reg-shift = <0x02>; clocks = <0x08 0x92 0x08 0x91>; clock-names = "baudclk\0apb_pclk"; resets = <0x21 0x53 0x21 0x54>; interrupts = <0x20>; status = "okay"; pinctrl-names = "default"; pinctrl-0 = <0x24>; }; ``` _Maybe the IRQ Numbers are different for NuttX vs Linux?_ We tried to enable a whole bunch of IRQs, but nothing got triggered... ```text up_enable_irq: irq=26 up_enable_irq: extirq=1, RISCV_IRQ_EXT=25 up_enable_irq: irq=27 up_enable_irq: extirq=2, RISCV_IRQ_EXT=25 up_enable_irq: irq=28 up_enable_irq: extirq=3, RISCV_IRQ_EXT=25 up_enable_irq: irq=29 ... up_enable_irq: irq=86 up_enable_irq: extirq=61, RISCV_IRQ_EXT=25 up_enable_irq: irq=87 up_enable_irq: extirq=62, RISCV_IRQ_EXT=25 up_enable_irq: irq=88 up_enable_irq: extirq=63, RISCV_IRQ_EXT=25 ``` So there's definitely a problem with our Interrupt Controller. _Maybe IRQ 32 is too high? (QEMU IRQ is only 10)_ [JH7110 Interrupt Connections](https://doc-en.rvspace.org/JH7110/TRM/JH7110_TRM/interrupt_connections.html) says that Global Interrupts are numbered 0 to 126 (127 total interrupts). That's a lot more than NuttX QEMU can handle. Let's fix NuttX Star64 to support more IRQs. From [qemu-rv/irq.h](https://github.com/lupyuen2/wip-pinephone-nuttx/blob/star64d/arch/risc-v/include/qemu-rv/irq.h#L31-L40): ```c /* Map RISC-V exception code to NuttX IRQ */ //// "JH7110 Interrupt Connections" says that Global Interrupts are 0 to 126 (127 total interrupts) //// https://doc-en.rvspace.org/JH7110/TRM/JH7110_TRM/interrupt_connections.html #define NR_IRQS (RISCV_IRQ_SEXT + 127) // Previously: ////#define QEMU_RV_IRQ_UART0 (RISCV_IRQ_MEXT + 10) ////#define NR_IRQS (QEMU_RV_IRQ_UART0 + 1) ``` From [qemu_rv_irq.c](https://github.com/lupyuen2/wip-pinephone-nuttx/blob/star64d/arch/risc-v/src/qemu-rv/qemu_rv_irq.c#L46-L72): ```c void up_irqinitialize(void) { ... /* Set priority for all global interrupts to 1 (lowest) */ int id; ////TODO: Why 52 PLIC Interrupts? for (id = 1; id <= NR_IRQS; id++) //// Changed 52 to NR_IRQS { putreg32(1, (uintptr_t)(QEMU_RV_PLIC_PRIORITY + 4 * id)); } ``` This is hardcoded to 64 IRQs, we should fix in future: [qemu_rv_irq.c](https://github.com/lupyuen2/wip-pinephone-nuttx/blob/star64d/arch/risc-v/src/qemu-rv/qemu_rv_irq.c#L143-L198) ```c void up_enable_irq(int irq) { ... else if (irq > RISCV_IRQ_EXT) { extirq = irq - RISCV_IRQ_EXT; _info("extirq=%d, RISCV_IRQ_EXT=%d\n", extirq, RISCV_IRQ_EXT);//// /* Set enable bit for the irq */ if (0 <= extirq && extirq <= 63) ////TODO: Why 63? { modifyreg32(QEMU_RV_PLIC_ENABLE1 + (4 * (extirq / 32)), 0, 1 << (extirq % 32)); } ``` Now we study the NuttX Code for Platform-Level Interrupt Controller... # Platform-Level Interrupt Controller for Star64 Read the article... - ["Star64 JH7110 + NuttX RTOS: RISC-V PLIC Interrupts and Serial I/O"](https://lupyuen.github.io/articles/plic) The Platform-Level Interrupt Controller (PLIC) handles Global Interrupts triggered by Peripherals (like UART). (PLIC works like Arm's Global Interrupt Controller) We update the [NuttX PLIC Code](https://github.com/lupyuen2/wip-pinephone-nuttx/blob/star64d/arch/risc-v/src/qemu-rv/qemu_rv_irq.c#L45-L214) based on these docs... - [SiFive U74-MC Core Complex Manual](https://starfivetech.com/uploads/u74mc_core_complex_manual_21G1.pdf) - [PLIC Spec](https://github.com/riscv/riscv-plic-spec/blob/master/riscv-plic.adoc) ![PLIC in JH7110 (U74) SoC](https://lupyuen.github.io/images/plic-title.jpg) _How to configure PLIC to forward Interrupts to the Harts?_ The PLIC Memory Map is below... From [SiFive U74-MC Core Complex Manual](https://starfivetech.com/uploads/u74mc_core_complex_manual_21G1.pdf) Page 193 (PLIC Memory Map) | Address | Width | Attr | Description |---------|-------|------|------------ | 0x0C00_0004 | 4B | RW | Source 1 priority | 0x0C00_0220 | 4B | RW | Source 136 priority | 0x0C00_1000 | 4B | RO | Start of pending array | 0x0C00_1010 | 4B | RO | Last word of pending array | 0x0C00_2100 | 4B | RW | Start Hart 1 S-Mode interrupt enables | 0x0C00_2110 | 4B | RW | End Hart 1 S-Mode interrupt enables | 0x0C00_2200 | 4B | RW | Start Hart 2 S-Mode interrupt enables | 0x0C00_2210 | 4B | RW | End Hart 2 S-Mode interrupt enables | 0x0C00_2300 | 4B | RW | Start Hart 3 S-Mode interrupt enables | 0x0C00_2310 | 4B | RW | End Hart 3 S-Mode interrupt enables | 0x0C00_2400 | 4B | RW | Start Hart 4 S-Mode interrupt enables | 0x0C00_2410 | 4B | RW | End Hart 4 S-Mode interrupt enables | 0x0C20_2000 | 4B | RW | Hart 1 S-Mode priority threshold | 0x0C20_2004 | 4B | RW | Hart 1 S-Mode claim/complete | 0x0C20_4000 | 4B | RW | Hart 2 S-Mode priority threshold | 0x0C20_4004 | 4B | RW | Hart 2 S-Mode claim/complete | 0x0C20_6000 | 4B | RW | Hart 3 S-Mode priority threshold | 0x0C20_6004 | 4B | RW | Hart 3 S-Mode claim/complete | 0x0C20_8000 | 4B | RW | Hart 4 S-Mode priority threshold | 0x0C20_8004 | 4B | RW | Hart 4 S-Mode claim/complete There are 5 Harts in JH7110... - __Hart 0:__ S7 Core (the limited core, unused) - __Harts 1 to 4:__ U7 Cores (the full cores) According to OpenSBI, we are now running on Hart 1. (Sounds right) (We pass the Hart ID to NuttX as Hart 0, since NuttX expects Hart ID to start at 0) Based on the above PLIC Memory Map, we fix the PLIC Addresses in NuttX to use Hart 1: [qemu_rv_plic.h](https://github.com/lupyuen2/wip-pinephone-nuttx/blob/star64d/arch/risc-v/src/qemu-rv/hardware/qemu_rv_plic.h#L33-L59) ```c // | 0x0C00_0004 | 4B | RW | Source 1 priority #define QEMU_RV_PLIC_PRIORITY (QEMU_RV_PLIC_BASE + 0x000000) // | 0x0C00_1000 | 4B | RO | Start of pending array #define QEMU_RV_PLIC_PENDING1 (QEMU_RV_PLIC_BASE + 0x001000) // Previously: // #define QEMU_RV_PLIC_PRIORITY (QEMU_RV_PLIC_BASE + 0x000000) // #define QEMU_RV_PLIC_PENDING1 (QEMU_RV_PLIC_BASE + 0x001000) #ifdef CONFIG_ARCH_USE_S_MODE // | 0x0C00_2100 | 4B | RW | Start Hart 1 S-Mode interrupt enables # define QEMU_RV_PLIC_ENABLE1 (QEMU_RV_PLIC_BASE + 0x002100) # define QEMU_RV_PLIC_ENABLE2 (QEMU_RV_PLIC_BASE + 0x002104) // | 0x0C20_2000 | 4B | RW | Hart 1 S-Mode priority threshold # define QEMU_RV_PLIC_THRESHOLD (QEMU_RV_PLIC_BASE + 0x202000) // | 0x0C20_2004 | 4B | RW | Hart 1 S-Mode claim/complete # define QEMU_RV_PLIC_CLAIM (QEMU_RV_PLIC_BASE + 0x202004) // Previously: // # define QEMU_RV_PLIC_ENABLE1 (QEMU_RV_PLIC_BASE + 0x002080) // # define QEMU_RV_PLIC_ENABLE2 (QEMU_RV_PLIC_BASE + 0x002084) // # define QEMU_RV_PLIC_THRESHOLD (QEMU_RV_PLIC_BASE + 0x201000) // # define QEMU_RV_PLIC_CLAIM (QEMU_RV_PLIC_BASE + 0x201004) ``` _What about the PLIC Base Address?_ According to [U74 Memory Map](https://doc-en.rvspace.org/JH7110/TRM/JH7110_TRM/u74_memory_map.html), the Base Addresses are: ```text 0x00_0200_0000 0x00_0200_FFFF RW A CLINT 0x00_0C00_0000 0x00_0FFF_FFFF RW A PLIC ``` Which are correct in NuttX: [qemu_rv_memorymap.h](https://github.com/lupyuen2/wip-pinephone-nuttx/blob/star64d/arch/risc-v/src/qemu-rv/hardware/qemu_rv_memorymap.h#L30-L32) ```c #define QEMU_RV_CLINT_BASE 0x02000000 #define QEMU_RV_PLIC_BASE 0x0c000000 ``` Note that there's a Core-Local Interruptor (CLINT) that handles Software Interrupt and Timer Interrupt... ![PLIC and CLINT in JH7110 (U74) SoC](https://lupyuen.github.io/images/plic-clint.jpg) TODO: Do we need to handle CLINT? Let's check that the RISC-V Interrupts are delegated correctly... # Delegate Machine-Mode Interrupts to Supervisor-Mode Read the article... - ["Star64 JH7110 + NuttX RTOS: RISC-V PLIC Interrupts and Serial I/O"](https://lupyuen.github.io/articles/plic) _NuttX runs in RISC-V Supervisor Mode, which can't handle Interrupts directly. (Needs Machine Mode) How can we be sure that the RISC-V Interrupts are correctly handled in Supervisor Mode?_ From [SiFive Interrupt Cookbook](https://sifive.cdn.prismic.io/sifive/0d163928-2128-42be-a75a-464df65e04e0_sifive-interrupt-cookbook.pdf), Page 15: > A CPU operating in Supervisor mode will trap to Machine mode upon the arrival of a Machine mode interrupt, unless the Machine mode interrupt has been delegated to Supervisor mode through the mideleg register. On the contrary, Supervisor interrupts will not immediately trigger if a CPU is in Machine mode. While operating in Supervisor mode, a CPU does not have visibility to configure Machine mode interrupts. According to the [RISC-V Spec](https://five-embeddev.com/riscv-isa-manual/latest/machine.html#machine-trap-delegation-registers-medeleg-and-mideleg), MIDELEG needs to be configured orrectly to delegate Machine Mode Interrupts to Supervisor Mode. From [OpenSBI Log](https://lupyuen.github.io/articles/linux#appendix-opensbi-log-for-star64), we see the value of MIDELEG... ```bash Boot HART MIDELEG: 0x0000000000000222 Boot HART MEDELEG: 0x000000000000b109 ``` MIDELEG is defined by the following bits: [csr.h](https://github.com/lupyuen2/wip-pinephone-nuttx/blob/star64d/arch/risc-v/include/csr.h#L343-L346): ```c #define MIP_SSIP (0x1 << 1) #define MIP_STIP (0x1 << 5) #define MIP_MTIP (0x1 << 7) #define MIP_SEIP (0x1 << 9) ``` So `Boot HART MIDELEG: 0x0000000000000222` means... - SSIP: Delegate Supervisor Software Interrupt - STIP: Delegate Supervisor Timer Interrupt - SEIP: Delegate Supervisor External Interrupt (But not MTIP: Delegate Machine Timer Interrupt) Thus we're good, the interrupts should be correctly delegated from Machine Mode to Supervisor Mode for NuttX. FYI: This is same for NuttX SBI: [nuttsbi/sbi_start.c](https://github.com/lupyuen2/wip-pinephone-nuttx/blob/star64d/arch/risc-v/src/nuttsbi/sbi_start.c#L91-L94) ```c /* Delegate interrupts */ reg = (MIP_SSIP | MIP_STIP | MIP_SEIP); WRITE_CSR(mideleg, reg); /* Delegate exceptions (all of them) */ reg = ((1 << RISCV_IRQ_IAMISALIGNED) | (1 << RISCV_IRQ_INSTRUCTIONPF) | (1 << RISCV_IRQ_LOADPF) | (1 << RISCV_IRQ_STOREPF) | (1 << RISCV_IRQ_ECALLU)); WRITE_CSR(medeleg, reg); ``` [SiFive Interrupt Cookbook](https://sifive.cdn.prismic.io/sifive/0d163928-2128-42be-a75a-464df65e04e0_sifive-interrupt-cookbook.pdf) states the Machine vs Supervisor Interrupt IDs: Machine Mode Interrupts: - Software Interrupt: Interrupt ID: 3 - Timer Interrupt: Interrupt ID: 7 - External Interrupt: Interrupt ID: 11 Supervisor Mode Interrupts: - Software Interrupt: Interrupt ID: 1 - Timer Interrupt: Interrupt ID: 5 - External Interrupt: Interrupt ID: 9 # NuttX Star64 handles UART Interrupts Read the article... - ["Star64 JH7110 + NuttX RTOS: RISC-V PLIC Interrupts and Serial I/O"](https://lupyuen.github.io/articles/plic) _After fixing PLIC Interrupts on Star64... Are UART Interrupts OK?_ UART Interrupts at RISC-V IRQ 32 (NuttX IRQ 57) are now OK yay! But still no UART Output though... ```text 123067BCnx_start: Entry up_irq_enable: up_enable_irq: irq=17 up_enable_irq: RISCV_IRQ_SOFT=17 uart_register: Registering /dev/console uart_register: Registering /dev/ttyS0 up_enable_irq: irq=57 up_enable_irq: extirq=32, RISCV_IRQ_EXT=25 $%^&riscv_doirq: irq=57 #*$%^&riscv_doirq: irq=57 #*$%^&riscv_doirq: irq=57 #*$%^&riscv_doirq: irq=57 #*$%^&riscv_doirq: irq=57 ... #*$%^&riscv_doirq: irq=57 #*$%^&riscv_doirq: irq=57 #*$%^&riscv_doirq: irq=57 #*$%^&riscv_doirq: irq=57 #*$%^&nx_start: CPU0: Beginning Idle Loop ``` And NuttX detects the UART Input Interrupts when we type yay! ```text 123067BCnx_start: Entry up_irq_enable: up_enable_irq: irq=17 up_enable_irq: RISCV_IRQ_SOFT=17 uart_register: Registering /dev/console uart_register: Registering /dev/ttyS0 up_enable_irq: irq=57 up_enable_irq: extirq=32, RISCV_IRQ_EXT=25 u16550_rxint: enable=1 056789056789056789056789056789056789056789056789056789056789056789056789056789056789056789056789056789056789056789056789056789056789056789056789056789056789056789056789056789056789056789056789056789056789056789056789w056789o056789r056789k056789_056789s056789t056789a056789r056789t056789_056789l056789o056789w056789p056789r056789i056789:056789 056789S056789t056789a056789r056789056789t056789i056789n056789g056789 056789l056789o056789w056789-056789p056789r056789i056789o056789r056789i056789t056789y056789 056789k056789e056789r056789n056789e056789l056789 056789w056789o056789r056789k056789e056789r056789 056789t056789h056789r056789e+056789a +++056789d++++056789(+++056789s+056789)056789 ``` [(`+` means UART Input Interrupt)](https://github.com/lupyuen2/wip-pinephone-nuttx/blob/star64d/drivers/serial/uart_16550.c#L965-L978) But why is UART Interrupt triggered repeatedly with [UART_IIR_INTSTATUS = 0](https://github.com/lupyuen2/wip-pinephone-nuttx/blob/star64d/drivers/serial/uart_16550.c#L954-L966)? Is it because we didn't Complete a RISC-V Interrupt correctly? _What happens if we don't Complete an Interrupt?_ Completing an Interrupt happens here: [qemu_rv_irq_dispatch.c](https://github.com/lupyuen2/wip-pinephone-nuttx/blob/star64d/arch/risc-v/src/qemu-rv/qemu_rv_irq_dispatch.c#L81-L88) ```c if (RISCV_IRQ_EXT <= irq) { /* Then write PLIC_CLAIM to clear pending in PLIC */ putreg32(irq - RISCV_IRQ_EXT, QEMU_RV_PLIC_CLAIM); } ``` If we don't Complete an Interrupt, we won't receive any subsequent Interrupts (like UART Input)... ```text 123067BCnx_start: Entry up_irq_enable: up_enable_irq: irq=17 up_enable_irq: RISCV_IRQ_SOFT=17 uart_register: Registering /dev/console uart_register: Registering /dev/ttyS0 up_enable_irq: irq=57 up_enable_irq: extirq=32, RISCV_IRQ_EXT=25 u16550_rxint: enable=1 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 uart_write (0xc0200428): 0000 2a 2a 2a 6d 61 69 6e 0a ***main. u16550_txint: enable=0 AAAAAAAAAu16550_txint: enable=1 Duart_write (0xc000a610): 0000 0a 4e 75 74 74 53 68 65 6c 6c 20 28 4e 53 48 29 .NuttShell (NSH) 0010 20 4e 75 74 74 58 2d 31 32 2e 30 2e 33 0a NuttX-12.0.3. u16550_txint: enable=0 AAAAAAAAAAAAAAAu16550_txint: enable=1 Duart_write (0xc0015338): 0000 6e 73 68 3e 20 nsh> u16550_txint: enable=0 AAAAAu16550_txint: enable=1 Duart_write (0xc0015310): 0000 1b 5b 4b .[K u16550_txint: enable=0 AAAu16550_txint: enable=1 Du16550_rxint: enable=0 u16550_rxint: enable=1 nx_start: CPU0: Beginning Idle Loop ``` (No response to UART Input) So it seems we are Completing Interrupts correctly. We checked the other RISC-V NuttX Ports, they Claim and Complete Interrupts the exact same way. _Are we Completing the Interrupt too soon? Maybe we should slow down?_ Let's slow down the Interrupt Completion with a Logging Delay: [qemu_rv_irq_dispatch.c](https://github.com/lupyuen2/wip-pinephone-nuttx/blob/star64d/arch/risc-v/src/qemu-rv/qemu_rv_irq_dispatch.c#L81-L88) ```c if (RISCV_IRQ_EXT <= irq) { _info("irq=%d, RISCV_IRQ_EXT=%d\n", irq, RISCV_IRQ_EXT);//// /* Then write PLIC_CLAIM to clear pending in PLIC */ putreg32(irq - RISCV_IRQ_EXT, QEMU_RV_PLIC_CLAIM); } ``` Seems to work better... ```text 123067BCnx_start: Entry up_irq_enable: up_enable_irq: irq=17 up_enable_irq: RISCV_IRQ_SOFT=17 uart_register: Registering /dev/console uart_register: Registering /dev/ttyS0 up_enable_irq: irq=57 up_enable_irq: extirq=32, RISCV_IRQ_EXT=25 u16550_rxint: enable=1 riscv_dispatch_irq: irq=57, RISCV_IRQ_EXT=25 056789riscv_dispatch_irq: irq=57, RISCV_IRQ_EXT=25 riscv_dispatch_irq: irq=57, RISCV_IRQ_EXT=25 riscv_dispatch_irq: irq=57, RISCV_IRQ_EXT=25 riscv_dispatch_irq: irq=57, RISCV_IRQ_EXT=25 ... riscv_dispatch_irq: irq=57, RISCV_IRQ_EXT=25 riscv_dispatch_irq: irq=57, RISCV_IRQ_EXT=25 riscv_dispatch_irq: irq=57, RISCV_IRQ_EXT=25 nx_start: CPU0: Beginning Idle Loop ``` Also we increase the System Delay (to match PinePhone): - System Type > Delay loops per millisecond = 116524 ```bash CONFIG_BOARD_LOOPSPERMSEC=116524 ``` [(Source)](https://github.com/lupyuen2/wip-pinephone-nuttx/blob/star64d/boards/risc-v/qemu-rv/rv-virt/configs/knsh64/defconfig#L47) _UART might need some time to warm up? Maybe we enable the IRQ later?_ Let's delay the enabling of IRQ to later... We comment out the Enable IRQ in [uart_16550.c](https://github.com/lupyuen2/wip-pinephone-nuttx/blob/star64d/drivers/serial/uart_16550.c#L860-L871): ```c static int u16550_attach(struct uart_dev_s *dev) { ... /* Attach and enable the IRQ */ ret = irq_attach(priv->irq, u16550_interrupt, dev); #ifndef CONFIG_ARCH_NOINTC if (ret == OK) { /* Enable the interrupt (RX and TX interrupts are still disabled * in the UART */ ////Enable Interrupt later: ////up_enable_irq(priv->irq); ``` And add it to `uart_write`: [serial.c](https://github.com/lupyuen2/wip-pinephone-nuttx/blob/star64d/drivers/serial/serial.c#L1177-L1188) ```c static ssize_t uart_write(FAR struct file *filep, FAR const char *buffer, size_t buflen) { static int count = 0; if (count++ == 3) { up_enable_irq(57); }//// ``` Seems better... ```text 123067BCnx_start: Entry up_irq_enable: up_enable_irq: irq=17 up_enable_irq: RISCV_IRQ_SOFT=17 uart_register: Registering /dev/console uart_register: Registering /dev/ttyS0 u16550_rxint: enable=1 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 uart_write (0xc0200428): 0000 2a 2a 2a 6d 61 69 6e 0a ***main. up_enable_irq: irq=57 up_enable_irq: extirq=32, RISCV_IRQ_EXT=25 056789056789056789056789056789056789u05678910567896056789505678950567890056789_056789t056789x056789i056789n056789t056789:056789 056789e056789n056789a056789b056789l056789e056789=0567890056789 056789056789056789A056789056789056789056789056789056789056789056789056789056789056789056789056789AAAAA056789AAA056789u05678910567896056789505678950567890056789_056789t056789x056789i056789n056789t056789:056789 056789e056789n056789a056789b056789l056789e056789=0567891056789 056789D-D-D-D-D-D-D-D-D-D-D-D-D-D-D-D-D-D-D-D-D-D-D-D-D-D-D-D- ``` After removing the logs, NSH works OK yay! Watch what happens when we enter `ls` at the NSH Shell... [(Watch the Demo on YouTube)](https://youtu.be/TdSJdiQFsv8) ![NSH on Star64](https://lupyuen.github.io/images/plic-nsh2.png) ```text 123067BCnx_start: Entry up_irq_enable: up_enable_irq: irq=17 up_enable_irq: RISCV_IRQ_SOFT=17 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 up_enable_irq: irq=57 up_enable_irq: extirq=32, RISCV_IRQ_EXT=25 ..***main NuttShell (NSH) NuttX-12.0.3 nsh> ......++.+.l......s...... ................................................p.o.s.i.x._.s.p.a.w.n..:. .p.i.d.=...0.x.c.0.2.0.2.9.7.8. .p.a.t.h.=..l.s. .f.i.l.e._.a.c.t.i.o.n.s.=...0.x.c.0.2.0.2.9.8.0. .a.t.t.r.=...0.x.c.0.2.0.2.9.8.8. .a.r.g.v.=...0.x.c.0.2.0.2.a.2.8. .........................................................e.x.e.c._.s.p.a.w.n.:. .E.R.R.O..R.:. .F.a.i.l.e.d. .t.o. .l.o.a.d. .p.r.o.g.r.a.m. .'..l.s.'.:. ..-.2. .......n.x.p.o.s.i.x._.s.p.a.w.n._.e.x.e.c.:. .E.R.R.O.R.:. .e.x.e.c. .f.a.i.l.e.d.:. ..2. ............................................................................................................../: ............................................................... dev......../ .............. proc......../ ............... system........./ .............................................................nsh> ...................n.x._.s.t.a.r.t.:. .C.P.U.0.:. .B.e.g.i.n.n.i.n.g. .I.d.l.e. .L.o.o.p. .......................... ``` (So amazing that NuttX Apps and Context Switching are OK... Even though we haven't implemented the RISC-V Timer!) But it's super slow. Each dot is 1 Million Calls to the UART Interrupt Handler, with UART Interrupt Status [UART_IIR_INTSTATUS = 0](https://github.com/lupyuen2/wip-pinephone-nuttx/blob/star64d/drivers/serial/uart_16550.c#L954-L966)! From [uart_16550.c](https://github.com/lupyuen2/wip-pinephone-nuttx/blob/star64d/drivers/serial/uart_16550.c#L948-L967): ```c /* Get the current UART status and check for loop * termination conditions */ status = u16550_serialin(priv, UART_IIR_OFFSET); /* The UART_IIR_INTSTATUS bit should be zero if there are pending * interrupts */ if ((status & UART_IIR_INTSTATUS) != 0) { /* Break out of the loop when there is no longer a * pending interrupt */ //// Print after every 1 million interrupts: static int i = 0; if (i++ % 1000000 == 1) { *(volatile uint8_t *)0x10000000 = '.'; ``` TODO: Why is UART Interrupt triggered repeatedly with [UART_IIR_INTSTATUS = 0](https://github.com/lupyuen2/wip-pinephone-nuttx/blob/star64d/drivers/serial/uart_16550.c#L954-L966)? _Maybe because OpenSBI is still handling UART Interrupts in Machine Mode?_ We tried to disable PLIC Interrupts for Machine Mode: [qemu_rv_irq.c](https://github.com/lupyuen2/wip-pinephone-nuttx/blob/star64d/arch/risc-v/src/qemu-rv/qemu_rv_irq.c#L58-L63) ```c // Disable All Global Interrupts for Hart 1 Machine-Mode // | 0x0C00_2080 | 4B | RW | Start Hart 1 M-Mode interrupt enables #define QEMU_RV_PLIC_ENABLE1_MMODE (QEMU_RV_PLIC_BASE + 0x002080) #define QEMU_RV_PLIC_ENABLE2_MMODE (QEMU_RV_PLIC_BASE + 0x002084) putreg32(0x0, QEMU_RV_PLIC_ENABLE1_MMODE); putreg32(0x0, QEMU_RV_PLIC_ENABLE2_MMODE); ``` But we still see spurious UART interrupts. TODO: How does OpenSBI handle UART I/O? Are the UART Interrupts still routed to OpenSBI? Can we remove them from OpenSBI? TODO: [Robert Lipe](https://twitter.com/robertlipe/status/1685830584688340992?t=wTD98qn0WfhUCDho6px6gw) suggests that we check for floating inputs on the control signals TODO: Throttle interrupts (for now) in [riscv_dispatch_irq](https://github.com/lupyuen2/wip-pinephone-nuttx/blob/star64d/arch/risc-v/src/qemu-rv/qemu_rv_irq_dispatch.c#L56-L91) TODO: Did we configure 16550 UART Interrupt Register correctly? TODO: Is NuttX 16550 UART Driver any different from Linux? # NuttX boots OK on Star64 JH7110 Read the article... - ["Star64 JH7110 + NuttX RTOS: Creating the First Release for the RISC-V SBC"](https://lupyuen.github.io/articles/release) From the previous section we saw that JH7110 triggers too many spurious UART interrupts... - ["Spurious UART Interrupts"](https://lupyuen.github.io/articles/plic#spurious-uart-interrupts) JH7110 uses a Synopsys DesignWare 8250 UART that has a peculiar problem with the Line Control Register (LCR)... If we write to LCR while the UART is busy, it will trigger spurious UART Interrupts. The fix is to wait for the UART to be not busy before writing to LCR. Here's my proposed patch for the NuttX 16550 UART Driver... - ["Fix the Spurious UART Interrupts"](https://lupyuen.github.io/articles/plic#appendix-fix-the-spurious-uart-interrupts) After fixing the spurious UART interrupts, now NuttX boots OK on Star64 yay! ![NuttX boots OK on Star64 JH7110](https://lupyuen.github.io/images/star64-bootok.png) ```text Starting kernel ... clk u5_dw_i2c_clk_core already disabled clk u5_dw_i2c_clk_apb already disabled BCnx_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 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 up_exit: TCB=0x40409890 exiting nx_start: CPU0: Beginning Idle Loop NuttShell (NSH) NuttX-12.0.3 nsh> uname -a posix_spawn: pid=0xc0202978 path=uname file_actions=0xc0202980 attr=0xc0202988 argv=0xc0202a28 exec_spawn: ERROR: Failed to load program 'uname': -2 nxposix_spawn_exec: ERROR: exec failed: 2 NuttX 12.0.3 7a92743-dirty Aug 3 2023 18:06:04 risc-v star64 nsh> ls -l posix_spawn: pid=0xc0202978 path=ls file_actions=0xc0202980 attr=0xc0202988 argv=0xc0202a28 exec_spawn: ERROR: Failed to load program 'ls': -2 nxposix_spawn_exec: ERROR: exec failed: 2 /: dr--r--r-- 0 dev/ dr--r--r-- 0 proc/ dr--r--r-- 0 system/ nsh> ``` [(Watch the Demo Video on YouTube)](https://youtu.be/6vQ-TXXojbQ) [(See the Complete Log)](https://gist.github.com/lupyuen/eef8de0817ceed2072b2bacc925cdd96) _How did we build NuttX for Star64?_ To build NuttX for Star64, [install the prerequisites](https://nuttx.apache.org/docs/latest/quickstart/install.html) and [clone the git repositories](https://nuttx.apache.org/docs/latest/quickstart/install.html) for ``nuttx`` and ``apps``. Before building NuttX for Star64, download the __RISC-V Toolchain riscv64-unknown-elf__ from [SiFive RISC-V Tools](https://github.com/sifive/freedom-tools/releases/tag/v2020.12.0). Add the downloaded toolchain `riscv64-unknown-elf-toolchain-.../bin` to the `PATH` Environment Variable. Check the RISC-V Toolchain: ```bash $ riscv64-unknown-elf-gcc -v ``` Configure the NuttX project and build the project: ```bash $ cd nuttx $ tools/configure.sh star64:nsh $ make $ riscv64-unknown-elf-objcopy -O binary nuttx nuttx.bin ``` This produces the NuttX Kernel ``nuttx.bin``. Next, build the NuttX Apps Filesystem: ```bash $ make export $ pushd ../apps $ tools/mkimport.sh -z -x ../nuttx/nuttx-export-*.tar.gz $ make import $ popd $ genromfs -f initrd -d ../apps/bin -V "NuttXBootVol" ``` This generates the Initial RAM Disk ``initrd``. Download the [Device Tree jh7110-visionfive-v2.dtb](https://github.com/starfive-tech/VisionFive2/releases/download/VF2_v3.1.5/jh7110-visionfive-v2.dtb) from [StarFive VisionFive2 Software Releases](https://github.com/starfive-tech/VisionFive2/releases) into the ``nuttx`` folder. Now we create a Bootable MicroSD... [(See the Build Outputs)](https://github.com/lupyuen2/wip-pinephone-nuttx/releases/tag/jh7110b-0.0.1) [(See the Build Steps)](https://github.com/lupyuen2/wip-pinephone-nuttx/releases/tag/jh7110b-0.0.1) [(See the Build Log)](https://gist.github.com/lupyuen/c6dc9aeec74d399029ebaf46ac16ef79) # Bootable MicroSD for NuttX Read the article... - ["Star64 JH7110 + NuttX RTOS: Creating the First Release for the RISC-V SBC"](https://lupyuen.github.io/articles/release) _How do we create a Bootable MicroSD for NuttX?_ From the previous section, we have the NuttX Kernel ``nuttx.bin``, Initial RAM Disk ``initrd`` and Device Tree `jh7110-visionfive-v2.dtb`. We'll pack all 3 files into a Flat Image Tree (FIT). Inside the ``nuttx`` folder, create a Text File named ``nuttx.its`` with the following content: [nuttx.its](https://github.com/lupyuen/nuttx-star64/blob/main/nuttx.its) ```text /dts-v1/; / { description = "NuttX FIT image"; #address-cells = <2>; images { vmlinux { description = "vmlinux"; data = /incbin/("./nuttx.bin"); type = "kernel"; arch = "riscv"; os = "linux"; load = <0x0 0x40200000>; entry = <0x0 0x40200000>; compression = "none"; }; ramdisk { description = "buildroot initramfs"; data = /incbin/("./initrd"); type = "ramdisk"; arch = "riscv"; os = "linux"; load = <0x0 0x46100000>; compression = "none"; hash-1 { algo = "sha256"; }; }; fdt { data = /incbin/("./jh7110-visionfive-v2.dtb"); type = "flat_dt"; arch = "riscv"; load = <0x0 0x46000000>; compression = "none"; hash-1 { algo = "sha256"; }; }; }; configurations { default = "nuttx"; nuttx { description = "NuttX"; kernel = "vmlinux"; fdt = "fdt"; loadables = "ramdisk"; }; }; }; ``` [(Based on visionfive2-fit-image.its)](https://github.com/starfive-tech/VisionFive2/blob/JH7110_VisionFive2_devel/conf/visionfive2-fit-image.its) Package the NuttX Kernel, Initial RAM Disk and Device Tree into a Flat Image Tree: ```bash ## For macOS: brew install u-boot-tools ## For Linux: sudo apt install u-boot-tools ## Generate FIT Image from `nuttx.bin`, `initrd` and `jh7110-visionfive-v2.dtb`. ## `nuttx.its` must be in the same directory as the NuttX binaries! mkimage \ -f nuttx.its \ -A riscv \ -O linux \ -T flat_dt \ starfiveu.fit ## To check FIT image mkimage -l starfiveu.fit ``` We will see... ```text → mkimage -f nuttx.its -A riscv -O linux -T flat_dt starfiveu.fit FIT description: NuttX FIT image Created: Fri Aug 4 23:20:52 2023 Image 0 (vmlinux) Description: vmlinux Created: Fri Aug 4 23:20:52 2023 Type: Kernel Image Compression: uncompressed Data Size: 2097800 Bytes = 2048.63 KiB = 2.00 MiB Architecture: RISC-V OS: Linux Load Address: 0x40200000 Entry Point: 0x40200000 Image 1 (ramdisk) Description: buildroot initramfs Created: Fri Aug 4 23:20:52 2023 Type: RAMDisk Image Compression: uncompressed Data Size: 8086528 Bytes = 7897.00 KiB = 7.71 MiB Architecture: RISC-V OS: Linux Load Address: 0x46100000 Entry Point: unavailable Hash algo: sha256 Hash value: 44b3603e6e611ade7361a936aab09def23651399d4a0a3c284f47082d788e877 Image 2 (fdt) Description: unavailable Created: Fri Aug 4 23:20:52 2023 Type: Flat Device Tree Compression: uncompressed Data Size: 50235 Bytes = 49.06 KiB = 0.05 MiB Architecture: RISC-V Load Address: 0x46000000 Hash algo: sha256 Hash value: 42767c996f0544f513280805b41f996446df8b3956c656bdbb782125ae8ffeec Default Configuration: 'nuttx220569' Configuration 0 (nuttx220569) Description: NuttX Kernel: vmlinux FDT: fdt Loadables: ramdisk ``` The Flat Image Tree ``starfiveu.fit`` will be copied to a microSD Card in the next step. To prepare the microSD Card, download the [microSD Image sdcard.img](https://github.com/starfive-tech/VisionFive2/releases/download/VF2_v3.1.5/sdcard.img) from [StarFive VisionFive2 Software Releases](https://github.com/starfive-tech/VisionFive2/releases) Write the downloaded image to a microSD Card with [Balena Etcher](https://www.balena.io/etcher/) or [GNOME Disks](https://wiki.gnome.org/Apps/Disks). Copy the file ``starfiveu.fit`` from the previous section and overwrite the file on the microSD Card. ```bash ## Copy to microSD cp starfiveu.fit "/Volumes/NO NAME" ls -l "/Volumes/NO NAME/starfiveu.fit" ## Unmount microSD ## TODO: Verify that /dev/disk2 is microSD diskutil unmountDisk /dev/disk2 ``` Check that Star64 is connected to our computer via a USB Serial Adapter. Insert the microSD Card into Star64 and power up Star64. NuttX boots on Star64 and NuttShell (nsh) appears in the Serial Console. To see the available commands in NuttShell: ```bash $ help ``` [Booting NuttX over TFTP](https://lupyuen.github.io/articles/tftp) is also supported on Star64. [(See the Build Outputs)](https://github.com/lupyuen2/wip-pinephone-nuttx/releases/tag/jh7110b-0.0.1) [(See the Build Steps)](https://github.com/lupyuen2/wip-pinephone-nuttx/releases/tag/jh7110b-0.0.1) [(See the Build Log)](https://gist.github.com/lupyuen/c6dc9aeec74d399029ebaf46ac16ef79) More about Flat Image Tree... - [Flattened Image Tree (FIT) Format](https://u-boot.readthedocs.io/en/latest/usage/fit/source_file_format.html) - [Single kernel and FDT blob](https://u-boot.readthedocs.io/en/latest/usage/fit/kernel_fdt.html) - [Multiple kernels, ramdisks and FDT blobs](https://u-boot.readthedocs.io/en/latest/usage/fit/multi.html) TODO: Why use sdcard.img # Add Star64 JH7110 Arch and Board to NuttX Read the article... - ["Star64 JH7110 + NuttX RTOS: Creating the First Release for the RISC-V SBC"](https://lupyuen.github.io/articles/release) _How did we add Star64 JH7110 to NuttX as a new Arch and Board?_ We added Star64 JH7110 to NuttX with 3 Pull Requests... 1. First we fix any dependencies needed by Star64 JH7110. This PR fixes the 16550 UART Driver used by JH7110... [Fix 16550 UART](https://github.com/apache/nuttx/pull/10019) 1. Next we submit the PR that implements the JH7110 SoC as a __NuttX Arch__... [Add support for JH7110 SoC](https://github.com/apache/nuttx/pull/10069) We add JH7110 to the Kconfig for the RISC-V SoCs: [arch/risc-v/Kconfig](https://github.com/apache/nuttx/pull/10069/files#diff-9c348f27c59e1ed0d1d9c24e172d233747ee09835ab0aa7f156da1b7caa6a5fb) And we create a Kconfig for JH7110: [arch/risc-v/src/jh7110/Kconfig](https://github.com/apache/nuttx/pull/10069/files#diff-36a3009882ced77a24e9a7fd7ce3cf481ded4655f1adc366e7722a87ceab293b) Then we add the source files for JH7110 at... [arch/risc-v/src/jh7110](https://github.com/apache/nuttx/tree/master/arch/risc-v/src/jh7110) 1. Finally we submit the PR that implements Star64 SBC as a __NuttX Board__... [Add support for Star64 SBC](https://github.com/lupyuen2/wip-pinephone-nuttx/pull/40) We add Star64 to the Kconfig for the NuttX Boards: [nuttx/boards/Kconfig](https://github.com/lupyuen2/wip-pinephone-nuttx/pull/38/files#diff-60cc096e3a9b22a769602cbbc3b0f5e7731e72db7b0338da04fcf665ed753b64) We create a Kconfig for Star64: [nuttx/boards/risc-v/jh7110/star64/Kconfig](https://github.com/lupyuen2/wip-pinephone-nuttx/pull/38/files#diff-76f41ff047f7cc79980a18f527aa05f1337be8416d3d946048b099743f10631c) And we add the source files for Star64 at... [boards/risc-v/jh7110/star64](https://github.com/apache/nuttx/tree/master/boards/risc-v/jh7110/star64) 1. In the same PR, update the __NuttX Docs__... Add JH7110 and Star64 to the list of supported platforms: [Documentation/introduction/detailed_support.rst](https://github.com/lupyuen2/wip-pinephone-nuttx/pull/38/files#diff-d8a0e68fcb8fcb7e919c4b01226b6a25f888ed297145b82c719875cf8e6f5ae4) ![Supported Platforms](https://lupyuen.github.io/images/release-doc3.png) Create a page for the JH7110 NuttX Arch: [Documentation/platforms/risc-v/jh7110/index.rst](https://github.com/lupyuen2/wip-pinephone-nuttx/pull/38/files#diff-79d8d013e3cbf7600551f1ac23beb5db8bd234a0067576bfe0997b16e5d5c148) ![JH7110 Arch](https://lupyuen.github.io/images/release-doc2.png) Under JH7110, create a page for the Star64 NuttX Board: [Documentation/platforms/risc-v/jh7110/boards/star64/index.rst](https://github.com/lupyuen2/wip-pinephone-nuttx/pull/38/files#diff-a57fa454397c544c8a717c35212a88d3e3e0c77c9c6e402f5bb52dfeb62e1349) ![Star64 Board](https://lupyuen.github.io/images/release-doc1.png) _Seems we need to copy a bunch of source files across branches?_ No sweat! Suppose we created a staging PR in our own repo... - [github.com/lupyuen2/wip-pinephone-nuttx/pull/40](https://github.com/lupyuen2/wip-pinephone-nuttx/pull/40) This command produces a list of changed files... ```bash ## TODO: Change this to your PR pr=https://github.com/lupyuen2/wip-pinephone-nuttx/pull/40 curl -L $pr.diff \ | grep "diff --git" \ | sort \ | cut -d" " -f3 \ | cut -c3- ``` Like this... ```text boards/risc-v/jh7110/star64/include/board.h boards/risc-v/jh7110/star64/include/board_memorymap.h boards/risc-v/jh7110/star64/scripts/Make.defs boards/risc-v/jh7110/star64/scripts/ld.script ``` That we can copy to another branch in a script... ```bash b=$HOME/new_branch mkdir -p $b/boards/risc-v/jh7110/star64/include mkdir -p $b/boards/risc-v/jh7110/star64/scripts a=boards/risc-v/jh7110/star64/include/board.h cp $a $b/$a a=boards/risc-v/jh7110/star64/include/board_memorymap.h cp $a $b/$a a=boards/risc-v/jh7110/star64/scripts/Make.defs cp $a $b/$a a=boards/risc-v/jh7110/star64/scripts/ld.script cp $a $b/$a ``` _How did we generate the NuttX Build Configuration?_ The NuttX Build Configuration for Star64 is at... [boards/risc-v/jh7110/star64/configs/nsh/defconfig](https://github.com/lupyuen2/wip-pinephone-nuttx/pull/40/files#diff-cdbd91013d0074f15d469491b707d1d6576752bd7b7b9ec6ed311edba8ab4b53) We generated the `defconfig` with this command... ```bash make menuconfig \ && make savedefconfig \ && grep -v CONFIG_HOST defconfig \ >boards/risc-v/jh7110/star64/configs/nsh/defconfig ``` During development, we should enable additional debug options... ```text CONFIG_DEBUG_ASSERTIONS=y CONFIG_DEBUG_ASSERTIONS_EXPRESSION=y CONFIG_DEBUG_BINFMT=y CONFIG_DEBUG_BINFMT_ERROR=y CONFIG_DEBUG_BINFMT_WARN=y CONFIG_DEBUG_ERROR=y CONFIG_DEBUG_FEATURES=y CONFIG_DEBUG_FS=y CONFIG_DEBUG_FS_ERROR=y CONFIG_DEBUG_FS_WARN=y CONFIG_DEBUG_FULLOPT=y CONFIG_DEBUG_INFO=y CONFIG_DEBUG_MM=y CONFIG_DEBUG_MM_ERROR=y CONFIG_DEBUG_MM_WARN=y CONFIG_DEBUG_SCHED=y CONFIG_DEBUG_SCHED_ERROR=y CONFIG_DEBUG_SCHED_INFO=y CONFIG_DEBUG_SCHED_WARN=y CONFIG_DEBUG_SYMBOLS=y CONFIG_DEBUG_WARN=y ``` - BINFMT is the Binary Loader, good for troubleshooting NuttX App ELF loading issues - SCHED is for Task Scheduler, which will show the spawning of NuttX App Tasks - MM is for Memory Management, for troubleshooting Memory Mapping issues - FS is for File System Before merging with NuttX Mainline, remove the BINFMT, FS, MM and SCHED debug options. TODO: GPIO next # StarFive VisionFive2 Software Release Read the article... - ["Star64 JH7110 + NuttX RTOS: Creating the First Release for the RISC-V SBC"](https://lupyuen.github.io/articles/release) StarFive VisionFive2 Software Releases seem to boot OK on Star64... - [VisionFive2 Software Releases](https://github.com/starfive-tech/VisionFive2/releases) - [SD Card Image](https://github.com/starfive-tech/VisionFive2/releases/download/VF2_v3.1.5/sdcard.img) [(See the Boot Log for Star64)](https://gist.github.com/lupyuen/030e4feb2fa95319290f3027032c24a8) Login with... ```text buildroot login: root Password: starfive ``` Based on the files above, we figured out how to generate the Flat Image Tree for NuttX: [Makefile](https://github.com/starfive-tech/VisionFive2/blob/JH7110_VisionFive2_devel/Makefile#L279-L283) Also we see the script that generates the SD Card Image: [genimage.sh](https://github.com/starfive-tech/VisionFive2/blob/JH7110_VisionFive2_devel/genimage.sh) ```bash genimage \ --rootpath "${ROOTPATH_TMP}" \ --tmppath "${GENIMAGE_TMP}" \ --inputpath "${INPUT_DIR}" \ --outputpath "${OUTPUT_DIR}" \ --config genimage-vf2.cfg ``` The SD Card Partitions are defined in [genimage-vf2.cfg](https://github.com/starfive-tech/VisionFive2/blob/JH7110_VisionFive2_devel/conf/genimage-vf2.cfg): ```text image sdcard.img { hdimage { gpt = true } partition spl { image = "work/u-boot-spl.bin.normal.out" partition-type-uuid = 2E54B353-1271-4842-806F-E436D6AF6985 offset = 2M size = 2M } partition uboot { image = "work/visionfive2_fw_payload.img" partition-type-uuid = 5B193300-FC78-40CD-8002-E86C45580B47 offset = 4M size = 4M } partition image { # partition-type = 0xC partition-type-uuid = EBD0A0A2-B9E5-4433-87C0-68B6B72699C7 image = "work/starfive-visionfive2-vfat.part" offset = 8M size = 292M } partition root { # partition-type = 0x83 partition-type-uuid = 0FC63DAF-8483-4772-8E79-3D69D8477DE4 image = "work/buildroot_rootfs/images/rootfs.ext4" offset = 300M bootable = true } } ``` Useful for creating our own SD Card Partitions! (We won't need the `spl`, `uboot` and `root` partitions for NuttX) # UART Clock for JH7110 Read the article... - ["Star64 JH7110 + NuttX RTOS: Creating the First Release for the RISC-V SBC"](https://lupyuen.github.io/articles/release) _How did we figure out the UART Clock for JH7110?_ ```bash CONFIG_16550_UART0_CLOCK=23040000 ``` [(Source)](https://github.com/lupyuen2/wip-pinephone-nuttx/blob/57d5bba4723b58c7bb947f9fa206be377c80c8d0/boards/risc-v/jh7110/star64/configs/nsh/defconfig#L10-L18) We logged the values of DLM and DLL in the UART Driver during startup... ```c uint32_t dlm = u16550_serialin(priv, UART_DLM_OFFSET); uint32_t dll = u16550_serialin(priv, UART_DLL_OFFSET); ``` [(We capture DLM and DLL only when DLAB=1)](https://github.com/apache/nuttx/blob/master/drivers/serial/uart_16550.c#L817-L851) (Be careful to print only when DLAB=0) According to our log, DLM is 0 and DLL is 13. Which means.. ```text dlm = 0 = (div >> 8) dll = 13 = (div & 0xff) ``` Which gives `div=13`. Now since `baud=115200` at startup... ```text div = (uartclk + (baud << 3)) / (baud << 4) 13 = (uartclk + 921600) / 1843200 uartclk = (13 * 1843200) - 921600 = 23040000 ``` Thus `uartclk=23040000`. And that's why we set... ```bash CONFIG_16550_UART0_CLOCK=23040000 ``` [(Source)](https://github.com/lupyuen2/wip-pinephone-nuttx/blob/57d5bba4723b58c7bb947f9fa206be377c80c8d0/boards/risc-v/jh7110/star64/configs/nsh/defconfig#L10-L18) # HDMI Display for Star64 JH7110 Read the article... - ["RISC-V Star64 JH7110: Inside the Display Controller"](https://lupyuen.github.io/articles/display2) _Will NuttX work with the HDMI Display on Star64?_ Let's find out! Maybe our HDMI code will be reused for PineTab-V's MIPI DSI Display Panel. Here are the official docs... - [Display Subsystem](https://doc-en.rvspace.org/JH7110/TRM/JH7110_TRM/display_subsystem.html) - [SDK for HDMI](http://doc-en.rvspace.org/VisionFive2/DG_HDMI/) - [Display Controller Developing Guide](http://doc-en.rvspace.org/VisionFive2/DG_Display/) - [GPU Developing and Porting Guide](http://doc-en.rvspace.org/VisionFive2/DG_GPU/) - [Multimedia Developing Guide](http://doc-en.rvspace.org/VisionFive2/DG_Multimedia/) - [MIPI LCD Developing and Porting Guide](http://doc-en.rvspace.org/VisionFive2/DG_LCD/) From the docs above we have the [Display Subsystem Block Diagram](https://doc-en.rvspace.org/JH7110/TRM/JH7110_TRM/block_diagram_display.html)... ![Display Subsystem Block Diagram](https://doc-en.rvspace.org/JH7110/TRM/Image/RD/JH7110/vout_block_diagram18.png) [(Source)](https://doc-en.rvspace.org/JH7110/TRM/JH7110_TRM/block_diagram_display.html) Which says that JH7110 uses a __DC8200 Dual Display Controller__ to drive the MIPI DSI and HDMI Displays. [(But the DC8200 docs are confidential sigh)](https://doc-en.rvspace.org/JH7110/TRM/JH7110_TRM/detail_info_display.html) And we have the [Display Subsystem Clock and Reset](https://doc-en.rvspace.org/JH7110/TRM/JH7110_TRM/clock_n_reset_display.html)... ![Display Subsystem Clock and Reset](https://doc-en.rvspace.org/JH7110/TRM/Image/RD/JH7110/vout_clkrst18.png) [(Source)](https://doc-en.rvspace.org/JH7110/TRM/JH7110_TRM/clock_n_reset_display.html) So to make HDMI work on JH7110, we need a create a NuttX Driver for the DC8200 Display Controller... # DC8200 Display Controller for Star64 JH7110 Read the article... - ["RISC-V Star64 JH7110: Inside the Display Controller"](https://lupyuen.github.io/articles/display2) Let's talk about the __DC8200 Dual Display Controller__. [(But the DC8200 docs are confidential sigh)](https://doc-en.rvspace.org/JH7110/TRM/JH7110_TRM/detail_info_display.html) Here's the [Block Diagram of DC8200 Display Controller](https://doc-en.rvspace.org/VisionFive2/DG_Display/JH7110_SDK/block_diagram_display.html)... ![Block Diagram of DC8200 Display Controller](https://doc-en.rvspace.org/VisionFive2/DG_Display/Image/JH7110_SDK/Display_Block_Diagram.png) [(Source)](https://doc-en.rvspace.org/VisionFive2/DG_Display/JH7110_SDK/block_diagram_display.html) (Display Devices refer to MIPI DPHY and HDMI, interchangeable) Here are the [Linux Drivers for DC8200 Display Controller](https://doc-en.rvspace.org/VisionFive2/DG_Display/JH7110_SDK/source_code_structure_display.html)... - [vs_dc.c](https://github.com/starfive-tech/linux/blob/JH7110_VisionFive2_devel/drivers/gpu/drm/verisilicon/vs_dc.c): Display Controller - [vs_dc_hw.c](https://github.com/starfive-tech/linux/blob/JH7110_VisionFive2_devel/drivers/gpu/drm/verisilicon/vs_dc_hw.c): Framebuffer and Overlay (similar to A64 Display Engine) - [vs_drv.c](https://github.com/starfive-tech/linux/blob/JH7110_VisionFive2_devel/drivers/gpu/drm/verisilicon/vs_drv.c): Device for Direct Rendering Manager - [vs_crtc.c](https://github.com/starfive-tech/linux/blob/JH7110_VisionFive2_devel/drivers/gpu/drm/verisilicon/vs_crtc.c): Display Pipeline (Colour / Gamma / LUT) - [vs_plane.c](https://github.com/starfive-tech/linux/blob/JH7110_VisionFive2_devel/drivers/gpu/drm/verisilicon/vs_plane.c): Display Plane - [vs_simple_enc.c](https://github.com/starfive-tech/linux/blob/JH7110_VisionFive2_devel/drivers/gpu/drm/verisilicon/vs_simple_enc.c): [Display Subsystem (DSS)](https://software-dl.ti.com/processor-sdk-linux/esd/docs/06_03_00_106/linux/Foundational_Components/Kernel/Kernel_Drivers/Display/DSS.html) Encoder - [vs_gem.c](https://github.com/starfive-tech/linux/blob/JH7110_VisionFive2_devel/drivers/gpu/drm/verisilicon/vs_gem.c): [Graphics Execution Manager](https://en.wikipedia.org/wiki/Direct_Rendering_Manager#Graphics_Execution_Manager) (Memory Management Framework) - [vs_virtual.c](https://github.com/starfive-tech/linux/blob/JH7110_VisionFive2_devel/drivers/gpu/drm/verisilicon/vs_virtual.c): Virtual Display - [vs_dc_dec.c](https://github.com/starfive-tech/linux/blob/JH7110_VisionFive2_devel/drivers/gpu/drm/verisilicon/vs_dc_dec.c): Bitmap Decompression [(See the Notes here)](https://github.com/starfive-tech/linux/tree/JH7110_VisionFive2_devel/drivers/gpu/drm/verisilicon) We'll see the Call Flow in a while. Are these used? - [vs_dc_mmu.c](https://github.com/starfive-tech/linux/blob/JH7110_VisionFive2_devel/drivers/gpu/drm/verisilicon/vs_dc_mmu.c): Memory Mapping - [vs_fb.c](https://github.com/starfive-tech/linux/blob/JH7110_VisionFive2_devel/drivers/gpu/drm/verisilicon/vs_fb.c): GEM Memory Mapping for Framebuffer Here's the (partial) [Linux Device Tree for DC8200](https://doc-en.rvspace.org/VisionFive2/DG_Display/JH7110_SDK/device_tree_config_display.html)... ```text &dc8200 { status = "okay"; dc_out: port { #address-cells = <1>; #size-cells = <0>; dc_out_dpi0: endpoint@0 { reg = <0>; remote-endpoint = <&hdmi_input0>; }; dc_out_dpi1: endpoint@1 { reg = <1>; remote-endpoint = <&hdmi_in_lcdc>; }; dc_out_dpi2: endpoint@2 { reg = <2>; remote-endpoint = <&mipi_in>; }; }; }; ``` [(Source)](https://doc-en.rvspace.org/VisionFive2/DG_Display/JH7110_SDK/device_tree_config_display.html) HDMI I2C Encoder in JH7110 is the [NXP Semiconductors TDA998X HDMI Encoder](https://doc-en.rvspace.org/VisionFive2/DG_Display/JH7110_SDK/kernel_menu_config_diplay.html) Next we need to create a NuttX Driver for the HDMI Controller... # HDMI Controller for Star64 JH7110 The HDMI Controller for JH7110 is [Inno HDMI 2.0 Transmitter For TSMC28HPC+](https://doc-en.rvspace.org/JH7110/TRM/JH7110_TRM/detail_info_display.html). Based on [JH7110 HDMI Developing Guide](https://doc-en.rvspace.org/VisionFive2/DG_HDMI/JH7110_SDK/source_code_structure_hdmi.html), the Linux Drivers for JH7110 HDMI (VeriSilicon Inno HDMI) are... - [inno_hdmi.c](https://github.com/starfive-tech/linux/blob/JH7110_VisionFive2_devel/drivers/gpu/drm/verisilicon/inno_hdmi.c) - [inno_hdmi.h](https://github.com/starfive-tech/linux/blob/JH7110_VisionFive2_devel/drivers/gpu/drm/verisilicon/inno_hdmi.h) The [Linux Device Tree](https://doc-en.rvspace.org/VisionFive2/DG_HDMI/JH7110_SDK/device_tree_hdmi.html) looks like... ```text hdmi: hdmi@29590000 { compatible = "starfive,jh7100-hdmi","inno,hdmi"; reg = <0x0 0x29590000 0x0 0x4000>; interrupts = <99>; status = "disabled"; clocks = <&clkvout JH7110_U0_HDMI_TX_CLK_SYS>, <&clkvout JH7110_U0_HDMI_TX_CLK_MCLK>, <&clkvout JH7110_U0_HDMI_TX_CLK_BCLK>, <&hdmitx0_pixelclk>; clock-names = "sysclk", "mclk","bclk","pclk"; resets = <&rstgen RSTN_U0_HDMI_TX_HDMI>; reset-names = "hdmi_tx"; }; ... &hdmi { status = "okay"; pinctrl-names = "default"; pinctrl-0 = <&inno_hdmi_pins>; hdmi_in: port { #address-cells = <1>; #size-cells = <0>; hdmi_in_lcdc: endpoint@0 { reg = <0>; remote-endpoint = <&dc_out_dpi1>; }; }; }; ``` [(Source)](https://doc-en.rvspace.org/VisionFive2/DG_HDMI/JH7110_SDK/device_tree_hdmi.html) We see the [HDMI Initialization Process](https://doc-en.rvspace.org/VisionFive2/DG_HDMI/JH7110_SDK/initialization_process.html)... ![HDMI Initialization Process](https://doc-en.rvspace.org/VisionFive2/DG_HDMI/Image/JH7110_SDK/HDMI_Init.svg) [(Source)](https://doc-en.rvspace.org/VisionFive2/DG_HDMI/JH7110_SDK/initialization_process.html) And the [HDMI Plug and Unplug Process](https://doc-en.rvspace.org/VisionFive2/DG_HDMI/JH7110_SDK/plug_n_unplug_process.html)... ![HDMI Plug and Unplug Process](https://doc-en.rvspace.org/VisionFive2/DG_HDMI/Image/JH7110_SDK/HDMI_Pug_Unplug.svg) [(Source)](https://doc-en.rvspace.org/VisionFive2/DG_HDMI/JH7110_SDK/plug_n_unplug_process.html) # Test HDMI for Star64 JH7110 _How will we test the HDMI Display for Star64 JH7110?_ We run the [`modetest` command to test HDMI](https://doc-en.rvspace.org/VisionFive2/DG_Display/JH7110_SDK/test_example_display.html)... ```bash modetest \ -M starfive \ -D 0 \ -a \ -s 116@31:1920x1080 \ -P 39@31:1920x1080@RG16 \ -Ftiles ``` [(Source)](https://doc-en.rvspace.org/VisionFive2/DG_Display/JH7110_SDK/test_example_display.html) `116@31:1920x1080` means `@: ` `39@31:1920x1080@RG16` means `@: @` [(CRTC "CRT Controller" refers to the Display Pipeline)](https://en.wikipedia.org/wiki/Direct_Rendering_Manager#KMS_device_model) See also... - [Before Debug](https://doc-en.rvspace.org/VisionFive2/DG_Display/JH7110_SDK/before_debug.html) - [Debug Display](https://doc-en.rvspace.org/VisionFive2/DG_Display/JH7110_SDK/debug_hdmi.html) TODO: What's inside the modetest app? [modetest.c](https://gitlab.freedesktop.org/mesa/drm/-/blob/main/tests/modetest/modetest.c) TODO: What parameters does modetest pass to the DC8200 Driver? TODO: Can we create a simpler modetest for our own testing on NuttX? # Direct Rendering Manager Driver for DC8200 Read the article... - ["RISC-V Star64 JH7110: Inside the Display Controller"](https://lupyuen.github.io/articles/display2) ![JH7110 Linux Display Driver](https://lupyuen.github.io/images/jh7110-display.jpg) Let's walk through the code in the Linux Driver for DC8200 Display Controller, to understand how we'll implement it in NuttX. The DRM Driver is named "starfive"... ```c // name = "starfive" static struct platform_driver vs_drm_platform_driver = { .probe = vs_drm_platform_probe, .remove = vs_drm_platform_remove, ... }; ``` [(Source)](https://github.com/starfive-tech/linux/blob/JH7110_VisionFive2_devel/drivers/gpu/drm/verisilicon/vs_drv.c#L448-L457) Here are the DRM Operations supported by the driver... ```c static struct drm_driver vs_drm_driver = { .driver_features = DRIVER_MODESET | DRIVER_ATOMIC | DRIVER_GEM, .lastclose = drm_fb_helper_lastclose, .prime_handle_to_fd = drm_gem_prime_handle_to_fd, .prime_fd_to_handle = drm_gem_prime_fd_to_handle, .gem_prime_import = vs_gem_prime_import, .gem_prime_import_sg_table = vs_gem_prime_import_sg_table, .gem_prime_mmap = vs_gem_prime_mmap, .dumb_create = vs_gem_dumb_create, #ifdef CONFIG_DEBUG_FS .debugfs_init = vs_debugfs_init, #endif .fops = &fops, .name = "starfive", .desc = "VeriSilicon DRM driver", .date = "20191101", .major = DRV_MAJOR, .minor = DRV_MINOR, }; ``` [(Source)](https://github.com/starfive-tech/linux/blob/JH7110_VisionFive2_devel/drivers/gpu/drm/verisilicon/vs_drv.c#L125-L143) The DRM Driver includes these Sub Drivers... ```c static struct platform_driver *drm_sub_drivers[] = { /* put display control driver at start */ &dc_platform_driver, /* connector */ #ifdef CONFIG_STARFIVE_INNO_HDMI &inno_hdmi_driver, #endif &simple_encoder_driver, #ifdef CONFIG_VERISILICON_VIRTUAL_DISPLAY &virtual_display_platform_driver, #endif }; ``` [(Source)](https://github.com/starfive-tech/linux/blob/JH7110_VisionFive2_devel/drivers/gpu/drm/verisilicon/vs_drv.c#L301-L315) (More about [dc_platform_driver](https://github.com/starfive-tech/linux/blob/JH7110_VisionFive2_devel/drivers/gpu/drm/verisilicon/vs_dc.c#L1642-L1649) in the next section) [vs_drm_init](https://github.com/starfive-tech/linux/blob/JH7110_VisionFive2_devel/drivers/gpu/drm/verisilicon/vs_drv.c#L459-L472) registers [drm_sub_drivers](https://github.com/starfive-tech/linux/blob/JH7110_VisionFive2_devel/drivers/gpu/drm/verisilicon/vs_drv.c#L301-L315) and [vs_drm_platform_driver](https://github.com/starfive-tech/linux/blob/JH7110_VisionFive2_devel/drivers/gpu/drm/verisilicon/vs_drv.c#L448-L457) at startup. Here are the File Operations supported by the DRM Driver... ```c static const struct file_operations fops = { .owner = THIS_MODULE, .open = drm_open, .release = drm_release, .unlocked_ioctl = drm_ioctl, .compat_ioctl = drm_compat_ioctl, .poll = drm_poll, .read = drm_read, .mmap = vs_gem_mmap, }; ``` [(Source)](https://github.com/starfive-tech/linux/blob/JH7110_VisionFive2_devel/drivers/gpu/drm/verisilicon/vs_drv.c#L54-L63) [(More about Direct Rendering Manager)](https://en.wikipedia.org/wiki/Direct_Rendering_Manager) [(__Justin / Fishwaldo__ suggests that we check out the simpler HDMI Driver in U-Boot)](https://fosstodon.org/@Fishwaldo/110902984442385966) # Call Flow for DC8200 Display Controller Driver Read the article... - ["RISC-V Star64 JH7110: Inside the Display Controller"](https://lupyuen.github.io/articles/display2) The DC8200 Controller Driver is named "vs-dc" (for VeriSilicon)... ```c // name = "vs-dc" struct platform_driver dc_platform_driver = { .probe = dc_probe, .remove = dc_remove, ... }; ``` [(Source)](https://github.com/starfive-tech/linux/blob/JH7110_VisionFive2_devel/drivers/gpu/drm/verisilicon/vs_dc.c#L1642-L1649) Probe for Display Controller is implemented here: [dc_probe](https://github.com/starfive-tech/linux/blob/JH7110_VisionFive2_devel/drivers/gpu/drm/verisilicon/vs_dc.c#L1595-L1629) We see the Component Functions exposed by the driver... ```c const struct component_ops dc_component_ops = { .bind = dc_bind, .unbind = dc_unbind, }; ``` [(Source)](https://github.com/starfive-tech/linux/blob/JH7110_VisionFive2_devel/drivers/gpu/drm/verisilicon/vs_dc.c#L1584-L1587) Bind to Display Controller is here... - [dc_bind](https://github.com/starfive-tech/linux/blob/JH7110_VisionFive2_devel/drivers/gpu/drm/verisilicon/vs_dc.c#L1421-L1573), which calls.. - [dc_init](https://github.com/starfive-tech/linux/blob/JH7110_VisionFive2_devel/drivers/gpu/drm/verisilicon/vs_dc.c#L644-L722), which calls... - [dc_hw_init](https://github.com/starfive-tech/linux/blob/JH7110_VisionFive2_devel/drivers/gpu/drm/verisilicon/vs_dc_hw.c#L1301-L1361) Here are the Display Pipeline [(CRTC)](https://www.kernel.org/doc/html/v4.15/gpu/drm-kms.html) functions exposed by the driver... ```c static const struct vs_crtc_funcs dc_crtc_funcs = { .enable = vs_dc_enable, .disable = vs_dc_disable, .mode_fixup = vs_dc_mode_fixup, .set_gamma = vs_dc_set_gamma, .enable_gamma = vs_dc_enable_gamma, .enable_vblank = vs_dc_enable_vblank, .commit = vs_dc_commit, }; ``` [(Source)](https://github.com/starfive-tech/linux/blob/JH7110_VisionFive2_devel/drivers/gpu/drm/verisilicon/vs_dc.c#L1400-L1408) Enable Display Pipeline is implemented here... - [vs_dc_enable](https://github.com/starfive-tech/linux/blob/JH7110_VisionFive2_devel/drivers/gpu/drm/verisilicon/vs_dc.c#L740-L826), which calls... - [dc_hw_setup_display](https://github.com/starfive-tech/linux/blob/JH7110_VisionFive2_devel/drivers/gpu/drm/verisilicon/vs_dc_hw.c#L1480-L1487) Enable Display Pipeline [vs_dc_enable](https://github.com/starfive-tech/linux/blob/JH7110_VisionFive2_devel/drivers/gpu/drm/verisilicon/vs_dc.c#L740-L826) is called by [vs_crtc_atomic_enable](https://github.com/starfive-tech/linux/blob/JH7110_VisionFive2_devel/drivers/gpu/drm/verisilicon/vs_crtc.c#L265-L276) Which is called by [drm_atomic_helper](https://github.com/starfive-tech/linux/blob/JH7110_VisionFive2_devel/drivers/gpu/drm/drm_atomic_helper.c#L1323-L1408) ```c static const struct drm_crtc_helper_funcs vs_crtc_helper_funcs = { .mode_fixup = vs_crtc_mode_fixup, .atomic_enable = vs_crtc_atomic_enable, .atomic_disable = vs_crtc_atomic_disable, .atomic_begin = vs_crtc_atomic_begin, .atomic_flush = vs_crtc_atomic_flush, }; ``` [(Source)](https://github.com/starfive-tech/linux/blob/JH7110_VisionFive2_devel/drivers/gpu/drm/verisilicon/vs_crtc.c#L338-L344) Commit Display Pipeline is here... - [vs_dc_commit](https://github.com/starfive-tech/linux/blob/JH7110_VisionFive2_devel/drivers/gpu/drm/verisilicon/vs_dc.c#L1381-L1398), which calls... - [dc_hw_commit](https://github.com/starfive-tech/linux/blob/JH7110_VisionFive2_devel/drivers/gpu/drm/verisilicon/vs_dc_hw.c#L2038-L2076) Commit Display Pipeline [vs_dc_commit](https://github.com/starfive-tech/linux/blob/JH7110_VisionFive2_devel/drivers/gpu/drm/verisilicon/vs_dc.c#L1381-L1398) is called by [vs_crtc_atomic_flush](https://github.com/starfive-tech/linux/blob/JH7110_VisionFive2_devel/drivers/gpu/drm/verisilicon/vs_crtc.c#L320-L336) Which is called by [drm_atomic_helper](https://github.com/starfive-tech/linux/blob/JH7110_VisionFive2_devel/drivers/gpu/drm/drm_atomic_helper.c#L2445-L2570) These are the exposed functions for the Display Plane... ```c static const struct vs_plane_funcs dc_plane_funcs = { .update = vs_dc_update_plane, .disable = vs_dc_disable_plane, .check = vs_dc_check_plane, }; ``` [(Source)](https://github.com/starfive-tech/linux/blob/JH7110_VisionFive2_devel/drivers/gpu/drm/verisilicon/vs_dc.c#L1410-L1414) Update Display Plane is here... - [vs_dc_update_plane](https://github.com/starfive-tech/linux/blob/JH7110_VisionFive2_devel/drivers/gpu/drm/verisilicon/vs_dc.c#L1262-L1280), which calls... - [update_plane](https://github.com/starfive-tech/linux/blob/JH7110_VisionFive2_devel/drivers/gpu/drm/verisilicon/vs_dc.c#L1153-L1196), which calls... - [dc_hw_update_plane](https://github.com/starfive-tech/linux/blob/JH7110_VisionFive2_devel/drivers/gpu/drm/verisilicon/vs_dc_hw.c#L1368-L1399) Update Display Plane [vs_dc_update_plane](https://github.com/starfive-tech/linux/blob/JH7110_VisionFive2_devel/drivers/gpu/drm/verisilicon/vs_dc.c#L1262-L1280) is called by [vs_plane_atomic_update](https://github.com/starfive-tech/linux/blob/JH7110_VisionFive2_devel/drivers/gpu/drm/verisilicon/vs_plane.c#L268-L301) Which is called by [drm_atomic_helper](https://github.com/starfive-tech/linux/blob/JH7110_VisionFive2_devel/drivers/gpu/drm/drm_atomic_helper.c#L2445-L2570) ```c const struct drm_plane_helper_funcs vs_plane_helper_funcs = { .atomic_check = vs_plane_atomic_check, .atomic_update = vs_plane_atomic_update, .atomic_disable = vs_plane_atomic_disable, }; ``` [(Source)](https://github.com/starfive-tech/linux/blob/JH7110_VisionFive2_devel/drivers/gpu/drm/verisilicon/vs_plane.c#L314-L318) Refer to [Linux DRM Internals](https://www.kernel.org/doc/html/v4.15/gpu/drm-internals.html) # Call Flow for DC8200 Display Hardware Driver Read the article... - ["RISC-V Star64 JH7110: Inside the Display Controller"](https://lupyuen.github.io/articles/display2) Display Planes Info: [dc_hw_planes](https://github.com/starfive-tech/linux/blob/JH7110_VisionFive2_devel/drivers/gpu/drm/verisilicon/vs_dc_hw.c#L472-L1084) Display Controller Info: [dc_info](https://github.com/starfive-tech/linux/blob/JH7110_VisionFive2_devel/drivers/gpu/drm/verisilicon/vs_dc_hw.c#L1086-L1150) Initialise Display Hardware: [dc_hw_init](https://github.com/starfive-tech/linux/blob/JH7110_VisionFive2_devel/drivers/gpu/drm/verisilicon/vs_dc_hw.c#L1301-L1361) - Read the Hardware Revision and Chip ID - Initialise every Layer / Display Plane - Initialise every Panel (Cursor) Commit Display Hardware: [dc_hw_commit](https://github.com/starfive-tech/linux/blob/JH7110_VisionFive2_devel/drivers/gpu/drm/verisilicon/vs_dc_hw.c#L2038-L2076) - Call [gamma_ex_commit](https://github.com/starfive-tech/linux/blob/JH7110_VisionFive2_devel/drivers/gpu/drm/verisilicon/vs_dc_hw.c#L1548-L1574) - Call [plane_ex_commit](https://github.com/starfive-tech/linux/blob/JH7110_VisionFive2_devel/drivers/gpu/drm/verisilicon/vs_dc_hw.c#L1768-L1863) - Update Cursor - Update QoS Update Display Plane: [dc_hw_update_plane](https://github.com/starfive-tech/linux/blob/JH7110_VisionFive2_devel/drivers/gpu/drm/verisilicon/vs_dc_hw.c#L1368-L1399) - Copy Framebuffer - Copy Scale - Copy Position - Copy Blend Display Hardware Functions: ```c static const struct dc_hw_funcs hw_func = { .gamma = &gamma_ex_commit, .plane = &plane_ex_commit, .display = setup_display_ex, }; ``` [(Source)](https://github.com/starfive-tech/linux/blob/JH7110_VisionFive2_devel/drivers/gpu/drm/verisilicon/vs_dc_hw.c#L2032-L2036) Setup Display (Upper Level): [dc_hw_setup_display](https://github.com/starfive-tech/linux/blob/JH7110_VisionFive2_devel/drivers/gpu/drm/verisilicon/vs_dc_hw.c#L1480-L1487) - Copy Display Struct - Call [setup_display_ex](https://github.com/starfive-tech/linux/blob/JH7110_VisionFive2_devel/drivers/gpu/drm/verisilicon/vs_dc_hw.c#L1971-L2030) Setup Display (Extended): [setup_display_ex](https://github.com/starfive-tech/linux/blob/JH7110_VisionFive2_devel/drivers/gpu/drm/verisilicon/vs_dc_hw.c#L1971-L2030) - Set Colour Format - Call [setup_display](https://github.com/starfive-tech/linux/blob/JH7110_VisionFive2_devel/drivers/gpu/drm/verisilicon/vs_dc_hw.c#L1865-L1969) Setup Display (Lower Level): [setup_display](https://github.com/starfive-tech/linux/blob/JH7110_VisionFive2_devel/drivers/gpu/drm/verisilicon/vs_dc_hw.c#L1865-L1969) - Set DPI Config - Disable Display Panel - Set Display Horizontal Resolution - Set Display Horizontal Sync - Set Display Vertical Resolution - Set Display Vertical Resolution Sync - Configure Framebuffer Sync Mode - Set Framebuffer Background Colour - Set Display Dither - Enable Display Panel - Set Overlay Config - Set Cursor Config Commit Display Plane (Extended): [plane_ex_commit](https://github.com/starfive-tech/linux/blob/JH7110_VisionFive2_devel/drivers/gpu/drm/verisilicon/vs_dc_hw.c#L1768-L1863) - Set Colour Space - Set Gamma - Call [plane_commit](https://github.com/starfive-tech/linux/blob/JH7110_VisionFive2_devel/drivers/gpu/drm/verisilicon/vs_dc_hw.c#L1576-L1766) Commit Display Plane: [plane_commit](https://github.com/starfive-tech/linux/blob/JH7110_VisionFive2_devel/drivers/gpu/drm/verisilicon/vs_dc_hw.c#L1576-L1766) - For every Layer / Display Plane: - Set Framebuffer YUV Address - Set Framebuffer YUV Stride - Set Framebuffer Width and Height - Clear Framebuffer - Set Primary Framebuffer Config - Set Non-Primary Framebuffer Config - Enable Framebuffer Scaling (X and Y) - Set Framebuffer Offset (X and Y) - Set Framebuffer Blending - Set Colour Key / Transparency - Set ROI [plane_ex_commit](https://github.com/starfive-tech/linux/blob/JH7110_VisionFive2_devel/drivers/gpu/drm/verisilicon/vs_dc_hw.c#L1768-L1863) and [gamma_ex_commit](https://github.com/starfive-tech/linux/blob/JH7110_VisionFive2_devel/drivers/gpu/drm/verisilicon/vs_dc_hw.c#L1548-L1574) are called by [dc_hw_commit](https://github.com/starfive-tech/linux/blob/JH7110_VisionFive2_devel/drivers/gpu/drm/verisilicon/vs_dc_hw.c#L2038-L2076) # Call Flow for DC8200 Framebuffer Driver Read the article... - ["RISC-V Star64 JH7110: Inside the Display Controller"](https://lupyuen.github.io/articles/display2) At startup, [vs_drm_bind](https://github.com/starfive-tech/linux/blob/JH7110_VisionFive2_devel/drivers/gpu/drm/verisilicon/vs_drv.c#L193-L271) calls [vs_mode_config_init](https://github.com/starfive-tech/linux/blob/JH7110_VisionFive2_devel/drivers/gpu/drm/verisilicon/vs_fb.c#L178-L191) to register the Framebuffer Driver: [vs_mode_config_funcs](https://github.com/starfive-tech/linux/blob/JH7110_VisionFive2_devel/drivers/gpu/drm/verisilicon/vs_fb.c#L166-L172). Which is defined as... ```c static const struct drm_mode_config_funcs vs_mode_config_funcs = { .fb_create = vs_fb_create, .get_format_info = vs_get_format_info, .output_poll_changed = drm_fb_helper_output_poll_changed, .atomic_check = drm_atomic_helper_check, .atomic_commit = drm_atomic_helper_commit, }; ``` [(Source)](https://github.com/starfive-tech/linux/blob/JH7110_VisionFive2_devel/drivers/gpu/drm/verisilicon/vs_fb.c#L166-L172) [vs_fb_create](https://github.com/starfive-tech/linux/blob/JH7110_VisionFive2_devel/drivers/gpu/drm/verisilicon/vs_fb.c#L60-L123) is called by [drm_framebuffer](https://github.com/starfive-tech/linux/blob/JH7110_VisionFive2_devel/drivers/gpu/drm/drm_framebuffer.c#L286-L329) [vs_get_format_info](https://github.com/starfive-tech/linux/blob/JH7110_VisionFive2_devel/drivers/gpu/drm/verisilicon/vs_fb.c#L155-L164) is called by [drm_fourcc](https://github.com/starfive-tech/linux/blob/JH7110_VisionFive2_devel/drivers/gpu/drm/drm_fourcc.c#L302-L325) Framebuffer Formats: [vs_formats](https://github.com/starfive-tech/linux/blob/JH7110_VisionFive2_devel/drivers/gpu/drm/verisilicon/vs_fb.c#L134-L139) # Call Flow for DC8200 Virtual Display Driver TODO ```c static const struct drm_connector_helper_funcs vd_connector_helper_funcs = { .get_modes = vd_get_modes, .mode_valid = vd_mode_valid, .best_encoder = vd_best_encoder, }; ``` [(Source)](https://github.com/starfive-tech/linux/blob/JH7110_VisionFive2_devel/drivers/gpu/drm/verisilicon/vs_virtual.c#L197-L201) Display Resolutions: [vd_get_modes](https://github.com/starfive-tech/linux/blob/JH7110_VisionFive2_devel/drivers/gpu/drm/verisilicon/vs_virtual.c#L153-L181) ```c static const struct drm_connector_funcs vd_connector_funcs = { .fill_modes = drm_helper_probe_single_connector_modes, .destroy = vd_connector_destroy, .detect = vd_connector_detect, .atomic_duplicate_state = drm_atomic_helper_connector_duplicate_state, .atomic_destroy_state = drm_atomic_helper_connector_destroy_state, .reset = drm_atomic_helper_connector_reset, }; ``` [(Source)](https://github.com/starfive-tech/linux/blob/JH7110_VisionFive2_devel/drivers/gpu/drm/verisilicon/vs_virtual.c#L215-L222) ```c const struct component_ops vd_component_ops = { .bind = vd_bind, .unbind = vd_unbind, }; ``` [(Source)](https://github.com/starfive-tech/linux/blob/JH7110_VisionFive2_devel/drivers/gpu/drm/verisilicon/vs_virtual.c#L311-L314) ```c // name = "vs-virtual-display" struct platform_driver virtual_display_platform_driver = { .probe = vd_probe, .remove = vd_remove, ... }; ``` [(Source)](https://github.com/starfive-tech/linux/blob/JH7110_VisionFive2_devel/drivers/gpu/drm/verisilicon/vs_virtual.c#L353-L360) Display Pipelines: ```c static const struct drm_crtc_funcs vs_crtc_funcs = { .set_config = drm_atomic_helper_set_config, .destroy = vs_crtc_destroy, .page_flip = drm_atomic_helper_page_flip, .reset = vs_crtc_reset, .atomic_duplicate_state = vs_crtc_atomic_duplicate_state, .atomic_destroy_state = vs_crtc_atomic_destroy_state, .atomic_set_property = vs_crtc_atomic_set_property, .atomic_get_property = vs_crtc_atomic_get_property, //.gamma_set = drm_atomic_helper_legacy_gamma_set, .late_register = vs_crtc_late_register, .enable_vblank = vs_crtc_enable_vblank, .disable_vblank = vs_crtc_disable_vblank, }; ``` [(Source)](https://github.com/starfive-tech/linux/blob/JH7110_VisionFive2_devel/drivers/gpu/drm/verisilicon/vs_crtc.c#L207-L220) Simple Encoder Driver: ```c // name = "vs-simple-encoder" struct platform_driver simple_encoder_driver = { .probe = encoder_probe, .remove = encoder_remove, ... }; ``` [(Source)](https://github.com/starfive-tech/linux/blob/JH7110_VisionFive2_devel/drivers/gpu/drm/verisilicon/vs_simple_enc.c#L300-L307) # Call Flow for HDMI Controller Driver TODO ```c // name = "innohdmi-starfive" struct platform_driver inno_hdmi_driver = { .probe = inno_hdmi_probe, .remove = inno_hdmi_remove, ... }; ``` [(Source)](https://github.com/starfive-tech/linux/blob/JH7110_VisionFive2_devel/drivers/gpu/drm/verisilicon/inno_hdmi.c#L1155-L1163) ```c static const struct drm_encoder_helper_funcs inno_hdmi_encoder_helper_funcs = { .enable = inno_hdmi_encoder_enable, .disable = inno_hdmi_encoder_disable, .mode_fixup = inno_hdmi_encoder_mode_fixup, .mode_set = inno_hdmi_encoder_mode_set, .atomic_check = inno_hdmi_encoder_atomic_check, }; ``` [(Source)](https://github.com/starfive-tech/linux/blob/JH7110_VisionFive2_devel/drivers/gpu/drm/verisilicon/inno_hdmi.c#L651-L657) MIPI DSI: ```c // name = "dw-mipi-dsi" struct platform_driver dw_mipi_dsi_driver = { .probe = dsi_probe, .remove = dsi_remove, ... }; ``` [(Source)](https://github.com/starfive-tech/linux/blob/JH7110_VisionFive2_devel/drivers/gpu/drm/verisilicon/dw_mipi_dsi.c#L1066-L1073) ```c // name = "cdns-dsi" static struct platform_driver cdns_dsi_platform_driver = { .probe = cdns_dsi_drm_probe, .remove = cdns_dsi_drm_remove, ... }; ``` [(Source)](https://github.com/starfive-tech/linux/blob/JH7110_VisionFive2_devel/drivers/gpu/drm/verisilicon/starfive_drm_dsi.c#L1679-L1687) ```c static const struct component_ops dsi_component_ops = { .bind = dsi_bind, .unbind = dsi_unbind, }; ``` [(Source)](https://github.com/starfive-tech/linux/blob/JH7110_VisionFive2_devel/drivers/gpu/drm/verisilicon/dw_mipi_dsi.c#L998-L1001) ```c static const struct drm_bridge_funcs dw_mipi_dsi_bridge_funcs = { .mode_set = bridge_mode_set, .enable = bridge_enable, .post_disable = bridge_post_disable, .attach = bridge_attach, .mode_fixup = bridge_mode_fixup, }; ``` [(Source)](https://github.com/starfive-tech/linux/blob/JH7110_VisionFive2_devel/drivers/gpu/drm/verisilicon/dw_mipi_dsi.c#L869-L875) # LCD Panel for Star64 JH7110 Also in the JH7110 Display Docs: How to connect an LCD Panel to JH7110. JH7110's MIPI DSI Controller is [Cadence MIPI DSI v1.3.1 TX Controller IP (DSITX)](https://doc-en.rvspace.org/VisionFive2/DG_LCD/JH7110_SDK/ic_specification_lcd.html)... ![Cadence MIPI DSI v1.3.1 TX Controller IP (DSITX)](https://doc-en.rvspace.org/VisionFive2/DG_LCD/Image/JH7110_SDK/ic_spec.png) [(Source)](https://doc-en.rvspace.org/VisionFive2/DG_LCD/JH7110_SDK/ic_specification_lcd.html) JH7110's MIPI DPHY Controller is [MIPI DPHY M31 (M31DPHYRX611TL028D_00151501)](https://doc-en.rvspace.org/VisionFive2/DG_LCD/JH7110_SDK/ic_specification_lcd.html)... ![MIPI DPHY M31 (M31DPHYRX611TL028D_00151501)](https://doc-en.rvspace.org/VisionFive2/DG_LCD/Image/JH7110_SDK/MIPI_DPHY_M31.png) [(Source)](https://doc-en.rvspace.org/VisionFive2/DG_LCD/JH7110_SDK/ic_specification_lcd.html) Refer to the... - [Linux Display Driver](https://doc-en.rvspace.org/VisionFive2/DG_LCD/JH7110_SDK/display_driver_locations_lcd.html) - [Linux Device Tree](https://doc-en.rvspace.org/VisionFive2/DG_LCD/JH7110_SDK/device_tree_configuration_lcd.html) See also the Sample LCD Panel Code for Seeed LCD Panel... - [Enable LCD](https://doc-en.rvspace.org/VisionFive2/DG_LCD/JH7110_SDK/enable_lcd.html) - [Disable LCD](https://doc-en.rvspace.org/VisionFive2/DG_LCD/JH7110_SDK/disable_lcd.html) - [Obtain LCD Information](https://doc-en.rvspace.org/VisionFive2/DG_LCD/JH7110_SDK/obtain_lcd_information.html) Here's the [LCD Initialisation Process](https://doc-en.rvspace.org/VisionFive2/DG_LCD/JH7110_SDK/initalization_lcd.html) ![LCD Initialisation Process](https://doc-en.rvspace.org/VisionFive2/DG_LCD/Image/JH7110_SDK/LCD_Init.svg) [(Source)](https://doc-en.rvspace.org/VisionFive2/DG_LCD/JH7110_SDK/initalization_lcd.html) And the [MIPI Parameter Configuration](https://doc-en.rvspace.org/VisionFive2/DG_LCD/JH7110_SDK/mipi_configuration.html) for 1C2L (2-lane MIPI DSI) and 1C4L (4-lane MIPI DSI). # HDMI Driver for U-Boot Bootloader Read the article... - ["RISC-V Star64 JH7110: Power Up the Display Controller with U-Boot Bootloader"](https://lupyuen.github.io/articles/display3) We're building the NuttX HDMI Driver for Star64 JH7110 RISC-V SBC. [__Justin (Fishwaldo)__](https://fosstodon.org/@Fishwaldo/110902984442385966) suggests that we check out the simpler HDMI Driver in __U-Boot Bootloader__... - [__U-Boot Display Driver for JH7110__](https://github.com/starfive-tech/u-boot/tree/JH7110_VisionFive2_devel/drivers/video/starfive) Here's our analysis of the Display Driver in U-Boot (which calls the HDMI Driver below): - [sf_vop_probe](https://github.com/starfive-tech/u-boot/blob/JH7110_VisionFive2_devel/drivers/video/starfive/sf_vop.c#L657-L699): Call [sf_vop_power](https://github.com/starfive-tech/u-boot/blob/JH7110_VisionFive2_devel/drivers/video/starfive/sf_vop.c#L76-L112), [vout_probe_resources_jh7110](https://github.com/starfive-tech/u-boot/blob/JH7110_VisionFive2_devel/drivers/video/starfive/sf_vop.c#L250-L367) and [sf_display_init](https://github.com/starfive-tech/u-boot/blob/JH7110_VisionFive2_devel/drivers/video/starfive/sf_vop.c#L369-L655). Flush the dcache. TODO: Who calls [sf_vop_probe](https://github.com/starfive-tech/u-boot/blob/JH7110_VisionFive2_devel/drivers/video/starfive/sf_vop.c#L657-L699)? - [sf_vop_power](https://github.com/starfive-tech/u-boot/blob/JH7110_VisionFive2_devel/drivers/video/starfive/sf_vop.c#L76-L112): Power on (Where is the code?) - [vout_probe_resources_jh7110](https://github.com/starfive-tech/u-boot/blob/JH7110_VisionFive2_devel/drivers/video/starfive/sf_vop.c#L250-L367): Enable Clocks noc_disp, vout_src, top_vout_axi, top_vout_ahb, dc_pix0, dc_pix1, dc_axi, dc_core, dc_ahb, hdmitx0_pixelclk. Deassert Resets noc_disp, rst_vout_src, dc8200_rst_axi, dc8200_rst_core, dc8200_rst_ahb. (Similar to [Linux Driver vs_dc_enable](https://lupyuen.github.io/articles/display2#vs_dc_enable)) - [sf_display_init](https://github.com/starfive-tech/u-boot/blob/JH7110_VisionFive2_devel/drivers/video/starfive/sf_vop.c#L369-L655): Call [dc_hw_init](https://github.com/starfive-tech/u-boot/blob/JH7110_VisionFive2_devel/drivers/video/starfive/sf_vop.c#L237-L248), [device_probe](https://github.com/starfive-tech/u-boot/blob/JH7110_VisionFive2_devel/drivers/core/device.c#L484-L596) and [display_enable](https://github.com/starfive-tech/u-boot/blob/JH7110_VisionFive2_devel/drivers/video/display-uclass.c#L23-L40). Set Clock pix_src as pix0's parent. Set HDMI Clock Rate pix_src. Write to HDMI Registers. ([device_probe](https://github.com/starfive-tech/u-boot/blob/JH7110_VisionFive2_devel/drivers/core/device.c#L484-L596) calls [inno_hdmi_probe](https://github.com/starfive-tech/u-boot/blob/JH7110_VisionFive2_devel/drivers/video/starfive/sf_hdmi.c#L500-L541), which is explained below) ([display_enable](https://github.com/starfive-tech/u-boot/blob/JH7110_VisionFive2_devel/drivers/video/display-uclass.c#L23-L40) calls [inno_hdmi_enable](https://github.com/starfive-tech/u-boot/blob/JH7110_VisionFive2_devel/drivers/video/starfive/sf_hdmi.c#L444-L457), which is explained below) - [dc_hw_init](https://github.com/starfive-tech/u-boot/blob/JH7110_VisionFive2_devel/drivers/video/starfive/sf_vop.c#L237-L248): Read Hardware Revision and Chip ID (similar to [Linux Driver dc_hw_init](https://lupyuen.github.io/articles/display2)) And here's our analysis of the HDMI Driver in U-Boot: - [inno_hdmi_probe](https://github.com/starfive-tech/u-boot/blob/JH7110_VisionFive2_devel/drivers/video/starfive/sf_hdmi.c#L500-L541): Enable HDMI Clocks sys_clk, mclk, bclk. Read HDMI Status. ([inno_hdmi_probe](https://github.com/starfive-tech/u-boot/blob/JH7110_VisionFive2_devel/drivers/video/starfive/sf_hdmi.c#L500-L541) is called by [device_probe](https://github.com/starfive-tech/u-boot/blob/JH7110_VisionFive2_devel/drivers/core/device.c#L484-L596), which is called by [sf_display_init](https://github.com/starfive-tech/u-boot/blob/JH7110_VisionFive2_devel/drivers/video/starfive/sf_vop.c#L369-L655)) - [inno_hdmi_enable](https://github.com/starfive-tech/u-boot/blob/JH7110_VisionFive2_devel/drivers/video/starfive/sf_hdmi.c#L444-L457): Configure 1920x1080 60Hz. Power on HDMI Tx Phy and HDMI TDMS. Start Data Sync. Call [inno_hdmi_detect](https://github.com/starfive-tech/u-boot/blob/JH7110_VisionFive2_devel/drivers/video/starfive/sf_hdmi.c#L32-L58) and [inno_hdmi_tx_phy_param_config](https://github.com/starfive-tech/u-boot/blob/JH7110_VisionFive2_devel/drivers/video/starfive/sf_hdmi.c#L389-L425) ([inno_hdmi_enable](https://github.com/starfive-tech/u-boot/blob/JH7110_VisionFive2_devel/drivers/video/starfive/sf_hdmi.c#L444-L457) is called by [display_enable](https://github.com/starfive-tech/u-boot/blob/JH7110_VisionFive2_devel/drivers/video/display-uclass.c#L23-L40), which is called by [sf_display_init](https://github.com/starfive-tech/u-boot/blob/JH7110_VisionFive2_devel/drivers/video/starfive/sf_vop.c#L369-L655)) - [inno_hdmi_detect](https://github.com/starfive-tech/u-boot/blob/JH7110_VisionFive2_devel/drivers/video/starfive/sf_hdmi.c#L32-L58): Enable Pre-PLL, Post-PLL, LDO, Serializer - [inno_hdmi_tx_phy_param_config](https://github.com/starfive-tech/u-boot/blob/JH7110_VisionFive2_devel/drivers/video/starfive/sf_hdmi.c#L389-L425): Call [inno_hdmi_config_1920x1080p60](https://github.com/starfive-tech/u-boot/blob/JH7110_VisionFive2_devel/drivers/video/starfive/sf_hdmi.c#L245-L288) and [inno_hdmi_tx_ctrl](https://github.com/starfive-tech/u-boot/blob/JH7110_VisionFive2_devel/drivers/video/starfive/sf_hdmi.c#L376-L387) - [inno_hdmi_config_1920x1080p60](https://github.com/starfive-tech/u-boot/blob/JH7110_VisionFive2_devel/drivers/video/starfive/sf_hdmi.c#L245-L288): Config PLL for 1080p, 60hz - [inno_hdmi_tx_ctrl](https://github.com/starfive-tech/u-boot/blob/JH7110_VisionFive2_devel/drivers/video/starfive/sf_hdmi.c#L376-L387): Config Video Format Identification Code. bist mode: 0x00, normal mode: 0x10, phy mode: 0x4 See [hdmi_read](https://github.com/starfive-tech/u-boot/blob/JH7110_VisionFive2_devel/drivers/video/starfive/sf_hdmi.c#L22-L30) and [hdmi_write](https://github.com/starfive-tech/u-boot/blob/JH7110_VisionFive2_devel/drivers/video/starfive/sf_hdmi.c#L22-L30) [U0_HDMITX Base Address](https://doc-en.rvspace.org/JH7110/TRM/JH7110_TRM/system_memory_map.html) is 0x2959_0000 TODO: Is sf_vop_priv.regs_hi = 0x2940_0000 or 0x2948_0000? Here's the Device Tree for U-Boot: - [starfive_visionfive2.dts](https://github.com/starfive-tech/u-boot/blob/JH7110_VisionFive2_devel/arch/riscv/dts/starfive_visionfive2.dts) Which includes [jh7110.dtsi](https://github.com/starfive-tech/u-boot/blob/JH7110_VisionFive2_devel/arch/riscv/dts/jh7110.dtsi) - [starfive_visionfive2-u-boot.dtsi](https://github.com/starfive-tech/u-boot/blob/JH7110_VisionFive2_devel/arch/riscv/dts/starfive_visionfive2-u-boot.dtsi) Which includes [jh7110-u-boot.dtsi](https://github.com/starfive-tech/u-boot/blob/JH7110_VisionFive2_devel/arch/riscv/dts/jh7110-u-boot.dtsi) From U-Boot Device Tree: [jh7110.dtsi](https://github.com/starfive-tech/u-boot/blob/JH7110_VisionFive2_devel/arch/riscv/dts/jh7110.dtsi#L1292-L1343) ```text dc8200: dc8200@29400000 { compatible = "starfive,sf-dc8200"; reg = <0x0 0x29400000 0x0 0x100>, <0x0 0x29400800 0x0 0x2000>; reg-names = "hi", "low"; status = "okay"; clocks = <&clkgen JH7110_NOC_BUS_CLK_DISP_AXI>, <&clkgen JH7110_VOUT_SRC>, <&clkgen JH7110_VOUT_TOP_CLK_VOUT_AXI>, <&clkgen JH7110_VOUT_TOP_CLK_VOUT_AHB>, <&clkvout JH7110_U0_DC8200_CLK_PIX0>, <&clkvout JH7110_U0_DC8200_CLK_PIX1>, <&clkvout JH7110_U0_DC8200_CLK_AXI>, <&clkvout JH7110_U0_DC8200_CLK_CORE>, <&clkvout JH7110_U0_DC8200_CLK_AHB>, <&clkvout JH7110_DOM_VOUT_TOP_LCD_CLK>, <&hdmitx0_pixelclk>, <&clkvout JH7110_DC8200_PIX0>, <&clkvout JH7110_U0_DC8200_CLK_PIX0_OUT>, <&clkvout JH7110_U0_DC8200_CLK_PIX1_OUT>; clock-names = "disp_axi","vout_src", "top_vout_axi","top_vout_ahb", "dc_pix0","dc_pix1", "dc_axi","dc_core","dc_ahb", "top_vout_lcd","hdmitx0_pixelclk","dc8200_pix0", "dc8200_pix0_out","dc8200_pix1_out"; resets = <&rstgen RSTN_U0_DOM_VOUT_TOP_SRC>, <&rstgen RSTN_U0_DC8200_AXI>, <&rstgen RSTN_U0_DC8200_AHB>, <&rstgen RSTN_U0_DC8200_CORE>, <&rstgen RSTN_U0_NOC_BUS_DISP_AXI_N>; reset-names = "rst_vout_src","rst_axi","rst_ahb","rst_core", "rst_noc_disp"; ``` U-Boot Clocks and Resets Mapped Nicely: | Clock Names | Clocks | |:------------|:-------| | disp_axi / noc_disp | JH7110_NOC_BUS_CLK_DISP_AXI | vout_src | JH7110_VOUT_SRC | top_vout_axi | JH7110_VOUT_TOP_CLK_VOUT_AXI | top_vout_ahb | JH7110_VOUT_TOP_CLK_VOUT_AHB | dc_pix0 | JH7110_U0_DC8200_CLK_PIX0 | dc_pix1 | JH7110_U0_DC8200_CLK_PIX1 | dc_axi | JH7110_U0_DC8200_CLK_AXI | dc_core | JH7110_U0_DC8200_CLK_CORE | dc_ahb | JH7110_U0_DC8200_CLK_AHB | top_vout_lcd | JH7110_DOM_VOUT_TOP_LCD_CLK | hdmitx0_pixelclk | hdmitx0_pixelclk | dc8200_pix0 | JH7110_DC8200_PIX0 | dc8200_pix0_out | JH7110_U0_DC8200_CLK_PIX0_OUT | dc8200_pix1_out | JH7110_U0_DC8200_CLK_PIX1_OUT | Reset Names | Resets | |:------------|:-------| | rst_vout_src | RSTN_U0_DOM_VOUT_TOP_SRC | rst_axi | RSTN_U0_DC8200_AXI | rst_ahb | RSTN_U0_DC8200_AHB | rst_core | RSTN_U0_DC8200_CORE | rst_noc_disp | RSTN_U0_NOC_BUS_DISP_AXI_N U-Boot Clocks are defined here: [clk-jh7110.c](https://github.com/starfive-tech/u-boot/blob/JH7110_VisionFive2_devel/drivers/clk/starfive/clk-jh7110.c) From the Linux Device Tree: [jh7110.dtsi](https://github.com/starfive-tech/linux/blob/JH7110_VisionFive2_devel/arch/riscv/boot/dts/starfive/jh7110.dtsi#L339-L360) ```text dc8200: dc8200@29400000 { compatible = "starfive,jh7110-dc8200","verisilicon,dc8200"; verisilicon,dss-syscon = <&dssctrl>;//20220624 panel syscon reg = <0x0 0x29400000 0x0 0x100>, <0x0 0x29400800 0x0 0x2000>, <0x0 0x17030000 0x0 0x1000>; interrupts = <95>; status = "disabled"; clocks = <&clkgen JH7110_NOC_BUS_CLK_DISP_AXI>, <&clkgen JH7110_VOUT_SRC>, <&clkgen JH7110_VOUT_TOP_CLK_VOUT_AXI>, <&clkgen JH7110_VOUT_TOP_CLK_VOUT_AHB>, <&clkvout JH7110_U0_DC8200_CLK_PIX0>, <&clkvout JH7110_U0_DC8200_CLK_PIX1>, <&clkvout JH7110_U0_DC8200_CLK_AXI>, <&clkvout JH7110_U0_DC8200_CLK_CORE>, <&clkvout JH7110_U0_DC8200_CLK_AHB>, <&clkgen JH7110_VOUT_TOP_CLK_VOUT_AXI>, <&clkvout JH7110_DOM_VOUT_TOP_LCD_CLK>, <&hdmitx0_pixelclk>, <&clkvout JH7110_DC8200_PIX0>, <&clkvout JH7110_U0_DC8200_CLK_PIX0_OUT>, <&clkvout JH7110_U0_DC8200_CLK_PIX1_OUT>; clock-names = "noc_disp","vout_src", "top_vout_axi","top_vout_ahb", "pix_clk","vout_pix1", "axi_clk","core_clk","vout_ahb", "vout_top_axi","vout_top_lcd","hdmitx0_pixelclk","dc8200_pix0", "dc8200_pix0_out","dc8200_pix1_out"; resets = <&rstgen RSTN_U0_DOM_VOUT_TOP_SRC>, <&rstgen RSTN_U0_DC8200_AXI>, <&rstgen RSTN_U0_DC8200_AHB>, <&rstgen RSTN_U0_DC8200_CORE>, <&rstgen RSTN_U0_NOC_BUS_DISP_AXI_N>; reset-names = "rst_vout_src","rst_axi","rst_ahb","rst_core", "rst_noc_disp"; }; clkvout: clock-controller@295C0000 { compatible = "starfive,jh7110-clk-vout"; reg = <0x0 0x295C0000 0x0 0x10000>; reg-names = "vout"; clocks = <&hdmitx0_pixelclk>, <&mipitx_dphy_rxesc>, <&mipitx_dphy_txbytehs>, <&clkgen JH7110_VOUT_SRC>, <&clkgen JH7110_VOUT_TOP_CLK_VOUT_AHB>; clock-names = "hdmitx0_pixelclk", "mipitx_dphy_rxesc", "mipitx_dphy_txbytehs", "vout_src", "vout_top_ahb"; resets = <&rstgen RSTN_U0_DOM_VOUT_TOP_SRC>; reset-names = "vout_src"; #clock-cells = <1>; power-domains = <&pwrc JH7110_PD_VOUT>; status = "okay"; }; ``` Linux Clocks and Resets Mapped Nicely: | Clock Names | Clocks | |:------------|:-------| |noc_disp | JH7110_NOC_BUS_CLK_DISP_AXI |vout_src | JH7110_VOUT_SRC |top_vout_axi | JH7110_VOUT_TOP_CLK_VOUT_AXI |top_vout_ahb | JH7110_VOUT_TOP_CLK_VOUT_AHB |pix_clk | JH7110_U0_DC8200_CLK_PIX0 |vout_pix1 | JH7110_U0_DC8200_CLK_PIX1 |axi_clk | JH7110_U0_DC8200_CLK_AXI |core_clk | JH7110_U0_DC8200_CLK_CORE |vout_ahb | JH7110_U0_DC8200_CLK_AHB |vout_top_axi | JH7110_VOUT_TOP_CLK_VOUT_AXI |vout_top_lcd | JH7110_DOM_VOUT_TOP_LCD_CLK |hdmitx0_pixelclk | hdmitx0_pixelclk |dc8200_pix0 | JH7110_DC8200_PIX0 |dc8200_pix0_out | JH7110_U0_DC8200_CLK_PIX0_OUT |dc8200_pix1_out | JH7110_U0_DC8200_CLK_PIX1_OUT | Reset Names | Resets | |:------------|:-------| |rst_vout_src | RSTN_U0_DOM_VOUT_TOP_SRC |rst_axi | RSTN_U0_DC8200_AXI |rst_ahb | RSTN_U0_DC8200_AHB |rst_core | RSTN_U0_DC8200_CORE |rst_noc_disp | RSTN_U0_NOC_BUS_DISP_AXI_N See the [JH7110 Memory Map](https://doc-en.rvspace.org/JH7110/TRM/JH7110_TRM/memory_map_display.html) and [Control Registers](https://doc-en.rvspace.org/JH7110/TRM/JH7110_TRM/controller_registers_display.html) Based on [DOM VOUT CRG](https://doc-en.rvspace.org/JH7110/TRM/JH7110_TRM/dom_vout_crg.html), here are the Clock Registers and the Reset Register.... - clk_u0_dc8200_clk_axi: Offset 0x10 - clk_u0_dc8200_clk_core: Offset 0x14 - clk_u0_dc8200_clk_ahb: Offset 0x18 - clk_u0_dc8200_clk_pix0: Offset 0x1c - clk_u0_dc8200_clk_pix1: Offset 0x20 - clk_dom_vout_top_lcd_clk: Offset 0x24 - clk_u0_cdns_dsiTx_clk_apb: Offset 0x28 - clk_u0_cdns_dsiTx_clk_sys: Offset 0x2c - clk_u0_cdns_dsiTx_clk_dpi: Offset 0x30 - clk_u0_cdns_dsiTx_clk_txesc: Offset 0x34 - clk_u0_mipitx_dphy_clk_txesc: Offset 0x38 - clk_u0_hdmi_tx_clk_mclk: Offset 0x3c - clk_u0_hdmi_tx_clk_bclk: Offset 0x40 - clk_u0_hdmi_tx_clk_sys: Offset 0x44 - Software_RESET_assert0_addr_assert_sel: Offset 0x38 - Bit [0] rstn_u0_dc8200_rstn_axi - 1: Assert reset - 0: De-assert reset - Bit [1] rstn_u0_dc8200_rstn_ahb - 1: Assert reset - 0: De-assert reset - Bit [2] rstn_u0_dc8200_rstn_core - 1: Assert reset - 0: De-assert reset - Bit [9] rstn_u0_hdmi_tx_rstn_hdmi - 1: Assert reset - 0: De-assert reset TODO: clk_u0_dc8200_clk_pix0: Offset 0x1c - Bits [24:29]: clk_mux_sel (Default 0) Clock multiplexing selector: - clk_dc8200_pix0 - clk_hdmitx0_pixelclk What are the values for clk_mux_sel? TODO: Can we read another Clock Mux to figure this out? From [clk-jh7110.c](https://github.com/starfive-tech/u-boot/blob/JH7110_VisionFive2_devel/drivers/clk/starfive/clk-jh7110.c#L77-L80): ```c static const char *u0_dc8200_clk_pix_sels[2] = { [0] = "dc8200_pix0", [1] = "hdmitx0_pixelclk", }; clk_dm(JH7110_U0_DC8200_CLK_PIX0, starfive_clk_composite(priv->vout, "u0_dc8200_clk_pix0", u0_dc8200_clk_pix_sels, ARRAY_SIZE(u0_dc8200_clk_pix_sels), VOUT_OFFSET(JH7110_U0_DC8200_CLK_PIX0), 1, 1, 0)); clk_dm(JH7110_U0_DC8200_CLK_PIX1, starfive_clk_composite(priv->vout, "u0_dc8200_clk_pix1", u0_dc8200_clk_pix_sels, ARRAY_SIZE(u0_dc8200_clk_pix_sels), VOUT_OFFSET(JH7110_U0_DC8200_CLK_PIX1), 1, 1, 0)); static struct clk *starfive_clk_mux(void __iomem *reg, const char *name, unsigned int offset, u8 width, const char * const *parent_names, u8 num_parents) { return clk_register_mux(NULL, name, parent_names, num_parents, 0, reg + offset, STARFIVE_CLK_MUX_SHIFT, width, 0); } static struct clk *starfive_clk_composite(void __iomem *reg, const char *name, const char * const *parent_names, unsigned int num_parents, unsigned int offset, unsigned int mux_width, unsigned int gate_width, unsigned int div_width) { struct clk *clk = ERR_PTR(-ENOMEM); struct clk_divider *div = NULL; struct clk_gate *gate = NULL; struct clk_mux *mux = NULL; int mask_arry[4] = {0x1, 0x3, 0x7, 0xF}; int mask; if (mux_width) { if (mux_width > 4) goto fail; else mask = mask_arry[mux_width-1]; mux = kzalloc(sizeof(*mux), GFP_KERNEL); if (!mux) goto fail; mux->reg = reg + offset; mux->mask = mask; mux->shift = STARFIVE_CLK_MUX_SHIFT; mux->num_parents = num_parents; mux->flags = 0; mux->parent_names = parent_names; } #define VOUT_OFFSET(id) (((id) - JH7110_CLK_VOUT_START) * 4) #define STARFIVE_CLK_MUX_SHIFT 24 /*[29:24]*/ ``` TODO: Reconcile the above drivers with the [Official Linux Driver](https://lupyuen.github.io/articles/display2) TODO: Test on NuttX TODO: Do we really need I2C for testing HDMI? # Poking the Star64 JH7110 Display Controller with U-Boot Bootloader ![Star64 JH7110 Display Controller is alive!](https://lupyuen.github.io/images/display3-title.png) Read the article... - ["RISC-V Star64 JH7110: Power Up the Display Controller with U-Boot Bootloader"](https://lupyuen.github.io/articles/display3) In the olden days we would `peek` and `poke` the Display Controller, to see weird and wonderful displays. Today (46 years later), we poke around the Display Controller of Star64 JH7110 SBC with a modern tool (not BASIC): U-Boot Bootloader! (Spoiler: No weird and wonderful displays for today!) Boot Star64 SBC without a microSD Card. Shut down our TFTP Server. We should see the U-Boot Prompt. The `md` command dumps the JH7110 Memory: RAM, ROM, even I/O Registers! ```text # md md - memory display Usage: md [.b, .w, .l, .q] address [# of objects] ``` We can dump the Boot ROM at `0x2A00` `0000`... ```text # md 2A000000 2a000000: 00000297 12628293 30529073 30005073 ......b.s.R0sP.0 2a000010: 30405073 41014081 42014181 43014281 sP@0.@.A.A.B.B.C 2a000020: 44014381 45014481 46014581 47014681 .C.D.D.E.E.F.F.G 2a000030: 48014781 49014881 4a014981 4b014a81 .G.H.H.I.I.J.J.K 2a000040: 4c014b81 4d014c81 4e014d81 4f014e81 .K.L.L.M.M.N.N.O 2a000050: 01974f81 8193d710 02970ae1 82930000 .O.............. 2a000060: 907305a2 02933052 85630010 50730402 ..s.R0....c...sP 2a000070: 22f37c14 02b2f140 d7102117 b8810113 .|."@....!...... 2a000080: 40510133 25734581 1d63f140 033704b5 3.Q@.Es%@.c...7. 2a000090: 23b70110 30230110 03210003 fe734de3 ...#..#0..!..Ms. 2a0000a0: f0018293 f0018313 00628e63 f2018393 ........c.b..... 2a0000b0: 00737a63 0002be03 01c33023 032102a1 czs.....#0....!. 2a0000c0: fe736ae3 f2018313 d7101397 a9838393 .js............. 2a0000d0: 00737763 00033023 4de30321 00effe73 cws.#0..!..Ms... 2a0000e0: a81d61d0 10734621 00733046 26731050 .a..!Fs.F0s.P.s& 2a0000f0: 8a213440 04b7da7d 25730200 1913f140 @4!.}.....s%@... ``` (Where is the Source Code for JH7110 Boot ROM?) We can dump the UART Registers at `0x1000` `0000`... ```text # md 10000000 10000000: 0000006d 00000000 000000c1 00000003 m............... 10000010: 00000003 00000000 00000000 00000000 ................ 10000020: 00000000 00000000 00000000 00000000 ................ 10000030: 00000064 00000064 00000064 00000064 d...d...d...d... 10000040: 00000064 00000064 00000064 00000064 d...d...d...d... 10000050: 00000064 00000064 00000064 00000064 d...d...d...d... 10000060: 00000064 00000064 00000064 00000064 d...d...d...d... 10000070: 00000000 00000000 00000000 00000003 ................ 10000080: 00000001 00000000 00000000 00000001 ................ 10000090: 00000000 00000000 00000001 00000000 ................ 100000a0: 00000000 00000000 00000000 00000000 ................ 100000b0: 00000000 00000000 00000000 00000000 ................ 100000c0: 00000000 00000000 00000000 00000000 ................ 100000d0: 00000000 00000000 00000000 00000000 ................ 100000e0: 00000000 00000000 00000000 00000000 ................ 100000f0: 00000000 00000000 3331342a 44570110 ........*413..WD ``` Let's write to UART. The `mw` command writes to JH7110 Memory... ```text # mw mw - memory write (fill) Usage: mw [.b, .w, .l, .q] address value [count] ``` (Hmmm sounds like a Security Risk) Let's poke the UART Transmit Register at `0x1000` `0000`... ```text # mw 10000000 2a 1 * ``` Yep it prints "`*`", which is ASCII Code 2A! Based on the [JH7110 System Memory Map](https://doc-en.rvspace.org/JH7110/TRM/JH7110_TRM/system_memory_map.html), the Display Controller Registers are at... - DC8200 AHB0: 0x2940_0000 - DC8200 AHB1: 0x2948_0000 - U0_HDMITX: 0x2959_0000 - VOUT_SYSCON: 0x295B_0000 - VOUT_CRG: 0x295C_0000 - DSI TX: 0x295D_0000 - MIPITX DPHY: 0x295E_0000 But the values are all zero! ```text # md 29400000 29400000: 00000000 00000000 00000000 00000000 ................ 29400010: 00000000 00000000 00000000 00000000 ................ # md 29480000 29480000: 00000000 00000000 00000000 00000000 ................ 29480010: 00000000 00000000 00000000 00000000 ................ # md 29590000 29590000: 00000000 00000000 00000000 00000000 ................ 29590010: 00000000 00000000 00000000 00000000 ................ # md 295B0000 295b0000: 00000000 00000000 00000000 00000000 ................ 295b0010: 00000000 00000000 00000000 00000000 ................ # md 295C0000 295c0000: 00000000 00000000 00000000 00000000 ................ 295c0010: 00000000 00000000 00000000 00000000 ................ ``` Maybe because the Display Controller is not powered up? Can we poke the PMU Registers and Clock and Reset Registers to power it up? Let's find out... # Power Management Registers for Star64 JH7110 Display Controller Read the article... - ["RISC-V Star64 JH7110: Power Up the Display Controller with U-Boot Bootloader"](https://lupyuen.github.io/articles/display3) _Can we poke the Power Management Registers to power up the Display Controller?_ From [JH7110 Power Management](https://doc-en.rvspace.org/JH7110/TRM/JH7110_TRM/overview_pm.html)... ![Power Management](https://doc-en.rvspace.org/JH7110/TRM/Image/RD/JH7110/power_stratey.png) Display Controller (vout) is powered by the Power Domains... - dom_dig - dom_vout _How to enable these Power Domains?_ We refer to the [Power Management Unit (PMU) Registers](https://doc-en.rvspace.org/JH7110/TRM/JH7110_TRM/register_info_pmu.html) From [System Memory Map](https://doc-en.rvspace.org/JH7110/TRM/JH7110_TRM/system_memory_map.html), PMU is at 0x1703_0000... ```text # md 17030000 17030000: 00000000 00000000 000000ff 00000003 ................ 17030010: 00000000 0000ffff 04008402 04010040 ............@... 17030020: 00010040 00000000 00000000 00000000 @............... 17030030: 00000000 00000000 00000000 00000000 ................ 17030040: 00000000 00000000 000007ff 00000012 ................ 17030050: 0000001f ffffffff 00000001 00000003 ................ 17030060: 00000000 00000000 00000080 00000000 ................ 17030070: 00000000 00000000 00000000 00000000 ................ 17030080: 00000003 00000000 00000203 00000000 ................ 17030090: 00000020 00000003 000001c3 00000000 ............... 170300a0: 00000000 00000000 00000000 00000000 ................ 170300b0: 00000000 00000000 00000000 00000000 ................ 170300c0: 00000000 00000000 00000000 00000000 ................ 170300d0: 00000000 00000000 00000000 00000000 ................ 170300e0: 00000000 00000000 00000000 00000000 ................ 170300f0: 00000000 00000000 00000000 00000000 ................ ``` _Which Power Domains are already enabled?_ The [Current Power Mode Register](https://doc-en.rvspace.org/JH7110/TRM/JH7110_TRM/register_info_pmu.html) is at Offset Address 0x80. (Address 0x17030080) From U-Boot Log above, Current Power Mode is 3, which means... - Bit 0 (systop_power_mode): SYSTOP Power is Enabled - Bit 1 (cpu_power_mode): CPU Power is Enabled - But Bit 4 (vout_power_mode): VOUT Power is Disabled! _How to enable VOUT Power?_ From [PMU Function Description](https://doc-en.rvspace.org/JH7110/TRM/JH7110_TRM/function_descript_pmu.html)... > SW encourage Turn-on sequence: > (1) Configure the register SW turn-on power mode (offset 0x0c), write the bit 1 which power domain will be turn-on, write the others 0; > (2) Write the SW Turn-on command sequence. Write the register Software encourage (offset 0x44) 0xff –> 0x05 –> 0x50 Which is implemented in the Linux Driver (with no delays in the Command Sequence): [jh7110_pmu.c](https://github.com/starfive-tech/linux/blob/JH7110_VisionFive2_devel/drivers/soc/starfive/jh7110_pmu.c#L132-L147) _What's a "Software Encourage"?_ Something got Lost in Translation. Let's assume it means "Software Trigger". We do it in U-Boot like this... ```text md 1703000c 1 mw 1703000c 0x10 1 md 1703000c 1 mw 17030044 0xff 1 mw 17030044 0x05 1 mw 17030044 0x50 1 ``` Then we check status if VOUT Power is on... ```text md 17030080 1 md 29400000 0x40 md 29480000 0x40 md 29590000 0x40 md 295B0000 0x40 md 295C0000 0x40 ``` Here's the U-Boot Log... ```text StarFive # md 1703000c 1 1703000c: 00000003 .... StarFive # mw 1703000c 0x10 1 StarFive # StarFive # md 1703000c 1 1703000c: 00000010 .... StarFive # md 17030080 1 17030080: 00000003 .... StarFive # mw 17030044 0xff 1 StarFive # StarFive # md 17030080 1 17030080: 00000003 .... StarFive # mw 17030044 0x05 1 StarFive # StarFive # md 17030080 1 17030080: 00000003 .... StarFive # mw 17030044 0x50 1 StarFive # StarFive # md 17030080 1 17030080: 00000013 .... ``` _Is the Display Controller now powered up?_ From U-Boot Log above, the [Current Power Mode](https://doc-en.rvspace.org/JH7110/TRM/JH7110_TRM/register_info_pmu.html) (0x17030080) is 0x13, which means... - Bit 0 (systop_power_mode): SYSTOP Power is Enabled - Bit 1 (cpu_power_mode): CPU Power is Enabled - Bit 4 (vout_power_mode): VOUT Power is Enabled But the Display Controller Registers are still missing... ```text StarFive # md 29400000 0x40 29400000: 00000000 00000000 00000000 00000000 ................ 29400010: 00000000 00000000 00000000 00000000 ................ 29400020: 00000000 00000000 00000000 00000000 ................ 29400030: 00000000 00000000 00000000 00000000 ................ 29400040: 00000000 00000000 00000000 00000000 ................ 29400050: 00000000 00000000 00000000 00000000 ................ 29400060: 00000000 00000000 00000000 00000000 ................ 29400070: 00000000 00000000 00000000 00000000 ................ 29400080: 00000000 00000000 00000000 00000000 ................ 29400090: 00000000 00000000 00000000 00000000 ................ 294000a0: 00000000 00000000 00000000 00000000 ................ 294000b0: 00000000 00000000 00000000 00000000 ................ 294000c0: 00000000 00000000 00000000 00000000 ................ 294000d0: 00000000 00000000 00000000 00000000 ................ 294000e0: 00000000 00000000 00000000 00000000 ................ 294000f0: 00000000 00000000 00000000 00000000 ................ StarFive # md 29480000 0x40 29480000: 00000000 00000000 00000000 00000000 ................ 29480010: 00000000 00000000 00000000 00000000 ................ 29480020: 00000000 00000000 00000000 00000000 ................ 29480030: 00000000 00000000 00000000 00000000 ................ 29480040: 00000000 00000000 00000000 00000000 ................ 29480050: 00000000 00000000 00000000 00000000 ................ 29480060: 00000000 00000000 00000000 00000000 ................ 29480070: 00000000 00000000 00000000 00000000 ................ 29480080: 00000000 00000000 00000000 00000000 ................ 29480090: 00000000 00000000 00000000 00000000 ................ 294800a0: 00000000 00000000 00000000 00000000 ................ 294800b0: 00000000 00000000 00000000 00000000 ................ 294800c0: 00000000 00000000 00000000 00000000 ................ 294800d0: 00000000 00000000 00000000 00000000 ................ 294800e0: 00000000 00000000 00000000 00000000 ................ 294800f0: 00000000 00000000 00000000 00000000 ................ StarFive # md 29590000 0x40 29590000: 00000000 00000000 00000000 00000000 ................ 29590010: 00000000 00000000 00000000 00000000 ................ 29590020: 00000000 00000000 00000000 00000000 ................ 29590030: 00000000 00000000 00000000 00000000 ................ 29590040: 00000000 00000000 00000000 00000000 ................ 29590050: 00000000 00000000 00000000 00000000 ................ 29590060: 00000000 00000000 00000000 00000000 ................ 29590070: 00000000 00000000 00000000 00000000 ................ 29590080: 00000000 00000000 00000000 00000000 ................ 29590090: 00000000 00000000 00000000 00000000 ................ 295900a0: 00000000 00000000 00000000 00000000 ................ 295900b0: 00000000 00000000 00000000 00000000 ................ 295900c0: 00000000 00000000 00000000 00000000 ................ 295900d0: 00000000 00000000 00000000 00000000 ................ 295900e0: 00000000 00000000 00000000 00000000 ................ 295900f0: 00000000 00000000 00000000 00000000 ................ StarFive # md 295B0000 0x40 295b0000: 00000000 00000000 00000000 00000000 ................ 295b0010: 00000000 00000000 00000000 00000000 ................ 295b0020: 00000000 00000000 00000000 00000000 ................ 295b0030: 00000000 00000000 00000000 00000000 ................ 295b0040: 00000000 00000000 00000000 00000000 ................ 295b0050: 00000000 00000000 00000000 00000000 ................ 295b0060: 00000000 00000000 00000000 00000000 ................ 295b0070: 00000000 00000000 00000000 00000000 ................ 295b0080: 00000000 00000000 00000000 00000000 ................ 295b0090: 00000000 00000000 00000000 00000000 ................ 295b00a0: 00000000 00000000 00000000 00000000 ................ 295b00b0: 00000000 00000000 00000000 00000000 ................ 295b00c0: 00000000 00000000 00000000 00000000 ................ 295b00d0: 00000000 00000000 00000000 00000000 ................ 295b00e0: 00000000 00000000 00000000 00000000 ................ 295b00f0: 00000000 00000000 00000000 00000000 ................ StarFive # md 295C0000 0x40 295c0000: 00000000 00000000 00000000 00000000 ................ 295c0010: 00000000 00000000 00000000 00000000 ................ 295c0020: 00000000 00000000 00000000 00000000 ................ 295c0030: 00000000 00000000 00000000 00000000 ................ 295c0040: 00000000 00000000 00000000 00000000 ................ 295c0050: 00000000 00000000 00000000 00000000 ................ 295c0060: 00000000 00000000 00000000 00000000 ................ 295c0070: 00000000 00000000 00000000 00000000 ................ 295c0080: 00000000 00000000 00000000 00000000 ................ 295c0090: 00000000 00000000 00000000 00000000 ................ 295c00a0: 00000000 00000000 00000000 00000000 ................ 295c00b0: 00000000 00000000 00000000 00000000 ................ 295c00c0: 00000000 00000000 00000000 00000000 ................ 295c00d0: 00000000 00000000 00000000 00000000 ................ 295c00e0: 00000000 00000000 00000000 00000000 ................ 295c00f0: 00000000 00000000 00000000 00000000 ................ StarFive # ``` Let's poke the Clock and Reset Registers... # Clock and Reset Registers for Star64 JH7110 Display Controller Read the article... - ["RISC-V Star64 JH7110: Power Up the Display Controller with U-Boot Bootloader"](https://lupyuen.github.io/articles/display3) _VOUT Power is already enabled via PMU. How do we poke the Clock and Reset Registers to power up the Display Controller?_ Based on the [JH7110 Clock Structure](https://doc-en.rvspace.org/JH7110/TRM/JH7110_TRM/clock_structure.html)... ![Clock Structure](https://doc-en.rvspace.org/JH7110/TRM/Image/Drawing/Clock_Structure.svg) Display Controller (vout_crg, top left) is clocked and reset by... - clk_vout_root - mipiphy ref clk - hdmiphy ref clk - clk_vout_src (1228.8M) - clk_vout_axi (614.4M) - clk_vout_ahb (204.8M)(clk_ahb1) - clk_mclk (51.2M) - rstn_vout _How to enable the above clocks and deassert the above reset?_ We refer to the [System Clock and Reset (SYS CRG) Registers](https://doc-en.rvspace.org/JH7110/TRM/JH7110_TRM/sys_crg.html). From [System Memory Map](https://doc-en.rvspace.org/JH7110/TRM/JH7110_TRM/system_memory_map.html), System CRG is at 0x1302_0000... ```text # md 13020000 13020000: 01000000 00000001 00000002 00000000 ................ 13020010: 01000002 01000000 00000003 00000003 ................ 13020020: 00000002 80000000 80000000 00000004 ................ 13020030: 80000000 00000002 00000002 00000002 ................ 13020040: 00000002 0000000c 00000000 00000000 ................ 13020050: 00000002 00000002 80000014 80000010 ................ 13020060: 8000000c 80000000 80000000 80000000 ................ 13020070: 80000000 80000000 80000000 00000006 ................ 13020080: 80000000 80000000 80000000 80000000 ................ 13020090: 80000000 80000000 80000000 80000000 ................ 130200a0: 00000002 00000002 00000002 01000000 ................ 130200b0: 80000000 00000003 00000000 00000000 ................ 130200c0: 00000000 0000000c 00000000 00000000 ................ 130200d0: 00000000 80000000 00000003 00000002 ................ 130200e0: 80000000 80000000 00000000 00000002 ................ 130200f0: 00000000 00000000 00000000 00000000 ................ ``` TODO: Which Clocks and Registers are already enabled / deasserted? Based on the [System Control Registers](https://doc-en.rvspace.org/JH7110/TRM/JH7110_TRM/sys_crg.html), we need to enable these clocks... - Clock AHB 1: Offset 0x28 - MCLK Out: Offset 0x4c - clk_u0_sft7110_noc_bus_clk_cpu_axi: Offset 0x98 - clk_u0_sft7110_noc_bus_clk_axicfg0_axi: Offset 0x9c - clk_u0_dom_vout_top_clk_dom_vout_top_clk_vout_src: Offset 0xe8 - Clock NOC Display AXI: Offset 0xf0 - Clock Video Output AHB: Offset 0xf4 - Clock Video Output AXI: Offset 0xf8 - Clock Video Output HDMI TX0 MCLK: Offset 0xfc (OK looks excessive, but better to enable more clocks than too few!) For the Clock Registers above, we should set Bit 31 (clk_icg) to 1... - 1: Clock enable - 0: Clock disable Here are the U-Boot Commands to enable the clocks... ```text md 13020028 1 mw 13020028 0x80000000 1 md 13020028 1 md 1302004c 1 mw 1302004c 0x80000000 1 md 1302004c 1 md 13020098 1 mw 13020098 0x80000000 1 md 13020098 1 md 1302009c 1 mw 1302009c 0x80000000 1 md 1302009c 1 md 130200e8 1 mw 130200e8 0x80000000 1 md 130200e8 1 md 130200f0 1 mw 130200f0 0x80000000 1 md 130200f0 1 md 130200f4 1 mw 130200f4 0x80000000 1 md 130200f4 1 md 130200f8 1 mw 130200f8 0x80000000 1 md 130200f8 1 md 130200fc 1 mw 130200fc 0x80000000 1 md 130200fc 1 ``` Here's the U-Boot Log... ```text StarFive # md 13020028 1 13020028: 80000000 .... StarFive # mw 13020028 0x80000000 1 StarFive # StarFive # md 13020028 1 13020028: 80000000 .... StarFive # md 1302004c 1 1302004c: 00000000 .... StarFive # mw 1302004c 0x80000000 1 StarFive # StarFive # md 1302004c 1 1302004c: 80000000 .... StarFive # md 13020098 1 13020098: 80000000 .... StarFive # mw 13020098 0x80000000 1 StarFive # StarFive # md 13020098 1 13020098: 80000000 .... StarFive # md 1302009c 1 1302009c: 80000000 .... StarFive # mw 1302009c 0x80000000 1 StarFive # StarFive # md 1302009c 1 1302009c: 80000000 .... StarFive # md 130200e8 1 130200e8: 00000000 .... StarFive # mw 130200e8 0x80000000 1 StarFive # StarFive # md 130200e8 1 130200e8: 80000000 .... StarFive # md 130200f0 1 130200f0: 00000000 .... StarFive # mw 130200f0 0x80000000 1 StarFive # StarFive # md 130200f0 1 130200f0: 80000000 .... StarFive # md 130200f4 1 130200f4: 00000000 .... StarFive # mw 130200f4 0x80000000 1 StarFive # StarFive # md 130200f4 1 130200f4: 80000000 .... StarFive # md 130200f8 1 130200f8: 00000000 .... StarFive # mw 130200f8 0x80000000 1 StarFive # StarFive # md 130200f8 1 130200f8: 80000000 .... StarFive # md 130200fc 1 130200fc: 00000000 .... StarFive # mw 130200fc 0x80000000 1 StarFive # StarFive # md 130200fc 1 130200fc: 80000000 .... StarFive # ``` _What about the Resets?_ Based on the [System Control Registers](https://doc-en.rvspace.org/JH7110/TRM/JH7110_TRM/sys_crg.html), we need to deassert these resets... - Software RESET 1 Address Selector: Offset 0x2fc Bit 11: rstn_u0_dom_vout_top_rstn_dom_vout_top_rstn_vout_src - 1: Assert reset - 0: De-assert reset - SYSCRG RESET Status 0: Offset 0x308 Bit 26: rstn_u0_sft7110_noc_bus_reset_disp_axi_n - 1: Assert reset - 0: De-assert reset - SYSCRG RESET Status 1: Offset 0x30c Bit 11: rstn_u0_dom_vout_top_rstn_dom_vout_top_rstn_vout_src - 1: Assert reset - 0: De-assert reset These are the U-Boot Commands to Deassert the above Resets... ```text md 130202fc 1 mw 130202fc 0x7e7f600 1 md 130202fc 1 md 13020308 1 mw 13020308 0xfb9fffff 1 md 13020308 1 md 1302030c 1 ``` (The last one is already deasserted, no need to change it) Then we check the Display Registers... ```text md 29400000 0x20 md 29480000 0x20 md 29590000 0x20 md 295B0000 0x20 md 295C0000 0x20 ``` Here's the U-Boot Log... ```text StarFive # md 1703000c 1 1703000c: 00000003 .... StarFive # mw 1703000c 0x10 1 StarFive # StarFive # md 1703000c 1 1703000c: 00000010 .... StarFive # mw 17030044 0xff 1 StarFive # StarFive # mw 17030044 0x05 1 StarFive # StarFive # mw 17030044 0x50 1 StarFive # StarFive # md 17030080 1 17030080: 00000013 .... StarFive # mw 13020028 0x80000000 1 StarFive # StarFive # mw 1302004c 0x80000000 1 StarFive # StarFive # mw 13020098 0x80000000 1 StarFive # StarFive # mw 1302009c 0x80000000 1 StarFive # StarFive # mw 130200e8 0x80000000 1 StarFive # StarFive # mw 130200f0 0x80000000 1 StarFive # StarFive # mw 130200f4 0x80000000 1 StarFive # StarFive # mw 130200f8 0x80000000 1 StarFive # StarFive # mw 130200fc 0x80000000 1 StarFive # StarFive # mw 130202fc 0x7e7f600 1 StarFive # StarFive # mw 13020308 0xfb9fffff 1 StarFive # StarFive # md 295B0000 0x40 295b0000: 00000000 000b0000 00000000 00000000 ................ 295b0010: 00000000 00000000 00000000 00000000 ................ 295b0020: 00000000 00000000 00000000 00000000 ................ 295b0030: 00000000 00000000 00000000 00000000 ................ 295b0040: 00000000 00000000 00000000 00000000 ................ 295b0050: 00000000 00000000 00000000 00000000 ................ 295b0060: 00000000 00000000 00000000 00000000 ................ 295b0070: 00000000 00000000 00000000 00000000 ................ 295b0080: 00000000 00000000 00000000 00000000 ................ 295b0090: 00000000 00000000 00000000 00000000 ................ 295b00a0: 00000000 00000000 00000000 00000000 ................ 295b00b0: 00000000 00000000 00000000 00000000 ................ 295b00c0: 00000000 00000000 00000000 00000000 ................ 295b00d0: 00000000 00000000 00000000 00000000 ................ 295b00e0: 00000000 00000000 00000000 00000000 ................ 295b00f0: 00000000 00000000 00000000 00000000 ................ StarFive # md 295C0000 0x40 295c0000: 00000004 00000004 00000004 0000000c ................ 295c0010: 00000000 00000000 00000000 00000000 ................ 295c0020: 00000000 00000000 00000000 00000000 ................ 295c0030: 00000000 00000000 00000000 00000000 ................ 295c0040: 00000000 00000000 00000fff 00000000 ................ 295c0050: 00000000 00000000 00000000 00000000 ................ 295c0060: 00000000 00000000 00000000 00000000 ................ 295c0070: 00000000 00000000 00000000 00000000 ................ 295c0080: 00000000 00000000 00000000 00000000 ................ 295c0090: 00000000 00000000 00000000 00000000 ................ 295c00a0: 00000000 00000000 00000000 00000000 ................ 295c00b0: 00000000 00000000 00000000 00000000 ................ 295c00c0: 00000000 00000000 00000000 00000000 ................ 295c00d0: 00000000 00000000 00000000 00000000 ................ 295c00e0: 00000000 00000000 00000000 00000000 ................ 295c00f0: 00000000 00000000 00000000 00000000 ................ StarFive # ``` The Display Controller Registers are now visible at VOUT_CRG (0x295C_0000) yay! Which means Display Controller is alive yay! ![Star64 JH7110 Display Controller is alive!](https://lupyuen.github.io/images/display3-title.png) The Default Values seem to match [DOM VOUT CRG](https://doc-en.rvspace.org/JH7110/TRM/JH7110_TRM/dom_vout_crg.html). (`clk_tx_esc` should have default `24'hc`, there is a typo in the doc: `24'h12`) ![`clk_tx_esc` should have default `24'hc`, there is a typo in the doc: `24'h12`](https://lupyuen.github.io/images/display3-typo.png) FYI: It hangs when we read U0_HDMITX at 0x2959_0000. Probably because Display Controller is actually powered up. ```text StarFive # mw 13020028 0x80000000 1 StarFive # StarFive # mw 1302004c 0x80000000 1 StarFive # StarFive # mw 13020098 0x80000000 1 StarFive # StarFive # mw 1302009c 0x80000000 1 StarFive # StarFive # mw 130200e8 0x80000000 1 StarFive # StarFive # mw 130200f0 0x80000000 1 StarFive # StarFive # mw 130200f4 0x80000000 1 StarFive # StarFive # mw 130200f8 0x80000000 1 StarFive # StarFive # mw 130200fc 0x80000000 1 StarFive # StarFive # mw 130202fc 0x7e7f600 1 StarFive # StarFive # mw 13020308 0xfb9fffff 1 StarFive # StarFive # md 29400000 0x40 29400000: 00000000 00000000 00000000 00000000 ................ 29400010: 00000000 00000000 00000000 00000000 ................ 29400020: 00000000 00000000 00000000 00000000 ................ 29400030: 00000000 00000000 00000000 00000000 ................ 29400040: 00000000 00000000 00000000 00000000 ................ 29400050: 00000000 00000000 00000000 00000000 ................ 29400060: 00000000 00000000 00000000 00000000 ................ 29400070: 00000000 00000000 00000000 00000000 ................ 29400080: 00000000 00000000 00000000 00000000 ................ 29400090: 00000000 00000000 00000000 00000000 ................ 294000a0: 00000000 00000000 00000000 00000000 ................ 294000b0: 00000000 00000000 00000000 00000000 ................ 294000c0: 00000000 00000000 00000000 00000000 ................ 294000d0: 00000000 00000000 00000000 00000000 ................ 294000e0: 00000000 00000000 00000000 00000000 ................ 294000f0: 00000000 00000000 00000000 00000000 ................ StarFive # md 29480000 0x40 29480000: 00000000 00000000 00000000 00000000 ................ 29480010: 00000000 00000000 00000000 00000000 ................ 29480020: 00000000 00000000 00000000 00000000 ................ 29480030: 00000000 00000000 00000000 00000000 ................ 29480040: 00000000 00000000 00000000 00000000 ................ 29480050: 00000000 00000000 00000000 00000000 ................ 29480060: 00000000 00000000 00000000 00000000 ................ 29480070: 00000000 00000000 00000000 00000000 ................ 29480080: 00000000 00000000 00000000 00000000 ................ 29480090: 00000000 00000000 00000000 00000000 ................ 294800a0: 00000000 00000000 00000000 00000000 ................ 294800b0: 00000000 00000000 00000000 00000000 ................ 294800c0: 00000000 00000000 00000000 00000000 ................ 294800d0: 00000000 00000000 00000000 00000000 ................ 294800e0: 00000000 00000000 00000000 00000000 ................ 294800f0: 00000000 00000000 00000000 00000000 ................ StarFive # md 29590000 0x40 ``` # Automate U-Boot to Power Up Display Controller Read the article... - ["RISC-V Star64 JH7110: Power Up the Display Controller with U-Boot Bootloader"](https://lupyuen.github.io/articles/display3) _That's a long list of U-Boot Commands. Can we automate this?_ ```text mw 1703000c 0x10 1 mw 17030044 0xff 1 mw 17030044 0x05 1 mw 17030044 0x50 1 mw 13020028 0x80000000 1 mw 1302004c 0x80000000 1 mw 13020098 0x80000000 1 mw 1302009c 0x80000000 1 mw 130200e8 0x80000000 1 mw 130200f0 0x80000000 1 mw 130200f4 0x80000000 1 mw 130200f8 0x80000000 1 mw 130200fc 0x80000000 1 mw 130202fc 0x7e7f600 1 mw 13020308 0xfb9fffff 1 md 295C0000 0x20 ``` Sure can! Run this in U-Boot... ```text ## Create the command to power up the Video Output setenv video_on 'mw 1703000c 0x10 1 ; mw 17030044 0xff 1 ; mw 17030044 0x05 1 ; mw 17030044 0x50 1 ; mw 13020028 0x80000000 1 ; mw 1302004c 0x80000000 1 ; mw 13020098 0x80000000 1 ; mw 1302009c 0x80000000 1 ; mw 130200e8 0x80000000 1 ; mw 130200f0 0x80000000 1 ; mw 130200f4 0x80000000 1 ; mw 130200f8 0x80000000 1 ; mw 130200fc 0x80000000 1 ; mw 130202fc 0x7e7f600 1 ; mw 13020308 0xfb9fffff 1 ; md 295C0000 0x20 ; ' ## Check that it's correct printenv video_on ## Save it for future reboots saveenv ## Run the command to power up the Video Output run video_on ``` (The `run` feels a bit like BASIC) We should see... ```text StarFive # run video_on 295c0000: 00000004 00000004 00000004 0000000c ................ 295c0010: 00000000 00000000 00000000 00000000 ................ 295c0020: 00000000 00000000 00000000 00000000 ................ 295c0030: 00000000 00000000 00000000 00000000 ................ 295c0040: 00000000 00000000 00000fff 00000000 ................ 295c0050: 00000000 00000000 00000000 00000000 ................ 295c0060: 00000000 00000000 00000000 00000000 ................ 295c0070: 00000000 00000000 00000000 00000000 ................ ``` So much easier! Maybe we could use this to render something to the HDMI Display! (Before converting to C for NuttX) _How will we test this in NuttX?_ Probably inside `board_late_initialize` like this: [jh7110_appinit.c](https://github.com/lupyuen2/wip-pinephone-nuttx/blob/hdmi/boards/risc-v/jh7110/star64/src/jh7110_appinit.c#L154-L215) ```c 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 // Verfy that Display Controller is down uint32_t val = getreg32(0x295C0000); DEBUGASSERT(val == 0); // Power up the Display Controller // TODO: Switch to constants putreg32(0x10, 0x1703000c); putreg32(0xff, 0x17030044); putreg32(0x05, 0x17030044); putreg32(0x50, 0x17030044); putreg32(0x80000000, 0x13020028); putreg32(0x80000000, 0x1302004c); putreg32(0x80000000, 0x13020098); putreg32(0x80000000, 0x1302009c); putreg32(0x80000000, 0x130200e8); putreg32(0x80000000, 0x130200f0); putreg32(0x80000000, 0x130200f4); putreg32(0x80000000, 0x130200f8); putreg32(0x80000000, 0x130200fc); // Software RESET 1 Address Selector: Offset 0x2fc // Clear Bit 11: rstn_u0_dom_vout_top_rstn_dom_vout_top_rstn_vout_src modifyreg32(0x130202fc, 1 << 11, 0); // Addr, Clear Bits, Set Bits // SYSCRG RESET Status 0: Offset 0x308 // Clear Bit 26: rstn_u0_sft7110_noc_bus_reset_disp_axi_n modifyreg32(0x13020308, 1 << 26, 0); // Addr, Clear Bits, Set Bits // Verfy that Display Controller is up val = getreg32(0x295C0000); DEBUGASSERT(val == 4); // Test HDMI int test_hdmi(void); int ret = test_hdmi(); DEBUGASSERT(ret == 0); } // Display Subsystem Base Address // https://doc-en.rvspace.org/JH7110/TRM/JH7110_TRM/memory_map_display.html #define DISPLAY_BASE_ADDRESS (0x29400000) // DOM VOUT Control Registers // https://doc-en.rvspace.org/JH7110/TRM/JH7110_TRM/memory_map_display.html #define CRG_BASE_ADDRESS (DISPLAY_BASE_ADDRESS + 0x1C0000) // Enable Clock // https://doc-en.rvspace.org/JH7110/TRM/JH7110_TRM/dom_vout_crg.html #define CLK_ICG (1 << 31) // https://doc-en.rvspace.org/JH7110/TRM/JH7110_TRM/dom_vout_crg.html #define clk_u0_dc8200_clk_pix0 (CRG_BASE_ADDRESS + 0x1c) // Test HDMI int test_hdmi(void) { ... } ``` # Read the Star64 JH7110 Display Controller Registers with U-Boot Bootloader Read the article... - ["RISC-V Star64 JH7110: Power Up the Display Controller with U-Boot Bootloader"](https://lupyuen.github.io/articles/display3) From [jh7110_appinit.c](https://github.com/lupyuen2/wip-pinephone-nuttx/blob/hdmi/boards/risc-v/jh7110/star64/src/jh7110_appinit.c#L154-L215): ```c // Display Subsystem Base Address // https://doc-en.rvspace.org/JH7110/TRM/JH7110_TRM/memory_map_display.html #define DISPLAY_BASE_ADDRESS (0x29400000) // DOM VOUT Control Registers // https://doc-en.rvspace.org/JH7110/TRM/JH7110_TRM/memory_map_display.html #define CRG_BASE_ADDRESS (DISPLAY_BASE_ADDRESS + 0x1C0000) // DOM VOUT Control Registers // https://doc-en.rvspace.org/JH7110/TRM/JH7110_TRM/dom_vout_crg.html #define clk_u0_dc8200_clk_axi (CRG_BASE_ADDRESS + 0x10) #define clk_u0_dc8200_clk_core (CRG_BASE_ADDRESS + 0x14) #define clk_u0_dc8200_clk_ahb (CRG_BASE_ADDRESS + 0x18) #define clk_u0_dc8200_clk_pix0 (CRG_BASE_ADDRESS + 0x1c) #define clk_u0_dc8200_clk_pix1 (CRG_BASE_ADDRESS + 0x20) #define clk_u0_hdmi_tx_clk_mclk (CRG_BASE_ADDRESS + 0x3c) #define clk_u0_hdmi_tx_clk_bclk (CRG_BASE_ADDRESS + 0x40) #define clk_u0_hdmi_tx_clk_sys (CRG_BASE_ADDRESS + 0x44) #define CLK_ICG (1 << 31) #define Software_RESET_assert0_addr_assert_sel (CRG_BASE_ADDRESS + 0x38) #define rstn_u0_dc8200_rstn_axi (1 << 0) #define rstn_u0_dc8200_rstn_ahb (1 << 1) #define rstn_u0_dc8200_rstn_core (1 << 2) #define rstn_u0_hdmi_tx_rstn_hdmi (1 << 9) ``` U-Boot Commands: ```text run video_on mw 295C0010 0x80000000 1 mw 295C0014 0x80000000 1 mw 295C0018 0x80000000 1 mw 295C001c 0x80000000 1 mw 295C0020 0x80000000 1 mw 295C003c 0x80000000 1 mw 295C0040 0x80000000 1 mw 295C0044 0x80000000 1 md 295C0038 1 mw 295C0038 0 1 md 295C0038 1 md 295C004c 1 mw 295C004c 0 1 md 295C004c 1 ## TODO: Why is Reset at 295C0048? md 295C0048 1 mw 295C0048 0 1 md 295C0048 1 md 29400000 0x20 md 29480000 0x20 md 295C0000 0x20 ``` U-Boot Log: ```text StarFive # run video_on 295c0000: 00000004 00000004 00000004 0000000c ................ 295c0010: 00000000 00000000 00000000 00000000 ................ 295c0020: 00000000 00000000 00000000 00000000 ................ 295c0030: 00000000 00000000 00000000 00000000 ................ 295c0040: 00000000 00000000 00000fff 00000000 ................ 295c0050: 00000000 00000000 00000000 00000000 ................ 295c0060: 00000000 00000000 00000000 00000000 ................ 295c0070: 00000000 00000000 00000000 00000000 ................ StarFive # mw 295C0010 0x80000000 1 StarFive # StarFive # mw 295C0014 0x80000000 1 StarFive # StarFive # mw 295C0018 0x80000000 1 StarFive # StarFive # mw 295C001c 0x80000000 1 StarFive # StarFive # mw 295C0020 0x80000000 1 StarFive # StarFive # mw 295C003c 0x80000000 1 StarFive # StarFive # mw 295C0040 0x80000000 1 StarFive # StarFive # mw 295C0044 0x80000000 1 StarFive # StarFive # md 29400000 0x20 29400000: 00000000 00000000 00000000 00000000 ................ 29400010: 00000000 00000000 00000000 00000000 ................ 29400020: 00000000 00000000 00000000 00000000 ................ 29400030: 00000000 00000000 00000000 00000000 ................ 29400040: 00000000 00000000 00000000 00000000 ................ 29400050: 00000000 00000000 00000000 00000000 ................ 29400060: 00000000 00000000 00000000 00000000 ................ 29400070: 00000000 00000000 00000000 00000000 ................ StarFive # mw 295C0038 0 1 StarFive # StarFive # md 29400000 0x20 29400000: 00000000 00000000 00000000 00000000 ................ 29400010: 00000000 00000000 00000000 00000000 ................ 29400020: 00000000 00000000 00000000 00000000 ................ 29400030: 00000000 00000000 00000000 00000000 ................ 29400040: 00000000 00000000 00000000 00000000 ................ 29400050: 00000000 00000000 00000000 00000000 ................ 29400060: 00000000 00000000 00000000 00000000 ................ 29400070: 00000000 00000000 00000000 00000000 ................ StarFive # mw 295C004c 0 1 StarFive # StarFive # md 29400000 0x20 29400000: 00000000 00000000 00000000 00000000 ................ 29400010: 00000000 00000000 00000000 00000000 ................ 29400020: 00000000 00000000 00000000 00000000 ................ 29400030: 00000000 00000000 00000000 00000000 ................ 29400040: 00000000 00000000 00000000 00000000 ................ 29400050: 00000000 00000000 00000000 00000000 ................ 29400060: 00000000 00000000 00000000 00000000 ................ 29400070: 00000000 00000000 00000000 00000000 ................ StarFive # mw 295C0048 0 1 StarFive # StarFive # md 29400000 0x20 29400000: 00000900 80010000 00222200 00000000 .........""..... 29400010: 00000000 00000004 14010000 000b4b41 ............AK.. 29400020: 00008200 00005720 20210316 16015600 .... W....! .V.. 29400030: 0000030e a0600084 00000000 00000000 ......`......... 29400040: 00000000 00000000 00000000 00000000 ................ 29400050: 00000000 00000000 00000000 00000000 ................ 29400060: 00000000 00000000 00000000 00000000 ................ 29400070: 00000000 08050000 00000002 00000000 ................ StarFive # md 29480000 0x20 29480000: 00000000 00000000 00000000 00000000 ................ 29480010: 00000000 00000000 00000000 00000000 ................ 29480020: 00000000 00000000 00000000 00000000 ................ 29480030: 00000000 00000000 00000000 00000000 ................ 29480040: 00000000 00000000 00000000 00000000 ................ 29480050: 00000000 00000000 00000000 00000000 ................ 29480060: 00000000 00000000 00000000 00000000 ................ 29480070: 00000000 00000000 00000000 00000000 ................ StarFive # md 295C0000 0x20 295c0000: 00000004 00000004 00000004 0000000c ................ 295c0010: 80000000 80000000 80000000 80000000 ................ 295c0020: 80000000 00000000 00000000 00000000 ................ 295c0030: 00000000 00000000 00000000 80000000 ................ 295c0040: 80000000 80000000 00000000 00000777 ............w... 295c0050: 00000000 00000000 00000000 00000000 ................ 295c0060: 00000000 00000000 00000000 00000000 ................ 295c0070: 00000000 00000000 00000000 00000000 ................ StarFive # ``` TODO: Why is Reset at 295C0048? TODO: Did we overwrite any default values for Clock Mux and Multiplier? [Revision and Chip ID](https://github.com/starfive-tech/linux/blob/JH7110_VisionFive2_devel/drivers/gpu/drm/verisilicon/vs_dc_hw.c#L1301-L1361) are at... ```c #define DC_HW_REVISION 0x0024 #define DC_HW_CHIP_CID 0x0030 ``` U-Boot Commands... ```text ## Dump the Hardware Revision md 29400024 1 ## Dump the Chip ID md 29400030 1 ``` We see... ```text StarFive # md 29400024 1 29400024: 00005720 W.. StarFive # md 29400030 1 29400030: 0000030e .... StarFive # ``` Based on [vs_dc_hw.c](https://github.com/starfive-tech/linux/blob/JH7110_VisionFive2_devel/drivers/gpu/drm/verisilicon/vs_dc_hw.c#L1301-L1361)... - revision = 0x5720 Which means hw.rev = DC_REV_0 - chip id = 0x30e Which looks correct yay! FYI: 295C004c shows 0x777, but can't be cleared. Why? Is it showing the current Reset Status? ```text StarFive # md 295C0038 1 295c0038: 00000000 .... StarFive # md 295C004c 1 295c004c: 00000777 w... StarFive # md 295C0048 1 295c0048: 00000000 .... StarFive # mw 295C004c 0 1 StarFive # StarFive # md 295C004c 1 295c004c: 00000777 w... StarFive # md 29400000 0x20 29400000: 00000900 80010000 00222200 00000000 .........""..... 29400010: 00000000 00000004 14010000 000b4b41 ............AK.. 29400020: 00008200 00005720 20210316 16015600 .... W....! .V.. 29400030: 0000030e a0600084 00000000 00000000 ......`......... 29400040: 00000000 00000000 00000000 00000000 ................ 29400050: 00000000 00000000 00000000 00000000 ................ 29400060: 00000000 00000000 00000000 00000000 ................ 29400070: 00000000 08050000 00000002 00000000 ................ StarFive # md 29480000 0x20 29480000: 00000000 00000000 00000000 00000000 ................ 29480010: 00000000 00000000 00000000 00000000 ................ 29480020: 00000000 00000000 00000000 00000000 ................ 29480030: 00000000 00000000 00000000 00000000 ................ 29480040: 00000000 00000000 00000000 00000000 ................ 29480050: 00000000 00000000 00000000 00000000 ................ 29480060: 00000000 00000000 00000000 00000000 ................ 29480070: 00000000 00000000 00000000 00000000 ................ StarFive # md 295C0000 0x20 295c0000: 00000004 00000004 00000004 0000000c ................ 295c0010: 80000000 80000000 80000000 80000000 ................ 295c0020: 80000000 00000000 00000000 00000000 ................ 295c0030: 00000000 00000000 00000000 80000000 ................ 295c0040: 80000000 80000000 00000000 00000777 ............w... 295c0050: 00000000 00000000 00000000 00000000 ................ 295c0060: 00000000 00000000 00000000 00000000 ................ 295c0070: 00000000 00000000 00000000 00000000 ................ StarFive # ``` U-Boot Script: ```text ## Create the command to power up the Display Controller setenv display_on 'mw 295C0010 0x80000000 1 ; mw 295C0014 0x80000000 1 ; mw 295C0018 0x80000000 1 ; mw 295C001c 0x80000000 1 ; mw 295C0020 0x80000000 1 ; mw 295C003c 0x80000000 1 ; mw 295C0040 0x80000000 1 ; mw 295C0044 0x80000000 1 ; mw 295C0048 0 1 ; md 29400000 0x20 ; ' ## Check that it's correct printenv display_on ## Save it for future reboots saveenv ## Run the command to power up the Video Output run video_on ## Run the command to power up the Display Controller run display_on ``` We should see... ```text StarFive # run video_on 295c0000: 00000004 00000004 00000004 0000000c ................ 295c0010: 00000000 00000000 00000000 00000000 ................ 295c0020: 00000000 00000000 00000000 00000000 ................ 295c0030: 00000000 00000000 00000000 00000000 ................ 295c0040: 00000000 00000000 00000fff 00000000 ................ 295c0050: 00000000 00000000 00000000 00000000 ................ 295c0060: 00000000 00000000 00000000 00000000 ................ 295c0070: 00000000 00000000 00000000 00000000 ................ StarFive # run display_on 29400000: 00000900 80010000 00222200 00000000 .........""..... 29400010: 00000000 00000004 14010000 000b4b41 ............AK.. 29400020: 00008200 00005720 20210316 16015600 .... W....! .V.. 29400030: 0000030e a0600084 00000000 00000000 ......`......... 29400040: 00000000 00000000 00000000 00000000 ................ 29400050: 00000000 00000000 00000000 00000000 ................ 29400060: 00000000 00000000 00000000 00000000 ................ 29400070: 00000000 08050000 00000002 00000000 ................ ``` TODO: NuttX Log ```text ## 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 BCboard_late_initialize: revision=0x5720, chip_id=0x30e NuttShell (NSH) NuttX-12.0.3 nsh> ``` # JH7110 System Configuration Registers TODO [SYS SYSCON: System Configuration Registers](https://doc-en.rvspace.org/JH7110/TRM/JH7110_TRM/sys_syscon.html) From [System Memory Map](https://doc-en.rvspace.org/JH7110/TRM/JH7110_TRM/system_memory_map.html), System SYSCON is at 0x1303_0000 ```text # md 13030000 13030000: 00000000 00000000 00000000 00000000 ................ 13030010: 00000000 00d54d54 034fea80 0000007d ....TM....O.}... 13030020: 45555555 042ba603 45e00000 00c7a60c UUUE..+....E.... 13030030: 45333333 00000002 00000000 00000000 333E............ 13030040: 00000000 00000000 00000000 00000000 ................ 13030050: 00000000 00000000 00000000 00000000 ................ 13030060: 00000002 00000000 2a000000 00000000 ...........*.... 13030070: 2a000000 00000000 2a000000 00000000 ...*.......*.... 13030080: 2a000000 01aa8000 00000d54 6aa00000 ...*....T......j 13030090: 00000004 00000000 00000000 00042600 .............&.. 130300a0: 00000000 00000000 00000000 00000000 ................ 130300b0: 00000000 00000000 00000000 00000000 ................ 130300c0: 00000000 00000000 00000000 00000000 ................ 130300d0: 00000000 00000000 00000000 00000000 ................ 130300e0: 00000000 00000000 00000000 00000000 ................ 130300f0: 00000000 00000000 00000000 00000000 ................ ``` TODO: Which SYSCON Registers are already configured? # JH7110 Bus Connection TODO From [Bus Connection](https://doc-en.rvspace.org/JH7110/TRM/JH7110_TRM/bus_connection.html): ![Bus Connection](https://doc-en.rvspace.org/JH7110/TRM/Image/RD/JH7110/stg_mtrx_connection17.png) TODO: Do we need to bother with Bus Connections? # Power Up the I2C Controller for Star64 JH7110 Let's explore the I2C Controller for Star64 JH7110. We need the I2C for controlling the [X-Powers AXP15060 PMIC](https://lupyuen.github.io/articles/display3#appendix-jh7110-display-driver), which will be used for HDMI Output. According to the [I2C Device Tree Configuration](https://doc-en.rvspace.org/VisionFive2/DG_I2C/JH7110_SDK/i2c_source_code.html): I2C Port 0 Base Address is 0x10030000, with range 0x10000. There are 7 I2C Ports: I2C0 to I2C6. From [JH7110 System Memory Map](https://doc-en.rvspace.org/JH7110/TRM/JH7110_TRM/system_memory_map.html)... | Start Address | End Address | Size | Attribute | Device/Description | |-|-|-|-|-| | 0x00_1003_0000 | 0x00_1003_FFFF | 64KB | RW A | I2C0 | 0x00_1004_0000 | 0x00_1004_FFFF | 64KB | RW A | I2C1 | 0x00_1005_0000 | 0x00_1005_FFFF | 64KB | RW A | I2C2 | 0x00_1203_0000 | 0x00_1203_FFFF | 64KB | RW A | I2C3 | 0x00_1204_0000 | 0x00_1204_FFFF | 64KB | RW A | I2C4 | 0x00_1205_0000 | 0x00_1205_FFFF | 64KB | RW A | I2C5 | 0x00_1206_0000 | 0x00_1206_FFFF | 64KB | RW A | I2C6 Let's dump the I2C Registers with U-Boot Bootloader... ```bash $ md 0x10030000 10030000: 00000000 00000000 00000000 00000000 ................ 10030010: 00000000 00000000 00000000 00000000 ................ 10030020: 00000000 00000000 00000000 00000000 ................ 10030030: 00000000 00000000 00000000 00000000 ................ 10030040: 00000000 00000000 00000000 00000000 ................ 10030050: 00000000 00000000 00000000 00000000 ................ 10030060: 00000000 00000000 00000000 00000000 ................ 10030070: 00000000 00000000 00000000 00000000 ................ 10030080: 00000000 00000000 00000000 00000000 ................ 10030090: 00000000 00000000 00000000 00000000 ................ 100300a0: 00000000 00000000 00000000 00000000 ................ 100300b0: 00000000 00000000 00000000 00000000 ................ 100300c0: 00000000 00000000 00000000 00000000 ................ 100300d0: 00000000 00000000 00000000 00000000 ................ 100300e0: 00000000 00000000 00000000 00000000 ................ 100300f0: 00000000 00000000 00000000 00000000 ................ ``` _The I2C Registers are empty?_ That's because we haven't powered up the I2C Controller. According to the [I2C Device Tree Configuration](https://doc-en.rvspace.org/VisionFive2/DG_I2C/JH7110_SDK/i2c_source_code.html), the I2C Clocks and I2C Resets are... ```text clocks = <&clkgen JH7110_I2C0_CLK_CORE>, <&clkgen JH7110_I2C0_CLK_APB>; clock-names = "ref", "pclk"; resets = <&rstgen RSTN_U0_DW_I2C_APB>; interrupts = <35>; ``` _How to enable Clocks JH7110_I2C0_CLK_CORE, JH7110_I2C0_CLK_APB?_ _And deassert Reset RSTN_U0_DW_I2C_APB?_ From [JH7110 Clock Structure](https://doc-en.rvspace.org/JH7110/TRM/JH7110_TRM/clock_structure.html) we see that... - I2C is clocked by clk_apb0 and clk_apb12 - There are 7 I2C Ports (named U0 to U6, as we'll soon see) From [JH7110 System CRG Clocks and Resets](https://doc-en.rvspace.org/JH7110/TRM/JH7110_TRM/sys_crg.html), here are the I2C Clocks and Resets... - System CRG Base Address: 0x00_1302_0000 - U0 Clock I2C APB (Offset 0x228) Bit 31: clk_icg (Default 0) - 1: Clock enable - 0: Clock disable - U1 Clock I2C APB (Offset 0x22c) Bit 31: clk_icg (Default 0) - 1: Clock enable - 0: Clock disable - U2 Clock I2C APB (Offset 0x230) Bit 31: clk_icg (Default 0) - 1: Clock enable - 0: Clock disable - U3 Clock I2C APB (Offset 0x234) Bit 31: clk_icg (Default 0) - 1: Clock enable - 0: Clock disable - U4 Clock I2C APB (Offset 0x238) Bit 31: clk_icg (Default 0) - 1: Clock enable - 0: Clock disable - U5 Clock I2C APB (Offset 0x23c) Bit 31: clk_icg (Default 0) - 1: Clock enable - 0: Clock disable - U6 Clock I2C APB (Offset 0x240) Bit 31: clk_icg (Default 0) - 1: Clock enable - 0: Clock disable - Software RESET 2 Address Selector (Offset 0x300) Bit 12: rstn_u0_i2c_rstn_apb (Default 0) - 1: Assert reset - 0: De-assert reset Bit 13: rstn_u1_i2c_rstn_apb (Default 1) - 1: Assert reset - 0: De-assert reset Bit 14: rstn_u2_i2c_rstn_apb (Default 1) - 1: Assert reset - 0: De-assert reset Bit 15: rstn_u3_i2c_rstn_apb (Default 1) - 1: Assert reset - 0: De-assert reset Bit 16: rstn_u4_i2c_rstn_apb (Default 1) - 1: Assert reset - 0: De-assert reset Bit 17: rstn_u5_i2c_rstn_apb (Default 1) - 1: Assert reset - 0: De-assert reset Bit 18: rstn_u6_i2c_rstn_apb (Default 1) - 1: Assert reset - 0: De-assert reset (What are U0 to U6? They represent the 7 I2C Ports in JH7110) In U-Boot, we inspect the System SYSCRG Registers for I2C... ```bash $ md 0x13020228 7 13020228: 00000000 00000000 00000000 00000000 ................ 13020238: 00000000 80000000 00000000 ............ $ md 0x13020300 1 13020300: ffe5efcc .... ``` Which says that... - All I2C Clocks are Disabled (except U5) - All I2C Resets are Asserted (except U0 and U5) TODO: Why is I2C Port 5 enabled? Is it used by U-Boot? Let's enable all the I2C Clocks and deassert all the I2C Resets... ```bash ## Enable the I2C Clocks $ mw 0x13020228 0x80000000 7 $ md 0x13020228 7 13020228: 80000000 80000000 80000000 80000000 ................ 13020238: 80000000 80000000 80000000 ............ ## Deassert the I2C Resets $ mw 0x13020300 0xffe00fcc 1 $ md 0x13020300 1 13020300: ffe00fcc .... ## Check the System SYSCON for I2C $ md 0x13030014 1 13030014: 00d54d54 TM.. $ md 0x1303009c 1 1303009c: 00042600 .&.. ## Dump the I2C Registers for I2C Port 0 $ md 0x10030000 0x40 10030000: 0000007f 00000055 00000055 00000001 ....U...U....... 10030010: 00000000 00000190 000001d6 0000003c ............<... 10030020: 00000082 00000006 00000010 00000000 ................ 10030030: 000008ff 00000000 00000000 00000000 ................ 10030040: 00000000 00000000 00000000 00000000 ................ 10030050: 00000000 00000000 00000000 00000000 ................ 10030060: 00000000 00000000 00000000 00000004 ................ 10030070: 00000006 00000000 00000000 00000001 ................ 10030080: 00000000 00000000 00000000 00000000 ................ 10030090: 00000000 00000064 00000001 00000000 ....d........... 100300a0: 00000005 00000001 00000000 00000000 ................ 100300b0: 00000000 00000000 00000000 00000000 ................ 100300c0: 00000000 00000000 00000000 00000000 ................ 100300d0: 00000000 00000000 00000000 00000000 ................ 100300e0: 00000000 00000000 00000000 00000000 ................ 100300f0: 00000000 001f1fee 3230302a 44570140 ........*002@.WD ``` Yep I2C Port 0 is alive yay! TODO: What about [SYS SYSCON](https://doc-en.rvspace.org/JH7110/TRM/JH7110_TRM/sys_syscon.html): - Base Address of System SYSCON: 0x00_1303_0000 - SYS SYSCONSAIF SYSCFG 20 (Offset 0x14, Default 0xd54d54) - Bit 25: u0_i2c_ic_en - I2C interface enable (Read-Only, default 0x0) - SYS SYSCONSAIF SYSCFG 156 (Offset 0x9c, Default 0x400) - Bit 00: u1_i2c_ic_en - I2C interface enable (Read-Only, Default 0) - Bit 15: u2_i2c_ic_en - I2C interface enable (Read-Only, Default 0) - Bit 16: u3_i2c_ic_en - I2C interface enable (Read-Only, Default 0) - Bit 17: u4_i2c_ic_en - I2C interface enable (Read-Only, Default 0) - Bit 18: u5_i2c_ic_en - I2C interface enable (Read-Only, Default 0) - Bit 19: u6_i2c_ic_en - I2C interface enable (Read-Only, Default 0) In U-Boot, we inspect the System SYSCON Registers for I2C... ```bash $ md 0x13030014 1 13030014: 00d54d54 TM.. $ md 0x1303009c 1 1303009c: 00042600 ``` Which says that... - I2C Port U5 is enabled - Other I2C Ports U0 to U6 are disabled TODO: Let's check again after enabling I2C FYI: I2C Port U0 is connected to these devices, according to the [JH7110 Device Tree](https://doc-en.rvspace.org/VisionFive2/DG_I2C/JH7110_SDK/i2c_source_code.html)... ```text &i2c0 { clock-frequency = <100000>; i2c-sda-hold-time-ns = <300>; i2c-sda-falling-time-ns = <510>; i2c-scl-falling-time-ns = <510>; auto_calc_scl_lhcnt; pinctrl-names = "default"; pinctrl-0 = <&i2c0_pins>; status = "disabled"; ac108_a: ac108@3b { // i2c_client 1 compatible = "x-power,ac108_0"; reg = <0x3b>; #sound-dai-cells = <0>; data-protocol = <0>; }; wm8960: codec@1a { //i2c_client 2 compatible = "wlf,wm8960"; reg = <0x1a>; #sound-dai-cells = <0>; wlf,shared-lrclk; }; }; ``` - X-Powers AC108 is the ADC for I2S Digital Audio - NXP WM8960 is the Stereo Codec for Digital Audio TODO: What are these messages from U-Boot? ```text clk u5_dw_i2c_clk_core already disabled clk u5_dw_i2c_clk_apb already disabled ``` [(Source)](https://gist.github.com/lupyuen/1e009a3343da70257d6f24400339053f#file-nuttx-scheme-star64-log-L160-L165) # Explore the I2C Controller for Star64 JH7110 TODO: Not linked in any section, appears when we search for "I2C": [I2C Register Description](https://doc-en.rvspace.org/JH7110/TRM/JH7110_TRM/register_descript_i2c.html) (Looks garbled) JH7110 I2C is based on [DesignWare I2C](https://github.com/torvalds/linux/blob/master/drivers/i2c/busses/i2c-designware-core.h) From [DesignWare DW_apb_i2c Databook](https://github.com/lupyuen/nuttx-star64/blob/main/designware-apb-i2c.pdf) Page 252: "Figure 8-2: Flowchart for DW_apb_i2c Master" ```text Write 0 to IC_ENABLE to disable DW_apb_i2c Program IC_CON register fields as required: 1. Set IC_SLAVE_DISABLE to 1 – Slave disabled 2. Set IC_RESTART_EN to 1 – Enable restart mode 3. Set IC_10BITADDR_MASTER to 0 – 7-bit addressing 4. Set IC_10BITADDR_SLAVE to 0 – 7-bit addressing 5. Set IC_MAX_SPEED_MODE to 1 – Standard mode 6. Set IC_MASTER_MODE to 1 – Master enabled Set address of target Slave by writing it to TAR Write to IC_SS_HCNT to set HIGH period of SCL Write to IC_SS_LCNT to set LOW period of SCL Write to IC_INTR_MASK to enable all interrupts Write to IC_RX_TL to set Rx FIFO threshold level Write to IC_TX_TL to set Tx FIFO threshold level Write 1 to IC_ENABLE to enable DW_apb_i2c Write to IC_DATA_CMD to push Write command and write data or Read command Tx FIFO Command is Write? N: RX_FULL interrupt asserted? Y: Read IC_DATA_CMD[7:0] to retrieve received byte Y: TX_EMPTY interrupt asserted? Y: More commands to send? Y: Write to IC_DATA_CMD... N: Is IC_STATUS[5] (MST_ACTIVITY) = 0? Y: Write 0 to IC_ENABLE to disable DW_apb_i2c ``` Which is implemented by this NuttX Driver: - [arch/arm/src/cxd56xx/cxd56_i2c.c](https://github.com/apache/nuttx/blob/master/arch/arm/src/cxd56xx/cxd56_i2c.c) - [arch/arm/src/cxd56xx/hardware/cxd56_i2c.h](https://github.com/apache/nuttx/blob/master/arch/arm/src/cxd56xx/hardware/cxd56_i2c.h) - Bit Bang Version: [cxd56_i2c_bitbang.c](https://github.com/apache/nuttx/blob/master/arch/arm/src/cxd56xx/cxd56_i2c_bitbang.c) and [cxd56_i2c_bitbang.h](https://github.com/apache/nuttx/blob/master/arch/arm/src/cxd56xx/cxd56_i2c_bitbang.h) TODO: Test this with U-Boot TODO: Read I2C Controller with U-Boot From [DesignWare DW_apb_i2c Databook](https://github.com/lupyuen/nuttx-star64/blob/main/designware-apb-i2c.pdf) Page 154: - IC_CON (Offset 0x00): I2C Control Actual Value: 0x7f - IC_TAR (Offset 0x04): I2C Target Address Default Value: 0x55 Actual Value: 0x55 - IC_SAR (Offset 0x08): I2C Slave Address Default Value: 0x55 Actual Value: 0x55 - IC_ENABLE (Offset 0x6C): I2C Enable Actual Value: 0 - IC_STATUS (Offset 0x70): I2C Status register Actual Value: 6 - IC_COMP_PARAM_1 (Offset 0xf4): Component Parameter Register Actual Value: 0x1f1fee - IC_COMP_VERSION (Offset 0xf8): Component Version ID Reset Value: "See the releases table in the AMBA 2 release notes" (?) Actual Value: 0x3230302a - IC_COMP_TYPE (Offset 0xfc): DesignWare Component Type Register Reset Value: 0x44570140 Actual Value: Yep correct - IC_DEVICE_ID (Offset 0xb8): I2C Device ID Reset Value: IC_DEVICE_ID_VALUE Actual Value: 0 The above Actual Values were retrieved from U-Boot... ```bash ## Omitted: Power Up the I2C Controller ... ## Dump the I2C Registers for I2C Port 0 $ md 0x10030000 0x40 10030000: 0000007f 00000055 00000055 00000001 ....U...U....... 10030010: 00000000 00000190 000001d6 0000003c ............<... 10030020: 00000082 00000006 00000010 00000000 ................ 10030030: 000008ff 00000000 00000000 00000000 ................ 10030040: 00000000 00000000 00000000 00000000 ................ 10030050: 00000000 00000000 00000000 00000000 ................ 10030060: 00000000 00000000 00000000 00000004 ................ 10030070: 00000006 00000000 00000000 00000001 ................ 10030080: 00000000 00000000 00000000 00000000 ................ 10030090: 00000000 00000064 00000001 00000000 ....d........... 100300a0: 00000005 00000001 00000000 00000000 ................ 100300b0: 00000000 00000000 00000000 00000000 ................ 100300c0: 00000000 00000000 00000000 00000000 ................ 100300d0: 00000000 00000000 00000000 00000000 ................ 100300e0: 00000000 00000000 00000000 00000000 ................ 100300f0: 00000000 001f1fee 3230302a 44570140 ........*002@.WD ``` TODO: Read PMIC with U-Boot TODO: Which I2C ports are in use? ```bash ## Omitted: Power Up the I2C Controller ... ## Dump the I2C Registers for I2C Port 1 $ md 0x10040000 0x40 10040000: 0000007f 00000055 00000055 00000001 ....U...U....... 10040010: 00000000 00000190 000001d6 0000003c ............<... 10040020: 00000082 00000006 00000010 00000000 ................ 10040030: 000008ff 00000000 00000000 00000000 ................ 10040040: 00000000 00000000 00000000 00000000 ................ 10040050: 00000000 00000000 00000000 00000000 ................ 10040060: 00000000 00000000 00000000 00000004 ................ 10040070: 00000006 00000000 00000000 00000001 ................ 10040080: 00000000 00000000 00000000 00000000 ................ 10040090: 00000000 00000064 00000001 00000000 ....d........... 100400a0: 00000005 00000001 00000000 00000000 ................ 100400b0: 00000000 00000000 00000000 00000000 ................ 100400c0: 00000000 00000000 00000000 00000000 ................ 100400d0: 00000000 00000000 00000000 00000000 ................ 100400e0: 00000000 00000000 00000000 00000000 ................ 100400f0: 00000000 001f1fee 3230302a 44570140 ........*002@.WD ## Dump the I2C Registers for I2C Port 2 $ md 0x10050000 0x40 10050000: 0000007f 00000055 00000055 00000001 ....U...U....... 10050010: 00000000 00000190 000001d6 0000003c ............<... 10050020: 00000082 00000006 00000010 00000000 ................ 10050030: 000008ff 00000000 00000000 00000000 ................ 10050040: 00000000 00000000 00000000 00000000 ................ 10050050: 00000000 00000000 00000000 00000000 ................ 10050060: 00000000 00000000 00000000 00000004 ................ 10050070: 00000006 00000000 00000000 00000001 ................ 10050080: 00000000 00000000 00000000 00000000 ................ 10050090: 00000000 00000064 00000001 00000000 ....d........... 100500a0: 00000005 00000001 00000000 00000000 ................ 100500b0: 00000000 00000000 00000000 00000000 ................ 100500c0: 00000000 00000000 00000000 00000000 ................ 100500d0: 00000000 00000000 00000000 00000000 ................ 100500e0: 00000000 00000000 00000000 00000000 ................ 100500f0: 00000000 001f1fee 3230302a 44570140 ........*002@.WD ## Dump the I2C Registers for I2C Port 3 $ md 0x12030000 0x40 12030000: 0000007f 00000055 00000055 00000001 ....U...U....... 12030010: 00000000 00000190 000001d6 0000003c ............<... 12030020: 00000082 00000006 00000010 00000000 ................ 12030030: 000008ff 00000000 00000000 00000000 ................ 12030040: 00000000 00000000 00000000 00000000 ................ 12030050: 00000000 00000000 00000000 00000000 ................ 12030060: 00000000 00000000 00000000 00000004 ................ 12030070: 00000006 00000000 00000000 00000001 ................ 12030080: 00000000 00000000 00000000 00000000 ................ 12030090: 00000000 00000064 00000001 00000000 ....d........... 120300a0: 00000005 00000001 00000000 00000000 ................ 120300b0: 00000000 00000000 00000000 00000000 ................ 120300c0: 00000000 00000000 00000000 00000000 ................ 120300d0: 00000000 00000000 00000000 00000000 ................ 120300e0: 00000000 00000000 00000000 00000000 ................ 120300f0: 00000000 001f1fee 3230302a 44570140 ........*002@.WD ## Dump the I2C Registers for I2C Port 4 $ md 0x12040000 0x40 12040000: 0000007f 00000055 00000055 00000001 ....U...U....... 12040010: 00000000 00000190 000001d6 0000003c ............<... 12040020: 00000082 00000006 00000010 00000000 ................ 12040030: 000008ff 00000000 00000000 00000000 ................ 12040040: 00000000 00000000 00000000 00000000 ................ 12040050: 00000000 00000000 00000000 00000000 ................ 12040060: 00000000 00000000 00000000 00000004 ................ 12040070: 00000006 00000000 00000000 00000001 ................ 12040080: 00000000 00000000 00000000 00000000 ................ 12040090: 00000000 00000064 00000001 00000000 ....d........... 120400a0: 00000005 00000001 00000000 00000000 ................ 120400b0: 00000000 00000000 00000000 00000000 ................ 120400c0: 00000000 00000000 00000000 00000000 ................ 120400d0: 00000000 00000000 00000000 00000000 ................ 120400e0: 00000000 00000000 00000000 00000000 ................ 120400f0: 00000000 001f1fee 3230302a 44570140 ........*002@.WD ## Dump the I2C Registers for I2C Port 5 $ md 0x12050000 0x40 12050000: 00000063 00000050 00000055 00000001 c...P...U....... 12050010: 000000ff 00000020 000000b0 0000003c .... .......<... 12050020: 00000082 00000006 00000010 00000200 ................ 12050030: 00000200 00000711 00000000 00000000 ................ 12050040: 00000001 00000000 00000000 00000000 ................ 12050050: 00000000 00000000 00000000 00000000 ................ 12050060: 00000000 00000000 00000000 00000001 ................ 12050070: 00000006 00000000 00000000 00000008 ................ 12050080: 00000000 00000000 00000000 00000000 ................ 12050090: 00000000 00000064 00000001 00000001 ....d........... 120500a0: 00000005 00000001 00000000 00000000 ................ 120500b0: 00000000 00000000 00000000 00000000 ................ 120500c0: 00000000 00000000 00000000 00000000 ................ 120500d0: 00000000 00000000 00000000 00000000 ................ 120500e0: 00000000 00000000 00000000 00000000 ................ 120500f0: 00000000 001f1fee 3230302a 44570140 ........*002@.WD ## Dump the I2C Registers for I2C Port 6 $ md 0x12060000 0x40 12060000: 0000007f 00000055 00000055 00000001 ....U...U....... 12060010: 00000000 00000190 000001d6 0000003c ............<... 12060020: 00000082 00000006 00000010 00000000 ................ 12060030: 000008ff 00000000 00000000 00000000 ................ 12060040: 00000000 00000000 00000000 00000000 ................ 12060050: 00000000 00000000 00000000 00000000 ................ 12060060: 00000000 00000000 00000000 00000004 ................ 12060070: 00000006 00000000 00000000 00000001 ................ 12060080: 00000000 00000000 00000000 00000000 ................ 12060090: 00000000 00000064 00000001 00000000 ....d........... 120600a0: 00000005 00000001 00000000 00000000 ................ 120600b0: 00000000 00000000 00000000 00000000 ................ 120600c0: 00000000 00000000 00000000 00000000 ................ 120600d0: 00000000 00000000 00000000 00000000 ................ 120600e0: 00000000 00000000 00000000 00000000 ................ 120600f0: 00000000 001f1fee 3230302a 44570140 ........*002@.WD ``` TODO: I2C Port 5 has I2C Target Address 0x50. Probably in use TODO: What are the I2C Controller Registers? TODO: What is I2C Target Address 0x50? Not AXP15060 PMIC, which has I2C Address 0x36 [(Page 7)](https://files.pine64.org/doc/datasheet/star64/AXP15060%20datasheet%20V0.1.pdf) TODO: Check the U-Boot Device Tree # Call OpenSBI from NuttX _How to call OpenSBI in NuttX?_ We run this __`ecall`__ to jump from NuttX (in RISC-V Supervisor Mode) to OpenSBI (in RISC-V Machine Mode)... - [__riscv_sbi.c: Calling OpenSBI in NuttX__](https://github.com/apache/nuttx/blob/master/arch/risc-v/src/common/supervisor/riscv_sbi.c#L52-L77) [(How __`ecall`__ works in OpenSBI)](https://www.thegoodpenguin.co.uk/blog/an-overview-of-opensbi/) [(More about OpenSBI)](https://courses.stephenmarz.com/my-courses/cosc562/risc-v/opensbi-calls/) Like this... From [jh7110_appinit.c](https://github.com/lupyuen2/wip-pinephone-nuttx/blob/sbi/boards/risc-v/jh7110/star64/src/jh7110_appinit.c#L155-L237) ```c // After NuttX boots on JH7110... void board_late_initialize(void) { ... // Make an ecall to OpenSBI int ret = test_opensbi(); DEBUGASSERT(ret == OK); } // Make an ecall to OpenSBI. Based on // https://github.com/riscv-software-src/opensbi/blob/master/firmware/payloads/test_main.c // https://www.thegoodpenguin.co.uk/blog/an-overview-of-opensbi/ int test_opensbi(void) { // Print `123` to Debug Console with Legacy Console Putchar. // Call sbi_console_putchar: EID 0x01, FID 0 // https://github.com/riscv-non-isa/riscv-sbi-doc/blob/master/src/ext-legacy.adoc sbi_ecall(SBI_EXT_0_1_CONSOLE_PUTCHAR, 0, '1', 0, 0, 0, 0, 0); sbi_ecall(SBI_EXT_0_1_CONSOLE_PUTCHAR, 0, '2', 0, 0, 0, 0, 0); sbi_ecall(SBI_EXT_0_1_CONSOLE_PUTCHAR, 0, '3', 0, 0, 0, 0, 0); ``` __`sbi_ecall`__ makes an __`ecall`__ to jump from NuttX (in RISC-V Supervisor Mode) to OpenSBI (in RISC-V Machine Mode)... From [jh7110_appinit.c](https://github.com/lupyuen2/wip-pinephone-nuttx/blob/sbi/boards/risc-v/jh7110/star64/src/jh7110_appinit.c#L268-L299) ```c // Make an ecall to OpenSBI. Based on // https://github.com/apache/nuttx/blob/master/arch/risc-v/src/common/supervisor/riscv_sbi.c#L52-L77 // https://github.com/riscv-software-src/opensbi/blob/master/firmware/payloads/test_main.c static struct sbiret sbi_ecall(unsigned int extid, unsigned int fid, uintptr_t parm0, uintptr_t parm1, uintptr_t parm2, uintptr_t parm3, uintptr_t parm4, uintptr_t parm5) { struct sbiret ret; register long r0 asm("a0") = (long)(parm0); register long r1 asm("a1") = (long)(parm1); register long r2 asm("a2") = (long)(parm2); register long r3 asm("a3") = (long)(parm3); register long r4 asm("a4") = (long)(parm4); register long r5 asm("a5") = (long)(parm5); register long r6 asm("a6") = (long)(fid); register long r7 asm("a7") = (long)(extid); asm volatile ( "ecall" : "+r"(r0), "+r"(r1) : "r"(r2), "r"(r3), "r"(r4), "r"(r5), "r"(r6), "r"(r7) : "memory" ); ret.error = r0; ret.value = r1; return ret; } ``` When we run our Modified NuttX Kernel on Star64 JH7110, we see `123` printed on the Debug Console. Yay! ```text Starting kernel ... 123 NuttShell (NSH) NuttX-12.0.3 nsh> ``` [(Source)](https://github.com/lupyuen2/wip-pinephone-nuttx/blob/sbi/boards/risc-v/jh7110/star64/src/jh7110_appinit.c#L300-L310) _But that's calling the Legacy Console Putchar Function. What about the newer Debug Console Functions?_ Let's call the newer [Debug Console Functions](https://github.com/riscv-non-isa/riscv-sbi-doc/blob/master/src/ext-debug-console.adoc) in OpenSBI... From [jh7110_appinit.c](https://github.com/lupyuen2/wip-pinephone-nuttx/blob/sbi/boards/risc-v/jh7110/star64/src/jh7110_appinit.c#L237-L265) ```c // TODO: Not supported by SBI v0.2, this will return SBI_ERR_NOT_SUPPORTED // Print `456` to Debug Console. // Call sbi_debug_console_write: EID 0x4442434E, FID 0 // https://github.com/riscv-non-isa/riscv-sbi-doc/blob/master/src/ext-debug-console.adoc#function-console-write-fid-0 const char *str = "456"; struct sbiret sret = sbi_ecall( SBI_EXT_DBCN, // Extension ID SBI_EXT_DBCN_CONSOLE_WRITE, // Function ID strlen(str), // Number of bytes (unsigned long)str, // Address Low 0, // Address High 0, 0, 0 // Unused ); _info("sret.value=%d, sret.error=%d\n", sret.value, sret.error); // DEBUGASSERT(sret.error == SBI_SUCCESS); // DEBUGASSERT(sret.value == strlen(str)); // TODO: Not supported by SBI v0.2, this will return SBI_ERR_NOT_SUPPORTED // Print `789` to Debug Console. // Call sbi_debug_console_write_byte: EID 0x4442434E, FID 2 // https://github.com/riscv-non-isa/riscv-sbi-doc/blob/master/src/ext-debug-console.adoc#function-console-write-byte-fid-2 sret = sbi_ecall(SBI_EXT_DBCN, SBI_EXT_DBCN_CONSOLE_WRITE_BYTE, '7', 0, 0, 0, 0, 0); sret = sbi_ecall(SBI_EXT_DBCN, SBI_EXT_DBCN_CONSOLE_WRITE_BYTE, '8', 0, 0, 0, 0, 0); sret = sbi_ecall(SBI_EXT_DBCN, SBI_EXT_DBCN_CONSOLE_WRITE_BYTE, '9', 0, 0, 0, 0, 0); _info("sret.value=%d, sret.error=%d\n", sret.value, sret.error); // DEBUGASSERT(sret.error == SBI_SUCCESS); // DEBUGASSERT(sret.value == strlen(str)); ``` But it fails with `SBI_ERR_NOT_SUPPORTED`... ```text Starting kernel ... test_opensbi: sret.value=0, sret.error=-2 test_opensbi: sret.value=0, sret.error=-2 ``` [(Source)](https://github.com/lupyuen2/wip-pinephone-nuttx/blob/sbi/boards/risc-v/jh7110/star64/src/jh7110_appinit.c#L300-L310) _Why is the Debug Console not supported on JH7110 OpenSBI?_ If we trace the StarFive JH7110 Source Code... 1. Browse to [github.com/starfive-tech/VisionFive2](https://github.com/starfive-tech/VisionFive2) 1. Click the [opensbi Folder](https://github.com/starfive-tech/opensbi/tree/c6a092cd80112529cb2e92e180767ff5341b22a3) It says... > OpenSBI fully supports SBI specification v0.2 Current version of SBI Spec is 2.0. OpenSBI implements a 4-year-old spec? But then we see this: [sbi_ecall.h](https://github.com/starfive-tech/opensbi/blob/c6a092cd80112529cb2e92e180767ff5341b22a3/include/sbi/sbi_ecall.h#L16-L17) ```c #define SBI_ECALL_VERSION_MAJOR 1 #define SBI_ECALL_VERSION_MINOR 0 ``` So it probably implements SBI v1.0. _What exactly is in SBI v0.2 and v1.0?_ Here are the SBI Specs... - [SBI Spec v0.2](https://github.com/riscv-non-isa/riscv-sbi-doc/blob/v0.2.0/riscv-sbi.adoc) - [SBI Spec v1.0](https://github.com/riscv-non-isa/riscv-sbi-doc/blob/v1.0.0/riscv-sbi.adoc) Definitely no Debug Console in there! Debug Console appears in [SBI Spec v2.0-rc1](https://github.com/riscv-non-isa/riscv-sbi-doc/blob/v2.0-rc1/riscv-sbi.adoc), but there isn't an official SBI Spec v2.0 yet. What exactly does JH7110 OpenSBI implement? The Debug Console code is already inside: [sbi_ecall_dbcn.c](https://github.com/starfive-tech/opensbi/blob/c6a092cd80112529cb2e92e180767ff5341b22a3/lib/sbi/sbi_ecall_dbcn.c) Why can't we call the Debug Console? Messy sigh. TODO: Call sbi_get_spec_version, sbi_get_impl_id, sbi_get_impl_version, sbi_probe_extension, sbi_get_mvendorid, sbi_get_marchid, sbi_get_mimpid # PineTab-V Factory Test Code The PineTab-V ships with [Factory Test Code](https://wiki.pine64.org/wiki/PineTab-V_Releases#Factory_releases). > ![PineTab-V Factory Test Code](https://lupyuen.github.io/images/pinetabv-factory.jpg) The factory test reference source code has been mirrored here (for quicker downloading)... - [PineTab-V_factorytestcode_SDK-20230725.tar.gz](https://drive.google.com/file/d/1OZkcGFVvyTn36JLebxT9_89blbOHM5ON/view?usp=drive_link) Let's look inside the Factory Test Code... # PineTab-V Device Tree [(__LCD Panel in PineTab-V__ is actually BOE TH101MB31IG002-28A)](https://fosstodon.org/@Fishwaldo/110902984462760802) Inside the PineTab-V Factory Test Code (from previous section), we see the PineTab-V Device Tree for Linux: [VisionFive2/linux/arch/riscv/boot/dts/starfive/jh7110-visionfive-v2.dtsi](https://github.com/lupyuen/nuttx-star64/blob/main/pinetabv-jh7110-visionfive-v2.dtsi) From the above Device Tree we see the LCD Panel (StarFive Jadard on MIPI DSI1) and Touch Panel (Goodix GT911)... ```text &i2c2 { clock-frequency = <100000>; i2c-sda-hold-time-ns = <300>; i2c-sda-falling-time-ns = <510>; i2c-scl-falling-time-ns = <510>; auto_calc_scl_lhcnt; status = "okay"; ... panel_radxa@19 { status = "okay"; compatible ="starfive_jadard"; reg = <0x19>; //reset-gpio = <&gpio 23 0>; blen-gpio = <&gpio 45 0>; enable-gpio = <&gpio 37 0>; port { panel_out1: endpoint { remote-endpoint = <&dsi1_output>; }; }; }; touchscreen@14 { status = "okay"; compatible = "goodix,gt911"; reg = <0x14>; irq-gpios = <&gpio 30 GPIO_ACTIVE_HIGH>; reset-gpios = <&gpio 31 GPIO_ACTIVE_HIGH>; }; }; ``` Which maps to this Linux Driver for StarFive Jadard Display Panel: (which one?) - er88577b: [VisionFive2/linux/drivers/gpu/drm/panel/panel-er88577b.c](https://github.com/lupyuen/nuttx-star64/blob/main/panel-er88577b.c) - jd9365da-h3: [VisionFive2/linux/drivers/gpu/drm/panel/panel-jadard-jd9365da-h3.c](https://github.com/lupyuen/nuttx-star64/blob/main/panel-jadard-jd9365da-h3.c) StarFive Jadard is also mentioned in [JH7110 MIPI LCD Developing and Porting Guide](https://doc-en.rvspace.org/VisionFive2/DG_LCD/JH7110_SDK/configuration_for_1c4l_port.html) We also see the Galaxycore GC02M2 Camera (on MIPI CSI)... ```text &gpio { ... gc02m2_pins: gc02m2_pins { gc02m2pins-pwdn { starfive,pins = ; starfive,pinmux = ; starfive,pin-ioconfig = ; starfive,pin-gpio-dout = ; starfive,pin-gpio-doen = ; }; gc02m2_pins-rst { starfive,pins = ; starfive,pinmux = ; starfive,pin-ioconfig = ; starfive,pin-gpio-dout = ; starfive,pin-gpio-doen = ; }; // gc02m2_pins-sw { // starfive,pins = ; // starfive,pinmux = ; // starfive,pin-ioconfig = ; // starfive,pin-gpio-dout = ; // starfive,pin-gpio-doen = ; // }; }; ... &i2c6 { clock-frequency = <100000>; i2c-sda-hold-time-ns = <300>; i2c-sda-falling-time-ns = <510>; i2c-scl-falling-time-ns = <510>; auto_calc_scl_lhcnt; pinctrl-names = "default"; pinctrl-0 = <&i2c6_pins>; status = "okay"; ... gc02m2: gc02m2@37 { status = "okay"; compatible = "galaxycore,gc02m2"; pinctrl-names = "default"; pinctrl-0 = <&gc02m2_pins>; reg = <0x37>; clocks = <&clk_ext_camera>; clock-names = "xvclk"; reset-gpio = <&gpio 33 GPIO_ACTIVE_HIGH>; pwdn-gpio = <&gpio 25 GPIO_ACTIVE_LOW>; // sw-gpios = <&gpio 53 GPIO_ACTIVE_LOW>; port { gc02m2_to_csi2rx0: endpoint { remote-endpoint = <&csi2rx0_from_gc02m2>; bus-type = <4>; /* MIPI CSI-2 D-PHY */ data-lanes = <1>; link-frequencies = /bits/ 64 <336000000>; }; }; }; }; ... &vin_sysctl { /* when use dvp open this pinctrl*/ status = "okay"; ports { #address-cells = <1>; #size-cells = <0>; port@1 { reg = <1>; #address-cells = <1>; #size-cells = <0>; ... csi2rx0_from_gc02m2: endpoint@1 { reg = <1>; remote-endpoint = <&gc02m2_to_csi2rx0>; bus-type = <4>; /* MIPI CSI-2 D-PHY */ clock-lanes = <0>; data-lanes = <2 1>; lane-polarities = <0 0 0>; status = "okay"; }; }; }; }; ``` # RAM Disk Address for RISC-V QEMU Read the article... - ["Star64 JH7110 + NuttX RTOS: RISC-V Semihosting and Initial RAM Disk"](https://lupyuen.github.io/articles/semihost) _Can we enable logging for RISC-V QEMU?_ Yep we use the `-trace "*"` option like this... ```bash qemu-system-riscv64 \ -semihosting \ -M virt,aclint=on \ -cpu rv64 \ -smp 8 \ -bios none \ -kernel nuttx \ -initrd initrd \ -nographic \ -trace "*" ``` 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: ```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` (`__ramdisk_start` from the previous section) Also we see that Kernel is loaded at `0x8000` `0000`, Device Tree at `0x8700` `0000`. # Device Tree for RISC-V QEMU Read the article... - ["Star64 JH7110 + NuttX RTOS: RISC-V Semihosting and Initial RAM Disk"](https://lupyuen.github.io/articles/semihost) To dump the Device Tree for QEMU RISC-V, we specify `dumpdtb`... ```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 ``` 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. # Scheme Interpreter crashes on NuttX _Why does Scheme Interpreter crash on NuttX?_ Let's run the Scheme Interpreter on NuttX QEMU, for easier debugging... [github.com/KenDickey/nuttx-umb-scheme](https://github.com/KenDickey/nuttx-umb-scheme) Here are the steps... ```bash pushd ../apps/interpreters git submodule add https://github.com/KenDickey/nuttx-umb-scheme umb-scheme popd ## We use the updated NuttX Config from: ## https://github.com/lupyuen2/wip-pinephone-nuttx/blob/malloc2a/boards/risc-v/qemu-rv/rv-virt/configs/knsh64/defconfig ## CONFIG_ARCH_SETJMP_H=y ## CONFIG_LIBM=y ## CONFIG_ARCH_PGPOOL_SIZE=16777216 ## And the updated Linker Script from: ## https://github.com/lupyuen2/wip-pinephone-nuttx/blob/malloc2a/boards/risc-v/qemu-rv/rv-virt/scripts/ld-kernel64.script#L25 ## pgram (rwx) : ORIGIN = 0x80400000, LENGTH = 16M make distclean tools/configure.sh rv-virt:knsh64 make ## Remember to build the Apps Filesystem ## Copy scheme prelude cp \ ../apps/interpreters/umb-scheme/prelude.scheme \ ../apps/bin/ qemu-system-riscv64 \ -semihosting \ -M virt,aclint=on \ -cpu rv64 \ -smp 8 \ -bios none \ -kernel nuttx \ -nographic ``` Here's the output... - [NuttX Scheme crashes on QEMU](https://gist.github.com/lupyuen/5c225a3a30086cd35455463955f5ff64) Our Crash Dump (see below) says... ```yaml EXCEPTION: Load page fault. MCAUSE: 000000000000000d, EPC: 00000000c000f366, MTVAL: 0000000000000030 ``` Exception Program Counter is `c000f366`. We disregard the `c000` because NuttX maps the Scheme App into the RISC-V User Memory Space at `c0000000`. So actual Program Counter is `f366` ```bash CONFIG_ARCH_DATA_NPAGES=128 CONFIG_ARCH_DATA_VBASE=0xC0100000 CONFIG_ARCH_HEAP_NPAGES=128 CONFIG_ARCH_HEAP_VBASE=0xC0200000 CONFIG_ARCH_TEXT_NPAGES=128 CONFIG_ARCH_TEXT_VBASE=0xC0000000 ``` [(Source)](https://github.com/apache/nuttx/blob/master/boards/risc-v/jh7110/star64/configs/nsh/defconfig#L25-L39) We look up `f366` in the disassembly... ```bash ## Dump the scheme disassembly to scheme.S riscv64-unknown-elf-objdump \ -t -S --demangle --line-numbers --wide \ ../apps/bin/scheme \ >scheme.S \ 2>&1 ``` And we see this disassembly... ```text 000000000000f360 <.L29>: apps/interpreters/umb-scheme/architecture.c:556 while (strcmp(name, Get_Symbol_Name(this_entry->Symbol)) != 0 f360: 0004b903 ld s2,0(s1) f364: 8522 mv a0,s0 /* Crashes here */ f366: 03093583 ld a1,48(s2) f36a: 00000097 auipc ra,0x0 f36e: 000080e7 jalr ra # f36a <.L29+0xa> ``` _Are we sure that's the crashing code?_ ```text /* Crashes here */ f366: 03093583 ld a1,48(s2) ``` Let's check the RISC-V Registers. Our Crash Dump (see below) says... ```text MTVAL: 0000000000000030 ``` Which means that our Scheme App crashed while accessing address `0x30`, which is invalid. _So `48(s2)` should equal `0x30`?_ The Crash Dump says... ```yaml up_dump_register: S0: 00000000c01017f8 S1: 00000000c0201ef0 S2: 0000000000000000 S3: 0000000000000000 ``` Which means Register S2 is 0. And `48(s2)` means S2 + 48, which is `0x30`. Yep we have the right line of crashing code! _Why did this fail? Is `this_entry` null?_ From [architecture.c](https://github.com/KenDickey/nuttx-umb-scheme/blob/main/architecture.c#L553-L556): ```c Public Object Intern_Name(String name) { Integer hash_value = Hash(name); if (hash_entry == NULL) { ... } else { // hash_entry is non-NULL Boolean at_end = FALSE; // this_entry becomes non-NULL Entry * this_entry = hash_entry; // Crashes here while (strcmp(name, Get_Symbol_Name(this_entry->Symbol)) != 0 { ... ``` Actually if we read the code above, we see that `this_entry` is on the Stack and can NEVER be NULL! _Maybe there's Stack Corruption?_ We see something interesting in the Crash Dump below... The App Stack is full! ```text STACKSIZE USED FILLED COMMAND 2048 2040 99.6%! irq 3056 1808 59.1% Idle_Task 1968 752 38.2% lpwork 0x802015f0 0x80201618 3008 744 24.7% /system/bin/init 2000 2000 100.0%! scheme ``` In the next section, we increase the Interrupt and App Stack Sizes. Meanwhile here's the [Complete Crash Dump](https://gist.github.com/lupyuen/5c225a3a30086cd35455463955f5ff64)... ```yaml riscv_exception: EXCEPTION: Load page fault. MCAUSE: 000000000000000d, EPC: 00000000c000f366, MTVAL: 0000000000000030 riscv_exception: PANIC!!! Exception = 000000000000000d _assert: Current Version: NuttX 12.3.0-RC0 322455f Oct 12 2023 15:48:34 risc-v _assert: Assertion failed panic: at file: common/riscv_exception.c:85 task: /system/bin/init process: /system/bin/init 0xc000004a up_dump_register: EPC: 00000000c000f366 up_dump_register: A0: 00000000c01017f8 A1: 000000000fffffff A2: 00000000c01017fe A3: 0000000000000000 up_dump_register: A4: 00000000000003f1 A5: 0000000000001610 A6: 00000000c0226380 A7: fffffffffffffff8 up_dump_register: T0: 000000008000729a T1: 00000000c0003af8 T2: 00000000000001ff T3: 000000000000006c up_dump_register: T4: 0000000000000068 T5: 0000000000000009 T6: 000000000000002a up_dump_register: S0: 00000000c01017f8 S1: 00000000c0201ef0 S2: 0000000000000000 S3: 0000000000000000 up_dump_register: S4: 00000000c001ea34 S5: 00000000c0166038 S6: 00000000c01017f8 S7: 000000000000003b up_dump_register: S8: 0000000000000021 S9: 0000000000000019 S10: 0000000000000000 S11: 0000000000000000 up_dump_register: SP: 00000000c0202210 FP: 00000000c01017f8 TP: 0000000000000000 RA: 00000000c000cbda dump_stack: User Stack: dump_stack: base: 0xc0202040 dump_stack: size: 00003008 dump_stack: sp: 0xc0202210 stack_dump: 0xc0202200: 00000063 00000000 c000c00e 00000000 c001ea34 00000000 c0165f78 00000000 stack_dump: 0xc0202220: c0166040 00000000 0000006c 00000000 c0265a50 00000000 c000cbda 00000000 stack_dump: 0xc0202240: c0265a50 00000000 00000019 00000000 00000021 00000000 0000003b 00000000 stack_dump: 0xc0202260: c01017f8 00000000 c0166038 00000000 c001ea34 00000000 c0265a50 00000000 stack_dump: 0xc0202280: c0166040 00000000 c0166048 00000000 c01660a0 00000000 c000cdc8 00000000 stack_dump: 0xc02022a0: c001ea34 00000000 c0265a50 00000000 c0166040 00000000 c0166048 00000000 stack_dump: 0xc02022c0: c01660a0 00000000 c000cdc8 00000000 c001ea34 00000000 c0165f78 00000000 stack_dump: 0xc02022e0: c0166040 00000000 c0166048 00000000 c0265a50 00000000 c000c6e0 00000000 stack_dump: 0xc0202300: c0166048 00000000 00000000 00000000 c0101cb0 00000000 c0165f98 00000000 stack_dump: 0xc0202320: c0165f18 00000000 c0165ff4 00000000 c0166048 00000000 c0265a50 00000000 stack_dump: 0xc0202340: c0101cb0 00000000 c0166048 00000000 c01660a0 00000000 c000cdc8 00000000 stack_dump: 0xc0202360: c0101cb0 00000000 c0265a50 00000000 c0101cb0 00000000 c0166048 00000000 stack_dump: 0xc0202380: c01660a0 00000000 c000cf0a 00000000 c0166048 00000000 c0265a50 00000000 stack_dump: 0xc02023a0: c0101cb0 00000000 c0166048 00000000 c01660a0 00000000 c000cf0a 00000000 stack_dump: 0xc02023c0: c0166048 00000000 c0265a50 00000000 c0101cb0 00000000 c0166048 00000000 stack_dump: 0xc02023e0: c01660a0 00000000 c000cdc8 00000000 c0101cb0 00000000 c0265a50 00000000 stack_dump: 0xc0202400: c0101cb0 00000000 c0166048 00000000 c01660a0 00000000 c000cf0a 00000000 stack_dump: 0xc0202420: c0166048 00000000 c0265a50 00000000 c0101cb0 00000000 c0166048 00000000 stack_dump: 0xc0202440: c01660a0 00000000 c000cf0a 00000000 c0166048 00000000 c0265a50 00000000 stack_dump: 0xc0202460: c0101cb0 00000000 c0166048 00000000 c01660a0 00000000 c000cf0a 00000000 stack_dump: 0xc0202480: c0166048 00000000 0000c34f 00000000 c0265a50 00000000 c01660a0 00000000 stack_dump: 0xc02024a0: 00000000 00000000 c00025fa 00000000 00000000 00000000 00000000 00000000 stack_dump: 0xc02024c0: 00000000 00000000 00000000 00000000 00000000 00000000 00000000 00000000 stack_dump: 0xc02024e0: 00000001 00000000 c0265a20 00000000 c0265a50 00000000 c0008112 00000000 stack_dump: 0xc0202500: 00000000 00000000 00000000 00000000 c0166048 00000000 c0002558 00000000 stack_dump: 0xc0202520: 00000000 00000000 c00028c4 00000000 deadbeef deadbeef 00000000 00000000 stack_dump: 0xc0202540: 6c756e28 2e2f296c 65686373 de00656d deadbeef deadbeef deadbeef deadbeef stack_dump: 0xc0202560: deadbeef deadbeef deadbeef deadbeef deadbeef deadbeef deadbeef deadbeef stack_dump: 0xc0202580: deadbeef deadbeef deadbeef deadbeef deadbeef deadbeef deadbeef deadbeef stack_dump: 0xc02025a0: deadbeef deadbeef deadbeef deadbeef deadbeef deadbeef deadbeef deadbeef stack_dump: 0xc02025c0: deadbeef deadbeef deadbeef deadbeef deadbeef deadbeef deadbeef deadbeef stack_dump: 0xc02025e0: deadbeef deadbeef deadbeef deadbeef deadbeef deadbeef deadbeef deadbeef stack_dump: 0xc0202600: deadbeef deadbeef deadbeef deadbeef deadbeef deadbeef deadbeef deadbeef stack_dump: 0xc0202620: deadbeef deadbeef deadbeef deadbeef deadbeef deadbeef deadbeef deadbeef stack_dump: 0xc0202640: 636c6557 20656d6f 55206f74 5320424d 6d656863 76202c65 69737265 33206e6f stack_dump: 0xc0202660: 2020322e 79706f43 68676972 63282074 39312029 312c3838 20363939 6c6c6957 stack_dump: 0xc0202680: 206d6169 61432052 6562706d 0a2e6c6c 20424d55 65686353 6320656d 73656d6f stack_dump: 0xc02026a0: 74697720 42412068 554c4f53 594c4554 204f4e20 52524157 59544e41 6854202e stack_dump: 0xc02026c0: 69207369 72662073 73206565 7774666f 20657261 0a646e61 20756f79 20657261 stack_dump: 0xc02026e0: 65657266 206f7420 69646572 69727473 65747562 20746920 65646e75 65632072 stack_dump: 0xc0202700: 69617472 6f63206e 7469646e 736e6f69 65530a2e 68742065 4d552065 63532042 stack_dump: 0xc0202720: 656d6568 6c655220 65736165 746f4e20 66207365 6420726f 69617465 202e736c stack_dump: 0xc0202740: 7079540a 28602065 74697865 6f206029 6f432072 6f72746e 20642d6c 65206f74 stack_dump: 0xc0202760: 2e746978 de000a0a deadbeef deadbeef deadbeef deadbeef deadbeef deadbeef stack_dump: 0xc0202780: deadbeef deadbeef deadbeef deadbeef deadbeef deadbeef deadbeef deadbeef stack_dump: 0xc02027a0: deadbeef deadbeef deadbeef deadbeef deadbeef deadbeef deadbeef deadbeef stack_dump: 0xc02027c0: deadbeef deadbeef deadbeef deadbeef deadbeef deadbeef 00000000 00000000 stack_dump: 0xc02027e0: 00000000 00000000 c00023f6 00000000 deadbeef deadbeef 8000a9f8 00000000 stack_dump: 0xc0202800: 00000000 00000000 00030d51 00000000 c0101060 00000000 c001d630 00000000 stack_dump: 0xc0202820: 00000000 00000000 c0101060 00000000 c001d640 00000000 00000000 00000000 stack_dump: 0xc0202840: c01011e0 00000000 c001d650 00000000 00000000 00000000 c01011e0 00000000 stack_dump: 0xc0202860: c001d660 00000000 00000000 00000000 c0101210 00000000 c001d670 00000000 stack_dump: 0xc0202880: 00000000 00000000 c01014b0 00000000 c001d420 00000000 c0202888 00000000 stack_dump: 0xc02028a0: 00000000 00000000 c0202870 00000000 00000000 00000000 c0200560 00000000 stack_dump: 0xc02028c0: c01014b0 00000000 c001d420 00000000 00000000 00000000 c0202888 00000000 stack_dump: 0xc02028e0: c0202870 00000000 00000000 00000000 c0200580 00000000 c01014b0 00000000 stack_dump: 0xc0202900: c001d420 00000000 c02028f8 00000000 c0202888 00000000 c0202870 00000000 stack_dump: 0xc0202920: 00000000 00000000 c02005b0 00000000 c01014b0 00000000 c001d420 00000000 stack_dump: 0xc0202940: c02028c0 00000000 c0202888 00000000 c0202870 00000000 00000000 00000000 stack_dump: 0xc0202960: c02005f0 00000000 c01014b0 00000000 c001d420 00000000 c02028c0 00000000 stack_dump: 0xc0202980: c0202888 00000000 c0202870 00000000 00000000 00000000 c0200630 00000000 stack_dump: 0xc02029a0: c01014b0 00000000 c001d420 00000000 c02028c0 00000000 c0202888 00000000 stack_dump: 0xc02029c0: c0202870 00000000 00000000 00000000 c0200670 00000000 c01014b0 00000000 stack_dump: 0xc02029e0: c001d420 00000000 c02028c0 00000000 c0202888 00000000 c0202870 00000000 stack_dump: 0xc0202a00: 00000000 00000000 c02006b0 00000000 c01014b0 00000000 c001d420 00000000 stack_dump: 0xc0202a20: c02028c0 00000000 c0202888 00000000 c0202870 00000000 00000000 00000000 stack_dump: 0xc0202a40: c02006f0 00000000 c01014b0 00000000 c001d420 00000000 c02028c0 00000000 stack_dump: 0xc0202a60: c0202888 00000000 c0202870 00000000 00000000 00000000 c0200730 00000000 stack_dump: 0xc0202a80: c01014b0 00000000 c001d420 00000000 c02028c0 00000000 c0202888 00000000 stack_dump: 0xc0202aa0: c0202870 00000000 00000000 00000000 c0200770 00000000 c01014b0 00000000 stack_dump: 0xc0202ac0: c001d420 00000000 c02028c0 00000000 c0202888 00000000 c0202870 00000000 stack_dump: 0xc0202ae0: 00000000 00000000 c02007b0 00000000 c0101360 00000000 c001d420 00000000 stack_dump: 0xc0202b00: 00000001 00000000 c0200308 00000000 c02007d0 00000000 c0101360 00000000 stack_dump: 0xc0202b20: c001d420 00000000 00000000 00000000 c02003a8 00000000 c02007f0 00000000 stack_dump: 0xc0202b40: c0101360 00000000 c001d420 00000000 00000000 00000000 c0200448 00000000 stack_dump: 0xc0202b60: c0200810 00000000 c0101270 00000000 c001d720 00000000 00000000 00000000 stack_dump: 0xc0202b80: c0101240 00000000 c001d420 00000000 c0202870 00000000 00000000 00000000 stack_dump: 0xc0202ba0: c0202870 00000000 00000000 00000000 00000000 00000000 c0101390 00000000 stack_dump: 0xc0202bc0: c001d420 00000000 c001f490 00000000 00000000 00000000 c000fc58 00000000 stack_dump: 0xc0202be0: 00000001 00000000 fffffffe ffffffff c01014b0 00000000 c001d420 00000000 dump_tasks: PID GROUP PRI POLICY TYPE NPX STATE EVENT SIGMASK STACKBASE STACKSIZE USED FILLED COMMAND dump_tasks: ---- --- --- -------- ------- --- ------- ---------- ---------------- 0x802002b0 2048 2040 99.6%! irq dump_task: 0 0 0 FIFO Kthread N-- Ready 0000000000000000 0x80206010 3056 1520 49.7% Idle_Task dump_task: 1 1 100 RR Kthread --- Waiting Semaphore 0000000000000000 0x8020a050 1968 752 38.2% lpwork 0x802015f0 0x80201618 dump_task: 2 2 100 RR Task --- Waiting Semaphore 0000000000000000 0xc0202040 3008 848 28.1% /system/bin/init dump_task: 3 3 100 RR Task --- Running 0000000000000000 0xc0202030 2000 2000 100.0%! scheme �F�0� r�������������������d���&���P����������\�������� ``` # Increase Stack Size for Scheme Interpreter _Scheme Interpreter crashes on NuttX because the App Stack Size is too small (2000 bytes)..._ ```text STACKSIZE USED FILLED COMMAND 2048 2040 99.6%! irq 3056 1808 59.1% Idle_Task 1968 752 38.2% lpwork 0x802015f0 0x80201618 3008 744 24.7% /system/bin/init 2000 2000 100.0%! scheme ``` _But we already set [CONFIG_INTERPRETERS_UMB_SCHEME_STACKSIZE=8192](https://github.com/KenDickey/nuttx-umb-scheme/blob/main/Kconfig)!_ Apparently it doesn't work. Maybe because we're running in NuttX Kernel Mode? (Instead of NuttX Flat Mode) This is how we increase the App Stack Size to 8192 in `.config`... ```bash CONFIG_POSIX_SPAWN_DEFAULT_STACKSIZE=8192 ``` After increasing the App Stack Size, the Scheme App doesn't crash any more! - [Before Setting CONFIG_POSIX_SPAWN_DEFAULT_STACKSIZE](https://gist.github.com/lupyuen/5c225a3a30086cd35455463955f5ff64) - [After Setting CONFIG_POSIX_SPAWN_DEFAULT_STACKSIZE](https://gist.github.com/lupyuen/572e6018ed982fe42b2b5ed40ffae505) - [Scheme runs OK on Star64 yay!](https://gist.github.com/lupyuen/1e009a3343da70257d6f24400339053f) To be safe, we increase the Default Stack Size, Interrupt Stack Size and other Kernel Stack Sizes: [knsh64/defconfig](https://github.com/lupyuen2/wip-pinephone-nuttx/commit/7b8ee95d2dfd848051da17ab7dd74b56ef59c94d) ```bash CONFIG_ARCH_INTERRUPTSTACK=8192 CONFIG_ARCH_KERNEL_STACKSIZE=8192 CONFIG_DEFAULT_TASK_STACKSIZE=8192 CONFIG_IDLETHREAD_STACKSIZE=8192 ## Use Default Values: ## CONFIG_INIT_STACKSIZE=8192 ## CONFIG_POSIX_SPAWN_DEFAULT_STACKSIZE=8192 ``` This shows that the Interrupt and Kernel Stacks have been increased... ```text STACKSIZE USED FILLED COMMAND 8192 8184 99.9%! irq 8176 1824 22.3% Idle_Task 8112 720 8.8% lpwork 0x80202df0 0x80202e18 8128 848 10.4% /system/bin/init ``` TODO: Why is Interrupt Stack full again? TODO: Explore [Run Time Stack Checking](https://cwiki.apache.org/confluence/plugins/servlet/mobile?contentId=139629451#content/view/139629451) Read on to find out what caused the App Stack to overflow... # Analyse the Stack Dump for Scheme Interpreter TODO: Enable "RTOS Features > Stack Backtrace" in menuconfig (SCHED_BACKTRACE) _In the previous section, we saw the Scheme App crashing on NuttX QEMU due to Stack Overflow..._ _But why? What are the Function Calls leading to the crash?_ Let's construct the Stack Backtrace. We search for other Code Addresses in the Stack Dump above... ```text → grep 'c00[0-1]....' --only-matching /tmp/a.log c000f366 c000004a c0003af8 c001ea34 c000cbda c000c00e c000cdc8 c000c6e0 c000cf0a c00025fa c0008112 c0002558 c00028c4 c00023f6 c001d630 c001d640 c001d650 c001d660 c001d670 c001d420 c001d720 c001f490 c000fc58 ``` We match the above Code Addresses to the Scheme App Disassembly (from the previous section). [(Remember to disregard the `c00` prefix because that's the RISC-V User Memory Space)](https://github.com/apache/nuttx/blob/master/boards/risc-v/jh7110/star64/configs/nsh/defconfig#L25-L39) Note that these Code Addresses are the __Return Addresses__. So we look up the Previous Instruction that appears in the Scheme App Disassembly... ```c c000f366: Intern_Name() calls Get_Symbol_Name() (The crashing code, from previous section) apps/interpreters/umb-scheme/architecture.c:556 Public Object Intern_Name(String name) { ... while (strcmp(name, Get_Symbol_Name(this_entry->Symbol)) != 0 c0003af8: (We skip this, not really meaningful) nuttx/libs/libc/stdio/lib_libfilelock.c:64 void funlockfile(FAR struct file_struct *stream) { nxrmutex_unlock(&stream->fs_lock); c000cbda: Read() calls Intern_Name() apps/interpreters/umb-scheme/io.c:336 /* Read a Scheme object from |Input_File|; leave it in Value_Register. */ Public void Read( FILE *Input_File ) { ... Value_Register = Intern_Name( Token_String ) ; c000c00e: Read_Symbol() calls Peek_Char() apps/interpreters/umb-scheme/io.c:726 /* We do a naive scan, scanning up to a delimiter. This allows for more than is described in the formal syntax. We do eliminate control characters. */ Private void Read_Symbol( Ch , Input_File ) Character Ch ; /* Leading character of symbol. */ FILE* Input_File ; /* Containing remaining chars. */ { ... while ( !Is_Delimiter( Peek_Char( Input_File ) ) ) c000cdc8: Read_List() calls Read() apps/interpreters/umb-scheme/io.c:446 /* Read list from Input_File and leave it in Value_Register. This allows the input `( . x )' (it treats it as equivalent to x), which is not strictly legal according to the manual. */ Private void Read_List( FILE* Input_File ) { ... Read( Input_File ) ; c000c6e0: Read() calls itself recursively apps/interpreters/umb-scheme/io.c:415 (discriminator 4) /* Read a Scheme object from |Input_File|; leave it in Value_Register. */ Public void Read( FILE* Input_File ) { ... case Comma_At_Token : Read( Input_File ) ; c000cf0a: Read_List() calls itself recursively apps/interpreters/umb-scheme/io.c:483 (discriminator 4) /* Read list from Input_File and leave it in Value_Register. This allows the input `( . x )' (it treats it as equivalent to x), which is not strictly legal according to the manual. */ Private void Read_List( FILE* Input_File ) { ... Read_List( Input_File ) ; c00025fa: Read_Eval_Print() calls Read() apps/interpreters/umb-scheme/steering.c:208 Public void Read_Eval_Print( input ) FILE* input; /* C file from which expressions are Read() */ { ... Read( input ); c0008112: Load() calls Read_Eval_Print() apps/interpreters/umb-scheme/primitive.c:1479 /* (load filename). Used to load the standard prelude, hence is public. */ Public void Load() { ... Read_Eval_Print(load_file); c0002558: Load_File() calls Load() apps/interpreters/umb-scheme/steering.c:197 (discriminator 2) Private void Load_File(String Filename) { ... Load(); c00028c4: Steering() calls Load_File() apps/interpreters/umb-scheme/steering.c:142 Private void Steering() { ... Load_File(STANDARD_PRELUDE_PATHNAME); c00023f6: NuttX _start() calls Scheme main() nuttx/arch/risc-v/src/common/crt0.c:187 void _start(int argc, char *argv[]) { ... /* Call the main() entry point passing argc and argv. */ ret = main(argc, argv); c000fc58: (We skip this, not really meaningful) apps/interpreters/umb-scheme/number.c:148 /* (number? object) */ Private void Number_Predicate() { Value_Register = Is_Number(Top(1)) ? The_True_Object ``` _What does this Stack Backtrace tell us?_ When we read the Function Calls from bottom to top... 1. [NuttX _start() calls Scheme main()](https://github.com/lupyuen2/wip-pinephone-nuttx/blob/malloc2a/arch/risc-v/src/common/crt0.c#L184-L187) (NuttX starts our Scheme App) 1. [main() calls Steering(), which calls Load_File()](https://github.com/KenDickey/nuttx-umb-scheme/blob/main/steering.c#L138-L143) (Scheme App loads the Prelude Script) 1. [LoadFile() calls Load()](https://github.com/KenDickey/nuttx-umb-scheme/blob/main/steering.c#L197) (Scheme App loads the Prelude Script) 1. [Load() calls Read_Eval_Print()](https://github.com/KenDickey/nuttx-umb-scheme/blob/main/primitive.c#L1477-L1479) (Scheme App executes the Prelude Script) 1. [Read_Eval_Print() calls Read()](https://github.com/KenDickey/nuttx-umb-scheme/blob/main/steering.c#L207-L208) (Scheme App reads the Prelude Script) 1. [Read() calls itself recursively](https://github.com/KenDickey/nuttx-umb-scheme/blob/main/io.c#L415) (Scheme App reads the Scheme Expressions recursively) 1. [Read() calls Intern_Name()](https://github.com/KenDickey/nuttx-umb-scheme/blob/main/io.c#L333-L337) (Scheme App maps a Symbol Token to the Internal Name) 1. [Intern_Name() calls Get_Symbol_Name()](https://github.com/KenDickey/nuttx-umb-scheme/blob/main/architecture.c#L553-L556) (Scheme App tries to get the Symbol Token's Internal Name and crashes here) Which says that the Scheme App tried to load the Scheme Prelude Script at startup. And crashed with a Stack Overflow. _So why did the Scheme App crash with a Stack Overflow?_ Remember that our NuttX App Stack was only 2000 bytes... ```text STACKSIZE USED FILLED COMMAND 2048 2040 99.6%! irq 3056 1808 59.1% Idle_Task 1968 752 38.2% lpwork 0x802015f0 0x80201618 3008 744 24.7% /system/bin/init 2000 2000 100.0%! scheme ``` And the Scheme App was recursively executing a Scheme Prelude Script. That's why it needs extra Stack Space to work! Mystery solved! # TODO 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 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: Handle Machine Exception https://github.com/lupyuen2/wip-pinephone-nuttx/blob/star64d/arch/risc-v/src/qemu-rv/qemu_rv_exception_m.S#L64 TODO: Check PolarFire Icicle https://lupyuen.github.io/articles/privilege#other-risc-v-ports-of-nuttx TODO: Check Linux Boot Code https://github.com/torvalds/linux/blob/master/arch/riscv/kernel/head.S TODO: Linux SBI Interface https://github.com/torvalds/linux/blob/master/arch/riscv/kernel/sbi.c # U-Boot Bootloader Log for TFTP ```text 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 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....... 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 # setenv tftp_server 192.168.x.x 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 Starting kernel ... clk u5_dw_i2c_clk_core already disabled clk u5_dw_i2c_clk_apb already disabled 123067DFAGHBC ``` # U-Boot Bootloader Log for Auto Network Boot ```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 OK 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 ... 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: *#################################################################  ###########################################################T ######T   #############  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 ###  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 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 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 Code: 879b 0277 d7b3 00f6 f793 1ff7 078e 95be (b023 0105) resetting ... reset not supported yet ### ERROR ### Please RESET the board ### ``` # NuttX Logs ## NuttX Star64 Binary Loader Log ```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 ``` ## NuttX QEMU Binary Loader Log ```text elf_symvalue: Other: 00005d76+c0000000=c0005d76 elf_read: Read 24 bytes from offset 1302760 elf_symvalue: Other: 00005d4c+c0000000=c0005d4c elf_read: Read 24 bytes from offset 1302784 elf_symvalue: Other: 00005d52+c0000000=c0005d52 elf_read: Read 24 bytes from offset 1302808 elf_symvalue: Other: 00005d56+c0000000=c0005d56 elf_read: Read 24 bytes from offset 1951160 elf_read: Read 24 bytes from offset 1770880 elf_symvalue: Other: 00000000+c0101028=c0101028 up_relocateadd: RISCV_64 at c0101028 [00000000] to sym=0x8020a850 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: 0x80209f60 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) up_exit: TCB=0x802088d0 exiting NuttShell (NSH) NuttX-12.0.3 nsh> nx_start: CPU0: Beginning Idle Loop ``` ## NuttX Star64 Memory Management Log ```text mm_free: Freeing 0x4040aa10 mm_free: Freeing 0x4040a9d0 mm_free: Freeing 0x4040a990 mm_free: Freeing 0x4040a950 mm_free: Freeing 0x4040a890 mm_malloc: Allocated 0x4040c810, size 6160 mm_malloc: Allocated 0x4040a890, size 64 mm_free: Freeing 0x4040c810 mm_free: Freeing 0x4040a890 mm_free: Freeing 0x40409290 mm_free: Freeing 0x40409fa0 mm_free: Freeing 0x40409250 mm_malloc: Allocated 0x40409250, size 368 mm_malloc: Allocated 0x404093c0, size 64 mm_initialize: Heap: name=(null), start=0xc0200000 size=528384 mm_addregion: [(null)] Region 1: base=0xc0200298 size=527712 mm_malloc: Allocated 0x4040a890, size 3088 mm_malloc: Allocated 0x4040b4a0, size 304 mm_malloc: Allocated 0x4040b5d0, size 32 mm_malloc: Allocated 0xc02002c0, size 624 mm_malloc: Allocated 0xc0200530, size 32 mm_malloc: Allocated 0xc0200550, size 32 mm_malloc: Allocated 0xc0200570, size 32 mm_malloc: Allocated 0x4040b5f0, size 32 mm_malloc: Allocated 0x4040b610, size 160 mm_malloc: Allocated 0xc0200590, size 19472 mm_free: Freeing 0x404093c0 nx_start_application: ret=3 mm_free: Freeing 0x40408b90 mm_free: Freeing 0x40408e80 mm_free: Freeing 0x40408e60 mm_free: Freeing 0x40408e40 mm_free: Freeing 0x40408e20 mm_free: Freeing 0x40408e00 mm_free: Freeing 0x40408b70 mm_free: Freeing 0x40408a40 up_exit: TCB=0x404088d0 exiting mm_free: Freeing 0x4040c000 mm_free: Freeing 0x404088d0 mm_malloc: Allocated 0xc0200590, size 848 nx_start: CPU0: Beginning Idle Loop ``` ## NuttX QEMU Memory Management Log ```text mm_free: Freeing 0x8020aa80 mm_free: Freeing 0x8020aa40 mm_free: Freeing 0x8020a9c0 mm_free: Freeing 0x8020a980 mm_free: Freeing 0x8020a940 mm_free: Freeing 0x8020a900 mm_free: Freeing 0x8020a840 mm_malloc: Allocated 0x8020c810, size 6160 mm_malloc: Allocated 0x8020a840, size 64 mm_free: Freeing 0x8020c810 mm_free: Freeing 0x8020a840 mm_free: Freeing 0x80209290 mm_free: Freeing 0x8020a810 mm_free: Freeing 0x80209250 mm_malloc: Allocated 0x80209250, size 368 mm_malloc: Allocated 0x802093c0, size 64 mm_initialize: Heap: name=(null), start=0xc0200000 size=528384 mm_addregion: [(null)] Region 1: base=0xc0200298 size=527712 mm_malloc: Allocated 0x8020a810, size 3088 mm_malloc: Allocated 0x80209400, size 304 mm_malloc: Allocated 0x80209fe0, size 32 mm_malloc: Allocated 0xc02002c0, size 624 mm_malloc: Allocated 0xc0200530, size 32 mm_malloc: Allocated 0xc0200550, size 32 mm_malloc: Allocated 0xc0200570, size 32 mm_malloc: Allocated 0x80209530, size 32 mm_malloc: Allocated 0x80209550, size 160 mm_malloc: Allocated 0xc0200590, size 19472 mm_free: Freeing 0x802093c0 mm_free: Freeing 0x80208b90 mm_free: Freeing 0x80208e80 mm_free: Freeing 0x80208e60 mm_free: Freeing 0x80208e40 mm_free: Freeing 0x80208e20 mm_free: Freeing 0x80208e00 mm_free: Freeing 0x80208b70 mm_free: Freeing 0x80208a40 up_exit: TCB=0x802088d0 exiting mm_free: Freeing 0x8020c000 mm_free: Freeing 0x802088d0 NuttShell (NSH) NuttX-12.0.3 nsh> nx_start: CPU0: Beginning Idle Loop ``` ## NuttX Star64 NSH Assertion Log No Console Output! ```text 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 1: Undefined symbol[0] has no name: -3 nx_start_application: ret=3 up_exit: TCB=0x404088d0 exiting _assert: Current Version: NuttX 12.0.3 46c1a0f Jul 27 2023 19:31:45 risc-v _assert: Assertion failed (_Bool)0: at file: nsh_main.c:54 task: /system/bin/init 0xc000001a up_dump_register: EPC: 000000004020fffc up_dump_register: A0: 0000000040401610 A1: 0000000000000036 A2: 00000000c0000e58 A3: 0000000000000000 up_dump_register: A4: 0000000000000000 A5: 0000000000000000 A6: 0000000000000036 A7: 00000000c0000e68 up_dump_register: T0: 0000000040211e9e T1: 00000000c000042c T2: 0000000000000000 T3: 0000000000000000 up_dump_register: T4: 0000000000000000 T5: 0000000000000000 T6: 0000000000000000 up_dump_register: S0: 0000000000000000 S1: 0000000040409a60 S2: 0000000040401748 S3: 00000000c0000e58 up_dump_register: S4: 00000000c0000e68 S5: 0000000000000036 S6: 0000000000000000 S7: 0000000000000000 up_dump_register: S8: 0000000000000000 S9: 0000000000000000 S10: 0000000000000000 S11: 0000000000000000 up_dump_register: SP: 000000004040b1d8 FP: 0000000000000000 TP: 0000000000000000 RA: 000000004020fffc dump_stack: Kernel Stack: dump_stack: base: 0x4040a810 dump_stack: size: 00003072 dump_stack: sp: 0x4040b1d8 stack_dump: 0x4040b1c0: 40409a60 00000000 00000000 00000000 4021021c 00000000 40406cf8 00000000 stack_dump: 0x4040b1e0: 00000219 00010000 c00003ee 00000000 00000000 00000000 4020029a 00000000 stack_dump: 0x4040b200: 7474754e 00000058 c000080e 00000000 c0202b90 00000000 ffffff83 ffffffff stack_dump: 0x4040b220: 00000000 00000000 402126ce 00000000 c000042c 2e323100 00332e30 00000000 stack_dump: 0x4040b240: c02003d8 00000000 363403d0 30613163 754a2066 3732206c 32303220 39312033 stack_dump: 0x4040b260: 3a31333a 00003534 00000000 00000000 00007fff 00000000 00000001 73697200 stack_dump: 0x4040b280: 00762d63 00000000 00000000 00000000 00000000 00000000 00000000 00000000 stack_dump: 0x4040b2a0: 00000000 00000000 00000000 00000000 00000000 00000000 00000000 00000000 stack_dump: 0x4040b2c0: 00000000 00000000 00000000 00000000 00000000 00000000 00000000 00000000 stack_dump: 0x4040b2e0: 40211eb0 00000000 00000000 00000000 402086f4 00000000 c000007c 00000000 stack_dump: 0x4040b300: 00040020 00000002 402086dc 00000000 c000007c 00000000 4040b308 00000000 stack_dump: 0x4040b320: 00000000 00000000 00000000 00000000 402126ce 00000000 c000042c 00000000 stack_dump: 0x4040b340: 00000000 00000000 00000000 00000000 00000000 00000000 00000001 00000000 stack_dump: 0x4040b360: c0000e68 00000000 00000036 00000000 c0000e58 00000000 00000000 00000000 stack_dump: 0x4040b380: c0000e58 00000000 00000036 00000000 c0000e68 00000000 00000000 00000000 stack_dump: 0x4040b3a0: 00000000 00000000 00000000 00000000 00000000 00000000 00000000 00000000 stack_dump: 0x4040b3c0: 00000000 00000000 00000000 00000000 00000000 00000000 00000000 00000000 stack_dump: 0x4040b3e0: 00000000 00000000 00000000 00000000 00000000 00000000 00000000 00000000 up_exit: TCB=0x40409a60 exiting nx_start: CPU0: Beginning Idle Loop ``` ## NuttX QEMU NSH Assertion Log Console Output OK ```text ABCnx_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 1: Undefined symbol[0] has no name: -3 up_exit: TCB=0x802088d0 exiting ***main _assert: Current Version: NuttX 12.0.3 3c99d2b-dirty Jul 27 2023 19:27:52 risc-v _assert: Assertion failed (_Bool)0: at file: nsh_main.c:54 task: /system/bin/init 0xc000001a up_dump_register: EPC: 0000000080001e82 up_dump_register: A0: 0000000080200f10 A1: 0000000000000036 A2: 00000000c0000e58 A3: 0000000000000000 up_dump_register: A4: 0000000000000000 A5: 0000000000000000 A6: 0000000000000036 A7: 00000000c0000e68 up_dump_register: T0: 0000000080006d36 T1: 00000000c000042c T2: 0000000000000000 T3: 0000000000000000 up_dump_register: T4: 0000000000000000 T5: 0000000000000000 T6: 0000000000000000 up_dump_register: S0: 0000000000000000 S1: 0000000080209a60 S2: 0000000080201750 S3: 00000000c0000e58 up_dump_register: S4: 00000000c0000e68 S5: 0000000000000036 S6: 0000000000000000 S7: 0000000000000000 up_dump_register: S8: 0000000000000000 S9: 0000000000000000 S10: 0000000000000000 S11: 0000000000000000 up_dump_register: SP: 000000008020b1d8 FP: 0000000000000000 TP: 0000000000000000 RA: 0000000080001e82 dump_stack: Kernel Stack: dump_stack: base: 0x8020a810 dump_stack: size: 00003072 dump_stack: sp: 0x8020b1d8 stack_dump: 0x8020b1c0: 80209a60 00000000 00000000 00000000 800020a2 00000000 80206cf8 00000000 stack_dump: 0x8020b1e0: 00000219 00010000 c00003ee 00000000 00000000 00000000 80000592 00000000 stack_dump: 0x8020b200: 7474754e 00000058 c000080e 00000000 c0202b90 00000000 c0200430 00000000 stack_dump: 0x8020b220: 00000000 00000000 80007566 00000000 c000042c 2e323100 00332e30 00000000 stack_dump: 0x8020b240: c02003d8 00000000 633303d0 32643939 69642d62 20797472 206c754a 32203732 stack_dump: 0x8020b260: 20333230 323a3931 32353a37 00000000 00007fff 00000000 00000001 73697200 stack_dump: 0x8020b280: 00762d63 00000000 00000000 00000000 00000000 00000000 00000000 00000000 stack_dump: 0x8020b2a0: 00000000 00000000 00000000 00000000 00000000 00000000 00000000 00000000 stack_dump: 0x8020b2c0: 00000000 00000000 00000000 00000000 00000000 00000000 00000000 00000000 stack_dump: 0x8020b2e0: 80006d48 00000000 00000000 00000000 80001264 00000000 c000007c 00000000 stack_dump: 0x8020b300: 00040020 00000002 8000124c 00000000 c000007c 00000000 8020b308 00000000 stack_dump: 0x8020b320: 00000000 00000000 00000000 00000000 80007566 00000000 c000042c 00000000 stack_dump: 0x8020b340: 00000000 00000000 00000000 00000000 00000000 00000000 00000001 00000000 stack_dump: 0x8020b360: c0000e68 00000000 00000036 00000000 c0000e58 00000000 00000000 00000000 stack_dump: 0x8020b380: c0000e58 00000000 00000036 00000000 c0000e68 00000000 00000000 00000000 stack_dump: 0x8020b3a0: 00000000 00000000 00000000 00000000 00000000 00000000 00000000 00000000 stack_dump: 0x8020b3c0: 00000000 00000000 00000000 00000000 00000000 00000000 00000000 00000000 stack_dump: 0x8020b3e0: 00000000 00000000 00000000 00000000 00000000 00000000 00000000 00000000 up_exit: TCB=0x80209a60 exiting nx_start: CPU0: Beginning Idle Loop ``` ## NuttX Star64 Console Output Log Console Output is stuck ```text 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 uart_write (0xc0200428): 0000 2a 2a 2a 6d 61 69 6e 0a ***main. uart_write (0xc000a610): 0000 0a 4e 75 74 74 53 68 65 6c 6c 20 28 4e 53 48 29 .NuttShell (NSH) 0010 20 4e 75 74 74 58 2d 31 32 2e 30 2e 33 0a NuttX-12.0.3. uart_write (0xc0015340): 0000 6e 73 68 3e 20 nsh> uart_write (0xc0015318): 0000 1b 5b 4b .[K nx_start: CPU0: Beginning Idle Loop ```