-
Notifications
You must be signed in to change notification settings - Fork 15.6k
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
ABI may depend on NDEBUG #9947
Comments
One possible fix would look like this: ~InternalMetadata() {
#if defined(NDEBUG) || defined(_MSC_VER)
if (HasMessageOwnedArenaTag()) {
delete reinterpret_cast<Arena*>(ptr_ - kMessageOwnedArenaTagMask);
}
#else
ActuallyDestroy();
#endif
}
private:
// Non-inlined destructor for debug builds,
// always defined for ABI reasons.
void ActuallyDestroy(); |
@pitrou, thanks for the issue report and possible fix. It looks like the addition of the Your proposed fix still violates the ODR, but I believe that in practice it may avoid the linker issue you reported and may appear to work. Generally speaking, we cannot guarantee that our C++ code will continue to work properly when the flags used to compile and install the library differ from the flags used to compile/link with it. However, we may want to special case |
For the record, my current workaround is to build both release and debug and in my app's CMakeLists.txt I have to do: if (NOT WIN32) ## this is a workaround for #9947 |
Supporting mixed compilation modes is not something we intend to do, as avoiding ODR violations under this kind of a context comes with significant performance costs that we are unwilling to pay. |
As I'm used to compiling under windows where you almost always have to compile things twice (1 debug, 1 release), I'm not shocked. The only nice thing that would help is then if FindProtobuf would find both release and debug libs on its own. |
I understand the desire not to support situations which are stricto sensu undefined behaviour. It also makes protobuf more difficult to use in situations where pre-built release binaries are available (e.g. Debian/Ubuntu packages), but no debug ones. |
@fowles am I missing something? |
Adding my voice here: I think it is a deviation from common practice to change the ABI based on whether [*] I am sure it is also common practice to install different libraries for debug and release. Both practices can be "common" at the same time. |
Where can we see the performance metrics this decision seems to be based on? |
As @cwharris wonders, I don't think fixing this without ODR violation costs such significant performance costs either. Modern processors with sophisticated memory hierarchy and branch predictors should be able to handle the branch depending on the debug switch in a few cycles, which is much cheaper than the core deallocation logic. Is there really no chance such costs can be tolerated? |
@cwharris the performance measurements we are using are only available inside google. They are essentially websearch performance tests. Even a half a percent of websearches overall performance foot print turns out to be a very large number. |
Sounds like the code is important for performance inside Google, but a hassle outside Google. Consider the following alternatives:
#if (defined(NDEBUG) || defined(_MSC_VER)) && defined(PROTOBUF_DANGEROUS_GOOGLE_INTERNAL_OPTIMIZATIONS)
~InternalMetadata() {
if (HasMessageOwnedArenaTag()) {
delete reinterpret_cast<Arena*>(ptr_ - kMessageOwnedArenaTagMask);
}
}
#else
~InternalMetadata();
#endif The documentation of this macro can explain why it is a bad idea to set it outside Google, and you can have it always set in the Google builds.
#if defined(_MSC_VER)
~InternalMetadata() {
if (HasMessageOwnedArenaTag()) {
delete reinterpret_cast<Arena*>(ptr_ - kMessageOwnedArenaTagMask);
}
}
#else
~InternalMetadata();
#endif
} |
@coryan that is only one of quite a few things that have ABI dependencies |
It's the only problem that seems to be breaking third-party downstream projects, though, so probably worth tackling? |
If someone has the cycles to create a PR, I can review it |
Re-opening since we would be willing to review a PR, but we cannot commit to fixing this in the near term (we are too busy with other commitments). |
Fix #9947: make the ABI identical between debug and non-debug builds
…n-debug builds (cherry picked from commit 0574167)
In changeset ab4585a, the
InternalMetadata
destructor changed to the following:This causes ABI problems between debug and non-debug in one specific case: when the
-fvisibility-inlines-hidden
compiler option is being used. According to the GCC wiki, that option has the following effect:However, when
libprotobuf.so
was compiled in non-debug mode with-fvisibility-inlines-hidden
enabled, any debug build of a third-party application will fail looking up~InternalMetadata()
What version of protobuf and what language are you using?
Version: v3.20.1
Language: C++
What runtime / compiler are you using (e.g., python version or gcc version)
gcc 10.3.0
The text was updated successfully, but these errors were encountered: