Skip to content

Building Linux for a Simulation and Running on nSIM

Yuriy Kolerov edited this page Jul 28, 2023 · 1 revision

Table of Content

UART Settings for Older Linux Kernels (v5.4 and Prior)

nSIM 2019.06 onwards supports DW UART (in addition to the legacy ARC UART). This paved the way to running Linux image built for HAPS FPGA on nSIM. So starting with kernel version v5.5-rc1, nsim_hs_smp_defconfig and nsim_hs_defconfig defconfigs were merged with corresponding haps defconfigs and everything was switched to DW UART.

For kernels v5.4 and prior (with ARC UART based configurations and DT) and/or older nSIM, the legacy ARC UART needs to be instantiated instead of -prop=nsim_mem-dev=<...> property presented in the guides:

  • For ARC770 use -prop=nsim_mem-dev=uart0,base=0xc0fc1000,irq=5
  • For ARC HS38 single-core use -prop=nsim_mem-dev=uart0,base=0xc0fc1000,irq=24
  • For ARC HS38 multi-core configuration with IDU, use -prop=nsim_mem-dev=uart0,base=0xc0fc1000,irq=0,use_connect=1

Prerequisites

Clone a repository and create a build directory:

# Clone the latest Buildroot
$ git clone https://git.busybox.net/buildroot

# ... or use a custom repository for support of ARCv3 targets
$ https://github.com/foss-for-synopsys-dwc-arc-processors/buildroot

# Create a build directory
$ mkdir buildroot/build
$ cd buildroot/build

In this guide we suppose that a toolchain is preinstalled in /tools/toolchains (2023.03 release). Also, only a limited set of all available toolchains is considered. All releases may be downloaded from the official releases page. Here is a table of toolchains, which are used in this guide:

ARC processors family Standard library Toolchain's installation path Version
ARC HS 6x glibc /tools/toolchains/arc64-linux-gnu 2023.03
ARC HS 5x uClibc-ng /tools/toolchains/arc32-linux-uclibc 2023.03
ARC HS 3x/4x glibc /tools/toolchains/arc-linux-gnu 2023.03
ARC HS 3x/4x uClibc-ng /tools/toolchains/arc-linux-uclibc 2023.03
ARC 700 uClibc-ng /tools/toolchains/arc700-linux-uclibc 2023.03

Building and Running a Kernel for ARC HS 3x/4x with glibc

We a going to use the latest Synopsys' development branch of the Linux kernel. Also, we use hasp_hs configuration file for the Linux kernel - it corresponds to a single core (UP) configuration. Save a custom configuration file in build/defconfig:

BR2_arcle=y
BR2_archs38=y
BR2_TOOLCHAIN_EXTERNAL=y
BR2_TOOLCHAIN_EXTERNAL_CUSTOM=y
BR2_TOOLCHAIN_EXTERNAL_PATH="/tools/toolchains/arc-linux-gnu"
BR2_TOOLCHAIN_EXTERNAL_HEADERS_5_16=y
BR2_TOOLCHAIN_EXTERNAL_CUSTOM_GLIBC=y
# BR2_TOOLCHAIN_EXTERNAL_INET_RPC is not set
BR2_TOOLCHAIN_EXTERNAL_CXX=y
BR2_TOOLCHAIN_EXTERNAL_FORTRAN=y
BR2_LINUX_KERNEL=y
BR2_LINUX_KERNEL_CUSTOM_GIT=y
BR2_LINUX_KERNEL_CUSTOM_REPO_URL="https://github.com/foss-for-synopsys-dwc-arc-processors/linux"
BR2_LINUX_KERNEL_CUSTOM_REPO_VERSION="arc64"
BR2_LINUX_KERNEL_DEFCONFIG="haps_hs"
BR2_LINUX_KERNEL_VMLINUX=y
BR2_TARGET_ROOTFS_INITRAMFS=y

Build a Linux image:

$ make -C .. O=$(pwd) defconfig DEFCONFIG=build/defconfig
$ make

Before running the image using nSIM create a corresponding configuration file hs4x.props (use nsim_fast=1 option if you have nSIM Pro license):

