2021-06-08 15:58:04 +03:00
[![builds.sr.ht status ](https://builds.sr.ht/~motiejus/bazel-zig-cc.svg )](https://builds.sr.ht/~motiejus/bazel-zig-cc)
2021-06-08 15:56:58 +03:00
2021-08-06 16:35:48 +03:00
# Bazel zig cc toolchain
2021-04-10 01:05:01 +03:00
2021-08-11 09:36:20 +03:00
This is a C/C++ toolchain that can (cross-)compile C/C++ programs. It contains
2021-12-22 10:46:49 +02:00
clang-13, musl, glibc (versions 2-2.34, selectable), all in a ~40MB package.
2021-08-11 09:36:20 +03:00
Read
[here ](https://andrewkelley.me/post/zig-cc-powerful-drop-in-replacement-gcc-clang.html )
about zig-cc; the rest of the README will present how to use this toolchain
from Bazel.
2021-08-06 17:19:54 +03:00
2021-08-06 14:30:25 +03:00
# Usage
2022-01-26 06:15:42 +02:00
Add this to your `WORKSPACE` :
2021-08-06 14:33:30 +03:00
```
2022-03-18 07:52:14 +02:00
BAZEL_ZIG_CC_VERSION = "v0.5.0"
2021-08-06 14:33:30 +03:00
http_archive(
name = "bazel-zig-cc",
2022-03-18 07:52:14 +02:00
sha256 = "7d7e2bcfe15fce3a6d46ab1ed6f06e36a4729ff5e75916023a09b425ef6f32dd",
2021-08-06 14:33:30 +03:00
strip_prefix = "bazel-zig-cc-{}".format(BAZEL_ZIG_CC_VERSION),
urls = ["https://git.sr.ht/~motiejus/bazel-zig-cc/archive/{}.tar.gz".format(BAZEL_ZIG_CC_VERSION)],
)
load("@bazel-zig-cc//toolchain:defs.bzl", zig_register_toolchains = "register_toolchains")
2022-02-08 14:09:30 +02:00
zig_register_toolchains()
2022-03-18 06:44:22 +02:00
# Or, if you are using this in production, you probably want more control:
zig_register_toolchains(
version = "< ... > ",
url_formats = [
"https://example.internal/zig/zig-{host_platform}-{version}.tar.xz",
],
host_platform_sha256 = { ... },
)
2021-08-11 09:36:20 +03:00
```
2022-01-26 06:15:42 +02:00
And this to `.bazelrc` :
```
build --incompatible_enable_cc_toolchain_resolution
2022-04-06 22:46:15 +03:00
build --extra_toolchains @zig_sdk//toolchain:linux_amd64_gnu .2.19
build --extra_toolchains @zig_sdk//toolchain:linux_arm64_gnu .2.28
build --extra_toolchains @zig_sdk//toolchain:darwin_amd64
build --extra_toolchains @zig_sdk//toolchain:darwin_arm64
2022-01-26 06:15:42 +02:00
```
2022-02-08 14:09:30 +02:00
The snippets above will download the zig toolchain and register it for the
2021-08-11 09:36:20 +03:00
following platforms:
2022-02-08 14:09:30 +02:00
- `x86_64-linux-gnu.2.19` for `["@platforms//os:linux", "@platforms//cpu:x86_64"]` .
2022-02-08 14:39:29 +02:00
- `x86_64-linux-gnu.2.28` for `["@platforms//os:linux", "@platforms//cpu:aarch64"]` .
2021-08-11 09:36:20 +03:00
- `x86_64-macos-gnu` for `["@platforms//os:macos", "@platforms//cpu:x86_64"]` .
2022-02-08 14:39:29 +02:00
- `aarch64-macos-gnu` for `["@platforms//os:macos", "@platforms//cpu:aarch64"]` .
2021-08-11 09:36:20 +03:00
Note that both Go and Bazel naming schemes are accepted. For convenience with
Go, the following Go-style toolchain aliases are created:
2022-02-08 14:39:29 +02:00
|Bazel (zig) name | Go name |
|---------------- | -------- |
|`x86_64` | `amd64` |
|`aarch64` | `arm64` |
|`macos` | `darwin` |
2021-08-11 09:36:20 +03:00
2021-12-22 10:52:15 +02:00
For example, the toolchain `linux_amd64_gnu.2.28` is aliased to
2021-08-11 09:40:16 +03:00
`x86_64-linux-gnu.2.28` . To find out which toolchains can be registered or
used, run:
2021-08-11 09:36:20 +03:00
```
2022-02-05 11:48:54 +02:00
$ bazel query @zig_sdk// ... | grep _toolchain$
2021-08-06 14:33:30 +03:00
```
2022-02-08 14:41:35 +02:00
## Specifying non-default toolchains
2022-02-02 15:11:10 +02:00
2022-02-08 14:41:35 +02:00
You may explicitly request Bazel to use a specific toolchain, even though a
different one is registered using `--extra_toolchains <toolchain>` in
`.bazelrc` . For example, if you wish to compile a specific binary (or run
tests) on linux/amd64/musl, you may specify:
2022-02-02 15:11:10 +02:00
```
2022-04-06 22:46:15 +03:00
--extra_toolchains @zig_sdk//toolchain:linux_amd64_musl
2022-02-02 15:11:10 +02:00
```
2022-02-05 11:16:16 +02:00
## UBSAN and "SIGILL: Illegal Instruction"
2021-12-09 10:40:04 +02:00
2022-02-05 11:16:16 +02:00
`zig cc` differs from "mainstream" compilers by [enabling UBSAN by
default][ubsan1]. Which means your program may compile successfully and crash
with:
2021-06-08 08:00:53 +03:00
2022-02-05 11:16:16 +02:00
```
SIGILL: illegal instruction
```
2021-12-22 10:46:49 +02:00
2022-02-05 11:16:16 +02:00
This is by design: it encourages program authors to fix the undefined behavior.
There are [many ways][ubsan2] to find the undefined behavior.
2021-12-22 10:46:49 +02:00
2022-02-05 11:16:16 +02:00
# Known Issues In bazel-zig-cc
2021-12-22 10:46:49 +02:00
2022-02-05 11:16:16 +02:00
These are the things you may stumble into when using bazel-zig-cc. I am
unlikely to implement them any time soon, but patches implementing those will
be accepted. See [Questions & Contributions ](#questions-amp-contributions ) on
how to contribute.
2021-12-22 10:46:49 +02:00
## Zig cache
2021-12-22 10:52:15 +02:00
Currently zig cache is in `$HOME` , so `bazel clean --expunge` does not clear
the zig cache. Zig's cache should be stored somewhere in the project's path.
2021-12-22 10:46:49 +02:00
2022-02-05 11:16:16 +02:00
## OSX: sysroot
For non-trivial programs (and for all darwin/arm64 cgo programs) MacOS SDK may
be necessary. Read [Jakub's comment][sysroot] about it. Support for OSX sysroot
is currently not implemented.
## OSX: different OS targets (Catalina -- Monterey)
[Zig 0.9.0 ](https://ziglang.org/download/0.9.0/release-notes.html#macOS ) may
target macos.10 (Catalina), macos.11 (Big Sur) or macos.12 (Monterey). It
currently targets the lowest version, without ability to change it.
2021-12-22 10:52:15 +02:00
# Known Issues In Upstream
2021-12-22 10:46:49 +02:00
2021-12-22 10:52:15 +02:00
This section lists issues that I've stumbled into when using `zig cc` , and is
outside of bazel-zig-cc's control.
2021-06-08 08:00:53 +03:00
2021-12-09 10:42:45 +02:00
## using glibc 2.27 or older
2021-10-20 07:31:40 +03:00
2021-08-11 13:33:32 +03:00
**Severity: Low**
2021-08-11 09:36:20 +03:00
Task: [ziglang/zig #9485 glibc 2.27 or older: fcntl64 not found, but zig's glibc headers refer it ](https://github.com/ziglang/zig/issues/9485 )
Background: when glibc 2.27 or older is selected, it may miss `fcntl64` . A
workaround is applied for `x86_64` , but not for aarch64. The same workaround
2021-08-11 09:43:13 +03:00
may apply to aarch64, but the author didn't find a need to test it (yet).
2021-08-11 09:36:20 +03:00
2021-12-22 10:52:15 +02:00
# Closed Upstream Issues
2021-06-08 15:56:58 +03:00
2022-01-17 17:17:14 +02:00
- [ziglang/zig #10386 zig cc regression in 0.9.0 ](https://github.com/ziglang/zig/issues/10386 )(CLOSED, thanks Xavier)
- [ziglang/zig #10312 macho: fail if requested -framework is not found ](https://github.com/ziglang/zig/pull/10312 ) (CLOSED, thanks kubkon)
- [ziglang/zig #10299 [darwin aarch64 cgo] regression](https://github.com/ziglang/zig/issues/10299) (CLOSED, thanks kubkon)
- [ziglang/zig #10297 [darwin x86_64 cgo] regression](https://github.com/ziglang/zig/issues/10297) (CLOSED, thanks kubkon)
2021-12-07 20:26:59 +02:00
- [ziglang/zig #9431 FileNotFound when compiling macos ](https://github.com/ziglang/zig/issues/9431 ) (CLOSED, thanks andrewrk)
2022-01-17 17:17:14 +02:00
- [ziglang/zig #9139 zig c++ hanging when compiling in parallel ](https://github.com/ziglang/zig/issues/9139 ) (CLOSED, thanks andrewrk)
2021-12-09 10:40:04 +02:00
- [ziglang/zig #9050 golang linker segfault ](https://github.com/ziglang/zig/issues/9050 ) (CLOSED, thanks kubkon)
2022-01-17 17:17:14 +02:00
- [ziglang/zig #7917 [meta] better c/c++ toolchain compatibility](https://github.com/ziglang/zig/issues/7917) (CLOSED, thanks andrewrk)
- [ziglang/zig #7915 ar-compatible command for zig cc ](https://github.com/ziglang/zig/issues/7915 ) (CLOSED, thanks andrewrk)
- [ziglang/zig #7667 misplaced relocated glibc stubs (pthread_sigmask) ](https://github.com/ziglang/zig/issues/7667 ) (CLOSED, thanks mjonaitis and andrewrk)
- [rules/go #2894 Per-arch_target linker flags ](https://github.com/bazelbuild/rules_go/issues/2894 ) (CLOSED, thanks mjonaitis)
2021-12-09 10:40:04 +02:00
- [golang/go #46644 cmd/link: with CC=zig: SIGSERV when cross-compiling to darwin/amd64 ](https://github.com/golang/go/issues/46644 ) (CLOSED, thanks kubkon)
2021-08-11 09:40:16 +03:00
# Testing
2022-02-05 11:16:16 +02:00
## build & run linux cgo + glibc
2021-08-11 09:40:16 +03:00
```
2022-04-06 22:16:53 +03:00
$ bazel build --platforms @zig_sdk//platform:linux_amd64 //test/go:go
2022-02-03 14:58:05 +02:00
$ file bazel-out/k8-opt-ST-d17813c235ce/bin/test/go/go_/go
bazel-out/k8-opt-ST-d17813c235ce/bin/test/go/go_/go: ELF 64-bit LSB executable, x86-64, version 1 (SYSV), dynamically linked, interpreter /lib64/ld-linux-x86-64.so.2, for GNU/Linux 2.0.0, Go BuildID=redacted, with debug_info, not stripped
2022-02-05 11:16:16 +02:00
$ bazel-out/k8-opt-ST-d17813c235ce/bin/test/go/go_/go
hello, world
2021-08-11 09:40:16 +03:00
```
2022-02-05 11:16:16 +02:00
## test linux cgo + musl on arm64 (under qemu-aarch64)
2021-08-11 09:40:16 +03:00
```
2022-02-05 11:16:16 +02:00
$ bazel test \
2022-02-08 14:42:32 +02:00
--config=qemu-aarch64 \
2022-04-06 22:16:53 +03:00
--platforms @zig_sdk//platform:linux_arm64 \
2022-04-06 22:46:15 +03:00
--extra_toolchains @zig_sdk//toolchain:linux_arm64_musl //test/...
2021-08-11 09:40:16 +03:00
...
2022-02-05 11:16:16 +02:00
INFO: Build completed successfully, 10 total actions
//test/go:go_test PASSED in 0.2s
2021-08-11 09:40:16 +03:00
```
## macos cgo
```
2022-04-06 22:16:53 +03:00
$ bazel build --platforms @zig_sdk//platform:darwin_amd64 //test/go:go
2021-08-11 09:40:16 +03:00
...
2022-02-03 14:58:05 +02:00
$ file bazel-out/k8-opt-ST-d17813c235ce/bin/test/go/go_/go
bazel-out/k8-opt-ST-d17813c235ce/bin/test/go/go_/go: Mach-O 64-bit x86_64 executable, flags:< NOUNDEFS | DYLDLINK | TWOLEVEL | PIE | HAS_TLV_DESCRIPTORS >
2021-08-11 09:40:16 +03:00
```
## Transient docker environment
```
2021-10-20 16:05:06 +03:00
$ docker run -e CC=/usr/bin/false -ti --rm -v $(pwd):/x -w /x debian:bullseye-slim
2022-02-11 07:45:01 +02:00
# apt update && apt install -y direnv git
2021-08-11 09:40:16 +03:00
# . .envrc
```
And run the `bazel build` commands above. Take a look at `.build.yml` and see
how CI does it.
2021-08-11 13:36:18 +03:00
2021-12-13 09:17:18 +02:00
# Questions & Contributions
2021-11-10 09:21:58 +02:00
2021-12-13 09:17:18 +02:00
Project's mailing list is [~motiejus/bazel-zig-cc][mailing-list]. The mailing
list is used for:
2021-11-10 09:21:58 +02:00
2021-12-13 09:17:18 +02:00
- announcements (I am aiming to send an email with every release).
- user discussions.
- raising issues.
- contributions.
I will generally respond to emails about issues. I may even be able to fix
them. However, no promises: you are much more likely (and welcome!) to get it
fixed by submitting a patch.
To contribute, send your patches to the mailing list, as described in
[git-send-email.io][git-send-email] or via [Sourcehut web UI][video].
2021-11-10 09:21:58 +02:00
2022-04-07 12:50:45 +03:00
Development is happening in the [next][next] branch. It may be incompatible
with main, but is expected to be landed "soon".
2021-11-10 09:21:58 +02:00
Copyright is retained by the contributors.
2021-12-09 10:40:04 +02:00
# Thanks
2021-08-11 13:36:18 +03:00
2021-11-10 09:21:58 +02:00
Many thanks to Adam Bouhenguel and his [bazel-zig-cc][ajbouh], the parent of
2021-12-09 10:40:04 +02:00
this repository. Also, the Zig team for making this all possible and handling
the issues promptly.
2021-11-10 09:21:58 +02:00
2021-12-13 09:17:18 +02:00
[mailing-list]: https://lists.sr.ht/~motiejus/bazel-zig-cc
2021-11-10 09:21:58 +02:00
[ajbouh]: https://github.com/ajbouh/bazel-zig-cc/
[git-send-email]: https://git-send-email.io/
[video]: https://spacepub.space/w/no6jnhHeUrt2E5ST168tRL
2021-12-09 10:40:04 +02:00
[sysroot]: https://github.com/ziglang/zig/issues/10299#issuecomment-989153750
2022-01-26 06:15:42 +02:00
[ubsan1]: https://github.com/ziglang/zig/issues/4830#issuecomment-605491606
[ubsan2]: https://github.com/ziglang/zig/issues/5163
2022-04-07 12:50:45 +03:00
[next]: https://git.sr.ht/~motiejus/bazel-zig-cc/tree/next