Stream: general

Topic: __funcs_on_exit indirect call type mismatch with wasi-sdk


view this post on Zulip Vladimir Vukicevic (Apr 28 2020 at 16:29):

Howdy! I'm trying to run a fairly large C/C++ test suite built on top of Catch2, compiled with wasi-sdk v10 and running with wasmtime. I'm getting a fail on exit:

===============================================================================
All tests passed (142033 assertions in 116 test cases)
Error: failed to run main module `/var/folders/zs/m4nj7dw15v54xf284j66fbth0000gp/T/Pram_IfXpoFVmPpbidgbO9vvp/g==/WasiRunner/16149EC0-pram-baselib-tests-debug.wasm/app/baselib-tests-debug.wasm`

Caused by:
    0: failed to invoke `_start`
    1: wasm trap: indirect call type mismatch, source location: @9e416a
       wasm backtrace:
         0: <unknown>!__funcs_on_exit
         1: <unknown>!__prepare_for_exit
         2: <unknown>!_start

There's no call to atexit in the code, so I'm guessing it's due to some generated __cxa_atexit with the wrong function signature. Is there any reasonable way for me to figure out which function has the wrong signature?

view this post on Zulip Yury Delendik (Apr 28 2020 at 16:39):

Is it possible to recompile the test suite with debug info and use a debugger with wasmtime?

view this post on Zulip Vladimir Vukicevic (Apr 28 2020 at 16:40):

mm. Good idea; I do have debug info, I didn't actually think to use a debugger here. Let me try that

view this post on Zulip Yury Delendik (Apr 28 2020 at 16:40):

/me is interested in assisting with troubleshooting any problems with running that setup

view this post on Zulip Yury Delendik (Apr 28 2020 at 16:41):

wasmtime transforms wasm DWARF if -g is provided (works by default with lldb/linux, though there is a PR for gdb)

view this post on Zulip Vladimir Vukicevic (Apr 28 2020 at 16:43):

do I just lldb on wasmtime itself, or is there a gdbserver-like thing to attach to? Also running with -g, I'm getting:

Error: failed to run main module `...`
Caused by:
    0: Debug information error
    1: Invalid opcode in DWARF expression

view this post on Zulip Yury Delendik (Apr 28 2020 at 16:44):

okay, looks like an old wasmtime

view this post on Zulip Vladimir Vukicevic (Apr 28 2020 at 16:44):

yeah, it's 0.15 -- I'm building from git right now

view this post on Zulip Yury Delendik (Apr 28 2020 at 16:44):

I recommend ;) https://github.com/bytecodealliance/wasmtime/pull/1572

There was a bug how value labels were resolved, which caused some DWARF expressions not be transformed, e.g. those are in the registers. Patch built on top of #1542 Documentation for ValueLocRange...

view this post on Zulip Vladimir Vukicevic (Apr 28 2020 at 16:45):

do I want your branch, or master -- or merge of the two?

view this post on Zulip Yury Delendik (Apr 28 2020 at 16:46):

I rebased on top of master today

view this post on Zulip Vladimir Vukicevic (Apr 28 2020 at 16:47):

ah there's a merge conflict with master still

view this post on Zulip Yury Delendik (Apr 28 2020 at 16:48):

right, I'll resolve that shortly, the branch can be used as is

view this post on Zulip Vladimir Vukicevic (Apr 28 2020 at 16:49):

erm, right. sorry, operating on too little sleep :)

view this post on Zulip Yury Delendik (Apr 28 2020 at 16:55):

/me rebased PRs just to have that clean from conflicts

view this post on Zulip Vladimir Vukicevic (Apr 28 2020 at 17:30):

Arg, I need a newer lldb on mac

view this post on Zulip Yury Delendik (Apr 28 2020 at 17:47):

that might be a problem, only version of brew's lldb can do it, I think

view this post on Zulip Yury Delendik (Apr 28 2020 at 17:49):

and just for MacOSX, ".lldbinit" needs settings set plugin.jit-loader.gdb.enable on

view this post on Zulip Yury Delendik (Apr 28 2020 at 17:50):

/me has lldb version 9.0.0 on mac

view this post on Zulip Vladimir Vukicevic (Apr 28 2020 at 18:07):

ah it was the lldbinit piece I was missing! lldb 10.0 from brew works:

