Stream: cranelift

Topic: ✔ Tricks with linkage internal / module external functions?


view this post on Zulip Adam Wick (Jan 06 2023 at 18:45):

Are there any tricks to declaring functions that are internal to the total linkage unit, but external to the current module? In particular, I'm trying to define a reference to a function that I'm going to declare in a different object file, but link statically.

It looks like cranelift is emitting full dynamic library calls / relocation info no matter what I do (I've tried declaring with Linkage::Import, and Linkage::Preemptible with a fake body), which gets me a linker warning on Linux and a bus error on MacOS.

Since the MacOS error is slightly more illuminating, I'll say that it pops up in dyld:

(lldb) bt
* thread #1, stop reason = EXC_BAD_ACCESS (code=2, address=0x100003f50)
  * frame #0: 0x00000001b096eb7c dyld`dyld4::fixupPage64(void*, mwl_info_hdr const*, dyld_chained_starts_in_segment const*, unsigned int, bool) + 132
    frame #1: 0x00000001b096e924 dyld`dyld4::dyld_map_with_linking_np(mwl_region const*, unsigned int, mwl_info_hdr const*, unsigned int) + 528
    frame #2: 0x00000001b096adc8 dyld`dyld4::setUpPageInLinkingRegions(dyld4::RuntimeState&, dyld4::Loader const*, unsigned long, unsigned short, unsigned short, bool, dyld3::Array<dyld4::PageInLinkingRange> const&, dyld3::Array<void const*> const&) + 832
    frame #3: 0x00000001b096a718 dyld`invocation function for block in dyld4::Loader::setUpPageInLinking(Diagnostics&, dyld4::RuntimeState&, unsigned long, unsigned long long, dyld3::Array<void const*> const&) const + 380
    frame #4: 0x00000001b096a48c dyld`dyld4::Loader::setUpPageInLinking(Diagnostics&, dyld4::RuntimeState&, unsigned long, unsigned long long, dyld3::Array<void const*> const&) const + 536
    frame #5: 0x00000001b096af10 dyld`dyld4::Loader::applyFixupsGeneric(Diagnostics&, dyld4::RuntimeState&, unsigned long long, dyld3::Array<void const*> const&, dyld3::Array<void const*> const&, bool, dyld3::Array<dyld4::Loader::MissingFlatLazySymbol> const&) const + 152
    frame #6: 0x00000001b09709ec dyld`dyld4::JustInTimeLoader::applyFixups(Diagnostics&, dyld4::RuntimeState&, dyld4::DyldCacheDataConstLazyScopedWriter&, bool) const + 680
    frame #7: 0x00000001b09548d0 dyld`dyld4::prepare(dyld4::APIs&, dyld3::MachOAnalyzer const*) + 2212
    frame #8: 0x00000001b0953dc4 dyld`start + 2404

The fault address (0x100003f50) is where I'd expect the call to be:

(lldb) disassemble -n gogogo
test`gogogo:
    0x100003f40 <+0>:  stp    x29, x30, [sp, #-0x10]!
    0x100003f44 <+4>:  mov    x29, sp
    0x100003f48 <+8>:  ldr    x0, #0x8                  ; <+16>
    0x100003f4c <+12>: b      0x100003f58               ; <+24>
    0x100003f50 <+16>: udf    #0x3f91
    0x100003f54 <+20>: .long  0x00280000                ; unknown opcode
    0x100003f58 <+24>: mov    x1, #0x4
    0x100003f5c <+28>: ldr    x3, #0x8                  ; <+36>
    0x100003f60 <+32>: b      0x100003f6c               ; <+44>
    0x100003f64 <+36>: udf    #0x3e80
    0x100003f68 <+40>: udf    #0x0
    0x100003f6c <+44>: blr    x3
    0x100003f70 <+48>: ldp    x29, x30, [sp], #0x10
    0x100003f74 <+52>: ret

I'm going to assume that the bus error is a write to read-only memory. (Although I'm curious what it would've written.) Also, for reference, the warning I get on Linux is:

/usr/bin/ld: output.o: warning: relocation in read-only section `.text'
/usr/bin/ld: warning: creating DT_TEXTREL in a PIE

Although the resultant binary does run.

This is a "simplified" version of what I'm running: https://github.com/acw/cranelift-hmmm
(makedo.sh will get you direct to lldb)

a cranelift example. Contribute to acw/cranelift-hmmm development by creating an account on GitHub.

view this post on Zulip bjorn3 (Jan 06 2023 at 18:53):

Try setting the is_pic flag when creating the target isa.

view this post on Zulip bjorn3 (Jan 06 2023 at 18:57):

Are there any tricks to declaring functions that are internal to the total linkage unit, but external to the current module?

Hidden visibility is not yet supported, but for PIE executables that shouldn't matter with respect to relocations as in PIE executables everything is handled as if the protected visibility is used, which prevents overriding of symbols by other DSO's. The problem here is that non-position independent code can't be linked into a position independent executable, so you need to tell cranelift to build position independent code using is_pic.

view this post on Zulip Adam Wick (Jan 06 2023 at 19:13):

bjorn3 said:

Try setting the is_pic flag when creating the target isa.

Interesting. That seems to fix the linker warning on Linux, but doesn't fix the bus error on macOS.

Diff for reference.

a cranelift example. Contribute to acw/cranelift-hmmm development by creating an account on GitHub.

view this post on Zulip bjorn3 (Jan 06 2023 at 19:33):

Can you show the output of objdump -dr output.o on macOS?

view this post on Zulip bjorn3 (Jan 06 2023 at 19:34):

(arguments may be slightly different on macOS. basically I want a disassembly (-d) with relocations (-r))

view this post on Zulip Adam Wick (Jan 06 2023 at 19:48):

bjorn3 said:

Can you show the output of objdump -dr output.o on macOS?

Oh, interesting:

% objdump -dr output.o

output.o:       file format mach-o arm64

Disassembly of section __TEXT,__text:

0000000000000020 <_gogogo>:
      20: fd 7b bf a9   stp     x29, x30, [sp, #-16]!
      24: fd 03 00 91   mov     x29, sp
      28: 40 00 00 58   ldr     x0, 0x30 <_gogogo+0x10>
      2c: 03 00 00 14   b       0x38 <_gogogo+0x18>
      30: 00 00 00 00   udf     #0
                0000000000000030:  ARM64_RELOC_UNSIGNED _variable-name-x
      34: 00 00 00 00   udf     #0
      38: 81 00 80 d2   mov     x1, #4
      3c: 43 00 00 58   ldr     x3, 0x44 <_gogogo+0x24>
      40: 03 00 00 14   b       0x4c <_gogogo+0x2c>
      44: 00 00 00 00   udf     #0
                0000000000000044:  ARM64_RELOC_UNSIGNED _print
      48: 00 00 00 00   udf     #0
      4c: 60 00 3f d6   blr     x3
      50: fd 7b c1 a8   ldp     x29, x30, [sp], #16
      54: c0 03 5f d6   ret

looks like the failure is actually do to a relocation from the string I'm declaring? hmmm.

view this post on Zulip bjorn3 (Jan 06 2023 at 22:15):

I think that is just because it is the first relocation. Neither should use an absolute ARM64_RELOC_UNSIGNED relocation (which has to be resolved by the dynamic linker), but instead use a pc relative relocation that gets resolved during linking.

view this post on Zulip bjorn3 (Jan 06 2023 at 22:16):

Could you please double check that you used the is_pic change on macOS too? If you did indeed use it, I don't understand why absolute relocations are used in the text segment.

view this post on Zulip bjorn3 (Jan 06 2023 at 22:19):

Apart from is_pic, my cranelift based backend for rustc doesn't set any flags or anything that could affect what kind of relocations are used, yet it works just fine on macOS.

view this post on Zulip Adam Wick (Jan 07 2023 at 03:37):

bjorn3 said:

Could you please double check that you used the is_pic change on macOS too? If you did indeed use it, I don't understand why absolute relocations are used in the text segment.

Yup, it's set. Ran .set("is_pic", "true") on the builder, and then .is_pic() on the flag returns true afterwards. It appears to be a MacOS thing, specifically, as the Linux build appears to be PC relative in both cases:

0000000000000000 <gogogo>:
   0:   55                      push   %rbp
   1:   48 89 e5                mov    %rsp,%rbp
   4:   48 8b 3d 00 00 00 00    mov    0x0(%rip),%rdi        # b <gogogo+0xb>
                        7: R_X86_64_GOTPCREL    variable-name-x-0x4
   b:   be 04 00 00 00          mov    $0x4,%esi
  10:   48 8b 0d 00 00 00 00    mov    0x0(%rip),%rcx        # 17 <gogogo+0x17>
                        13: R_X86_64_GOTPCREL   print-0x4
  17:   ff d1                   call   *%rcx
  19:   48 89 ec                mov    %rbp,%rsp
  1c:   5d                      pop    %rbp
  1d:   c3                      ret

Full source code that I'm using, if anyone sees anything odd. (Apologies for it being a trace long.)

a cranelift example. Contribute to acw/cranelift-hmmm development by creating an account on GitHub.

view this post on Zulip bjorn3 (Jan 07 2023 at 10:47):

Looks like the AArch64 backend doesn't support emitting pc-relative GOT relocations when trying to get the address of an external symbol.

view this post on Zulip bjorn3 (Jan 07 2023 at 10:53):

@Adam Wick Opened https://github.com/bytecodealliance/wasmtime/issues/5544.

.clif Test Case Any code which uses global_value or a call to a function not marked as colocated. Steps to Reproduce Compile with is_pic for AArch64 Expected Results Object file has a GOT relocatio...

view this post on Zulip Afonso Bordado (Jan 08 2023 at 16:37):

:wave: Hey @Adam Wick ,

I've been working on adding pc relative GOT relocations, could you try your code with this branch of cranelift and let me know if it works?

I tested it on AArch64 Linux and it seems to work, not sure if we need anything else for MacOS.

Standalone JIT-style runtime for WebAssembly, using Cranelift - GitHub - afonso360/wasmtime at aarch64-got

view this post on Zulip Afonso Bordado (Jan 08 2023 at 17:29):

hmm, just noticed I only had implemented the ELF relocations but not the MachO ones, when I added them it now crashes in the object crate

view this post on Zulip Afonso Bordado (Jan 08 2023 at 17:56):

It looks like we are in luck and the upstream object crate already supports this! I was able to compile it for MachO and dump it with llvm-objdump after updating it. Let me know if it works for you.

view this post on Zulip Adam Wick (Jan 10 2023 at 20:08):

Afonso Bordado said:

It looks like we are in luck and the upstream object crate already supports this! I was able to compile it for MachO and dump it with llvm-objdump after updating it. Let me know if it works for you.

Hey, just going to confirm in thread (just in case someone finds this via search) that Afonso's fix in PR 5550 resolves this issue.

👋 Hey, This PR adds support for using the GOT in PIC mode on AArch64. It implements the relocations for both ELF and MachO. In order to support the MachO relocations we need to update the object cr...

view this post on Zulip Notification Bot (Jan 10 2023 at 20:09):

Adam Wick has marked this topic as resolved.


Last updated: Jan 24 2025 at 00:11 UTC