This project includes the scripts and additional files required to build and publish the xPack GNU RISC-V Embedded GCC binaries.
It follows the official SiFive releases, and it is planned to make a new release after each future SiFive release.
The build scripts use the xPack Build Box (XBB), a set of elaborate build environments based on a recent GCC (Docker containers for GNU/Linux and Windows or a custom folder for MacOS).
There are two types of builds:
- local/native builds, which use the tools available on the host machine; generally the binaries do not run on a different system distribution/version; intended mostly for development purposes;
- distribution builds, which create the archives distributed as binaries; expected to run on most modern systems.
This page documents the distribution builds.
For native builds, see the build-native.sh
script.
The build scripts use a set of local repositories, to accommodate the
small changes required by the riscv-none-embed-
prefix:
These repositories were forked from the SiFive repositories:
GDB was upstreamed and does not require SiFive specific patches, so the original FSF repository is used:
git://sourceware.org/git/binutils-gdb.git
However, to accommodate the custom prefix, a separate branch is created
in xpack-dev-tools/riscv-binutils-gdb
.
xpack
- the updated content, used during buildsxpack-develop
- the updated content, used during developmentmaster
- empty, not used.
The prerequisites are common to all binary builds. Please follow the instructions from the separate Prerequisites for building xPack binaries page and return when ready.
Note: Building the Arm binaries requires an Arm machine.
The build scripts are available in the scripts
folder of the
xpack-dev-tools/riscv-none-embed-gcc-xpack
Git repo.
To download them, issue the following commands:
rm -rf ${HOME}/Work/riscv-none-embed-gcc-xpack.git; \
git clone \
https://github.com/xpack-dev-tools/riscv-none-embed-gcc-xpack.git \
${HOME}/Work/riscv-none-embed-gcc-xpack.git; \
git -C ${HOME}/Work/riscv-none-embed-gcc-xpack.git submodule update --init --recursive
Note: the repository uses submodules; for a successful build it is mandatory to recurse the submodules.
For development purposes, clone the xpack-develop
branch:
rm -rf ${HOME}/Work/riscv-none-embed-gcc-xpack.git; \
git clone \
--branch xpack-develop \
https://github.com/xpack-dev-tools/riscv-none-embed-gcc-xpack.git \
${HOME}/Work/riscv-none-embed-gcc-xpack.git; \
git -C ${HOME}/Work/riscv-none-embed-gcc-xpack.git submodule update --init --recursive
The script creates a temporary build Work/riscv-none-embed-gcc-${version}
folder in the user home. Although not recommended, if for any reasons
you need to change the location of the Work
folder,
you can redefine WORK_FOLDER_PATH
variable before invoking the script.
Due to the limitations of make
, builds started in folders with
spaces in names are known to fail.
If on your system the work folder is in such a location, redefine it in a
folder without spaces and set the WORK_FOLDER_PATH
variable before invoking
the script.
There are many other settings that can be redefined via
environment variables. If necessary,
place them in a file and pass it via --env-file
. This file is
either passed to Docker or sourced to shell. The Docker syntax
is not identical to shell, so some files may
not be accepted by bash.
The version string is an extension to semver, the format looks like 11.3.0-1
.
It includes the three digits with the original GCC version, a fourth
digit with the SiFive release, a fifth digit with the xPack release number.
When publishing on the npmjs.com server, a sixth digit is appended.
Compared to the original SiFive distribution, there should be no major functional changes, perhaps only bug fixes.
The actual changes for each version are documented in the corresponding release pages:
The details on how to prepare the development environment for native build
are in the
README-DEVELOP.md
file.
The builds currently run on 5 dedicated machines (Intel GNU/Linux, Arm 32 GNU/Linux, Arm 64 GNU/Linux, Intel macOS and Arm macOS.
The current platform for Intel GNU/Linux and Windows production builds is a
Debian 10, running on an Intel NUC8i7BEH mini PC with 32 GB of RAM
and 512 GB of fast M.2 SSD. The machine name is xbbli
.
caffeinate ssh xbbli
Before starting a build, check if Docker is started:
docker info
Before running a build for the first time, it is recommended to preload the docker images.
bash ${HOME}/Work/riscv-none-embed-gcc-xpack.git/scripts/helper/build.sh preload-images
The result should look similar to:
$ docker images
REPOSITORY TAG IMAGE ID CREATED SIZE
ilegeul/ubuntu amd64-18.04-xbb-v3.4 ace5ae2e98e5 4 weeks ago 5.11GB
It is also recommended to Remove unused Docker space. This is mostly useful after failed builds, during development, when dangling images may be left by Docker.
To check the content of a Docker image:
docker run --interactive --tty ilegeul/ubuntu:amd64-18.04-xbb-v3.4
To remove unused files:
docker system prune --force
Since the build takes a while, use screen
to isolate the build session
from unexpected events, like a broken
network connection or a computer entering sleep.
screen -S arm
sudo rm -rf ~/Work/riscv-none-embed-gcc-*-*
bash ${HOME}/Work/riscv-none-embed-gcc-xpack.git/scripts/helper/build.sh --develop --disable-multilib --linux64 --win64
or, for development builds:
sudo rm -rf ~/Work/riscv-none-embed-gcc-*-*
bash ${HOME}/Work/riscv-none-embed-gcc-xpack.git/scripts/helper/build.sh --develop --disable-tests --disable-multilib --arm64 --arm32
When ready, run the build on the production machine (xbbli
):
To detach from the session, use Ctrl-a
Ctrl-d
; to reattach use
screen -r arm
; to kill the session use Ctrl-a
Ctrl-k
and confirm.
About 5 hours later, the output of the build script is a set of 4 files and
their SHA signatures, created in the deploy
folder:
$ ls -l ~/Work/riscv-none-embed-gcc-*/deploy
total 1429180
-rw-rw-r-- 1 ilg ilg 361419776 Oct 24 20:30 xpack-riscv-none-embed-gcc-11.3.0-1-linux-x64.tar.gz
-rw-rw-r-- 1 ilg ilg 120 Oct 24 20:30 xpack-riscv-none-embed-gcc-11.3.0-1-linux-x64.tar.gz.sha
-rw-rw-r-- 1 ilg ilg 375055827 Oct 24 21:04 xpack-riscv-none-embed-gcc-11.3.0-1-win32-x64.zip
-rw-rw-r-- 1 ilg ilg 117 Oct 24 21:04 xpack-riscv-none-embed-gcc-11.3.0-1-win32-x64.zip.sha
The supported Arm architectures are:
armhf
for 32-bit devicesaarch64
for 64-bit devices
The current platform for Arm GNU/Linux production builds is Raspberry Pi OS,
running on a pair of Raspberry Pi4s, for separate 64/32 binaries.
The machine names are xbbla64
and xbbla32
.
caffeinate ssh xbbla64
caffeinate ssh xbbla32
Before starting a multi-platform build, check if Docker is started:
docker info
Before running a build for the first time, it is recommended to preload the docker images.
bash ${HOME}/Work/riscv-none-embed-gcc-xpack.git/scripts/helper/build.sh preload-images
The result should look similar to:
$ docker images
REPOSITORY TAG IMAGE ID CREATED SIZE
hello-world latest 46331d942d63 6 weeks ago 9.14kB
ilegeul/ubuntu arm64v8-18.04-xbb-v3.4 4e7f14f6c886 4 months ago 3.29GB
ilegeul/ubuntu arm32v7-18.04-xbb-v3.4 a3718a8e6d0f 4 months ago 2.92GB
Since the build takes a while, use screen
to isolate the build session
from unexpected events, like a broken
network connection or a computer entering sleep.
screen -S arm
sudo rm -rf ~/Work/riscv-none-embed-gcc-*-*
bash ${HOME}/Work/riscv-none-embed-gcc-xpack.git/scripts/helper/build.sh --develop --disable-multilib --arm64 --arm32
or, for development builds:
screen -S arm
sudo rm -rf ~/Work/riscv-none-embed-gcc-*-*
bash ${HOME}/Work/riscv-none-embed-gcc-xpack.git/scripts/helper/build.sh --develop --disable-tests --disable-multilib --arm64 --arm32
To detach from the session, use Ctrl-a
Ctrl-d
; to reattach use
screen -r arm
; to kill the session use Ctrl-a
Ctrl-k
and confirm.
About 15 hours later, the output of the build script is a set of 2
archives and their SHA signatures, created in the deploy
folder:
$ ls -l ~/Work/riscv-none-embed-gcc-*/deploy
total 702592
-rw-rw-r-- 1 ilg ilg 361351380 Oct 24 23:42 xpack-riscv-none-embed-gcc-11.3.0-1-linux-arm64.tar.gz
-rw-rw-r-- 1 ilg ilg 122 Oct 24 23:42 xpack-riscv-none-embed-gcc-11.3.0-1-linux-arm64.tar.gz.sha
-rw-rw-r-- 1 ilg ilg 358086677 Oct 25 07:31 xpack-riscv-none-embed-gcc-11.3.0-1-linux-arm.tar.gz
-rw-rw-r-- 1 ilg ilg 120 Oct 25 07:31 xpack-riscv-none-embed-gcc-11.3.0-1-linux-arm.tar.gz.sha
Note: on some versions, the 32-bit Arm binaries could not be built, since the RVV support requires more than 4 GB or RAM.
The current platforms for macOS production builds are:
- a macOS 10.13.6 running on a MacBook Pro 2011 with 32 GB of RAM and
a fast SSD; the machine name is
xbbmi
- a macOS 11.6.1 running on a Mac Mini M1 2020 with 16 GB of RAM;
the machine name is
xbbma
caffeinate ssh xbbmi
caffeinate ssh xbbma
To build the latest macOS version:
screen -S arm
sudo rm -rf ~/Work/riscv-none-embed-gcc-*-*
caffeinate bash ${HOME}/Work/riscv-none-embed-gcc-xpack.git/scripts/helper/build.sh --develop --disable-multilib --macos
or, for development builds:
sudo rm -rf ~/Work/riscv-none-embed-gcc-*-*
caffeinate bash ${HOME}/Work/riscv-none-embed-gcc-xpack.git/scripts/helper/build.sh --develop --disable-tests --disable-multilib --macos
To detach from the session, use Ctrl-a
Ctrl-d
; to reattach use
screen -r arm
; to kill the session use Ctrl-a
Ctrl-k
and confirm.
In about 4 hours, the output of the build script is a compressed archive
and its SHA signature, created in the deploy
folder:
$ ls -l ~/Work/riscv-none-embed-gcc-*/deploy
total 698240
-rw-r--r-- 1 ilg staff 357492352 Oct 24 22:51 xpack-riscv-none-embed-gcc-11.3.0-1-darwin-x64.tar.gz
-rw-r--r-- 1 ilg staff 121 Oct 24 22:51 xpack-riscv-none-embed-gcc-11.3.0-1-darwin-x64.tar.gz.sha
Instead of --all
, you can use any combination of:
--linux64 --win64
On Arm, instead of --all
, you can use any combination of:
--arm64 --arm32
Please note that, due to the specifics of the GCC build process, the
Windows build requires the corresponding GNU/Linux build, so --win64
should be run after or together with --linux64
.
To remove most build files, use:
bash ${HOME}/Work/riscv-none-embed-gcc-xpack.git/scripts/helper/build.sh clean
To also remove the repository and the output files, use:
bash ${HOME}/Work/riscv-none-embed-gcc-xpack.git/scripts/helper/build.sh cleanall
For production builds it is recommended to completely remove the build folder.
For performance reasons, the actual build folders are internal to each Docker run, and are not persistent. This gives the best speed, but has the disadvantage that interrupted builds cannot be resumed.
For development builds, it is possible to define the build folders in the host file system, and resume an interrupted build.
For development builds, it is also possible to create everything
with -g -O0
and be able to run debug sessions.
By default, the build steps use all available cores. If, for any reason, parallel builds fail, it is possible to reduce the load.
For development builds, to save time, it is recommended to build the toolchain without multilib.
The Docker scripts run with root privileges. This is generally not a problem, since at the end of the script the output files are reassigned to the actual user.
However, for an interrupted build, this step is skipped, and files in
the install folder will remain owned by root. Thus, before removing the
build folder, it might be necessary to run a recursive chown
.
A simple test is performed by the script at the end, by launching the executables to check if all shared/dynamic libraries are correctly used.
For a true test you need to unpack the archive in a temporary location
(like ~/Downloads
) and then run the
program from there. For example on macOS the output should
look like:
$ .../xpack-riscv-none-embed-gcc/bin/riscv-none-embed-gcc --version
riscv-none-embed-gcc (xPack RISC-V Embedded GCC, 64-bit) 11.3.0
After install, the package should create a structure like this (only the first two depth levels are shown):
$ tree -L 2 /Users/ilg/Library/xPacks/\@xpack-dev-tools/riscv-none-embed-gcc/11.3.0-1/.content/
/Users/ilg/Library/xPacks/\@xpack-dev-tools/riscv-none-embed-gcc/11.3.0-1/.content/
├── README.md
├── bin
│ ├── riscv-none-embed-addr2line
│ ├── riscv-none-embed-ar
│ ├── riscv-none-embed-as
│ ├── riscv-none-embed-c++
│ ├── riscv-none-embed-c++filt
│ ├── riscv-none-embed-cpp
│ ├── riscv-none-embed-elfedit
│ ├── riscv-none-embed-g++
│ ├── riscv-none-embed-gcc
│ ├── riscv-none-embed-gcc-11.3.0
│ ├── riscv-none-embed-gcc-ar
│ ├── riscv-none-embed-gcc-nm
│ ├── riscv-none-embed-gcc-ranlib
│ ├── riscv-none-embed-gcov
│ ├── riscv-none-embed-gcov-dump
│ ├── riscv-none-embed-gcov-tool
│ ├── riscv-none-embed-gdb
│ ├── riscv-none-embed-gdb-add-index
│ ├── riscv-none-embed-gdb-add-index-py3
│ ├── riscv-none-embed-gdb-py3
│ ├── riscv-none-embed-gprof
│ ├── riscv-none-embed-ld
│ ├── riscv-none-embed-ld.bfd
│ ├── riscv-none-embed-lto-dump
│ ├── riscv-none-embed-nm
│ ├── riscv-none-embed-objcopy
│ ├── riscv-none-embed-objdump
│ ├── riscv-none-embed-ranlib
│ ├── riscv-none-embed-readelf
│ ├── riscv-none-embed-size
│ ├── riscv-none-embed-strings
│ └── riscv-none-embed-strip
├── distro-info
│ ├── CHANGELOG.md
│ ├── licenses
│ ├── patches
│ └── scripts
├── include
│ └── gdb
├── lib
│ ├── bfd-plugins
│ ├── gcc
│ ├── libcc1.0.so
│ ├── libcc1.so -> libcc1.0.so
│ └── python3.7
├── libexec
│ ├── gcc
│ ├── libcrypt.2.dylib
│ ├── libcrypto.1.1.dylib
│ ├── libexpat.1.6.7.dylib
│ ├── libexpat.1.dylib -> libexpat.1.6.7.dylib
│ ├── libgcc_s.1.dylib
│ ├── libgmp.10.dylib
│ ├── libiconv.2.dylib
│ ├── libintl.8.dylib
│ ├── libisl.15.dylib
│ ├── liblzma.5.dylib
│ ├── libmpc.3.dylib
│ ├── libmpfr.4.dylib
│ ├── libncurses.6.dylib
│ ├── libpanel.6.dylib
│ ├── libpython3.7m.dylib
│ ├── libreadline.8.0.dylib
│ ├── libreadline.8.dylib -> libreadline.8.0.dylib
│ ├── libsqlite3.0.dylib
│ ├── libssl.1.1.dylib
│ ├── libstdc++.6.dylib
│ ├── libz.1.2.8.dylib
│ └── libz.1.dylib -> libz.1.2.8.dylib
├── riscv-none-embed
│ ├── bin
│ ├── include
│ ├── lib
│ └── share
└── share
├── doc
└── gcc-riscv-none-embed
21 directories, 58 files
No other files are installed in any system folders or other locations.
The binaries are distributed as portable archives; thus they do not need to run a setup and do not require an uninstall.
The XBB build scripts use a local cache such that files are downloaded only during the first run, later runs being able to use the cached files.
However, occasionally some servers may not be available, and the builds may fail.
The workaround is to manually download the files from an alternate
location (like
https://github.com/xpack-dev-tools/files-cache/tree/master/libs),
place them in the XBB cache (Work/cache
) and restart the build.
The build process is split into several scripts. The build starts on the
host, with build.sh
, which runs container-build.sh
several times,
once for each target, in one of the two docker containers. Both scripts
include several other helper scripts. The entire process is quite complex,
and an attempt to explain its functionality in a few words would not
be realistic. Thus, the authoritative source of details remains the source
code.