(lldb) bt
* thread #1, queue = 'com.apple.main-thread', stop reason = EXC_BAD_INSTRUCTION (code=EXC_I386_INVOP, subcode=0x0)
  * frame #0: 0x000000016ec53806 JIT(0x17d700000)`__funcs_on_exit at <gen-1>.wasm:10371434
    frame #1: 0x000000016ec53695 JIT(0x17d700000)`__prepare_for_exit at <gen-1>.wasm:10371335
    frame #2: 0x000000016da007be JIT(0x17d700000)`_start at <gen-1>.wasm:47534
    frame #3: 0x000000016ec6ace3
    frame #4: 0x000000010013b33a wasmtime`wasmtime::func::Func::call::_$u7b$$u7b$closure$u7d$$u7d$::h519bce4b59bf0f7c at func.rs:566:20
    frame #5: 0x0000000100161ba8 wasmtime`wasmtime_runtime::traphandlers::catch_traps::call_closure::h991a454e37488ebb(payload="P\x95���") at traphandlers.rs:385:17
    frame #6: 0x00000001005d0fd6 wasmtime`RegisterSetjmp(buf_storage=0x00007ffeefbf9370, body=(wasmtime`wasmtime_runtime::traphandlers::catch_traps::call_closure::h991a454e37488ebb at traphandlers.rs:381), payload=0x00007ffeefbf9230) at helpers.c:12:3

view this post on Zulip Vladimir Vukicevic (Apr 28 2020 at 18:07):

of course now I need a libc with debug info

view this post on Zulip Vladimir Vukicevic (Apr 28 2020 at 18:09):

because it's not clear how to figure out what it's actually trying to call -- I'm guessing with debug symbols, I'll be able to inspect the actual function pointer value, which should tell me the actual bad function?

view this post on Zulip Yury Delendik (Apr 28 2020 at 18:09):

yeah, that's a hope

view this post on Zulip Yury Delendik (Apr 28 2020 at 18:10):

you can also dis and check registers: maybe some value from params will pop

view this post on Zulip Vladimir Vukicevic (Apr 28 2020 at 18:11):

this is actually a little annoying. cranelift knows it's a bad signature. would be handy if it could show me what the "bad" one was (and expected, though in this case I know what's expected)

view this post on Zulip Yury Delendik (Apr 28 2020 at 18:12):

under the hood it is just an (signature) ID

view this post on Zulip Vladimir Vukicevic (Apr 28 2020 at 18:13):

sure, but that id is described in the wasm module

view this post on Zulip Yury Delendik (Apr 28 2020 at 18:14):

not really, more like engine assigned id/handle

view this post on Zulip Yury Delendik (Apr 28 2020 at 18:15):

I wonder if frame vars are inspectable

view this post on Zulip Vladimir Vukicevic (Apr 28 2020 at 18:15):

they don't seem to be

view this post on Zulip Yury Delendik (Apr 28 2020 at 18:16):

in wasmtime you can choose cranelift optimization e.g. via --opt-level 0

view this post on Zulip Vladimir Vukicevic (Apr 28 2020 at 18:18):

function "pointer" indices are direct indices into a table, right? funcptr value "1" == index 1

view this post on Zulip Vladimir Vukicevic (Apr 28 2020 at 18:18):

I think I can use the wat output and figure out all the calls to cxa_atexit

view this post on Zulip Yury Delendik (Apr 28 2020 at 18:18):

right

view this post on Zulip Yury Delendik (Apr 28 2020 at 18:19):

fwiw, 10371335 is bytecode offset in wasm in <gen-1>.wasm:10371335

view this post on Zulip Yury Delendik (Apr 28 2020 at 18:20):

/me is not sure https://wasdk.github.io/wasmcodeexplorer/ can handle that large file though

view this post on Zulip Vladimir Vukicevic (Apr 28 2020 at 18:20):

how do I get the wasm heap base address in lldb?

view this post on Zulip Yury Delendik (Apr 28 2020 at 18:21):

is there __vmctx defined?

view this post on Zulip Yury Delendik (Apr 28 2020 at 18:21):

if yes, __vmctx->memory shall do it

view this post on Zulip Vladimir Vukicevic (Apr 28 2020 at 18:21):

mm I can try, I was just using wasm2wat

view this post on Zulip Vladimir Vukicevic (Apr 28 2020 at 18:21):

__vmctx wasn't available, trying with opt-level 0 though

view this post on Zulip Vladimir Vukicevic (Apr 28 2020 at 18:22):

(thanks for your help btw!)

view this post on Zulip Vladimir Vukicevic (Apr 28 2020 at 18:25):

heh, --opt-level 0 made things worse :) (I have vmctx from a different frame turns out) With it I lose some local vars:

(int) var0 = <variable not available>
(int) var1 = 211424
(int) var2 = <variable not available>

without (default):

(int) var0 = 52080
(int) var1 = 211424
(int) var2 = 277217280

view this post on Zulip Yury Delendik (Apr 28 2020 at 18:26):

hmm... yeah... okay, I need to fix more stuff then

view this post on Zulip Yury Delendik (Apr 28 2020 at 18:28):

you can try setting a breakpoint (on symbols?), see if non-trap will give you more info

view this post on Zulip Vladimir Vukicevic (Apr 28 2020 at 18:28):

my strategy of "let me read ints from memory at various local var addresses (also double-dereferenced) in hopes of finding the function pointer index" did not succeed

view this post on Zulip Yury Delendik (Apr 28 2020 at 18:34):

right, having source code / DWARF for wasi-sdk might give you more clues for its location

view this post on Zulip Vladimir Vukicevic (Apr 28 2020 at 19:35):

I just realized I can also just.. put a printf inside funcs_atexit to print the indices as they're called

view this post on Zulip bjorn3 (Apr 28 2020 at 19:47):

printf: my favourite debugging technique when debugging miscompilations of cg_clif.

view this post on Zulip Vladimir Vukicevic (Apr 28 2020 at 20:26):

It would be _really nice_ to have some kind of wasm intrinsics for working with function signatures. I'd love to be able to add an assert(__wasm_signature_type(funcptr) == __wasm_signature_type(KnownGoodFunction)) in __cxa_atexit and give the user a good warning

view this post on Zulip Vladimir Vukicevic (Apr 28 2020 at 20:46):

That was extremely painful, but it looks like the culprit is $std::__2::vector<Baselib_Memory_PageState__std::__2::allocator<Baselib_Memory_PageState>_>::~vector__

view this post on Zulip Vladimir Vukicevic (Apr 28 2020 at 20:47):

which returns an i32, instead of returning void

view this post on Zulip Vladimir Vukicevic (Apr 28 2020 at 20:57):

   // Under some ABIs, destructors return this instead of void, and cannot be
   // passed directly to __cxa_atexit if the target does not allow this
   // mismatch.

view this post on Zulip Vladimir Vukicevic (Apr 28 2020 at 20:58):

er.. this seems like a llvm bug, because canCallMismatchedFunctionType should definitely be false for wasm

view this post on Zulip Vladimir Vukicevic (Apr 28 2020 at 21:00):

now I have 3hrs of meetings but I will fix this :)

view this post on Zulip Dan Gohman (Apr 28 2020 at 21:01):

@Vladimir Vukicevic it should be returning false already: https://github.com/llvm/llvm-project/blob/master/clang/lib/CodeGen/ItaniumCXXABI.cpp#L517

The LLVM Project is a collection of modular and reusable compiler and toolchain technologies. Note: the repository does not accept github pull requests at this moment. Please submit your patches at...

view this post on Zulip Vladimir Vukicevic (Apr 28 2020 at 21:49):

does wasm/wasi use the itanium ABI?

view this post on Zulip Vladimir Vukicevic (Apr 28 2020 at 21:50):

i'm trying to create a small repro

view this post on Zulip Dan Gohman (Apr 28 2020 at 21:50):

it uses a variant of the Itanium ABI

view this post on Zulip Dan Gohman (Apr 28 2020 at 21:52):

https://github.com/llvm/llvm-project/blob/master/clang/include/clang/Basic/TargetCXXABI.h#L91 if anyone is curious :-)

The LLVM Project is a collection of modular and reusable compiler and toolchain technologies. Note: the repository does not accept github pull requests at this moment. Please submit your patches at...

view this post on Zulip Vladimir Vukicevic (Apr 28 2020 at 22:58):

@Dan Gohman you may know this (hi, btw!) -- in my broken case, the bad __cxa_atexit destructor call setup is happening from a __cxx_global_var_init.77 function. Does that seem sane for global_var_init to do that?

view this post on Zulip Dan Gohman (Apr 28 2020 at 23:00):

Yeah. Wasm doesn't have an equivalent of .fini or .dtors in other platforms, so the global var inits have to register their associated destructor calls with __cxa_atexit

view this post on Zulip Dan Gohman (Apr 28 2020 at 23:01):

Also, hi! :grinning:

view this post on Zulip Vladimir Vukicevic (Apr 28 2020 at 23:04):

but I see global dtors also being called from __cxx_global_array_dtor functions too

view this post on Zulip Dan Gohman (Apr 28 2020 at 23:07):

__cxx_global_array_dtor is the function that gets registered with __cxa_atexit, which calls the dtors

view this post on Zulip Vladimir Vukicevic (Apr 28 2020 at 23:12):

well -- no, in my case I see a dtor being _directly_ registered with __cxa_atexit from a global_var_init

view this post on Zulip Vladimir Vukicevic (Apr 28 2020 at 23:13):

  (func $__cxx_global_var_init.77 (type $t9)
    (local $l0 i32) (local $l1 i32) (local $l2 i32) (local $l3 i32) (local $l4 i32)
    i32.const 2171
    local.set $l0
    i32.const 0
    local.set $l1
    i32.const 1024
    local.set $l2
    i32.const 211420
    local.set $l3
    i32.const 151088
    local.set $l4
    local.get $l3
    local.get $l4
    call $std::__2::basic_string<char__std::__2::char_traits<char>__std::__2::allocator<char>_>::basic_string<std::nullptr_t>_char_const*_
    drop
    local.get $l0
    local.get $l1
    local.get $l2
    call $__cxa_atexit
    drop
    return)

in my case $l0 = 2171 is the bad-signature vector dtor being passed to cxa_atexit

view this post on Zulip Dan Gohman (Apr 28 2020 at 23:17):

Hmm. That does seem odd. wasm destructors return i32, so I wouldn't think they could be registered with __cxa_atexit like that

view this post on Zulip Vladimir Vukicevic (Apr 28 2020 at 23:17):

yep, that's the bug :)

view this post on Zulip Dan Gohman (Apr 28 2020 at 23:19):

Is that a std::string? When I try a simple testcase with std::string, it registers __cxx_global_array_dtor with __cxa_atexit, and __cxx_global_array_dtor calls the dtor

view this post on Zulip Vladimir Vukicevic (Apr 28 2020 at 23:19):

I need to try -fregister-global-dtors-with-atexit but I really want to get to a small repro, this massive unified test runner is not great to work with :)

view this post on Zulip Vladimir Vukicevic (Apr 28 2020 at 23:21):

it's also totally possible that the wrong function pointer is being used here somehow too

view this post on Zulip Dan Gohman (Apr 28 2020 at 23:23):

Oh, awkward. It looks like -fregister-global-dtors-with-atexit does the same thing as what the wasm backend does with global dtors

view this post on Zulip Vladimir Vukicevic (Apr 28 2020 at 23:29):

Ooh. ignore my WAT paste above, the actual number is 2170 (which is still the dtor)

view this post on Zulip Vladimir Vukicevic (Apr 28 2020 at 23:30):

ok, 2170 is the same problem, just in a different __cxx_global_var_init.5.37. the ctor right before it is call $std::__2::vector<Baselib_Memory_PageState__std::__2::allocator<Baselib_Memory_PageState>_>::vector_std::initializer_list<Baselib_Memory_PageState>_, let me see if I can create something like this

view this post on Zulip Vladimir Vukicevic (Apr 28 2020 at 23:37):

Baselib_Memory_PageState is just an enum, nothing crazy.

view this post on Zulip Vladimir Vukicevic (Apr 28 2020 at 23:44):

Yeah, I can't reproduce this in a small testcase. The thing that's wrong is the only vector dtor that's in the function ptr table (lots of other dtors, I assume due to vtables). In a clang link command that outputs wasm, how do I get it to output bitcode or LLMV IR?

view this post on Zulip Dan Gohman (Apr 28 2020 at 23:58):

I don't know of a way to get it to emit llvm ir when producing wasm. It doesn't produce a linked LLVM IR image unless you're using -flto

view this post on Zulip Vladimir Vukicevic (Apr 29 2020 at 00:15):

I'm now at a bit of a loss how to proceed. I _think_ the most useful next steps will be one of:

  1. see if I can get some kind of whole-program/LTO bitcode that I can decompile to IR, and see if the problem is visible there
  2. see if I can find in one of the object files the actual issue
  3. ???

view this post on Zulip Dan Gohman (Apr 29 2020 at 00:16):

Can you determine which source file defines the object with the destructor in question?

view this post on Zulip Vladimir Vukicevic (Apr 29 2020 at 00:17):

Yeah I'm pretty sure I can

view this post on Zulip Dan Gohman (Apr 29 2020 at 00:20):

If you can compile that source file to .o, we could see if the problem is present there, or if it gets introduced at link time

view this post on Zulip Vladimir Vukicevic (Apr 29 2020 at 00:28):

define internal void @__cxx_global_var_init.27() #0 !dbg !13235 {
  %1 = alloca %"class.std::initializer_list", align 4
  %2 = alloca [6 x i32], align 4
  %3 = getelementptr inbounds [6 x i32], [6 x i32]* %2, i32 0, i32 0, !dbg !13236
  %4 = bitcast [6 x i32]* %2 to i8*, !dbg !13236
  call void @llvm.memcpy.p0i8.p0i8.i32(i8* align 4 %4, i8* align 4 bitcast ([6 x i32]* @constinit to i8*), i32 24, i1 false), !dbg !13236
  %5 = getelementptr inbounds %"class.std::initializer_list", %"class.std::initializer_list"* %1, i32 0, i32 0, !dbg !13236
  %6 = getelementptr inbounds [6 x i32], [6 x i32]* %2, i32 0, i32 0, !dbg !13236
  store i32* %6, i32** %5, align 4, !dbg !13236
  %7 = getelementptr inbounds %"class.std::initializer_list", %"class.std::initializer_list"* %1, i32 0, i32 1, !dbg !13236
  store i32 6, i32* %7, align 4, !dbg !13236
  %8 = call %"class.std::__2::vector.6"* @_ZNSt3__26vectorI24Baselib_Memory_PageStateNS_9allocatorIS1_EEEC2ESt16initializer_listIS1_E(%"class.std::__2::vector.6"* @pageStatesAll, %"class.std::initializer_list"* byval(%"class.std::initializer_list") align 4 %1), !dbg !13236
  %9 = call i32 @__cxa_atexit(void (i8*)* @__cxx_global_array_dtor.28, i8* null, i8* @__dso_handle) #4, !dbg !13236
  ret void, !dbg !13236
}

view this post on Zulip Vladimir Vukicevic (Apr 29 2020 at 00:28):

pretty sure that's it, and it looks fine :|

view this post on Zulip Dan Gohman (Apr 29 2020 at 00:29):

Is this an optimized build?

view this post on Zulip Dan Gohman (Apr 29 2020 at 00:29):

And if so, do you have wasm-opt in your PATH?

view this post on Zulip Vladimir Vukicevic (Apr 29 2020 at 00:29):

debug build (-g -O0)

view this post on Zulip Dan Gohman (Apr 29 2020 at 00:30):

Can you do wasm2wat on the .o file?

view this post on Zulip Vladimir Vukicevic (Apr 29 2020 at 00:30):

tried, I get "0001e60: error: invalid section code: 12"

view this post on Zulip Vladimir Vukicevic (Apr 29 2020 at 00:30):

(brb, meeting, sigh)

view this post on Zulip Dan Gohman (Apr 29 2020 at 00:30):

or llvm-objdump -d (which works now \o/)

view this post on Zulip Vladimir Vukicevic (Apr 29 2020 at 00:32):

I can do llvm-objdump.. but not sure how to tell if the result is useful, I don't get any symbol/function names

view this post on Zulip Dan Gohman (Apr 29 2020 at 00:33):

wasm2wat may need --enable-bulk-memory

view this post on Zulip Dan Gohman (Apr 29 2020 at 00:34):

Or if your wasm2wat is new enough, --enable-all should do it

view this post on Zulip Vladimir Vukicevic (Apr 29 2020 at 00:34):

yep that worked. let's see

view this post on Zulip Vladimir Vukicevic (Apr 29 2020 at 00:35):

  (func $__cxx_global_var_init.27 (type $t0)
    (local $l0 i32) (local $l1 i32) (local $l2 i32) (local $l3 i32) (local $l4 i32) (local $l5 i32) (local $l6 i32) (local $l7 i32) (local $l8 i32) (local $l9 i32) (local $l10 i64) (local $l11 i32) (local $l12 i32) (local $l13 i64) (local $l14 i64) (local $l15 i64) (local $l16 i32) (local $l17 i32) (local $l18 i32) (local $l19 i32) (local $l20 i32) (local $l21 i32) (local $l22 i32) (local $l23 i32)
    global.get $env.__stack_pointer
    local.set $l0
    i32.const 48
    local.set $l1
    local.get $l0
    local.get $l1
    i32.sub
    local.set $l2
    local.get $l2
    global.set $env.__stack_pointer
    i32.const 6
    local.set $l3
    i32.const 16
    local.set $l4
    local.get $l2
    local.get $l4
    i32.add
    local.set $l5
    local.get $l5
    local.set $l6
    i32.const 16
    local.set $l7
    local.get $l6
    local.get $l7
    i32.add
    local.set $l8
    i32.const 0
    local.set $l9
    local.get $l9
    i64.load offset=648 align=4
    local.set $l10
    local.get $l8
    local.get $l10
    i64.store align=4
    i32.const 8
    local.set $l11
    local.get $l6
    local.get $l11
    i32.add
    local.set $l12
    local.get $l9
    i64.load offset=640 align=4
    local.set $l13
    local.get $l12
    local.get $l13
    i64.store align=4
    local.get $l9
    i64.load offset=632 align=4
    local.set $l14
    local.get $l6
    local.get $l14
    i64.store align=4
    local.get $l2
    local.get $l6
    i32.store offset=40
    local.get $l2
    local.get $l3
    i32.store offset=44
    local.get $l2
    i64.load offset=40
    local.set $l15
    local.get $l2
    local.get $l15
    i64.store offset=8
    i32.const 620
    local.set $l16
    i32.const 8
    local.set $l17
    local.get $l2
    local.get $l17
    i32.add
    local.set $l18
    local.get $l16
    local.get $l18
    call $_ZNSt3__26vectorI24Baselib_Memory_PageStateNS_9allocatorIS1_EEEC2ESt16initializer_listIS1_E
    drop
    i32.const 16
    local.set $l19
    i32.const 0
    local.set $l20
    i32.const 0
    local.set $l21
    i32.const 620
    drop
    local.get $l19
    local.get $l20
    local.get $l21
    call $env.__cxa_atexit
    drop
    i32.const 48
    local.set $l22
    local.get $l2
    local.get $l22
    i32.add
    local.set $l23
    local.get $l23
    global.set $env.__stack_pointer
    return)

view this post on Zulip Vladimir Vukicevic (Apr 29 2020 at 00:35):

gah sorry, long paste

view this post on Zulip Dan Gohman (Apr 29 2020 at 00:36):

ok, so it has a 0, so there's presumably a relocation to fill it iin

view this post on Zulip Dan Gohman (Apr 29 2020 at 00:36):

llvm-objdump -d -r on the .o file should show the relocations

view this post on Zulip Vladimir Vukicevic (Apr 29 2020 at 01:11):

(ok meetings all done) What am I looking for in the -d -r output? the tail end looks like

    f598: 10 d4 81 80 80 00             call    212
            0000f599:  R_WASM_FUNCTION_INDEX_LEB    _ZNSt3__26vectorI24Baselib_Memory_PageStateNS_9allocatorIS1_EEEC2ESt16initializer_listIS1_E+0
    f59e: 1a                            drop
    f59f: 41 90 80 80 80 00             i32.const   16
            0000f5a0:  R_WASM_TABLE_INDEX_SLEB  __cxx_global_array_dtor.28+0
    f5a5: 21 13                         local.set   19
    f5a7: 41 00                         i32.const   0
    f5a9: 21 14                         local.set   20
    f5ab: 41 80 80 80 80 00             i32.const   0
            0000f5ac:  R_WASM_MEMORY_ADDR_SLEB  __dso_handle+0
    f5b1: 21 15                         local.set   21
    f5b3: 41 ec 84 80 80 00             i32.const   620
            0000f5b4:  R_WASM_MEMORY_ADDR_SLEB  pageStatesAll+0
    f5b9: 1a                            drop
    f5ba: 20 13                         local.get   19
    f5bc: 20 14                         local.get   20
    f5be: 20 15                         local.get   21
    f5c0: 10 80 80 80 80 00             call    0
            0000f5c1:  R_WASM_FUNCTION_INDEX_LEB    __cxa_atexit+0
    f5c6: 1a                            drop
    f5c7: 41 30                         i32.const   48
    f5c9: 21 16                         local.set   22
    f5cb: 20 02                         local.get   2
    f5cd: 20 16                         local.get   22
    f5cf: 6a                            i32.add
    f5d0: 21 17                         local.set   23
    f5d2: 20 17                         local.get   23
    f5d4: 24 80 80 80 80 00             global.set  0
            0000f5d5:  R_WASM_GLOBAL_INDEX_LEB  __stack_pointer+0
    f5da: 0f                            return
    f5db: 0b                            end

which looks correct unfortunately

view this post on Zulip Vladimir Vukicevic (Apr 29 2020 at 01:14):

I just audited all __cxa_atexit calls in this file and everything looks fine. This is the only file that the relevant type (the Baselib_Memory_PageState) enum/vector appears in

view this post on Zulip Vladimir Vukicevic (Apr 29 2020 at 01:40):

Here's what I've got. All the global_var_inits from this function set up a cxa_atexit with a global_array_dtor function, except for the bad one. The bad one is the "right" (consistent) index:

(func $_GLOBAL__sub_I_Baselib_Memory_Tests_Wasi.cpp (type $t9)
    call $__cxx_global_var_init.76     -> cxa_atexit with 2167      __cxx_global_array_dtor.78
    call $__cxx_global_var_init.1.46   -> cxa_atexit with 2168      __cxx_global_array_dtor.2.46
    call $__cxx_global_var_init.3.46   -> cxa_atexit with 2169      __cxx_global_array_dtor.4.46
    call $__cxx_global_var_init.5.37   -> cxa_atexit with ... 2170  $std::__2::vector<Baselib_Memory_PageState__std::__2::allocator<Baselib_Memory_PageState>_>::~vector__
    return)

(with annotations of what funcptr index they call and what the actual thing is)

view this post on Zulip Vladimir Vukicevic (Apr 29 2020 at 01:46):

Ok, I had the wrong things earlier. Based on that, the issue is in var_init.5 in that file. The .o file indeed has the wrong thing for var_init.5:

...
     705: 41 84 80 80 80 00             i32.const   4
            00000706:  R_WASM_TABLE_INDEX_SLEB  _ZNSt3__26vectorI24Baselib_Memory_PageStateNS_9allocatorIS1_EEED2Ev+0
     70b: 21 0d                         local.set   13
     70d: 20 0d                         local.get   13
     70f: 21 0e                         local.set   14
     711: 20 0c                         local.get   12
     713: 21 0f                         local.set   15
     715: 41 80 80 80 80 00             i32.const   0
            00000716:  R_WASM_MEMORY_ADDR_SLEB  __dso_handle+0
     71b: 21 10                         local.set   16
     71d: 20 0e                         local.get   14
     71f: 20 0f                         local.get   15
     721: 20 10                         local.get   16
     723: 10 80 80 80 80 00             call    0
            00000724:  R_WASM_FUNCTION_INDEX_LEB    __cxa_atexit+0

view this post on Zulip Vladimir Vukicevic (Apr 29 2020 at 01:49):

And the LLVM IR looks wrong (extra indentation in the cxa_atexit call):

