Skip to content
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

[libc++] regression: new/delete symbol overrides broken on macOS #123224

Open
tycho opened this issue Jan 16, 2025 · 7 comments
Open

[libc++] regression: new/delete symbol overrides broken on macOS #123224

tycho opened this issue Jan 16, 2025 · 7 comments
Assignees
Labels
libc++ libc++ C++ Standard Library. Not GNU libstdc++. Not libc++abi.

Comments

@tycho
Copy link
Member

tycho commented Jan 16, 2025

This is referring to commit 8418955.

In a shared library which statically links libc++ (ANGLE's libEGL in this case), the symbols for new and new[] are, as of the above commit, suddenly exposed as global, but the corresponding delete and delete[] operators are not.

Before the above commit:

$ nm -g -C --defined-only contrib/angle/angle/out/macOS-Debug-arm64/libEGL.dylib | grep -e new -e delete

After:

$ nm -g -C --defined-only contrib/angle/angle/out/macOS-Debug-arm64/libEGL.dylib | grep -e new -e delete
00000000001e9460 T operator new[](unsigned long)
00000000001e9740 T operator new[](unsigned long, std::align_val_t)
00000000001e9328 T operator new(unsigned long)
00000000001e95e0 T operator new(unsigned long, std::align_val_t)

This causes applications like mine with custom allocators (mimalloc in this case) to provide the implementations for the operator new symbols, but not the delete symbols, which inevitably causes a crash within the shared library when it tries to free memory.

@llvmbot llvmbot added the libc++ libc++ C++ Standard Library. Not GNU libstdc++. Not libc++abi. label Jan 16, 2025
@ShabbyX
Copy link

ShabbyX commented Jan 16, 2025

Attn @nico

@tycho
Copy link
Member Author

tycho commented Jan 16, 2025

I think it's this line specifically that suddenly made them visible globally. Strange that this change was only done to new/new[]...

@ldionne
Copy link
Member

ldionne commented Jan 16, 2025

@petrhosek Could you have a look?

@petrhosek
Copy link
Member

@petrhosek Could you have a look?

Yes, I was already looking at the macOS implementation in the context of #122983 so I can look into this.

@tycho Do you have instructions for building libEGL.dylib so I can reproduce this locally?

@tycho
Copy link
Member Author

tycho commented Jan 16, 2025

ANGLE's build directions:

https://chromium.googlesource.com/angle/angle/+/main/doc/DevSetup.md

The TL;DR is something like:

  • clone ANGLE
  • clone depot_tools
  • add depot_tools to PATH
  • enter ANGLE source tree, gclient sync
  • mkdir -p out/debug; cd out/debug
  • make an args.gn file
  • gn gen .
  • ninja libEGL libGLESv2

and you should get a libEGL + libGLESv2. Their version of libc++ comes from the third_party/libc++/src directory, which gets autorolled from Chromium periodically.

This is the args.gn I am using for arm64 debug builds of ANGLE at the moment, and what I used for the repro in the OP:

target_cpu = "arm64"
angle_enable_gl = true
angle_enable_gl_desktop_backend = true
angle_enable_d3d11 = false
angle_enable_d3d9 = false
angle_enable_glsl = true
angle_enable_metal = true
angle_enable_null = false
angle_enable_swiftshader = false
angle_enable_trace = false
angle_enable_annotator_run_time_checks = true
angle_enable_vulkan_validation_layers = true
angle_enable_vulkan_gpu_trace_events = true
angle_enable_vulkan = true
angle_enable_wgpu = false
angle_expose_non_conformant_extensions_and_versions = true
angle_enable_custom_vulkan_cmd_buffers = false
angle_enable_overlay = true
angle_enable_share_context_lock = false
angle_enable_global_mutex_recursion = true
angle_has_astc_encoder = false
is_component_build = false
enable_stripping = false
dcheck_always_on = true
angle_use_wayland = false
is_debug = true
use_custom_libcxx = true
is_clang = true
use_thin_lto = false
treat_warnings_as_errors = false
angle_use_custom_libvulkan = true
is_asan = false
is_tsan = false

# macOS-specific configuration
mac_deployment_target = "11.0"
angle_shared_libvulkan = true
enable_dsyms = true

# Support ccache wrapper for faster repeated builds
cc_wrapper = "/opt/homebrew/bin/ccache"

@tycho
Copy link
Member Author

tycho commented Jan 16, 2025

This allows me to build and run with libEGL+libGLESv2 without crashing:

libc++.patch

It basically makes all the delete operators visible (and overridable) too. But naturally it doesn't have the extra checks that the new operator ones have to notice partially overridden symbols.

But that did lead me to notice that not all of the new operators are exposed. libc++ is making these visible:

operator new[](unsigned long)
operator new[](unsigned long, std::align_val_t)
operator new(unsigned long)
operator new(unsigned long, std::align_val_t)

But it misses the nothrow versions, for some reason (the following are covered by mimalloc, but libc++ doesn't expose them?):

operator new[](unsigned long, std::nothrow_t const&)
operator new[](unsigned long, std::align_val_t, std::nothrow_t const&)
operator new(unsigned long, std::nothrow_t const&)
operator new(unsigned long, std::align_val_t, std::nothrow_t const&)

petrhosek added a commit to petrhosek/llvm-project that referenced this issue Jan 21, 2025
This roughly matches the ELF version and addresses the issue llvm#123224.
@petrhosek
Copy link
Member

The issue isn't .globl which is always emitted by Clang for operator new and operator delete, even in the Chromium build. Rather, Chromium build uses the -fvisibility-global-new-delete=force-hidden option which is lowered to .private_extern directive which allows the linker to internalize the symbol and it is why those symbols end up hidden in Chromium binaries.

We don't want to use _LIBCPP_OVERRIDABLE_FUNCTION for operator delete since we don't need the overriding detection for those symbols, rather we want to ensure that operator new is marked as .private_extern when -fvisibility-global-new-delete=force-hidden is set.

I have reworked the implementation to avoid the use of assembly altogether in #122983 which should address this issue once landed.

petrhosek added a commit that referenced this issue Jan 28, 2025
Reverts #120805

This change while desirable has two issues we discovered:

- It is incompatible with `-funique-internal-linkage-names`, see
#120805 (comment)
- It is incompatible with `-fvisibility-global-new-delete=force-hidden`,
see
#123224 (comment)

We were hoping to address both of these issues with
#122983, but that change has
other issues we haven't yet managed to resolve. For now, we have decided
to revert the change to avoid shipping a broken feature in LLVM 20, and
we plan to follow up with a new approach post branch.
github-actions bot pushed a commit to arm/arm-toolchain that referenced this issue Jan 28, 2025
…n" (#124431)

Reverts llvm/llvm-project#120805

This change while desirable has two issues we discovered:

- It is incompatible with `-funique-internal-linkage-names`, see
llvm/llvm-project#120805 (comment)
- It is incompatible with `-fvisibility-global-new-delete=force-hidden`,
see
llvm/llvm-project#123224 (comment)

We were hoping to address both of these issues with
llvm/llvm-project#122983, but that change has
other issues we haven't yet managed to resolve. For now, we have decided
to revert the change to avoid shipping a broken feature in LLVM 20, and
we plan to follow up with a new approach post branch.
oneseer pushed a commit to oneseer/libcxxabi that referenced this issue Mar 26, 2025
Reverts llvm/llvm-project#120805

This change while desirable has two issues we discovered:

- It is incompatible with `-funique-internal-linkage-names`, see
llvm/llvm-project#120805 (comment)
- It is incompatible with `-fvisibility-global-new-delete=force-hidden`,
see
llvm/llvm-project#123224 (comment)

We were hoping to address both of these issues with
llvm/llvm-project#122983, but that change has
other issues we haven't yet managed to resolve. For now, we have decided
to revert the change to avoid shipping a broken feature in LLVM 20, and
we plan to follow up with a new approach post branch.

NOKEYCHECK=True
GitOrigin-RevId: 4167ea2cb082a2acb00b8b1dc09aa780dc0e3110
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
libc++ libc++ C++ Standard Library. Not GNU libstdc++. Not libc++abi.
Projects
None yet
Development

No branches or pull requests

5 participants