Best practices for consuming C libraries

Say I wanted to create a library, that itself requires an SSH client. I’m a bit lost how to do that “properly” (Questions at the end).

I think I have these options:

  1. Dynamically link with an installed host package (e.g. libssh)
  2. Statically link with an installed host package
  3. Download and build libssh from build.zig and link statically
  4. Port it to Zig or write it from scratch

(1) is what you would normally do (outside zig world), in order to benefit from updates to libssh. The only painful aspect is that your deliverable has a runtime dependency, that you have to maintain if you’re packaging or you just silently depend on the library and your program fails.

In Zig world, doing that makes it impossible or hard (I never tried) to cross compile, because you don’t have the headers (or they may contain definitions specific to the compilation host or something).

A small inconvenience is also that you had to install development packages for libssh on the compilation host (not an issue for libssh, because it only has a dev package, but it’s just an example).

(2) Is like (1), but you wouldn’t have the runtime dependency and the benefit of the update. You would however use a curated build from the OS packaging and benefit from whatever build settings they used. That of course is of no value if you wanted to cross compile.

(3) Here you would let build.zig download the sources, build them and link them, maybe provide a wrapper to make it look like zig.

That of course is easier said than done, especially if you want to cross compile.

It’s quite a while since I build C projects, but most used something like autoconf where part of the build process consists of analyzing the host operating system and checking for various features. That doesn’t play well with cross compilation, so you would need to run the build in a VM or container if you want to use the project build process.

Because of that, it seems to be better to build via build.zig. But that seems to be a huge effort for complex projects. You had to do all the autoconf/cmake analysis mostly from scratch.

Then there is the problem of possible upstream changes breaking the zig build. So this needs to be maintained.

(4) Obviously the preferred choice, but time.

Neither of these four options look especially appealing to me, if my priority would be to just get access to some library. Since I’m currently thinking about a project that needs quite a lot of such libraries and need (well want really hard) the cross compilation feature, this is a bit scary.

There are some build.zig wrappers for libraries, but not many and off those, they do not appear to be maintained (someone did what needed to be done for a project, and that’s the end of it. And I really wouldn’t want to maintain each wrapper I might write in the future either).

This looks a lot like we would need something like homebrew, ports, debian source packages, or AUR to make using C dependencies feasible for all but the most hard core enthusiasts. This is a bit heartbreaking considering how much work all of these projects spend to get where they are, and yet here is a new beginning.

So here are my questions:

What would currently be canonical way to consume C libraries? Especially in the scenario where a zig library wraps/extends C and is then consumed by other zig projects.

Does Zig world have a strategy in mind how to handle this problem in the future? On one hand, it’s clear that Zig is aiming at replacing C progressively, but until there are zig ports for important functionality and Zig becomes much more popular, this is not helping yet.

Build.zig is a great concept, because it’s more powerful than all alternatives I know and still easy to use. But the real value of existing build tools for C lies in that field of autoconf and also in the work that has been done by distributions and package managers preconfiguring builds for target platforms. No matter how good the build tool is, the amount of work to be done to get access to libraries supporting zig features (that’s basically cross compilation insofar C libraries are concerned) is quite intimidating.

Lastly, I saw various videos and short tutorials explaining how to interface with C. But it would be great if there was a single detailed documentation showing the different options and demo code. Is there something like that (something that shows more than typical or trivial use cases)?

FYI linux headers can be downloaded from ubuntu mirror, I do link to libcurl and cross-compile from macos to linux and it works fine.

Here’s how I do it, but obviously it will require some adjustments for your case. IIRC the important part is to have .pc files in the correct path.

mkdir -p zig-out/usr/{lib,include}
cd zig-out

curl -L http://security.ubuntu.com/ubuntu/pool/main/c/curl/libcurl4_7.81.0-1ubuntu1.18_amd64.deb -o libcurl.deb
curl -L http://security.ubuntu.com/ubuntu/pool/main/c/curl/libcurl4-openssl-dev_7.81.0-1ubuntu1.18_amd64.deb -o libcurl-dev.deb

ar -x libcurl-dev.deb data.tar.zst
tar --use-compress-program=unzstd -xvf data.tar.zst
cp -r usr/include/x86_64-linux-gnu/* usr/include/

ar -x libcurl.deb data.tar.zst
tar --use-compress-program=unzstd -xvf data.tar.zst
cp -r usr/lib/x86_64-linux-gnu/* usr/lib/

cd ..
zig build -Doptimize=ReleaseSafe -Dtarget=x86_64-linux-gnu --search-prefix zig-out/usr
1 Like

You might find this interesting: How to zig a C project?

1 Like

I’m late to the party, but since it’s been bumped:
You question has nothing to do with Zig. Zig is just a language, it doesn’t change your design decisions regarding static or dynamic libraries, it just make it very easy to implement whichever option you think is best. Zig has standard and easy ways for consuming both dynamic and static libraries.
So, the best answer to your question is: do the exact same thing that you would in C.

1 Like

Thanks Luke, that was in deed interesting!

C is much less ambitious than Zig in some fields, such as cross compilation, where Zig even goes so far as to provide libc’s for target platforms in order to support seamless cross compilation. The more you do “the C thing”, the less relevant this advantage becomes, because every Zig project depending on something will break the seamless cross compilation feature (when using that Zig project as dependency).

I think it’s crucial for Zig’s long term success, that its scope can grow (here by adding functionality) without Zig loosing its characteristics not only as a language but also as an ecosystem. If over time popular libraries emerge that contain the kind of dependencies that C libraries have, Zig can hardly claim to be any more cross platform capable than the libraries people typically use.

It’s quite a lot of effort to replace the build system of a C library with build.zig. If you are right, then it would be stupid to even try. Nobody in C world would. But then Zig not only is like C, it also limits itself to be no more than C (in the scope of consuming dependencies).

I just got a reminder on my work side of life of how important existing libraries are. I am currently working in a Kotlin/Java project, the first time in a very long time. I completely forgot how many powerful libraries are available for Java, how easy it is to use them and how good their overall quality is. There are reasons for that, and it’s not only that Java was the default choice in the “enterprise”, because other languages had that too and they don’t have this rich and uniform pool of libraries.

C had a huge head start over Java (as did C++). While there are tons of libraries for C, they are not readily usable for a wide variety of reasons.

At the beginning, when Java was “just a language”, it had a rather rich standard library, compared to what ANSI C was and even compared to what the Zig std library has to offer. Because of that, Java could evolve based on that strong foundation and also because of the uniform VM. In C, every extension of scope requires a sustained maintenance effort. Use UNIX, cope with BSD, SYSV, Linux, Posix, … Use Windows cope with that beast. Use A that depends on B which depends on C, better have a dependency analyzer to help you. That problem is part of the very nature of C being just a language with an interface at the object file level.

I find Zig’s approach to stick to the C abstraction on one hand, but create a cross platform abstraction with its standard library and compile time fascinating, because I believe that this can actually work and provide a rich ecosystem such as that of Java with all the benefits of being a thin layer of abstraction (or none at all) on the actual operating system and hardware.

But that can only work, if such libraries can grow preserving Zigs abstraction of “a platform”. If you inherit C’s view, there is no chance to ever get that kind of growth.