nsim_fast=0
nsim_isa_family=av2hs
nsim_isa_core=3
chipid=0xffff
nsim_isa_atomic_option=1
nsim_isa_ll64_option=1
nsim_mmu=4
mmu_pagesize=8192
mmu_super_pagesize=2097152
mmu_stlb_entries=16
mmu_ntlb_ways=4
mmu_ntlb_sets=128
icache=32768,64,4,0
dcache=16384,64,2,0
nsim_isa_shift_option=2
nsim_isa_swap_option=1
nsim_isa_bitscan_option=1
nsim_isa_sat=1
nsim_isa_div_rem_option=1
nsim_isa_mpy_option=9
nsim_isa_enable_timer_0=1
nsim_isa_enable_timer_1=1
nsim_isa_number_of_interrupts=32
nsim_isa_number_of_external_interrupts=32
isa_counters=1
nsim_isa_pct_counters=8
nsim_isa_pct_size=48
nsim_isa_pct_interrupt=1
nsim_mem-dev=uart0,kind=dwuart,base=0xf0000000,irq=24
nsim_isa_aps_feature=1
nsim_isa_num_actionpoints=4
nsim_isa_rtc_option=1
nsim_isa_fpud_option=1

Then run the image using nSIM:

$ nsimdrv -propsfile hs4x.props images/vmlinux

Run the images without using a property file:

$ nsimdrv -prop nsim_isa_family=av2hs -prop nsim_isa_core=3 -prop chipid=0xffff -prop nsim_isa_atomic_option=1 \
          -prop nsim_isa_ll64_option=1 -prop nsim_mmu=4 -prop mmu_pagesize=8192 -prop mmu_super_pagesize=2097152 \
          -prop mmu_stlb_entries=16 -prop mmu_ntlb_ways=4 -prop mmu_ntlb_sets=128 -prop icache=32768,64,4,0 \
          -prop dcache=16384,64,2,0 -prop nsim_isa_shift_option=2 -prop nsim_isa_swap_option=1 -prop nsim_isa_bitscan_option=1 \
          -prop nsim_isa_sat=1 -prop nsim_isa_div_rem_option=1 -prop nsim_isa_mpy_option=9 -prop nsim_isa_enable_timer_0=1 \
          -prop nsim_isa_enable_timer_1=1 -prop nsim_isa_number_of_interrupts=32 -prop nsim_isa_number_of_external_interrupts=32 \
          -prop isa_counters=1 -prop nsim_isa_pct_counters=8 -prop nsim_isa_pct_size=48 -prop nsim_isa_pct_interrupt=1 \
          -prop nsim_mem-dev=uart0,kind=dwuart,base=0xf0000000,irq=24 -prop nsim_isa_aps_feature=1 \
          -prop nsim_isa_num_actionpoints=4 -prop nsim_isa_rtc_option=1 -prop nsim_isa_fpud_option=1 \
          images/vmlinux

Run the image using a predefined TCF file:

$ nsimdrv -tcf $NSIM_HOME/etc/tcf/templates/hs38_full.tcf \
          -prop nsim_mem-dev=uart0,kind=dwuart,base=0xf0000000,irq=24 \
          images/vmlinux

You will see an kernel's initialization output:

Console now belongs to UART, hit CRTL-] to return to simulator.
Linux version 5.16.0 (ykolerov@SNPS-HRlPxd6IgG) (arc-linux-gcc (ARC HS GNU/Linux glibc toolchain - build 1360) 12.2.1 20230306, GNU ld (ARC HS GNU/Linux glibc toolchain - build 1360) 2.40.50.20230314) #2 PREEMPT Tue Jul 25 18:27:27 +04 2023
Memory @ 80000000 [1024M]
Memory @ 100000000 [1024M] Not used
OF: fdt: Machine model: snps,zebu_hs
earlycon: uart8250 at MMIO32 0xf0000000 (options '115200n8')
printk: bootconsole [uart8250] enabled

IDENTITY        : ARCVER [0x53] ARCNUM [0x0] CHIPID [0xffff]
processor [0]   : HS38 R3.0 (ARCv2 ISA)
ISA Extn        : atomic ll64 unalign mpy[opt 9] div_rem   FPU: sp dp
BPU             : partial match, cache:2048, Predict Table:16384 Return stk: 8
MMU [v4]        : 8k/2M, swalk 2 lvl, JTLB 128x4, uDTLB 8, uITLB 4
I-Cache         : 32K, 4way/set, 64B Line, VIPT
D-Cache         : 16K, 2way/set, 64B Line, PIPT
Peripherals     : 0xc0000000
Timers          : Timer0 Timer1 RTC [UP 64-bit]
Vector Table    : 0x80000000 [64-bit]
DEBUG           : ActionPoint 4/full

...

Welcome to the HAPS Development Platform
zebu_hs login:

Username is root without a password. To halt target system issue halt command.

Building and Running a Kernel for ARC HS 3x/4x with uClibc-ng

We use haps_hs configuration file for the Linux kernel - it corresponds to a single core (UP) configuration. Here is a corresponding configuration file for Buildroot:

BR2_arcle=y
BR2_archs38=y
BR2_TOOLCHAIN_EXTERNAL=y
BR2_TOOLCHAIN_EXTERNAL_CUSTOM=y
BR2_TOOLCHAIN_EXTERNAL_PATH="/tools/toolchains/arc-linux-uclibc"
BR2_TOOLCHAIN_EXTERNAL_LOCALE=y
BR2_TOOLCHAIN_EXTERNAL_HAS_SSP=y
BR2_TOOLCHAIN_EXTERNAL_CXX=y
BR2_TOOLCHAIN_EXTERNAL_HEADERS_5_16=y
BR2_LINUX_KERNEL=y
BR2_LINUX_KERNEL_CUSTOM_GIT=y
BR2_LINUX_KERNEL_CUSTOM_REPO_URL="https://github.com/foss-for-synopsys-dwc-arc-processors/linux"
BR2_LINUX_KERNEL_CUSTOM_REPO_VERSION="arc64"
BR2_LINUX_KERNEL_DEFCONFIG="haps_hs"
BR2_LINUX_KERNEL_VMLINUX=y
BR2_TARGET_ROOTFS_INITRAMFS=y

A process of building and running of the Linux kernel for ARC HS38 based on uClibc-ng is the same as for glibc version.

Building and Running a Kernel for ARC 770 with uClibc-ng

We use nsim_700 configuration file for the Linux kernel - it corresponds to a single core (UP) configuration. Here is a corresponding configuration file for Buildroot:

BR2_arcle=y
BR2_arc770d=y
BR2_TOOLCHAIN_EXTERNAL=y
BR2_TOOLCHAIN_EXTERNAL_CUSTOM=y
BR2_TOOLCHAIN_EXTERNAL_PATH="/tools/toolchains/arc700-linux-uclibc"
BR2_TOOLCHAIN_EXTERNAL_LOCALE=y
BR2_TOOLCHAIN_EXTERNAL_HAS_SSP=y
BR2_TOOLCHAIN_EXTERNAL_CXX=y
BR2_TOOLCHAIN_EXTERNAL_HEADERS_5_16=y
BR2_GCC_VERSION_12_X=y
BR2_LINUX_KERNEL=y
BR2_LINUX_KERNEL_CUSTOM_GIT=y
BR2_LINUX_KERNEL_CUSTOM_REPO_URL="https://github.com/foss-for-synopsys-dwc-arc-processors/linux"
BR2_LINUX_KERNEL_CUSTOM_REPO_VERSION="arc64"
BR2_LINUX_KERNEL_DEFCONFIG="nsim_700"
BR2_LINUX_KERNEL_VMLINUX=y
BR2_TARGET_ROOTFS_INITRAMFS=y

Build a Linux image:

$ make -C .. O=$(pwd) defconfig DEFCONFIG=build/defconfig
$ make

Before running the image using nSIM create a corresponding configuration file arc700.props (use nsim_fast=1 option if you have nSIM Pro license):

nsim_fast=0
nsim_isa_family=a700
nsim_isa_atomic_option=1
nsim_mmu=3
icache=32768,64,2,0
dcache=32768,64,4,0
nsim_isa_dpfp=none
nsim_isa_shift_option=2
nsim_isa_swap_option=1
nsim_isa_bitscan_option=1
nsim_isa_sat=1
nsim_isa_mpy32=1
nsim_isa_enable_timer_0=1
nsim_isa_enable_timer_1=1
nsim_mem-dev=uart0,kind=dwuart,base=0xf0000000,irq=24
isa_counters=1
nsim_isa_pct_counters=8
nsim_isa_pct_size=48

Then run the image using nSIM:

$ nsimdrv -propsfile arc700.props images/vmlinux

Run the image without a property file:

$ nsimdrv -prop nsim_isa_family=a700 -prop nsim_isa_atomic_option=1 -prop nsim_mmu=3 -prop icache=32768,64,2,0 \
          -prop dcache=32768,64,4,0 -prop nsim_isa_dpfp=none -prop nsim_isa_shift_option=2 -prop nsim_isa_swap_option=1 \
          -prop nsim_isa_bitscan_option=1 -prop nsim_isa_sat=1 -prop nsim_isa_mpy32=1 -prop nsim_isa_enable_timer_0=1 \
          -prop nsim_isa_enable_timer_1=1 -prop nsim_mem-dev=uart0,kind=dwuart,base=0xf0000000,irq=24 -prop isa_counters=1 \
          -prop nsim_isa_pct_counters=8 -prop nsim_isa_pct_size=48 images/vmlinux

Run the image using a predefined TCF file:

$ nsimdrv -tcf $NSIM_HOME/etc/tcf/templates/arc770d.tcf -prop nsim_isa_number_of_interrupts=32 \
          -prop nsim_mem-dev=uart0,kind=dwuart,base=0xf0000000,irq=24 -prop nsim_isa_dpfp=none \
          -prop icache=32768,64,2,0 -prop dcache=32768,64,4,0 -prop nsim_isa_dpfp=none \
          -prop nsim_isa_swap_option=1 -prop nsim_isa_bitscan_option=1 -prop nsim_isa_sat=1 \
          -prop isa_counters=1 -prop nsim_isa_pct_counters=8 -prop nsim_isa_pct_size=48 \
          images/vmlinux

Building and Running a Kernel for ARC HS5x with uClibc-ng

ℹ️ Linux kernels for ARCv3 are loaded differently in comparison to kernels for ARCv1 and ARCv2. Kernels for ARCv3 are always loaded from a physical address 0x0 instead of 0x80000000. A custom loader image is generated to distinguish it from a default vmlinux (vmlinux uses 0x80000000 as a starting address in physical and virtual spaces).

We use haps_hs5x configuration file for the Linux kernel - it corresponds to a single core (UP) configuration. Here is a corresponding configuration file for Buildroot:

BR2_PACKAGE_HOST_LINUX_HEADERS_CUSTOM_5_16=y
BR2_ROOTFS_POST_IMAGE_SCRIPT="board/synopsys/arc64/post-image.sh"
BR2_ROOTFS_POST_SCRIPT_ARGS="$(LINUX_DIR)"
BR2_LINUX_KERNEL=y
BR2_LINUX_KERNEL_CUSTOM_GIT=y
BR2_LINUX_KERNEL_CUSTOM_REPO_URL="https://github.com/foss-for-synopsys-dwc-arc-processors/linux.git"
BR2_LINUX_KERNEL_CUSTOM_REPO_VERSION="arc64"
BR2_LINUX_KERNEL_DEFCONFIG="haps_hs5x"
BR2_LINUX_KERNEL_IMAGE_TARGET_CUSTOM=y
BR2_LINUX_KERNEL_IMAGE_TARGET_NAME="loader"
BR2_TOOLCHAIN_EXTERNAL=y
BR2_TOOLCHAIN_EXTERNAL_CUSTOM=y
BR2_TOOLCHAIN_EXTERNAL_PATH="/tools/toolchains/arc32-linux-uclibc"
BR2_TOOLCHAIN_EXTERNAL_HEADERS_5_16=y
BR2_TOOLCHAIN_EXTERNAL_LOCALE=y
BR2_TOOLCHAIN_EXTERNAL_HAS_SSP=y
BR2_TOOLCHAIN_EXTERNAL_CXX=y
BR2_TOOLCHAIN_EXTERNAL_INET_RPC=n
BR2_TOOLCHAIN_EXTERNAL_GCC_12=y
BR2_TARGET_ROOTFS_INITRAMFS=y

