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

[EH] Add __cxa_init_primary_exception to cxa_noexception.cpp #22027

Merged
merged 4 commits into from
May 31, 2024

Conversation

aheejin
Copy link
Member

@aheejin aheejin commented May 31, 2024

tl;dr:

This adds __cxa_init_primary_exception to cxa_noexception.cpp.

Other targets build cxa_exception.cpp in -fignore-exceptions mode and build cxa_noexception.cpp only in -fno-exeptions mode. But we build cxa_noexception.cpp in -fignore-exceptions mode, which means it needs the definition for __cxa_init_primary_exception for linking to succeed when no EH argument is given (which means -fignore-exceptions).


Long version (Feel free to skip):

Background:

After #21638, __cxa_init_primary_exception was added in libcxxabi:

__cxa_exception* __cxa_init_primary_exception(void* object, std::type_info* tinfo,
#ifdef __wasm__
// In Wasm, a destructor returns its argument
void *(_LIBCXXABI_DTOR_FUNC* dest)(void*)) throw() {
#else
void(_LIBCXXABI_DTOR_FUNC* dest)(void*)) throw() {
#endif
__cxa_exception* exception_header = cxa_exception_from_thrown_object(object);
exception_header->referenceCount = 0;
exception_header->unexpectedHandler = std::get_unexpected();
exception_header->terminateHandler = std::get_terminate();
exception_header->exceptionType = tinfo;
exception_header->exceptionDestructor = dest;
setOurExceptionClass(&exception_header->unwindHeader);
exception_header->unwindHeader.exception_cleanup = exception_cleanup_func;
return exception_header;
}
__cxa_exception* __cxa_init_primary_exception(void* object, std::type_info* tinfo,
void *(_LIBCXXABI_DTOR_FUNC* dest)(void*)) throw() {
__cxa_exception* exception_header = cxa_exception_from_thrown_object(object);
exception_header->referenceCount = 0;
exception_header->exceptionType = tinfo;
exception_header->exceptionDestructor = dest;
return exception_header;
}

Currently the files containing __cxa_init_primary_exception, cxa_exception.cpp and cxa_exception_emscripten.cpp, are only compiled when any of the EH mode is specified. cxa_exception.cpp is compiled when Wasm EH is selected, and cxa_exception_emscripten.cpp is compiled when Emscripten EH is selected:

if self.eh_mode == Exceptions.NONE:
filenames += ['cxa_noexception.cpp']
elif self.eh_mode == Exceptions.EMSCRIPTEN:
filenames += ['cxa_exception_emscripten.cpp']
elif self.eh_mode == Exceptions.WASM:
filenames += [
'cxa_exception_storage.cpp',
'cxa_exception.cpp',
'cxa_personality.cpp'
]

and this function is called from make_exception_ptr in libcxx:

