lcat
My personal /var/log
  • Home
  • Contact
  • HackMe!

Logbook cross-compiling dari Linux untuk FreeBSD

Motivasi awal melakukan cross-compiling ini adalah karena saya ingin mencoba menulis ulang blog ini menggunakan Rust. Ide ini terhenti sebentar setelah sadar bahwa NFSN, perusahaan hosting blog saya sekarang, menggunakan host FreeBSD 14.3.

### 2025-08-11

Saya sebenarnya sudah berhasil cross-compiling dengan bantuan [cross](https://github.com/cross-rs/cross). Tapi rasanya seperti jalan pintas dan saya juga masih belum paham apa yang sebenarnya terjadi. Akhirnya saya memutuskan untuk mencoba cross-compiling sendiri tanpa Docker dan cross.

Saya baca salah satu komentar di [sini](https://users.rust-lang.org/t/build-rustc-properly-for-cross-targets-linux-freebsd/38059) untuk langsung coba dengan `rustup target add x` lalu `cargo build --target x`, namun tidak langsung bisa. Ada error gagal menemukan `libexecinfo` yang ternyata setelah saya pelajari adalah library yang ada di FreeBSD.

### 2025-08-12

Saya banyak browsing-browsing.. sepertinya perlu mengunduh sysroot FreeBSD, saya coba ambil dari [sini](https://download.freebsd.org/ftp/releases/amd64/14.3-RELEASE/). Untuk kompilasi sepertinya perlu override build script yang dimention di [sini](https://rust-lang.github.io/rustup/cross-compilation.html). Awalnya saya coba mengubah `rustflags` pada `~/.cargo/config.toml` untuk menambah `-L ~/opt/freebsd-sysroot/lib`. Meski terlihat berhasil, sepertinya saya mengacaukan proses compile-nya. Yang benar adalah menambahkan opsi `-C link-arg=--sysroot=/home/cat/freebsd-14.3-sysroot`. Namun, ini masih belum berhasil karena gagal di proses linking, beberapa hal tidak berhasil ditemukan oleh linker.

Salah satu bacaan yang saya temukan menarik adalah yang dari [OS Dev Wiki tentang GCC Cross-Compiler](https://wiki.osdev.org/GCC_Cross-Compiler). Saya jadi baca-baca artikel wiki lainnya, misalnya yang membahas System V ABI, red zone pada x86-64, sistem penamaan target triplet yang ambigu.

### 2025-08-12

Saya menemukan [artikel menarik](https://marcelog.github.io/articles/cross_freebsd_compiler_in_linux.html) tentang cross-compiling ke FreeBSD dari Linux. Sedang saya coba ikuti, mulai dari proses build binutils dan gcc. Menarik ya.. compiling a compiler :))

Sudah berhasil compile binutils (ketika configure, yang paling menarik mungkin adalah opsi `--target`). Sekarang mau coba langsung compile GCC, ternyata memang perlu GMP, MPFR, dan MPC. Saya ragu kalau saya configure build gcc tanpa compile dulu untuk GMP, MPFR, dan MPC, jadi saya stick to the tutorial saja.

Sebelum bisa compile, di tutorial *Following the GNU toolchain convention*, perlu membuat direktori `$prefix/x86_64-unknown-freebsd` dulu. Saya kutip ulang yang dikutip artikelnya.

> By convention, the GNU toolchain will store all its data in ${prefix}/${target} where the prefix is the installation prefix for the toolchain (i.e: /usr) and target is the name of the target for which this toolchain generates binaries (i.e: /usr/x86_64-pc-freebsd7, /usr/x86_64-pc-linux-gnu, etc). Also, the names for the linker, compiler, etc, will be named in the form ${target}-gcc, ${target}-ar, ${target}-ld, etc. So you can have multiple toolchains for multiple targets residing in the same prefix installation directory. Inside each target directory, you would find /include, /lib, and /bin which helps to keep things clear, since each target has its own specific libraries, includes, stubs for executables, etc.

Saya sudah berhasil compile dan install semua dependensi GCC yaitu GMP, MPFR, dan MPC. Saya coba untuk configure GMP, MPFR, dan MPC dengan tidak mengikuti tutorial, tapi hanya dengan `--prefix $prefix` (dalam kasus saya, prefix adalah `/home/cat/opt/cross-freebsd`) dan `--host=x86_64-unknown-freebsd`. Sekarang, ketika compile GCC, configure-nya juga mirip, tapi `--host` diganti `--target`, sama ketika mengkompilasi binutils. Kalau untuk GCC, saya harus menspesifikkan target ke major version FreeBSD-nya, sehingga menjadi `x86_64-unknown-freebsd14`. Semoga ini tidak mengharuskan saya mengulang dari awal karena sejak awal tanpa suffix `14` :)

### 2025-08-13

Hari sudah berganti. GCC sedang compiling, kata Google sekitar 10-15 menit.. mungkin harus ditinggal istirahat dulu 😁. Malam ini saya belajar banyak hal dan sepertinya masih banyak yang masih belum sepenuhnya bisa saya pahami. Beberapa di antaranya adalah konsep soal linker, proses build untuk program C menggunakan configure, m4, dan kawan-kawannya yang asing saya lihat.

^ Later discovered: Ternyata build/distribution system yang dipakai adalah GNU autotools. Dari review yang saya baca, GNU autotools ini katanya lumayan robust dan fleksibel. Semoga bisa saya coba di masa depan.

### 2025-08-14

Saya sudah agak luang. Namun compile GCC belum juga berhasil meski sudah saya ulang dengan target triplet yang benar `x86_64-unknown-freebsd14`. Entah apa yang salah, saya sudah set `LD_LIBRARY_PATH` dengan benar.
 
```
/home/cat/opt/cross-freebsd/x86_64-unknown-freebsd14/bin/ld: cannot find /lib/libc.so.7: No such file or directory
/home/cat/opt/cross-freebsd/x86_64-unknown-freebsd14/bin/ld: cannot find /usr/lib/libc_nonshared.a: No such file or directory
collect2: error: ld returned 1 exit status
make[2]: *** [Makefile:1009: libgcc_s.so] Error 1
make[2]: Leaving directory '/home/cat/opt/cross-freebsd/scratch/gcc-15.2.0/build/x86_64-unknown-freebsd14/libgcc'
make[1]: *** [Makefile:14309: all-target-libgcc] Error 2
make[1]: Leaving directory '/home/cat/opt/cross-freebsd/scratch/gcc-15.2.0/build'
make: *** [Makefile:1061: all] Error 2
```

Saya akan coba ulang lagi dari awal, kali ini mengikuti [tutorial ini](https://preshing.com/20141119/how-to-build-a-gcc-cross-compiler/).
Mungkin ada yang membaca post ini lalu dalam hatinya menyarankan "kenapa gak chatgpt aja?" Saya rasa saya jelek sekali dalam *prompting*, karena bertanya ke ChatGPT malah membuat saya semakin bingung 🥲. Saya baru tahu dari tutorial yang saya ikuti ini, bahwa tidak perlu meng-compile GMP, MPFR, dan MPC satu per satu, melainkan bisa sekalian dengan GCC asalkan dibuatkan salinan/symlink dulu di source dir GCC. Saya juga baru tahu setelah baca-baca dari [dokumentasi GCC](https://gcc.gnu.org/install/download.html), kalau binutils juga bisa sekalian di-build :)

> If you also intend to build binutils (either to upgrade an existing installation or for use in place of the corresponding tools of your OS), unpack the binutils distribution either in the same directory or a separate one. In the latter case, add symbolic links to any components of the binutils you intend to build alongside the compiler (bfd, binutils, gas, gprof, ld, opcodes, …) to the directory containing the GCC sources.

> Likewise the GMP, MPFR, MPC and Gettext libraries can be automatically built together with GCC. You may simply run the contrib/download_prerequisites script in the GCC source directory to set up everything. Otherwise unpack the GMP, MPFR, MPC and/or Gettext source distributions in the directory containing the GCC sources and rename their directories to gmp, mpfr, mpc and gettext, respectively (or use symbolic links with the same name).

Setelah menunggu sekitar setengah jam.. **it's a success**! Ternyata tidak perlu juga provide `LD_LIBRARY_PATH`. Mungkin saja pada step sebelumnya saya gagal karena ada hal minor yang miss contohnya typo/salah eja. Atau mungkin karena kali ini target `make` saya adalah `all-gcc` bukan `make` saja (default target). Kalau dirangkum, history command saya kurang lebih seperti ini.

```fish
set -x PREFIX $HOME/opt/cross
set -x TARGET x86_64-unknown-freebsd14

# setup sources to compile
mkdir cross-freebsd-src
cd cross-freebsd-src
wget all binutils and all gcc dependencies (GMP, MPFR, MPC) -> shortcut: use contrib/download_prerequisites
for f in *.tar.*; tar xvf $f; end

# copy downloaded freebsd14.3 sysroot to $PREFIX/$TARGET
cp -r path/to/freebsd14.3-sysroot/* $PREFIX/$TARGET/

# compile binutils
cd binutils-2.45/
mkdir build
cd build
../configure --prefix=$PREFIX --target=$TARGET --disable-multilib
make -j(nproc)
make install

# compile gcc
cd ../gcc-15.2.0/
ln -s ../mpfr-4.2.2 mpfr
ln -s ../gmp-6.3.0 gmp
ln -s ../mpc-1.3.1 mpc
mkdir build
cd build
../configure --prefix=$PREFIX --target=$TARGET --disable-multilib --enable-languages=c,c++
make -j(nproc) all-gcc
make install-gcc
```


Ternyata ini masih belum selesai, saya masih belum bisa meng-compile program hello world sederhana karena tidak ada header standar seperti `stdio.h`. Mungkin ini karena target yang `all-gcc`. Karena next step di tutorialnya agak panjang, saya akan coba `make` biasa, kalau tidak bisa, saya akan ikuti sesuai tutorial. Gagal.

Setelah baca lebih cermat, di tutorial ini ternyata sudah dibahas:

> All of the remaining steps involve building GCC and Glibc. The trick is that there are parts of GCC which depend on parts of Glibc already being built, and vice versa. We can’t build either package in a single step; we need to go back and forth between the two packages and build their components in a way that satisfies their dependencies.

![ilustrasi di web preshing.com](https://preshing.com/images/cross-gcc-steps.png)

Setelah baca-baca lagi, ternyata di FreeBSD tidak pakai Glibc, itu sebabnya saya gagal compile Glibc. Saya juga baru ingat kalau sysroot FreeBSD 14.3 yang saya download sudah include pre-compiled libc. Setelah berbagai percobaan dan membaca [artikel SO ini](https://stackoverflow.com/questions/23011367/arm-linux-gnu-gcc-fatal-error-stdio-h-no-such-file-or-directory), saya berhasil compile hello world.

![hasil cross-compile dan run hello world]()

Selanjutnya, saya akan coba mengkonfigurasi config cargo untuk menggunakan gcc yang sudah dicompile sebagai linker. Isi file `~/.cargo/config.toml` saya sebagai berikut.

```toml
[target.x86_64-unknown-freebsd]
linker = "/home/cat/opt/cross/bin/x86_64-unknown-freebsd14-gcc"
rustflags = [
  "-C",
  "link-arg=--sysroot=/home/cat/opt/cross/x86_64-unknown-freebsd14",
]
```

Untuk program hello world, kompilasi berjalan lancar. Namun, ketika mencoba cross compile program untuk blog ini yang menggunakan library `rusqlite`, kompilasinya gagal. Sebabnya adalah karena pustaka `rusqlite` melakukan kompilasi sendiri yang terpisah ketika proses build menggunakan [pustaka cc-rs](https://docs.rs/cc/latest/cc/). Menambahkan environment variable CFLAGS membuat kompilasi berhasil.

```
env CFLAGS="--sysroot=$SYSROOT -I $SYSROOT/usr/include" cargo build --target=x86_64-unknown-freebsd
```

### Penutup

Terima kasih untuk pembaca, jika ada, yang membaca sampai akhir coret-coretan saya. Ternyata, tidak mudah untuk melakukan cross compiling, terutama jika harus build semuanya from source. Berikutnya saya akan menggunakan tool-tool pembantu seperti [cross-rs](https://github.com/cross-rs/cross) atau cross-compiler native saja agar tidak repot. Setidaknya sudah pernah mencoba. Mungkin lain kali saya akan mencoba mengkompilasi Glibc yang saat ini tidak saya butuhkan.

#### Notes

- **Luangkan waktu untuk membaca dokumentasi** sebelum mengikuti tutorial, misalnya untuk memahami target `make` yang tersedia atau flag-flag yang diterima ketika melakukan konfigurasi build (misalnya dengan `./configure`)
- Untuk meminimalkan kesalahan karena typo, gunakan environment variable, misalnya untuk `PREFIX`, `TARGET`, dan `SYSROOT`
- Kompilasi `gcc` sebaiknya dilakukan dengan `make all-gcc` saja karena target default sepertinya punya efek samping, misalnya membutuhkan Glibc yang tidak kita punya jika melakukan cross-compiling
- Proses build `gcc` bisa dilakukan dalam waktu yang lebih sedikit, dengan meringkas tahapan mendownload dan build dependensi, yaitu dengan bantuan script [contrib/download_prerequisites](https://gcc.gnu.org/wiki/InstallingGCC)
Created: 2025-08-12 15:54:49, Updated: 2025-08-14 13:09:43, ID: b62da227-754a-4a51-82c7-520e5d0e76a6