Build a Linux image:

$ make -C .. O=$(pwd) defconfig DEFCONFIG=build/defconfig
$ make

Before running the image using nSIM create a corresponding configuration file hs5x.props (use nsim_fast=1 option if you have nSIM Pro license):

nsim_fast=0
nsim_isa_family=av3hs
nsim_isa_dc_hw_prefetch=1
nsim_isa_dual_issue_option=1
nsim_isa_atomic_option=2
nsim_isa_m128_option=0
nsim_isa_ll64_option=1
nsim_isa_mpy_option=9
nsim_isa_div_rem_option=2
nsim_isa_enable_timer_0=1
nsim_isa_enable_timer_1=1
nsim_isa_rtc_option=1
icache=16384,64,4
dcache=16384,64,2
mmu_version=16
mmu_pagesize=4096
mmu_address_space=32
nsim_mem-dev=uart0,kind=dwuart,base=0xf0000000,irq=24
nsim_isa_number_of_interrupts=32
nsim_isa_number_of_external_interrupts=32
nsim_isa_has_fp=1
nsim_isa_fp_dds_option=1
nsim_isa_fp_div_option=1
nsim_isa_fp_dp_option=1
nsim_isa_fp_hp_option=1
nsim_isa_fp_vec_option=1
nsim_isa_fp_wide_option=1

Then run the image using nSIM:

$ nsimdrv -propsfile hs5x.props images/loader

Run the image without a property file:

$ nsimdrv -prop nsim_isa_family=av3hs -prop nsim_isa_dc_hw_prefetch=1 -prop nsim_isa_dual_issue_option=1 \
          -prop nsim_isa_atomic_option=2 -prop nsim_isa_m128_option=0 -prop nsim_isa_ll64_option=1 -prop nsim_isa_mpy_option=9 \
          -prop nsim_isa_div_rem_option=2 -prop nsim_isa_enable_timer_0=1 -prop nsim_isa_enable_timer_1=1 \
          -prop nsim_isa_rtc_option=1 -prop icache=16384,64,4 -prop dcache=16384,64,2 -prop mmu_version=16 -prop mmu_pagesize=4096 \
          -prop mmu_address_space=32 -prop nsim_mem-dev=uart0,kind=dwuart,base=0xf0000000,irq=24 -prop nsim_isa_number_of_interrupts=32 \
          -prop nsim_isa_number_of_external_interrupts=32 -prop nsim_isa_has_fp=1 -prop nsim_isa_fp_dds_option=1 \
          -prop nsim_isa_fp_div_option=1 -prop nsim_isa_fp_dp_option=1 -prop nsim_isa_fp_hp_option=1 -prop nsim_isa_fp_vec_option=1 \
          -prop nsim_isa_fp_wide_option=1 images/loader

Run the image using a predefined TCF file:

$ nsimdrv -tcf $NSIM_HOME/etc/tcf/templates/hs58_full.tcf -prop icache=16384,64,4 -prop dcache=16384,64,2 \
          -prop nsim_mem-dev=uart0,kind=dwuart,base=0xf0000000,irq=24 -prop nsim_isa_number_of_external_interrupts=32 \
          -prop nsim_isa_has_fp=1 -prop nsim_isa_fp_dds_option=1 -prop nsim_isa_fp_div_option=1 -prop nsim_isa_fp_dp_option=1 \
          -prop nsim_isa_fp_hp_option=1 -prop nsim_isa_fp_vec_option=1 -prop nsim_isa_fp_wide_option=1 \
          images/loader

Building and Running a Kernel for ARC HS6x with glibc

We use haps_arc64 configuration file for the Linux kernel - it corresponds to a single core (UP) configuration. Here is a corresponding configuration file for Buildroot:

