Static linking crt

so on rust, you can static link crt, this is useful especially for windows since there’s a requirement for c dlls that you have to install manually. this isn’t good for my game

with that said, i also wanna keep libc dynamically linked on linux so no issues arise.

i’m also intending on keeping full cross compilation support

i understand that this post is probably a question from the fact that it’s in the Help category and describes the edges of a problem statement.

however, there isn’t actually a question here, which might make it more difficult for people to help you. for example, are you asking whether it’s possible to statically link libc (is this the same as crt for you?) in a Zig project? or are you asking for pointers on how to go about setting your build.zig to take advantage of this possibility? have you gotten stuck somewhere along the process?

I don’t know what the question is either, but I’ll go ahead and say that static linking the crt on Windows doesn’t work at the moment, because Zig uses MinGW, which only supports dynamic linking. Furthermore, Windows, overall, only supports dynamic linking, because syscalls aren’t stable.
Static linking on Linux is feasible and highly recommended.
So it looks like you got it backwards.

I have some great news for you :slight_smile:

  • Rust standard library requires C runtime library (crt) to operate.
    Zig standard library calls the operating system directly without using the C runtime library for both linux and windows. But if you have C dependencies you need to link to the C runtime.
  • Starting with Windows 10 the universal C runtime is shipped with Windows and is updated by the system using windows update. And zig uses the universal C runtime :slight_smile:
1 Like

my understanding is that zig can link against MSVC libc! unfortunately i forgot whether there was some obstruction for cross-compiling to link against MSVC that required manual intervention or if there are licensing issues preventing Zig from linking against it out of the box…

but is it set to static

[target.x86_64-pc-windows-msvc]
rustflags = ["-C", "target-feature=+crt-static"]

see: How to link against static msvc libc on Windows? - #2 by dimdin

2 Likes

The question seems to already be answered, but I’m not as familiar with Windows so I went exploring when I saw the initial question.

When I ran

zig build-exe -target x86_64-windows-msvc -lc --show-builtin

I got

// ...
pub const link_mode = std.builtin.LinkMode.static;
// ...

Looking in the compiler source code at src/link/Coff.zig I found:

// ...
const lib_str = switch (comp.config.link_mode) {
    .dynamic => "",
    .static => "lib",
};
// ...
try argv.append(try allocPrint(arena, "{s}vcruntime{s}.lib", .{ lib_str, d_str }));
try argv.append(try allocPrint(arena, "{s}ucrt{s}.lib", .{ lib_str, d_str }));
// ...

So it seems that when link_mode is .static (which is the default for the generic windows-msvc target), Zig is statically linking the crt libraries (as described here).

If you are still concerned, you can take a look at dumpbin.exe /imports your_exe.exe to check the resulting executable. My experiments show the same results for a printf hello world program as when compiling with cl.exe /MT except Zig also links against ntdll.dll

2 Likes

AFAIK this is not recommended anymore since around Windows 10 and the ‘Universal C Runtime’ (UCRT), e.g. the UCRT DLL installation is now managed centrally by the Windows Update Process, and not expected to be done by an application (unless you need to support older Windows versions than 10)

I still prefer linking the CRT statically though, it’s just less hassle all around.

I just tried it:

main.zig

pub fn main() void {
    _ = std.c.printf("fgsdhjfd\n");
}

const std = @import("std");

build.zig

const std = @import("std");

pub fn build(b: *std.Build) void {
    const target = b.standardTargetOptions(.{});
    const exe_mod = b.createModule(.{
        .root_source_file = b.path("src/main.zig"),
        .target = target,
        .link_libc = true,
        .strip = true,
        .optimize = .ReleaseFast,
    });

    const exe = b.addExecutable(.{
        .name = "fsdfs",
        .root_module = exe_mod,
        .link_libc = true,
        .strip = true,
        .optimize = .ReleaseFast,
        .linkage = .static,
    });
    b.installArtifact(exe);
}

Output:

Edit: Ops, sorry. I forgot to set the target to msvc (the default is mingw). With msvc, indeed, it does statically link.

is that the master branch? i only use the bleeding edge version

are these gnu dlls installed on windows by default or do you need to run the visual c runtime installer?

It’s on “Bleeding edge” :grin: (trunk in less dramatic terminology).
These dlls I showed are the normal vcredist. Mingw will link with them.

No gnu anything is installed on Windows by default.

But as far as I understand, binaries built with _-windows-gnu build and statically link the mingw “gnu” part, and just link against the universal crt. So there is no need to install anything to run the produced binaries (at least since Windows 10).

1 Like