-
Notifications
You must be signed in to change notification settings - Fork 13k
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Allow export of Rust symbols from a C shared object via a staticlib #73295
Comments
FYI @japaric @alexcrichton @retep998 in case you have a suggestion as the linkage experts. Bottom line:
|
If you want a shared library to be usable as a Rust crate it has to be built using the The supported method is to export a function using the normal stable C ABI and then link against it as you would any other normal C ABI function. Using Rust types in that function signature will work if and only if you ensure that both sides are Rust where the Rust types comes from the exact same crate (as the Rust ABI is unstable, even a simple rebuild can change things) |
Controlling the visibility of a symbol is defintely something that's desired in Rust, but AFAIK it's just something no one's had enough time to sit down and design/RFC/etc. You might be able to play around on nightly with the The real problem here is that third bullet point, which is pretty nontrivial. The only thing I'd add is that in general it would only work for very simple datatypes. Even if the symbols were all hooked up you'd still have two Rust standard libraries linked in, and you can't, even with the same matching versions and all, pass datatypes between those two Rust standard libraries. There's some discussion of this on #63338 and related issues. Theoretically to get this working you'll need to make sure all the Rust dependencies are shared. There's Rust crates (including libstd) used by the Rust code between libbase.so and libservices.so. All those shared Rust crates need to be included exactly once and exported from libbase.so which all of libservices.so would then use. That's just how this could work linkage-wise, though, I'm not really sure how this could be set up build-wise beyond hand-crafting all the rustc invocations with various flags. |
Thank you very much for the comments, and yes I feared this would be difficult around that third bullet. In our build system (which is not cargo-based) we could in theory ensure there's exactly one copy of each crate — in the example given this would mean ensuring that
Even if worked through the first three points, the binary size growth from the final point feels like it would be more than just duplicating our own Rust code into lots of different Thanks again for the comments! I'm happy to leave this issue open or close it, whichever's preferred. |
For Gecko, we'd need the opposite: Without changing the Rust code, from the top level of Rust compilation, making FFI symbols visible to the static link step that combines Rust and C++ into libxul without leaving the FFI symbols visible to outside libxul. Should this issue be expected to result in that case getting addressed, or should I open a new issue for that? |
@hsivonen I think you want the same as @RReverser here: #54135 (comment). (I'm not the best person to answer whether that should be in a new issue, I just wanted to make sure it was noted that you're not the only one to request that.) |
Thanks. I filed a new issue. |
I believe both are essentially the same issue (a way to untangle linker visibility from |
For people looking for a way to export symbols from a
This works with |
For Chrome's purposes, we will be linking rlibs (in the future, --emit=obj or similar via #73632) together into a .so with clang. Thus no use of |
Also related to controlling symbol visibility: #33221 and #37530 We're now at a point in Chromium where we have the opposite problem too, which is covered by the above two issues, that all Rust symbols are exported when we don't want them to be (no_mangle or not). We really need some symbol visibility control for shared libs that is independent of pub and no_mangle. |
Are you using the staticlib, dylib or cdylib crate type? Cdylib should only export |
Neither, we are building rlibs and linking them all with C++ in the final link step to produce an exe. However on Mac, the final link step produces a Framework DLL, which then has too many symbols (all the Rust symbols are exported). Aside: We also have a component build that splits the project into DLLs, and there's potentially Rust rlibs linked into each of those, which presents many other problems (using Rust across cdylibs (essentially, but linked externally to Rust) to be addressed in the future separately. |
Would a version script solve the issue for frameworks? That is what rustc uses to limit exported symbols. Using symbol visibility hidden would break linking rlibs into rust dylibs, so we can't do that for the standard library. For the component builds you can use |
We can give this a try. For what it's worth, I think we would prefer to never link rust dylibs since we'll have small bits of Rust appearing amongst C++ all over the codebase for some time, and we'd be fine to not export anything from the stdlib as a result. That said we don't care about binary size in this build mode, it's for developers.
We are already building our stdlib (with panic=abort). It's not uncommon for us to have some of our C++ code end up in multiple DLLs in our component build, and I would have expected Rust to be similar. Requiring every Rust library to be built as a dylib in this mode is an option I hadn't really considered as a result, since it's so dissimilar. But I don't see anything particularly wrong with that idea off the top of my head. Thanks, I'll try things and come back. |
A version script can work for Linux, an export list can work for Mac, but a DEF file does not work for Windows as it can only add exported symbols. It is possible that a much more complicated linking setup may work that generates EXP files for linking against, but fundamentally this is kinda broken, and we should not need to maintain a list of exported symbols on each platform to make this work. After more investigation, there are different behaviours seen on Linux/Mac and Windows. For Windows, we are only seeing For Mac/Linux we are seeing all Rust symbols get exported. I don't know why Windows does not do this. I think that is #37530? |
This issue continues to be a problem for us in Chromium: https://bugs.chromium.org/p/chromium/issues/detail?id=1471542 Here the issue is another DLL that links some part of Chromium's C++ code, and now also links in some Rust code. The C++ code is compiled with On MSVC-Windows, the platform default is hidden, so symbols aren't exported there. We would really like:
cc: @bridiver |
rust-lang/compiler-team#656 should address the first bullet here, but has yet to be implemented. |
#131519 has fixed default-hidden-visibility so that's great! Applying it in Chromium we notice that we're still getting unexpected public symbols from Rust, but they are coming from symbols that are avoiding mangling (for reasons other than wanting to be public). But currently |
I strongly agree that we should have a way to separate the notions of controlling mangling and controlling visibility. |
clang offers the ability to mark a symbol as exported or non-exported from a dynamic shared object (DSO) using
__attribute__((visibility ("default")))
and__attribute__ ((visibility ("hidden")))
. This issue requests the same in Rust, though it may be a bit more complex in the Rust case.Another way of looking at it is to cleave the dual purpose of
#[no_mangle]
where it both sets visibility and alters the exported name.The rest of this issue explains why this would be useful.
I am working in a pre-existing complex C++ codebase. It produces multiple DSOs, let’s say:
libbase.so
libservices.so
We want to build Rust code into each of those DSOs. Internally, the C++ and Rust code is intermixed freely (for example, the C++ JSON APIs within
libbase.so
call into a serde-based parser, which calls back into C++ to instantiate objects, etc.) There is no possibility of splitting these DSOs into separate DSOs for Rust and C++ code.We build Rust code into these DSOs in the approved way, which is to aggregate a bunch of Rust libraries (rlibs) into a separate Rust
staticlib
for each of the DSOs. (For example,libbase_rust_deps.a
andlibservices_rust_deps.a
). The final C++ linker links exactly one of these staticlibs together with the C++ .a and .o in the final construction of the DSO. That should be fine, since the whole point of astaticlib
Rust target is to contain the Rust code and all its dependencies.libservices.so
depends onlibbase.so
. In C++,libservices.so
uses symbols exposed fromlibbase.so
. For example, there may be aCPP::Log
function used by C++ code within bothlibbase.so
andlibservices.so
. That’ll be exported fromlibbase.so
using an__attribute__((visibility ("default")))
annotation in the source code forCPP::Log
.So far so good.
But now we want to put a Rust wrapper around that
CPP::Log
function, to make, say,rs_log
. Ideally, there would be just a single implementation ofrs_log
inlibbase.so
. We want to be able to call that Rust wrapper from within the Rust parts oflibbase.so
orlibservices.so
.This seems to be impossible.
Our options are:
dylib
using rustc, which exposes all public Rust APIs from the DSO. We can’t do that for the reasons cited above; we need to build astaticlib
as the Rust code is just a small component of existing DSOs. Rustc can’t be in control of the final linking.rs_log
as#[no_mangle]
which successfully exports it fromlibbase.so
, but using a C ABI which prevents Rust discovering and using this symbol even if downstream crates are built using--extern rust_base=./libbase.so
(this latter case gives
)
Solution?
It’s a problem that
#[no_mangle]
has twin effects: specify a particular name for C linkage purposes, and mark the symbol for DSO export. We want to do the latter but not the former. #54135 (comment) wants to do the former but not the latter. Can we separate those?For our needs, ideally we’d mark
rs_log
with something like an#[dso_export]
annotation. This would:__attribute__((visibility ("default")))
, such that as it moves through the rlib, then staticlib, then the DSO, it is successfully exported from the DSO.#[no_mangle]
it wouldn’t change the symbol name and it would remain callable by Rust.staticlib
, determine if there are any such exported Rust APIs. If so, include whatever information is required by Rust to considerlibbase.so
to be a valid crate equivalent to adylib
. (I don’t know how Rust metadata is structured; whether this is just an extra few exported symbols or if there are entire linker sections within adylib
that would somehow need to be replicated. I strongly suspect the latter so I don’t even know if this is possible without changing the final C++ linker command).Related work
#54135 requests something similar, but using
#[used]
on static variables - so I think it’s not quite the same thing.dtolnay/cxx#219 proposes the workaround which we might need to do meanwhile (but isn’t ideal because, in the above example,
RS::Log
would be duplicated between two DSOs)The text was updated successfully, but these errors were encountered: