<div dir="ltr"><div dir="ltr">I think you hit the nail on the head: LLVM will try to use the native stack to spill stuff. It's also going to generate prologues and epilogues to update and restore it.</div><div dir="ltr"><br></div><div dir="ltr">IIRC, GHC attempts to generate code that is almost "stackless": most functions are leaf functions that simply tail call to the next function, so there is nothing to spill and no stack frame. LLVM always assumes it has permission to use the stack for something. Things may work, but I don't think anyone can make any promises.</div><br><div class="gmail_quote"><div dir="ltr" class="gmail_attr">On Fri, Jul 30, 2021 at 9:17 AM Ben Gamari via llvm-dev <<a href="mailto:llvm-dev@lists.llvm.org">llvm-dev@lists.llvm.org</a>> wrote:<br></div><blockquote class="gmail_quote" style="margin:0px 0px 0px 0.8ex;border-left:1px solid rgb(204,204,204);padding-left:1ex">tl;dr. GHC wants to change its calling convention to use the native<br>
       stack pointer register (e.g. $rsp on x86-64) for its (segmented)<br>
       stack. Is this going to be a problem for LLVM?<br>
<br>
<br>
Hi all,<br>
<br>
As you likely know, the Glasgow Haskell Compiler (GHC, [ghc]) has for many<br>
years supported LLVM as a code generation backend. This is facilitated<br>
via an LLVM calling convention (e.g. [ghc-cc]) which has been upstream since<br>
LLVM 3.4 (IIRC). This calling convention matches the convention<br>
used by GHC's internal code generator (the so-called NCG), allowing<br>
object code produced by LLVM to be linked against that produced by the<br>
NCG.<br>
<br>
GHC's LLVM backend implements a simple lowering of C-- (GHC's C-like<br>
imperative IR, built around GHC's STG abstract machine) to LLVM IR. C--<br>
procedures are lowered to LLVM procedures, with the STG machine state<br>
(namely register values) being passsed via arguments. The GHC calling<br>
convention maps the STG machine's registers to physical registers of the<br>
target machine.<br>
<br>
One of these registers (Sp) points to the STG stack, which is a<br>
segmented stack which must be traversable by GHC's garbage collector.<br>
On x86-64 Sp is currently mapped to $rbp. However, recently we have been<br>
considering [native-sp] modifying GHC's calling convention to rather use<br>
the native stack pointer (e.g. $rsp) for Sp, allowing stack-sampling<br>
performance measurement tools like `perf` to be used on Haskell<br>
programs.<br>
<br>
One of the open questions surrounding this change is how it will affect<br>
the LLVM backend. It seems that LLVM makes rather strong assumptions<br>
about the native stack register (e.g. freely using it for spilling).<br>
Moreover, in looking through LLVM's other calling conventions I have yet<br>
to find an example of a convention that uses the native stack pointer.<br>
<br>
So I ask:<br>
<br>
 * Can a TableGen-defined calling-convention in LLVM claim the native<br>
   stack register for use by arguments?<br>
<br>
 * Is there any way to affect LLVM's spilling behavior to spill to a<br>
   stack pointed to be a register other than the native stack pointer<br>
   register?<br>
<br>
Thanks in advance for your help.<br>
<br>
Cheers,<br>
<br>
- Ben<br>
<br>
<br>
<br>
[ghc]: <a href="https://www.haskell.org/ghc/" rel="noreferrer" target="_blank">https://www.haskell.org/ghc/</a><br>
[ghc-cc]: <a href="https://github.com/llvm/llvm-project/blob/8c8dbc10825cf099607af3da58b839e10c68320f/llvm/lib/Target/X86/X86CallingConv.td#L692[native-sp" rel="noreferrer" target="_blank">https://github.com/llvm/llvm-project/blob/8c8dbc10825cf099607af3da58b839e10c68320f/llvm/lib/Target/X86/X86CallingConv.td#L692<br>
[native-sp</a>]: <a href="https://well-typed.com/blog/2021/07/ghc-sp-profiling/" rel="noreferrer" target="_blank">https://well-typed.com/blog/2021/07/ghc-sp-profiling/</a><br>
_______________________________________________<br>
LLVM Developers mailing list<br>
<a href="mailto:llvm-dev@lists.llvm.org" target="_blank">llvm-dev@lists.llvm.org</a><br>
<a href="https://lists.llvm.org/cgi-bin/mailman/listinfo/llvm-dev" rel="noreferrer" target="_blank">https://lists.llvm.org/cgi-bin/mailman/listinfo/llvm-dev</a><br>
</blockquote></div></div>