template <class _Ep>
_LIBCPP_HIDE_FROM_ABI exception_ptr make_exception_ptr(_Ep __e) _NOEXCEPT {
# ifndef _LIBCPP_HAS_NO_EXCEPTIONS
# if _LIBCPP_AVAILABILITY_HAS_INIT_PRIMARY_EXCEPTION && __cplusplus >= 201103L
using _Ep2 = __decay_t<_Ep>;
void* __ex = __cxxabiv1::__cxa_allocate_exception(sizeof(_Ep));
# ifdef __wasm__
// In Wasm, a destructor returns its argument
(void)__cxxabiv1::__cxa_init_primary_exception(__ex, const_cast<std::type_info*>(&typeid(_Ep)), [](void* __p) -> void* {
# else
(void)__cxxabiv1::__cxa_init_primary_exception(__ex, const_cast<std::type_info*>(&typeid(_Ep)), [](void* __p) {
# endif

And make_exception_ptr is called from std::promise's destructor:

template <class _Rp>
promise<_Rp>::~promise() {
if (__state_) {
if (!__state_->__has_value() && __state_->use_count() > 1)
__state_->set_exception(make_exception_ptr(future_error(make_error_code(future_errc::broken_promise))));
__state_->__release_shared();
}
}


Bug:

Currently any program that calls std::promise's destructor without specifying any exception-related arguments fails, saying undefined symbol: __cxa_init_primary_exception. Not specifying any exception arguments, meaning not specifying any among -fno-exceptions, -fwasm-exceptions, -fexceptions, or
-sDISABLE_EXCEPTION_CATCHING=0, defaults to -fignore-exceptions, which allows throwing but not catching.


Analysis:

The callsite of __cxa_init_primary_exception in make_exception_ptr is guarded with #ifndef _LIBCPP_HAS_NO_EXCEPTIONS, so it seems it is supposed to be included only when exceptions are enabled. This _LIBCPP_HAS_NO_EXCEPTIONS is defined when __cpp_exceptions is defined:

# if !defined(__cpp_exceptions) || __cpp_exceptions < 199711L
# define _LIBCPP_HAS_NO_EXCEPTIONS
# endif

And that __cpp_exceptions is defined in clang, when -fcxx-exceptions is given:
https://github.com/llvm/llvm-project/blob/be566d2eacdaed972b90d2eeb1e66d732c9fe7c1/clang/lib/Frontend/InitPreprocessor.cpp#L638-L639

And that -fcxx-exceptions can be specified in command line, but it is also programmatically added here:
https://github.com/llvm/llvm-project/blob/be566d2eacdaed972b90d2eeb1e66d732c9fe7c1/clang/lib/Driver/ToolChains/Clang.cpp#L371-L388

You can see it is added by default unless the arch is XCore or PS4/PS5, which means for Wasm it has been always added so far, unless -fno-exceptions is explicitly specified.

So I tried checking the arguments there in Clang and adding -fcxx-exceptions only if either of -fwasm-exceptions or -enable-emscripten-cxx-exceptions is given. But this fails when none of EH is selected (which means -fignore-exceptions), because if -fcxx-exceptions is not added, we can't use keywords like throw.

So basically the problem is, other targets build cxa_exception.cpp in -fignore-exceptions mode and build cxa_noexception.cpp only in -fno-exeptions mode. But we build cxa_noexception.cpp in -fignore-exceptions mode, which means it needs the definition for __cxa_init_primary_exception for linking to succeed, because _LIBCPP_HAS_NO_EXCEPTIONS cannot be defined in -fignore-exceptions (because we couldn't disable -fcxx-exceptions above in Clang to use throw)

So this adds __cxa_init_primary_exception to cxa_noexception.cpp.

tl;dr:

This adds `__cxa_init_primary_exception` to `cxa_noexception.cpp`.

Other targets build `cxa_exception.cpp` in `-fignore-exceptions` mode
and build `cxa_noexception.cpp` only in `-fno-exeptions` mode. But we
build `cxa_noexception.cpp` in `-fignore-exceptions` mode, which means
it needs the definition for `__cxa_init_primary_exception` for linking
to succeed when no EH argument is given (which means
`-fignore-exceptions`).

---

Long version (Feel free to skip):

Background:

After emscripten-core#21638, `__cxa_init_primary_exception` was added in libcxxabi:
https://github.com/emscripten-core/emscripten/blob/28c10a1c5e2862edadd68ab627478204ae96d134/system/lib/libcxxabi/src/cxa_exception.cpp#L209-L226
https://github.com/emscripten-core/emscripten/blob/28c10a1c5e2862edadd68ab627478204ae96d134/system/lib/libcxxabi/src/cxa_exception_emscripten.cpp#L155-L162

Currently the files containing `__cxa_init_primary_exception`,
`cxa_exception.cpp` and `cxa_exception_emscripten.cpp`, are only
compiled when any of the EH mode is specified. `cxa_exception.cpp` is
compiled when Wasm EH is selected, and `cxa_exception_emscripten.cpp` is
compiled when Emscripten EH is selected:
https://github.com/emscripten-core/emscripten/blob/28c10a1c5e2862edadd68ab627478204ae96d134/tools/system_libs.py#L1599-L1608

and this function is called from `make_exception_ptr` in libcxx:
https://github.com/emscripten-core/emscripten/blob/28c10a1c5e2862edadd68ab627478204ae96d134/system/lib/libcxx/include/__exception/exception_ptr.h#L87-L99

And `make_exception_ptr` is called from `std::promise`'s destructor:
https://github.com/emscripten-core/emscripten/blob/28c10a1c5e2862edadd68ab627478204ae96d134/system/lib/libcxx/include/future#L1161-L1168

---

Bug:

Currently any program that calls `std::promise`'s destructor without
specifying any exception-related arguments fails, saying `undefined
symbol: __cxa_init_primary_exception`. Not specifying any exception
arguments, meaning not specifying any among `-fno-exceptions`,
`-fwasm-exceptions`, `-fexceptions`, or
`-sDISABLE_EXCEPTION_CATCHING=0`, defaults to `-fignore-exceptions`,
which allows throwing but not catching.

---

Analysis:

The callsite of `__cxa_init_primary_exception` in `make_exception_ptr`
is guarded with `#ifndef _LIBCPP_HAS_NO_EXCEPTIONS`, so it seems it is
supposed to be included only when exceptions are enabled. This
`_LIBCPP_HAS_NO_EXCEPTIONS` is defined when `__cpp_exceptions` is
defined:
https://github.com/emscripten-core/emscripten/blob/28c10a1c5e2862edadd68ab627478204ae96d134/system/lib/libcxx/include/__config#L644-L646

And that `__cpp_exceptions` is defined in clang, when `-fcxx-exceptions`
is given:
https://github.com/llvm/llvm-project/blob/be566d2eacdaed972b90d2eeb1e66d732c9fe7c1/clang/lib/Frontend/InitPreprocessor.cpp#L638-L639

And that `-fcxx-exceptions` can be specified in command line, but it is
also programmatically added here:
https://github.com/llvm/llvm-project/blob/be566d2eacdaed972b90d2eeb1e66d732c9fe7c1/clang/lib/Driver/ToolChains/Clang.cpp#L371-L388

You can see it is added by default unless the arch is XCore or PS4/PS5,
which means for Wasm it has been always added so far, unless
`-fno-exceptions` is explicitly specified.

So I tried checking the arguments there in Clang and adding
`-fcxx-exceptions` only if either of `-fwasm-exceptions` or
`-enable-emscripten-cxx-exceptions` is given. But this fails when none
of EH is selected (which means `-fignore-exceptions`), because if
`-fcxx-exceptions` is not added, we can't use keywords like `throw`.

So basically the problem is, other targets build `cxa_exception.cpp`
in `-fignore-exceptions` mode and build `cxa_noexception.cpp` only in
`-fno-exeptions` mode. But we build `cxa_noexception.cpp` in
`-fignore-exceptions` mode, which means it needs the definition for
`__cxa_init_primary_exception` for linking to succeed, because
`_LIBCPP_HAS_NO_EXCEPTIONS` cannot be defined in `-fignore-exceptions`
(because we couldn't disable `-fcxx-exceptions` above in Clang to use
 `throw`)

So this adds `__cxa_init_primary_exception` to `cxa_noexception.cpp`.
Copy link
Collaborator

@sbc100 sbc100 left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I assume this change is probably suitable for upstreaming?

@aheejin
Copy link
Member Author

aheejin commented May 31, 2024

Not sure. I've been upstreaming the changes that are likely necessary for upstream CMake users, and unless they use our system_libs.py, this is not gonna be necessary.

Also, cxa_noexceptions.cpp contains other emscripten-only functions too. Do you think we should upstream all these?

#if __EMSCRIPTEN__
// Under emscripten this code is also linked when building when
// DISABLE_EXCEPTION_CATCHING is set but DISABLE_EXCEPTION_THROWING is not.
// TODO(sbc): Perhaps just call std::terminate here. It could
// just be some test code that needs updating.
void *__cxa_allocate_exception(size_t thrown_size) throw() {
char* allocation = (char*)malloc(thrown_size + sizeof(__cxa_exception));
return allocation + sizeof(__cxa_exception);
}
static
inline
__cxa_exception*
cxa_exception_from_thrown_object(void* thrown_object)
{
return static_cast<__cxa_exception*>(thrown_object) - 1;
}
// Free a __cxa_exception object allocated with __cxa_allocate_exception.
void __cxa_free_exception(void *thrown_object) throw() {
// Compute the size of the padding before the header.
char *raw_buffer =
((char *)cxa_exception_from_thrown_object(thrown_object));
free((void *)raw_buffer);
}
#endif

@mzukovec
Copy link

Thanks for looking into this!

@sbc100
Copy link
Collaborator

sbc100 commented May 31, 2024

Not sure. I've been upstreaming the changes that are likely necessary for upstream CMake users, and unless they use our system_libs.py, this is not gonna be necessary.

Also, cxa_noexceptions.cpp contains other emscripten-only functions too. Do you think we should upstream all these?

#if __EMSCRIPTEN__
// Under emscripten this code is also linked when building when
// DISABLE_EXCEPTION_CATCHING is set but DISABLE_EXCEPTION_THROWING is not.
// TODO(sbc): Perhaps just call std::terminate here. It could
// just be some test code that needs updating.
void *__cxa_allocate_exception(size_t thrown_size) throw() {
char* allocation = (char*)malloc(thrown_size + sizeof(__cxa_exception));
return allocation + sizeof(__cxa_exception);
}
static
inline
__cxa_exception*
cxa_exception_from_thrown_object(void* thrown_object)
{
return static_cast<__cxa_exception*>(thrown_object) - 1;
}
// Free a __cxa_exception object allocated with __cxa_allocate_exception.
void __cxa_free_exception(void *thrown_object) throw() {
// Compute the size of the padding before the header.
char *raw_buffer =
((char *)cxa_exception_from_thrown_object(thrown_object));
free((void *)raw_buffer);
}
#endif

I guess my question is, is this __cxa_init_primary_exception (or the need for it) emscripten-specific? How come calls to it are being generated by the wasm backend but not other backends?

@aheejin aheejin merged commit b0ec4e1 into emscripten-core:main May 31, 2024
29 checks passed
@aheejin aheejin deleted the init_primary_ex branch May 31, 2024 16:14
@aheejin
Copy link
Member Author

aheejin commented May 31, 2024

I guess my question is, is this __cxa_init_primary_exception (or the need for it) emscripten-specific? How come calls to it are being generated by the wasm backend but not other backends?

Calls to __cxa_init_primary_exception are generated in other backends too. But they use cxa_exceptions.cpp in case of -fignore-exceptions so it's not a problem. They use cxa_noexceptions.cpp only when -fno-exceptions is given, in which case _LIBCPP_HAS_NO_EXCEPTIONS is defined and the calls to __cxa_init_primary_exception are not compiled:

template <class _Ep>
_LIBCPP_HIDE_FROM_ABI exception_ptr make_exception_ptr(_Ep __e) _NOEXCEPT {
# ifndef _LIBCPP_HAS_NO_EXCEPTIONS
# if _LIBCPP_AVAILABILITY_HAS_INIT_PRIMARY_EXCEPTION && __cplusplus >= 201103L
using _Ep2 = __decay_t<_Ep>;
void* __ex = __cxxabiv1::__cxa_allocate_exception(sizeof(_Ep));
# ifdef __wasm__
// In Wasm, a destructor returns its argument
(void)__cxxabiv1::__cxa_init_primary_exception(__ex, const_cast<std::type_info*>(&typeid(_Ep)), [](void* __p) -> void* {
# else
(void)__cxxabiv1::__cxa_init_primary_exception(__ex, const_cast<std::type_info*>(&typeid(_Ep)), [](void* __p) {
# endif

But we use cxa_noexception.cpp in case of -fignore-exceptions, which is our default. That's why I added the function there.

@sbc100
Copy link
Collaborator

sbc100 commented May 31, 2024

I guess my question is, is this __cxa_init_primary_exception (or the need for it) emscripten-specific? How come calls to it are being generated by the wasm backend but not other backends?

Calls to __cxa_init_primary_exception are generated in other backends too. But they use cxa_exceptions.cpp in case of -fignore-exceptions so it's not a problem. They use cxa_noexceptions.cpp only when -fno-exceptions is given, in which case _LIBCPP_HAS_NO_EXCEPTIONS is defined and the calls to __cxa_init_primary_exception is not compiled:

template <class _Ep>
_LIBCPP_HIDE_FROM_ABI exception_ptr make_exception_ptr(_Ep __e) _NOEXCEPT {
# ifndef _LIBCPP_HAS_NO_EXCEPTIONS
# if _LIBCPP_AVAILABILITY_HAS_INIT_PRIMARY_EXCEPTION && __cplusplus >= 201103L
using _Ep2 = __decay_t<_Ep>;
void* __ex = __cxxabiv1::__cxa_allocate_exception(sizeof(_Ep));
# ifdef __wasm__
// In Wasm, a destructor returns its argument
(void)__cxxabiv1::__cxa_init_primary_exception(__ex, const_cast<std::type_info*>(&typeid(_Ep)), [](void* __p) -> void* {
# else
(void)__cxxabiv1::__cxa_init_primary_exception(__ex, const_cast<std::type_info*>(&typeid(_Ep)), [](void* __p) {
# endif

But we use cxa_noexception.cpp in case of -fignore-exceptions, which is our default. That's why I added the function there.

I see.. so the issue stems from the different/special way that we handle -fignore-exceptions. Might be worth adding a comment here to that effect.

@aheejin
Copy link
Member Author

aheejin commented May 31, 2024

I see.. so the issue stems from the different/special way that we handle -fignore-exceptions. Might be worth adding a comment here to that effect.

#22034

msorvig pushed a commit to msorvig/emscripten that referenced this pull request Jun 4, 2024
…ten-core#22027)

tl;dr:

This adds `__cxa_init_primary_exception` to `cxa_noexception.cpp`.

Other targets build `cxa_exception.cpp` in `-fignore-exceptions` mode
and build `cxa_noexception.cpp` only in `-fno-exeptions` mode. But we
build `cxa_noexception.cpp` in `-fignore-exceptions` mode, which means
it needs the definition for `__cxa_init_primary_exception` for linking
to succeed when no EH argument is given (which means
`-fignore-exceptions`).

---

Long version (Feel free to skip):

Background:

After emscripten-core#21638, `__cxa_init_primary_exception` was added in libcxxabi:
https://github.com/emscripten-core/emscripten/blob/28c10a1c5e2862edadd68ab627478204ae96d134/system/lib/libcxxabi/src/cxa_exception.cpp#L209-L226
https://github.com/emscripten-core/emscripten/blob/28c10a1c5e2862edadd68ab627478204ae96d134/system/lib/libcxxabi/src/cxa_exception_emscripten.cpp#L155-L162

Currently the files containing `__cxa_init_primary_exception`,
`cxa_exception.cpp` and `cxa_exception_emscripten.cpp`, are only
compiled when any of the EH mode is specified. `cxa_exception.cpp` is
compiled when Wasm EH is selected, and `cxa_exception_emscripten.cpp` is
compiled when Emscripten EH is selected:
https://github.com/emscripten-core/emscripten/blob/28c10a1c5e2862edadd68ab627478204ae96d134/tools/system_libs.py#L1599-L1608

and this function is called from `make_exception_ptr` in libcxx:
https://github.com/emscripten-core/emscripten/blob/28c10a1c5e2862edadd68ab627478204ae96d134/system/lib/libcxx/include/__exception/exception_ptr.h#L87-L99

And `make_exception_ptr` is called from `std::promise`'s destructor:
https://github.com/emscripten-core/emscripten/blob/28c10a1c5e2862edadd68ab627478204ae96d134/system/lib/libcxx/include/future#L1161-L1168

---

Bug:

Currently any program that calls `std::promise`'s destructor without
specifying any exception-related arguments fails, saying `undefined
symbol: __cxa_init_primary_exception`. Not specifying any exception
arguments, meaning not specifying any among `-fno-exceptions`,
`-fwasm-exceptions`, `-fexceptions`, or
`-sDISABLE_EXCEPTION_CATCHING=0`, defaults to `-fignore-exceptions`,
which allows throwing but not catching.

---

Analysis:

The callsite of `__cxa_init_primary_exception` in `make_exception_ptr`
is guarded with `#ifndef _LIBCPP_HAS_NO_EXCEPTIONS`, so it seems it is
supposed to be included only when exceptions are enabled. This
`_LIBCPP_HAS_NO_EXCEPTIONS` is defined when `__cpp_exceptions` is
defined:
https://github.com/emscripten-core/emscripten/blob/28c10a1c5e2862edadd68ab627478204ae96d134/system/lib/libcxx/include/__config#L644-L646

And that `__cpp_exceptions` is defined in clang, when `-fcxx-exceptions`
is given:
https://github.com/llvm/llvm-project/blob/be566d2eacdaed972b90d2eeb1e66d732c9fe7c1/clang/lib/Frontend/InitPreprocessor.cpp#L638-L639

And that `-fcxx-exceptions` can be specified in command line, but it is
also programmatically added here:
https://github.com/llvm/llvm-project/blob/be566d2eacdaed972b90d2eeb1e66d732c9fe7c1/clang/lib/Driver/ToolChains/Clang.cpp#L371-L388

You can see it is added by default unless the arch is XCore or PS4/PS5,
which means for Wasm it has been always added so far, unless
`-fno-exceptions` is explicitly specified.

So I tried checking the arguments there in Clang and adding
`-fcxx-exceptions` only if either of `-fwasm-exceptions` or
`-enable-emscripten-cxx-exceptions` is given. But this fails when none
of EH is selected (which means `-fignore-exceptions`), because if
`-fcxx-exceptions` is not added, we can't use keywords like `throw`.

So basically the problem is, other targets build `cxa_exception.cpp`
in `-fignore-exceptions` mode and build `cxa_noexception.cpp` only in
`-fno-exeptions` mode. But we build `cxa_noexception.cpp` in
`-fignore-exceptions` mode, which means it needs the definition for
`__cxa_init_primary_exception` for linking to succeed, because
`_LIBCPP_HAS_NO_EXCEPTIONS` cannot be defined in `-fignore-exceptions`
(because we couldn't disable `-fcxx-exceptions` above in Clang to use
 `throw`)

So this adds `__cxa_init_primary_exception` to `cxa_noexception.cpp`.
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

4 participants