BR2_arcle=y
BR2_arc64=y
# BR2_STRIP_strip is not set
BR2_PACKAGE_HOST_LINUX_HEADERS_CUSTOM_5_16=y
BR2_ROOTFS_POST_IMAGE_SCRIPT="board/synopsys/arc64/post-image.sh"
BR2_ROOTFS_POST_SCRIPT_ARGS="$(LINUX_DIR)"
BR2_LINUX_KERNEL=y
BR2_LINUX_KERNEL_CUSTOM_GIT=y
BR2_LINUX_KERNEL_CUSTOM_REPO_URL="https://github.com/foss-for-synopsys-dwc-arc-processors/linux.git"
BR2_LINUX_KERNEL_CUSTOM_REPO_VERSION="arc64"
BR2_LINUX_KERNEL_DEFCONFIG="haps_arc64"
BR2_LINUX_KERNEL_IMAGE_TARGET_CUSTOM=y
BR2_LINUX_KERNEL_IMAGE_TARGET_NAME="loader"
BR2_TOOLCHAIN_EXTERNAL=y
BR2_TOOLCHAIN_EXTERNAL_CUSTOM=y
BR2_TOOLCHAIN_EXTERNAL_CUSTOM_GLIBC=y
BR2_TOOLCHAIN_EXTERNAL_PATH="/tools/toolchains/arc64-linux-gnu"
BR2_TOOLCHAIN_EXTERNAL_HEADERS_5_16=y
BR2_TOOLCHAIN_EXTERNAL_LOCALE=y
BR2_TOOLCHAIN_EXTERNAL_HAS_SSP=y
BR2_TOOLCHAIN_EXTERNAL_CXX=y
BR2_TOOLCHAIN_EXTERNAL_INET_RPC=n
BR2_TOOLCHAIN_EXTERNAL_GCC_12=y
BR2_TARGET_ROOTFS_INITRAMFS=y

Before running the image using nSIM create a corresponding configuration file hs6x.props (use nsim_fast=1 option if you have nSIM Pro license):

nsim_fast=0
nsim_isa_dual_issue_option=1
nsim_isa_has_hw_pf=1
nsim_isa_m128_option=1
nsim_isa_has_hw_pf=1
nsim_isa_vec64=1
nsim_isa_family=arc64
nsim_isa_enable_timer_0=1
nsim_isa_enable_timer_1=1
nsim_isa_rtc_option=1
nsim_isa_addr_size=64
nsim_isa_pc_size=64
icache=16384,64,4,o
dcache=16384,64,4,o
mmu_version=16
mmu_pagesize=4096
mmu_address_space=48
nsim_mem-dev=uart0,kind=dwuart,base=0xf0000000,irq=24
nsim_isa_mpy_option=9
nsim_isa_mpy64=1
nsim_isa_div64_option=1
nsim_isa_div_rem_option=2
nsim_isa_atomic_option=2
nsim_isa_has_fp=1
nsim_isa_fp_dds_option=1
nsim_isa_fp_div_option=1
nsim_isa_fp_dp_option=1
nsim_isa_fp_hp_option=1
nsim_isa_fp_vec_option=1
nsim_isa_fp_wide_option=1

Then run the image using nSIM:

$ nsimdrv -propsfile hs6x.props images/loader

Also, you can run the images without a property file:

$ nsimdrv -prop nsim_isa_dual_issue_option=1 -prop nsim_isa_has_hw_pf=1 -prop nsim_isa_m128_option=1 \
          -prop nsim_isa_vec64=1 -prop nsim_isa_family=arc64 -prop nsim_isa_enable_timer_0=1 -prop nsim_isa_enable_timer_1=1 \
          -prop nsim_isa_rtc_option=1 -prop nsim_isa_addr_size=64 -prop nsim_isa_pc_size=64 -prop icache=16384,64,4,o \
          -prop dcache=16384,64,4,o -prop mmu_version=16 -prop mmu_pagesize=4096 -prop mmu_address_space=48 \
          -prop nsim_mem-dev=uart0,kind=dwuart,base=0xf0000000,irq=24 -prop nsim_isa_mpy_option=9 -prop nsim_isa_mpy64=1 \
          -prop nsim_isa_div64_option=1 -prop nsim_isa_div_rem_option=2 -prop nsim_isa_atomic_option=2 -prop nsim_isa_has_fp=1 \
          -prop nsim_isa_fp_dds_option=1 -prop nsim_isa_fp_div_option=1 -prop nsim_isa_fp_dp_option=1 -prop nsim_isa_fp_hp_option=1 \
          -prop nsim_isa_fp_vec_option=1 -prop nsim_isa_fp_wide_option=1 \
          images/loader
Clone this wiki locally