1. Bootstrap the Rust Language
This section builds the prerequisite(s) for the rust programming language. All of these prerequisites are ephemera; after the final rust compiler is built and installed, everything built here will be destroyed.
We need this process because the rust compiler is written in rust; the rust project team makes sure that each minor version of the rust compiler can be built by the prior minor version. At the time I’m writing this, in March of 2025, the current version of rust — which is what the rust blueprint builds — is 1.85.0, which can be built by the 1.84 release, and so on. That being the case, I only know of two paths to bootstrap rust entirely from source code:
-
you can start with OCaml,[1] use it to compile rust 0.7, and then step up through all the versions of rust to 1.84.1; or
-
you can use a C++ compiler to build mrustc, use mrustc to build rust 1.74.0, and then step up through the versions from there to 1.84.1.
Obviously, we’re going to use the latter approach.
The creation of this blueprint was enormously simplified
by David Tolnay’s rust bootstrap project
at https://github.com/dtolnay/bootstrap,
which does essentially the same thing we’re doing here.
(The main difference being,
that project runs on a minimal debian userspace
constructed with debootstrap
,
while this one runs on Little Blue Linux.
This process is also a lot less generous with storage space:
you can perform the entire build with only about 25 GiB
of space avaialble for it.)
I doubt I would have attempted
to support the rust language in LB Linux at all
without having that project to use as a starting point.
Be aware that this process takes quite a while! I run it on an AMD64 virtual machine with 18 CPU cores. The LibreSSL build takes only a couple of minutes; mrustc takes almost an hour; and each of the rustc stages from 1.75 to 1.84 takes about half an hour. Figure around six hours for the full bootstrap. So far, I’ve been unable to get the bootstrap to work on my Orange Pi ARM computer, so I can’t give timing details for that.
[bootstrap-sources-rustc] == Source Code For The Rust Bootstrap
As I described earlier, in [bootstrap-sources-go], my practice is to package all the source code for ephemeral versions of packages needed for these bootstrap processes in a tarred-up git repository. Here’s how to create the bootstrap source archive for rust, if you want to do that rather than simply using the one I’ve built.
First, create an empty repository
in a bootstrap-rustc-1.85.0
directory.
I always create an empty first commit, so I will do that here.
mkdir bootstrap-rustc-1.85.0; cd bootstrap-rustc-1.85.0 git init git commit --allow-empty -m "commit zero"
Now, for each version of each package that you need during the bootstrap, create a branch and import it.
The mrustc compiler can be found at https://github.com/thepowersgang/mrustc.
All versions of the rust compiler sources can be found
at https://static.rust-lang.org/dist/rustc-x.yy.z-src.tar.xz
(where x.yy.z is the release number you want).
You can also obtain GPG signatures by adding .asc
to that URL.
These are not the same as the source archives you can find
at https://github.com/rust-lang/rust — as far as I’m aware, those archives won’t work properly with this process.
git checkout -b mrustc-0.11.2 tar -x --strip-components=1 -f /full/path/to/mrustc-0.11.2.tar.* git add -f . git commit -m "mrustc 0.11.2" git checkout master
Repeat those steps for rustc-1.74.0 through rustc-1.84.1.
There are two patches to apply for each rustc version; my practice is to apply those patches as commits on the branches in this git repository, because there’s really no reason to do anything else.
First, one of the vendored libraries included with rustc is openssl-sys
,
and old versions of that library — some of which are present in the rustc distribution files — have a version check that says they do not work with LibreSSL 4.0.
Since that’s the version used in CBL,
we either have to include an old version of LibreSSL for use by rustc,
or modify the version check so that it allows LibreSSL 4.0 to be used.
I prefer the latter approach,
so that’s a patch I apply to each version of rustc.
To compose those patches, modify all
versions[2]
of vendor/openssl-sys*/build/main.rs
.
In the validate_headers
function, there’s a list of LibreSSL versions
that includes things like:
(3, 7, _) => ('3', '7', 'x'), (3, 8, 0) => ('3', '8', '0'),
Just add these lines:
(4, 0, 0) => ('4', '0', '0'), (4, 0, _) => ('4', '0', 'x'),
in that same block.
- There’s one other irritation related to the vendored source packages
-
each package under vendor contains a file
.cargo-checksum.json
that contains SHA256 checksums of all the files in that package, and the build process checks to make sure the checksums match. So in addition to changing thebuild/main.rs
files, you need to calculate the new checksums and modify the.cargo-checksum.json
file as well. There is probably be a more idiomatic way to make changes like this; the message given if the checksum does not match is:directory sources are not intended to be edited, if modifications are required then it is recommended that `[patch]` is used with a forked copy of the source
But I don’t know exactly what that means or how to do it. If you do, let me know!
After you have made this change on one branch, you might be able to save effort by cherry-picking it onto each other branch, although you’ll need to resolve conflicts and double-check the results, including the checksum changes, each time.
In addition to the openssl-sys patch,
for each rustc version except the first one,
you should also create a config.toml
file
(and put it on the same branch, as a second commit).
This is used to configure the build process.
For the first several versions of rustc,
The config.toml
file should look like this:
change-id = SOME_CHANGE_ID_NUMBER_SEE_BELOW [build] cargo = "/FULL/PATH/TO/PREVIOUS/CARGO" rustc = "/FULL/PATH/TO/PREVIOUS/RUSTC" docs = false vendor = true extended = true tools = ["cargo"] [rust] channel = "stable"
The cargo
and rustc
lines are used to find the version of those programs
from the previous build of rustc.
The first intermediate version of rustc will be built by mrustc
and does not need a config.toml
file at all,
because the mrustc build process will define how it should be configured
and how it should find the bootstrap versions of cargo
and rustc
.
The second intermediate version of rustc,
which does need a config.toml
,
should specify /usr/src/rustc/bs/prev_rust/prefix/bin/cargo
and /usr/src/rustc/bs/prev_rust/prefix/bin/rustc
for these,
to use the versions of those programs produced by the mrustc build.
For all other intermediate rustc builds — and the final system rustc build — config.toml
should specify cargo as
/usr/src/rustc/bs/prev_rust/stage1-tools-bin/cargo
and rustc as
/usr/src/rustc/bs/prev_rust/stage1/bin/rustc
,
because that’s where this bootstrap build process will put them..
The other config.toml
line that needs to differ for each version
is the change-id
.
For rustc 1.75, there is a specific value for change-id
in the config.example.toml
file in the distribution,
so just use that one.
For all the later versions,
the example config file has a note saying that the latest change ID
in src/bootstrap/src/utils/change_tracker.rs
should be used;
so you should look at the CONFIG_CHANGE_HISTORY
section of that file
to find the largest change_id
value, and use that.
Later versions of rustc require additional configuration directives. For the rustc 1.83 build and later, you should add an additional stanza:
[llvm] download-ci-llvm = false
Also, for rustc 1.84 and later,
you should add the directive download-rustc = false
to the [rust]
stanza.
Once all the branches have been added
(and patched to include the config.toml
file),
check out the empty master
branch
and repack the repository to save as much space as possible:
rm .git/hooks/*.sample git reflog expire --expire=0 --all git gc git prune git pack-refs git repack -A -d -f --depth=500 --window=500
Now create a tar archive containing bootstrap-rustc-1.85.0
, compress it,
and put it in the TARFILE_DIR
location.
That’s the source code bundle that this blueprint will use.
1.1. Rustc Package User
The convention used in CBL, when constructing dependencies for a package that needs a process like this, is to do those builds as the same package user that will then be used for the desired package. We therefore need to ensure that package user exists.
add-package-user 'Rust Compiler' rustc
We’ll build the bootstrap ephemera in a bs
(for "BootStrap")
subdirectory of the package user’s home directory
so that it will be easy to get rid of it all when we’re done with it.
su -l -c 'mkdir -p $HOME/bs/' rustc
su -l -c "tar -x -C ~rustc/bs -f ${TARFILE_DIR}/bootstrap-rustc*tar.lz" \
rustc
1.2. Building mrustc
For each stage in our bootstrap build, we’ll start with a guard clause so that if we restart the bootstrap, it won’t necessarily need to start over from the beginning.
#!/bin/bash
set -e
if [ -f $HOME/bs/.mrustc-complete ]
then
echo "mrustc has already been built, skipping it"
exit
fi
The first part of the mrustc build simply builds mrustc.
cd $HOME/bs/bootstrap-rustc-*
git checkout mrustc-0.11.2
export LIBSSH2_SYS_USE_PKG_CONFIG=true
export MRUSTC_TARGET_VER="1.74"
export PARLEVEL="$(nproc)"
export RUSTC_VERSION="1.74.0"
make -j$(nproc) -f minicargo.mk bin/mrustc
mrustc has Makefile targets for building a full rustc, and that’s what we’ll do next.
First, we’ll need to have the first intermediate version of rustc itself.
This will be the latest version that mrustc is able to build successfully,
currently 1.74.
The Makefile target RUSTCSRC
expects to find the rustc
source distribution tar file in its top-level directory,
and will try to download it if it’s not there;
Since we don’t ever want to have anything downloaded at build time,
we’ll regenerate the source archive from our bootstrap git repository.
The RUSTCSRC
target will then unpack it and apply a patch
(presumably to make it easier for mrustc to build it),
git archive --prefix=rustc-1.74.0-src/ rustc-1.74.0 |
gzip > rustc-1.74.0-src.tar.gz
make -j$(nproc) -f minicargo.mk RUSTCSRC
I believe the next steps build a basic rustc
and cargo
,
and then use those tools to build a full version of rustc
including the rust standard library.
make -j$(nproc) -f minicargo.mk "output-1.74.0/rustc"
make -j$(nproc) -f minicargo.mk "output-1.74.0/cargo"
pushd run_rustc
make -j$(nproc)
Each subsequent rustc build will use the rustc
and cargo
programs
found in the prev_rust
directory, so that’s where we’ll put the
version built by mrustc.
THe rustc
built here is a shell script that sets LD_LIBRARY_PATH
and then runs the actual rustc
compiler.
The directories in that shell script specify the location of the libraries
where they were just built;
since we’re moving them, we have to adjust that shell script.
mv output-1.74.0 $HOME/bs/prev_rust
popd
oldpath='/usr/src/rustc/bs/bootstrap-rustc-1.85.0/run_rustc/output-1.74.0'
newpath="$HOME/bs/prev_rust"
sed -i -e "s@${oldpath}@${newpath}@g" $HOME/bs/prev_rust/prefix/bin/rustc
unset oldpath newpath
At the end of each build stage,
we’ll create the file checked by the guard clause,
and then put our local git repository back at its base state
(empty master
branch checked out, no files in the workspace).
For the mrustc build, we touch the guard file for rustc 1.74 as well,
because that was done as part of the build-mrustc
script.
touch $HOME/bs/.mrustc-complete
touch $HOME/bs/.rustc-1.74.0-complete
git checkout master
git clean -f -d .
su -l -c "bash /tmp/build-mrustc" rustc
1.3. Building The Remaining Intermediate Versions of Rustc
Now we’ll loop through all the rustc-
branches
in the bootstrap source repository,
to build all the versions of rustc
we need for this bootstrap.
The git for-each-ref
command will include
the version of rustc that mrustc just built,
but since we touched the complete
file for it,
the guard clause here will cause it to be skipped.
#!/bin/bash
set -e
export LIBSSH2_SYS_USE_PKG_CONFIG=true
export CARGO_HOME=$HOME/bs/.cargo
unset SUDO_USER
cd $HOME/bs/bootstrap-rustc-*
git for-each-ref --format='%(refname:short)' 'refs/*/rustc*' |
sort |
while read VERSION
do
if [ -f $HOME/bs/.${VERSION}-complete ]
then
echo "$VERSION has already been built, skipping it"
else
The rustc distribution includes a script, x.py
,
that actually builds the package.
The same script is normally used to install the package as well,
but we’re going to skip the installation process and just copy
the build output we need to prev_rust
.
Before we do that, we’ll wipe out the existing prev_rust
directory,
since we will no longer need it.
Then we’ll create the complete
file and tidy up the workspace as usual.
git checkout $VERSION
python x.py build
rm -rf $HOME/bs/prev_rust
archdir=$(dirname $(find build -name stage1))
mv $archdir $HOME/bs/prev_rust
unset archdir
touch $HOME/bs/.${VERSION}-complete
git checkout master
git clean -f -d .
fi
done
The final intermediate stage is finally complete, so we can clean up the bootstrap source repository and proceed with the final system rustc build!
cd $HOME/bs
rm -rf bootstrap-rustc-*
su -l -c "bash /tmp/build-intermediate-rustcs" rustc
1.4. Complete text of files
1.4.1. /tmp/build-intermediate-rustcs
#!/bin/bash
set -e
export LIBSSH2_SYS_USE_PKG_CONFIG=true
export CARGO_HOME=$HOME/bs/.cargo
unset SUDO_USER
cd $HOME/bs/bootstrap-rustc-*
git for-each-ref --format='%(refname:short)' 'refs/*/rustc*' |
sort |
while read VERSION
do
if [ -f $HOME/bs/.${VERSION}-complete ]
then
echo "$VERSION has already been built, skipping it"
else
git checkout $VERSION
python x.py build
rm -rf $HOME/bs/prev_rust
archdir=$(dirname $(find build -name stage1))
mv $archdir $HOME/bs/prev_rust
unset archdir
touch $HOME/bs/.${VERSION}-complete
git checkout master
git clean -f -d .
fi
done
cd $HOME/bs
rm -rf bootstrap-rustc-*
1.4.2. /tmp/build-mrustc
#!/bin/bash
set -e
if [ -f $HOME/bs/.mrustc-complete ]
then
echo "mrustc has already been built, skipping it"
exit
fi
cd $HOME/bs/bootstrap-rustc-*
git checkout mrustc-0.11.2
export LIBSSH2_SYS_USE_PKG_CONFIG=true
export MRUSTC_TARGET_VER="1.74"
export PARLEVEL="$(nproc)"
export RUSTC_VERSION="1.74.0"
make -j$(nproc) -f minicargo.mk bin/mrustc
git archive --prefix=rustc-1.74.0-src/ rustc-1.74.0 |
gzip > rustc-1.74.0-src.tar.gz
make -j$(nproc) -f minicargo.mk RUSTCSRC
make -j$(nproc) -f minicargo.mk "output-1.74.0/rustc"
make -j$(nproc) -f minicargo.mk "output-1.74.0/cargo"
pushd run_rustc
make -j$(nproc)
mv output-1.74.0 $HOME/bs/prev_rust
popd
oldpath='/usr/src/rustc/bs/bootstrap-rustc-1.85.0/run_rustc/output-1.74.0'
newpath="$HOME/bs/prev_rust"
sed -i -e "s@${oldpath}@${newpath}@g" $HOME/bs/prev_rust/prefix/bin/rustc
unset oldpath newpath
touch $HOME/bs/.mrustc-complete
touch $HOME/bs/.rustc-1.74.0-complete
git checkout master
git clean -f -d .
2. rustc
Name |
Rust Compiler |
---|---|
Version |
1.85.0 |
Project URL |
|
SCM URL |
|
Download URL |
(unknown) |
Patches |
|
Dependencies |
Rust is a programming language. Some people quite like it, to the point that there are people trying to make it easy to write parts of the Linux kernel in rust rather than C.
This package provides a compiler for the rust language, rustc
.
It requires a fairly complex bootstrap process.
The source distribution file for this package expands
into a directory rustc-x.yy.z-src
(where x.yy.z
is the version number)
so it must be repackaged for CBL.
The distribution can be downloaded from
https://static.rust-lang.org/dist/rustc-x.yy.z-src.tar.xz,
and a GPG signature can be obtained by adding `.asc`to that URL.
That site doesn’t provide a list of available files, though,
so you have to know the full version number of the release you want.
- Dependencies
2.1. How The Rust Programming Language Works in GNU/Linux
I don’t know much about rust,
aside from the fact that it’s really irritating to build
unless you’re willing to start by downloading a binary version of it.
About the only things I can tell you are that
rustc
is the rust compiler,
and it comes with a program called cargo
that manages dependencies and is perhaps also a build system.
Rust libraries are called "crates," I think?
If you’re interested in learning more about rust, there’s a lot of documentation — including entire books about it! — at https://www.rust-lang.org/.
This blueprint will install rust to /opt/rust
,
so if you want to use it you will probably want
to add /opt/rust/bin
to your PATH
.
You may also need to adjust LD_LIBRARY_PATH
;
I’m not sure.
2.2. Building Rust
The rust source distribution has a vendor
directory
that uses almost two gigabytes of storage space,
with multiple versions of almost every package in it.
This is annoying, but not really problematic.
One thing in it that is problematic is the vendored openssl-sys
package:
rustc 1.85 includes four versions of that, ranging from 0.9.72 to 0.9.104,
and the versions before 0.9.102 have a version check
that prohibits use of LibreSSL 4.0.0,
which is what LB Linux includes.
Rather than providing an old version of LibreSSL just for rust,
I opt to modify the version check to accept LibreSSL 4.0.0.
-
rustc-1.85.0-modify-openssl-version-check-1.patch
The rustc source code includes a configure
script,
but it’s not anything like an autotools-generated configure
.
All it actually does is write a config.toml
file
that actually controls how the build is done.
Here, we’ll write a config.toml directly.
This will be quite similar to the config.toml
files
we used to build all the bootstrap versions of rustc.
We’ll tell it to build the documentation, though,
and we’ll build more of the available tools than just cargo
.
If you’re going to be doing a lot of rust development,
you might want to set sanitizers
and profiler
to true
;
those will build additional runtimes that might be helpful.
You might also want to build documentation by setting docs
to true
.
You can read about these and other options in the config.example.toml
that is part of the source distribution.
There’s probably other documetnation available as well,
but I’m not sure where to find it.
CT=config.toml
echo 'change-id = 134809' > $CT
echo '[build]' >> $CT
echo 'cargo = "/usr/src/rustc/bs/prev_rust/stage1-tools-bin/cargo"' >> $CT
echo 'rustc = "/usr/src/rustc/bs/prev_rust/stage1/bin/rustc"' >> $CT
echo 'docs = false' >> $CT
echo 'vendor = true' >> $CT
echo 'extended = true' >> $CT
echo 'tools = ["cargo", "clippy", "rustdoc", "rustfmt"]' >> $CT
echo '[rust]' >> $CT
echo 'channel = "stable"' >> $CT
echo 'download-rustc = false' >> $CT
echo '[llvm]' >> $CT
echo 'download-ci-llvm = false' >> $CT
Since we actually want to install this version of rustc as a system package,
we also want an install
section
that will put things in the right place.
Unfortunately, the build fails if we tell it to install to /usr
and /etc
directly — it probably tries to do something to the /usr
directory per se,
which fails since that is not writable by package users.
We can install it to /opt/rust
instead.
echo '[install]' >> $CT
echo 'prefix = "/opt/rust"' >> $CT
echo 'sysconfdir = "/opt/rust/etc"' >> $CT
unset CT
Building and installing Rust is done using the x.py
script.
python x.py build
(none)
python x.py install
Now that we’ve actually installed everything, we can also clean up the bootstrap directory.
rm -rf $HOME/bs