define internal void @__cxx_global_var_init.5() #0 !dbg !3799 {
  %1 = alloca %"class.std::initializer_list", align 4
  %2 = alloca [1 x i32], align 4
  %3 = getelementptr inbounds [1 x i32], [1 x i32]* %2, i32 0, i32 0, !dbg !3800
  store i32 4, i32* %3, align 4, !dbg !3800
  %4 = getelementptr inbounds %"class.std::initializer_list", %"class.std::initializer_list"* %1, i32 0, i32 0, !dbg !3800
  %5 = getelementptr inbounds [1 x i32], [1 x i32]* %2, i32 0, i32 0, !dbg !3800
  store i32* %5, i32** %4, align 4, !dbg !3800                                                                                                      %6 = getelementptr inbounds %"class.std::initializer_list", %"class.std::initializer_list"* %1, i32 0, i32 1, !dbg !3800
  store i32 1, i32* %6, align 4, !dbg !3800
  %7 = call %"class.std::__2::vector.6"* @_ZNSt3__26vectorI24Baselib_Memory_PageStateNS_9allocatorIS1_EEEC2ESt16initializer_listIS1_E(%"class.std::__2::vector.6"* @_ZGR39Baselib_Test_Memory_SupportedPageStates_, %"class.std::initializer_list"* byval(%"class.std::initializer_list") align 4 %1), !dbg !3800
  %8 = call i32 @__cxa_atexit(
            void (i8*)* bitcast (%"class.std::__2::vector.6"* (%"class.std::__2::vector.6"*)* @_ZNSt3__26vectorI24Baselib_Memory_PageStateNS_9allocatorIS1_EEED2Ev to void (i8*)*),
            i8* bitcast (%"class.std::__2::vector.6"* @_ZGR39Baselib_Test_Memory_SupportedPageStates_ to i8*),
            i8* @__dso_handle) #4, !dbg !3800  store %"class.std::__2::vector.6"* @_ZGR39Baselib_Test_Memory_SupportedPageStates_, %"class.std::__2::vector.6"** @Baselib_Test_Memory_SupportedPageStates, align 4, !dbg !3800
  ret void, !dbg !3801
}

view this post on Zulip Dan Gohman (Apr 29 2020 at 04:15):

It looks like it might be related to global reference temporaries

view this post on Zulip Dan Gohman (Apr 29 2020 at 04:19):

struct B { B(); ~B(); };
namespace test {
  const B b1 = B();
  const B &b2 = B();
}

shows the problem

view this post on Zulip Vladimir Vukicevic (Apr 29 2020 at 04:31):

@Dan Gohman yep. I literally just tracked it down to here: https://github.com/llvm-mirror/clang/blob/master/lib/CodeGen/CGExpr.cpp#L347

Mirror kept for legacy. Moved to https://github.com/llvm/llvm-project - llvm-mirror/clang

view this post on Zulip Vladimir Vukicevic (Apr 29 2020 at 04:32):

That call doesn't go through all the same code that happens https://github.com/llvm-mirror/clang/blob/master/lib/CodeGen/CGDeclCXX.cpp#L143

Mirror kept for legacy. Moved to https://github.com/llvm/llvm-project - llvm-mirror/clang

view this post on Zulip Vladimir Vukicevic (Apr 29 2020 at 04:34):

@Dan Gohman can you fix, or want me to give it a go?

view this post on Zulip Dan Gohman (Apr 29 2020 at 04:36):

I'm not at a computer where I can debug clang easily, so if yyou want to have a go, go for ito

view this post on Zulip Vladimir Vukicevic (Apr 29 2020 at 04:37):

I may poke you tomorrow for help in actually getting a patch submitted :put_litter_in_its_place:

view this post on Zulip Vladimir Vukicevic (Apr 29 2020 at 04:42):

@Dan Gohman I take it back. I don't have anywhere near enough context to do this :)

view this post on Zulip Vladimir Vukicevic (Apr 29 2020 at 04:51):

Yeah definitely can't do this, not without spending a large chunk of time on it that I wish I had. Let me know if I should file it somewhere.

view this post on Zulip Vladimir Vukicevic (Apr 29 2020 at 04:58):

And armed with all this info, I see this code in our tests:

const std::vector<Baselib_Memory_PageState>& Baselib_Test_Memory_SupportedPageStates = { Baselib_Memory_PageState_ReadWrite };

which, if I explicitly declare a vector with the right value and then assign that to the reference, the problem goes away.

view this post on Zulip Dan Gohman (Apr 29 2020 at 05:12):

Alternatively, could you just remove the & there and create a regular global instead of a global reference to a temporary?

view this post on Zulip Vladimir Vukicevic (Apr 29 2020 at 05:14):

not in this case, this is a thing that's an extern elsewhere and it's a per-platform vector where most other platforms are initialized from a global vector

view this post on Zulip Vladimir Vukicevic (Apr 29 2020 at 05:15):

also, hooray for tracking down something that likely would have been a pain in the butt at some random point later on :)

view this post on Zulip Dan Gohman (Apr 29 2020 at 05:15):

Looking at the code in clang, I have an idea of what might be wrong, but I'll need to debug more to confirm. But I'll have to pick this up tomorrow.

view this post on Zulip Vladimir Vukicevic (Apr 29 2020 at 05:15):

Yep, no problem. Thanks for looking into it!

view this post on Zulip Vladimir Vukicevic (Apr 30 2020 at 14:55):

@Dan Gohman any luck on this? (I was out yesterday)

view this post on Zulip Dan Gohman (Apr 30 2020 at 14:56):

Not yet; I got busy with other things yesterday, but I'm going to take another look soon

view this post on Zulip Vladimir Vukicevic (Apr 30 2020 at 14:56):

np, thanks!

view this post on Zulip Dan Gohman (Apr 30 2020 at 20:20):

The dtor bitcast is coming from here:

view this post on Zulip Dan Gohman (Apr 30 2020 at 20:20):

https://github.com/llvm/llvm-project/blob/master/clang/lib/CodeGen/ItaniumCXXABI.cpp#L2403

The LLVM Project is a collection of modular and reusable compiler and toolchain technologies. Note: the repository does not accept github pull requests at this moment. Please submit your patches at...

view this post on Zulip Dan Gohman (Apr 30 2020 at 20:20):

So one option is to make it call EmitDeclDestroy, or factor out code from EmitDeclDestroy for it to use.

view this post on Zulip Dan Gohman (Apr 30 2020 at 20:21):

That would let it use a __cxx_global_array_dtor wrapper, which should fix the problem.

view this post on Zulip Dan Gohman (Apr 30 2020 at 20:21):

There's also a plausible workaround:

view this post on Zulip Dan Gohman (Apr 30 2020 at 20:21):

It turns out we already have code in the LLVM wasm backend which attempts to emulate function pointer casts by inserting wrapper functions.

view this post on Zulip Dan Gohman (Apr 30 2020 at 20:21):

The only reason that code isn't saving us here is that it's looking for casted functions which are immediately called,

view this post on Zulip Dan Gohman (Apr 30 2020 at 20:21):

and here we have a function being casted and passed as an argument to __cxa_atexit.

view this post on Zulip Dan Gohman (Apr 30 2020 at 20:21):

So we could tweak this:

view this post on Zulip Dan Gohman (Apr 30 2020 at 20:22):

https://github.com/llvm/llvm-project/blob/master/llvm/lib/Target/WebAssembly/WebAssemblyFixFunctionBitcasts.cpp#L75

The LLVM Project is a collection of modular and reusable compiler and toolchain technologies. Note: the repository does not accept github pull requests at this moment. Please submit your patches at...

view this post on Zulip Dan Gohman (Apr 30 2020 at 20:22):

to recognize uses which are "the first operand of a __cxa_atexit call"

view this post on Zulip Dan Gohman (Apr 30 2020 at 20:22):

which would then cause it to insert a wrapper, which ought to generate working code.

view this post on Zulip Vladimir Vukicevic (May 01 2020 at 17:21):

It seems like both paths to __cxa_atexit should go through something common

view this post on Zulip Vladimir Vukicevic (May 01 2020 at 17:24):

But maybe not. Either one of your suggestions seems pretty reasonable, but fixing it in a non-WASM specific way seems better because this is technically a problem on any ABI that can't call mismatched functions. Which is probably just wasm, but still.

view this post on Zulip Vladimir Vukicevic (May 06 2020 at 19:07):

@Dan Gohman any luck on this? (Or, alternatively, is there a bug filed I can track -- or do you want me to file one somewhere? :)

view this post on Zulip Dan Gohman (May 06 2020 at 19:22):

No, I've not made any further progress yet.

view this post on Zulip Dan Gohman (May 06 2020 at 19:23):

I poked at it a bit, but I didn't see any super easy paths here. If you wanted to file a bug with the small testcase above and the findings about it, that'd be great!

view this post on Zulip Vladimir Vukicevic (May 07 2020 at 21:36):

Where's the best place to file the bug? Against wasi-sdk or llvm?

view this post on Zulip Jakub Konka (May 07 2020 at 21:38):

For wasi-sdk, that'd be here: https://github.com/webassembly/wasi-sdk/issues

WASI-enabled WebAssembly C/C++ toolchain. Contribute to WebAssembly/wasi-sdk development by creating an account on GitHub.

view this post on Zulip Vladimir Vukicevic (May 07 2020 at 21:56):

Yeah, the issue is a llvm bug though for the wasm backend; wasn't sure if filing that in wasi-sdk was the right place

view this post on Zulip Dan Gohman (May 07 2020 at 22:26):

For clang/llvm bugs, use the llvm tracker: https://bugs.llvm.org/

view this post on Zulip Vladimir Vukicevic (May 11 2020 at 16:44):

Filed https://bugs.llvm.org/show_bug.cgi?id=45876


Last updated: Jan 24 2025 at 00:11 UTC