<html><head><meta http-equiv="Content-Type" content="text/html charset=utf-8"></head><body style="word-wrap: break-word; -webkit-nbsp-mode: space; -webkit-line-break: after-white-space;" class=""><div class="">The code in X86TargetLowering::IsEligibleForTailCallOptimization() has this part:</div><div class=""><br class=""></div><div class=""><div class=""> // The callee has to preserve all registers the caller needs to preserve.</div><div class=""> const X86RegisterInfo *TRI = Subtarget.getRegisterInfo();</div><div class=""> const uint32_t *CallerPreserved = TRI->getCallPreservedMask(MF, CallerCC);</div><div class=""> if (!CCMatch) {</div><div class=""> const uint32_t *CalleePreserved = TRI->getCallPreservedMask(MF, CalleeCC);</div><div class=""> if (!TRI->regmaskSubsetEqual(CallerPreserved, CalleePreserved))</div><div class=""> return false;</div><div class=""> }</div></div><div class=""><br class=""></div><div class="">which usually checks that this is fine. Maybe that code looks at the regmask of the calling convention rather than the new regmask computed by IPRA?</div><div class=""><br class=""></div><div class="">- Matthias</div><div class=""><br class=""></div><div class="">On Aug 4, 2016, at 9:22 PM, vivek pandya via llvm-dev <<a href="mailto:llvm-dev@lists.llvm.org" class="">llvm-dev@lists.llvm.org</a>> wrote:</div><div><blockquote type="cite" class=""><br class="Apple-interchange-newline"><div class=""><div dir="ltr" class="">Hello all,<div class=""><br class=""></div><div class="">Adding MatzeB this may be interesting for him.</div><div class=""><br class=""></div><div class="">I have tried out following way to save most of the registers for cold functions, in RegisterUsagePropagation.cpp</div><div class=""><br class=""></div><div class=""><div class="">if (PSI->isColdFunction(F) && F->doesNotAccessMemory() && !F->hasLocalLinkage()) {</div><div class=""> dbgs() << "Cold Function : " << F->getName() << "\n";</div><div class=""> F->setCallingConv(CallingConv::CXX_FAST_TLS);</div><div class=""> }</div></div><div class=""><br class=""></div><div class="">previously I was using CallingConv::PreserveMost but it also saves RAX and that generated bug for functions which returns address to global objects or some thing similar. CXX_FAST_TLS is very similar to PreserveMost but it excludes RAX and </div><div class=""><div class="gmail_extra">RDI. It also excludes XMM*. There is also one more interesting bug ( actually it is very similar to what we faced while optimizing function for not saving registers)</div><div class="gmail_extra"><br class=""></div><div class="gmail_extra">See in SPASS application there is a very simple function :</div><div class="gmail_extra"><div class="gmail_extra"><br class=""></div><div class="gmail_extra"><span class="" style="white-space:pre"> </span>BOOL opts_IdIsNull(OPTID Id)</div><div class="gmail_extra">{</div><div class="gmail_extra"> return opts_IdEqual(opts_IdNull(), Id);</div><div class="gmail_extra">}</div><div class="gmail_extra"><br class=""></div><div class="gmail_extra">This is cold function for a particular input and that is why it is getting selected for the optimization. But with CXX_FAST_TLS it is getting generated as follows :</div><div class="gmail_extra"><br class=""></div><div class="gmail_extra"><div class="gmail_extra">_opts_IdIsNull: ## @opts_IdIsNull</div><div class="gmail_extra"><span class="" style="white-space:pre"> </span>.cfi_startproc</div><div class="gmail_extra">## BB#0: ## %entry</div><div class="gmail_extra"><span class="" style="white-space:pre"> </span>pushq<span class="" style="white-space:pre"> </span>%rbp</div><div class="gmail_extra">Ltmp9:</div><div class="gmail_extra"><span class="" style="white-space:pre"> </span>.cfi_def_cfa_offset 16</div><div class="gmail_extra">Ltmp10:</div><div class="gmail_extra"><span class="" style="white-space:pre"> </span>.cfi_offset %rbp, -16</div><div class="gmail_extra"><span class="" style="white-space:pre"> </span>movq<span class="" style="white-space:pre"> </span>%rsp, %rbp</div><div class="gmail_extra">Ltmp11:</div><div class="gmail_extra"><span class="" style="white-space:pre"> </span>.cfi_def_cfa_register %rbp</div><div class="gmail_extra"><span class="" style="white-space:pre"> </span>pushq<span class="" style="white-space:pre"> </span>%rsi</div><div class="gmail_extra"><span class="" style="white-space:pre"> </span>pushq<span class="" style="white-space:pre"> </span>%rcx</div><div class="gmail_extra">Ltmp12:</div><div class="gmail_extra"><span class="" style="white-space:pre"> </span>.cfi_offset %rcx, -32</div><div class="gmail_extra">Ltmp13:</div><div class="gmail_extra"><span class="" style="white-space:pre"> </span>.cfi_offset %rsi, -24</div><div class="gmail_extra"><span class="" style="white-space:pre"> </span>movl<span class="" style="white-space:pre"> </span>%edi, %eax</div><div class="gmail_extra"><span class="" style="white-space:pre"> </span>movl<span class="" style="white-space:pre"> </span>$-1, %edi</div><div class="gmail_extra"><span class="" style="white-space:pre"> </span>popq<span class="" style="white-space:pre"> </span>%rcx</div><div class="gmail_extra"><span class="" style="white-space:pre"> </span>popq<span class="" style="white-space:pre"> </span>%rsi</div><div class="gmail_extra"><span class="" style="white-space:pre"> </span>popq<span class="" style="white-space:pre"> </span>%rbp</div><div class="gmail_extra"><span class="" style="white-space:pre"> </span>jmp<span class="" style="white-space:pre"> </span>_opts_IdEqual ## TAILCALL</div><div class="gmail_extra"><span class="" style="white-space:pre"> </span>.cfi_endproc</div></div><div class="gmail_extra"><br class=""></div><div class="gmail_extra">Here before tailcall due to odl values are returned back result is generated wrong. </div><div class="gmail_extra">I am also confused why rcx and rsi is saved/restored ? Is this normal ?</div><div class="gmail_extra">Any suggestion to handle such case? </div><div class="gmail_extra"><br class=""></div><div class="gmail_extra">Also if we do not want to use a CC directly then can you help me find proper place such that we can iterate over parameter list (also how to iterate over only register which are used for params only?) and do not put this in regMask and also not have save/restore for them. Some hints will speed up my work.</div><div class="gmail_extra"><br class=""></div><div class="gmail_extra">Sincerely,</div><div class="gmail_extra">Vivek</div><div class="gmail_extra"><br class=""></div><div class="gmail_extra"><br class=""></div><div class="gmail_extra"><br class=""></div></div><div class="gmail_extra"><div class="gmail_quote">On Fri, Jul 29, 2016 at 11:06 PM, Hal Finkel <span dir="ltr" class=""><<a href="mailto:hfinkel@anl.gov" target="_blank" class="">hfinkel@anl.gov</a>></span> wrote:<br class=""><blockquote class="gmail_quote" style="margin:0px 0px 0px 0.8ex;border-left-width:1px;border-left-color:rgb(204,204,204);border-left-style:solid;padding-left:1ex"><div class=""><div style="font-family: arial, helvetica, sans-serif; font-size: 10pt;" class=""><br class=""><hr class=""><blockquote style="border-left-width: 2px; border-left-style: solid; border-left-color: rgb(16, 16, 255); margin-left: 5px; padding-left: 5px; font-weight: normal; font-style: normal; text-decoration: none; font-family: Helvetica, Arial, sans-serif; font-size: 12pt;" class=""><b class="">From: </b>"vivek pandya" <<a href="mailto:vivekvpandya@gmail.com" target="_blank" class="">vivekvpandya@gmail.com</a>><br class=""><b class="">To: </b>"Hal Finkel" <<a href="mailto:hfinkel@anl.gov" target="_blank" class="">hfinkel@anl.gov</a>><br class=""><b class="">Cc: </b>"llvm-dev" <<a href="mailto:llvm-dev@lists.llvm.org" target="_blank" class="">llvm-dev@lists.llvm.org</a>>, "Quentin Colombet" <<a href="mailto:qcolombet@apple.com" target="_blank" class="">qcolombet@apple.com</a>>, "Mehdi Amini" <<a href="mailto:mehdi.amini@apple.com" target="_blank" class="">mehdi.amini@apple.com</a>><br class=""><b class="">Sent: </b>Friday, July 29, 2016 5:02:44 AM<span class=""><br class=""><b class="">Subject: </b>Re: A thought to improve IPRA<br class=""><br class=""></span><div dir="ltr" class=""><br class=""><div class="gmail_extra"><br class=""><div class="gmail_quote"><span class="">On Fri, Jul 29, 2016 at 9:01 AM, Hal Finkel <span dir="ltr" class=""><<a href="mailto:hfinkel@anl.gov" target="_blank" class="">hfinkel@anl.gov</a>></span> wrote:<br class=""></span><blockquote class="gmail_quote" style="margin:0px 0px 0px 0.8ex;border-left-width:1px;border-left-style:solid;border-left-color:rgb(204,204,204);padding-left:1ex"><span class=""><hr class=""><div class=""><div class=""><br class="">
> From: "vivek pandya" <<a href="mailto:vivekvpandya@gmail.com" target="_blank" class="">vivekvpandya@gmail.com</a>><br class="">
> To: "Mehdi Amini" <<a href="mailto:mehdi.amini@apple.com" target="_blank" class="">mehdi.amini@apple.com</a>><br class="">
> Cc: "llvm-dev" <<a href="mailto:llvm-dev@lists.llvm.org" target="_blank" class="">llvm-dev@lists.llvm.org</a>>, "Hal Finkel" <<a href="mailto:hfinkel@anl.gov" target="_blank" class="">hfinkel@anl.gov</a>>, "Quentin Colombet" <<a href="mailto:qcolombet@apple.com" target="_blank" class="">qcolombet@apple.com</a>><br class="">
> Sent: Thursday, July 28, 2016 2:59:02 PM<br class="">
> Subject: Re: A thought to improve IPRA<br class="">
><br class="">
><br class="">
> I have been working on PGO driven IPRA and I want to measure if this<br class="">
> help to reduce execution time. So as mentioned earlier the idea is<br class="">
> to make cold function register usage free i.e saving and restoring<br class="">
> all used register by such cold function so caller of that function<br class="">
> will have more free registers. So here I am changing standard callee<br class="">
> saved registers set to a set which will be decided dynamically based<br class="">
> on the actual register usage.<br class="">
><br class="">
> I am facing few problems to get this working:<br class="">
> 1 ) While generating CFI for such function it requires to map Dwarf<br class="">
> register to LLVM register and even if we force LLVM to use Dwarf<br class="">
> register number for CFI then also it will be wrong for some register<br class="">
> for which currently we don't have such mapping for example R8D<br class="">
> register on X86 (when dealing with actual register usage info we may<br class="">
> have such case where R8D is being used)<br class="">
> To fix this I tried to filter the functions which will be optimized<br class="">
> by putting a constraints that it should have attribute NoUnwind but<br class="">
> that does not help. Is it possible to disable CFI generation?<br class="">
<br class="">
</div></div></span><div class=""><div class="">Disabling CFI generation does not seem like the right solution. If the R8D definition, and similar, need DWARF register numbers, then we should fix that (you can try rearranging things and using DwarfRegAlias, or at least for testing, add the same DwarfRegNum as for R8).<br class="">
<span class="">Adding DwarfRegNum as for R8 does not work because the mapping is currently generated as a sorted array on the first value of the key for DwarfLLVMRegPair. So in the build directory in file X86GenRegisterInfo.inc this will add 2 different entries for mapping LLVM Reg to Dwarf number i.e R8 -> 8 and R8D -> 8 but in the array of mapping Dwarf to LLVM Reg there is only entry as it will add 8 only once. </span> </div></div></blockquote><div class=""><div class=""><blockquote class="gmail_quote" style="margin:0px 0px 0px 0.8ex;border-left-width:1px;border-left-style:solid;border-left-color:rgb(204,204,204);padding-left:1ex"><span class="">
><br class="">
><br class="">
> 2) R8D is a 48 bit register<br class="">
<br class="">
</span>Why do you say that? For one thing, it is in a register class GR32 and holds only 32-bit values.<br class=""></blockquote><div class="">Sorry this is my bad R8D is 32 bit value. To get this working changing CC for cold functions to "preserve_all" seems to be easy and safe way. Let me know your thought about this.</div></div></div></div></div></div></blockquote>Sounds like a reasonable thing to try.<span class=""><font color="#888888" class=""><br class=""><br class=""> -Hal</font></span><div class=""><div class=""><br class=""><blockquote style="border-left-width: 2px; border-left-style: solid; border-left-color: rgb(16, 16, 255); margin-left: 5px; padding-left: 5px; font-weight: normal; font-style: normal; text-decoration: none; font-family: Helvetica, Arial, sans-serif; font-size: 12pt;" class=""><div dir="ltr" class=""><div class="gmail_extra"><div class="gmail_quote"><div class=""></div><div class=""><br class=""></div><div class="">-Vivek</div><blockquote class="gmail_quote" style="margin:0px 0px 0px 0.8ex;border-left-width:1px;border-left-style:solid;border-left-color:rgb(204,204,204);padding-left:1ex">
<br class="">
-Hal<br class="">
<div class=""><div class=""><br class="">
> but pushing and popping such register is<br class="">
> not allowed and current implementation for CalleeSaved Register also<br class="">
> uses either 64 bit or 32 bit version of X86 instruction according to<br class="">
> target. So here I think it may be good to push/pop R8 for R8D (i.e I<br class="">
> don't want to change current implementation which inserts MI for<br class="">
> CSR) for that I need to find biggest register for which given<br class="">
> register is alias like R8 has R8D as alias. How can I find that?<br class="">
> I tried to use getMatchingSuperReg(unsigned Reg, unsigned SubIdx,<br class="">
> const TargetRegisterClass *RC) but here I don't know what will be<br class="">
> SubIdx for given Reg in given RC.<br class="">
><br class="">
><br class="">
> So for example if a function which should be optimized for above<br class="">
> optimization is having following set of clobbered registers:<br class="">
> R8D,R8, ECX, EAX, RAX, ESI It should push/pop R8, RCX, RAX, RSI.<br class="">
><br class="">
><br class="">
> Please help!<br class="">
> - Vivek<br class="">
><br class="">
><br class="">
><br class="">
><br class="">
> On Sat, Jul 9, 2016 at 12:26 AM, vivek pandya <<br class="">
> <a href="mailto:vivekvpandya@gmail.com" target="_blank" class="">vivekvpandya@gmail.com</a> > wrote:<br class="">
><br class="">
><br class="">
><br class="">
><br class="">
><br class="">
><br class="">
><br class="">
> On Sat, Jul 9, 2016 at 12:18 AM, Mehdi Amini < <a href="mailto:mehdi.amini@apple.com" target="_blank" class="">mehdi.amini@apple.com</a><br class="">
> > wrote:<br class="">
><br class="">
><br class="">
><br class="">
><br class="">
><br class="">
><br class="">
><br class="">
><br class="">
> On Jul 8, 2016, at 11:41 AM, vivek pandya < <a href="mailto:vivekvpandya@gmail.com" target="_blank" class="">vivekvpandya@gmail.com</a> ><br class="">
> wrote:<br class="">
><br class="">
><br class="">
><br class="">
><br class="">
> On Fri, Jul 8, 2016 at 11:46 PM, Mehdi Amini < <a href="mailto:mehdi.amini@apple.com" target="_blank" class="">mehdi.amini@apple.com</a><br class="">
> > wrote:<br class="">
><br class="">
><br class="">
><br class="">
><br class="">
><br class="">
><br class="">
><br class="">
><br class="">
> On Jul 8, 2016, at 11:12 AM, vivek pandya < <a href="mailto:vivekvpandya@gmail.com" target="_blank" class="">vivekvpandya@gmail.com</a> ><br class="">
> wrote:<br class="">
><br class="">
><br class="">
><br class="">
><br class="">
><br class="">
> Hello LLVM Developers,<br class="">
><br class="">
><br class="">
> I have a thought to improve IPRA and I would like summaries<br class="">
> discussion on IRC regarding that so we can develop an idea out of<br class="">
> that if it really helps.<br class="">
><br class="">
><br class="">
> So idea is to have more callee saved registers at infrequently called<br class="">
> leaf procedures and try provide more registers to procedures which<br class="">
> are in upper region of the call graph. But as pointed out by Quentin<br class="">
> this optimization may help in context of "true" IPRA but in our case<br class="">
> we may not require this. But I think that it can improve performance<br class="">
> in current IPRA. I explain both arguments ( Quentin's and mine) with<br class="">
> following example.<br class="">
><br class="">
><br class="">
> Consider following call sequence A->B->C , here C is very less time<br class="">
> called leaf procedure while A is called frequently and B may call C<br class="">
> based on some condition now while propagating actual register usage<br class="">
> information from C to A we almost clobbered most of the registers so<br class="">
> in this case as per Quentin's point we does not hurt the performance<br class="">
> as we fall back to CC but I think we can improve the performance as<br class="">
> follows:<br class="">
> If we mark every register preserved by C (i.e having more spill<br class="">
> reloads at procedure entry and exit ) and if this can help at A.<br class="">
> Suppose A requires more number of distinct registers than CC can<br class="">
> provide and if not provided it will spill variables to memory. Now<br class="">
> if we can provide more registers at A by having more spills at C<br class="">
> then we can save spill at A which can be beneficial because A is<br class="">
> frequently called but C is less frequently called and thus reducing<br class="">
> total number of spill/restore in program execution.<br class="">
><br class="">
><br class="">
> However again effect of this optimization will be limited by the<br class="">
> scope of current IPRA (i.e one Module only) because we can' really<br class="">
> propagate the details about more callee saved registers to caller<br class="">
> which is defined in other module, but still it may helpful.<br class="">
><br class="">
><br class="">
> Any thoughts on this ?<br class="">
><br class="">
><br class="">
><br class="">
><br class="">
> I think it is interesting, have you considered:<br class="">
><br class="">
><br class="">
> - the code size impact? (C will have a lot of spills)<br class="">
> Yes, this needs to be address with some heuristics based on call<br class="">
> frequency to C and no of clobbers it has. Also can we say that a<br class="">
> function which does not have any kind of call instruction in it's<br class="">
> body will have less clobbers ?<br class="">
><br class="">
><br class="">
> I am not sure what you mean.<br class="">
> A function which may do lots of computation but does not required to<br class="">
> call any other function may not have too many simultaneous live<br class="">
> ranges thus with very few registers it can be compiled.<br class="">
><br class="">
><br class="">
><br class="">
><br class="">
><br class="">
><br class="">
><br class="">
><br class="">
><br class="">
><br class="">
><br class="">
><br class="">
><br class="">
> - what if C is cold but all (most) of its call sites are located in<br class="">
> different modules?<br class="">
> Can we user Uses to get no of call site in current module and based<br class="">
> on that we decide to optimize? Again some heuristics .<br class="">
><br class="">
><br class="">
> Of course, but what I’m mentioning is exactly what does not work with<br class="">
> that.<br class="">
><br class="">
><br class="">
><br class="">
><br class="">
><br class="">
><br class="">
><br class="">
><br class="">
><br class="">
><br class="">
><br class="">
><br class="">
><br class="">
><br class="">
><br class="">
> - an alternative approach where we would break the CGSCC ordering to<br class="">
> codegen B and A before C, so we would be able to spill minimally<br class="">
> when performing the codegen for C?<br class="">
> Do you here mean marking all preserve for C while code gen for B and<br class="">
> then when we come to C (top-down) we may avoid some spills if C can<br class="">
> use regs which are not really used by B?<br class="">
><br class="">
><br class="">
> Yes, but it may be harder to implement for not much gain after all.<br class="">
><br class="">
><br class="">
><br class="">
><br class="">
><br class="">
> Also this can be applied to a function which is less frequently<br class="">
> called and which may not be a leaf function. It may help.<br class="">
><br class="">
><br class="">
> Sure, you can just refer to this as “PGO driven IPRA”.<br class="">
> Ok I will look into this.<br class="">
><br class="">
><br class="">
> Vivek<br class="">
><br class="">
><br class="">
><br class="">
><br class="">
><br class="">
> —<br class="">
> Mehdi<br class="">
><br class="">
><br class="">
><br class="">
><br class="">
<br class="">
</div></div><span class=""><font color="#888888" class="">--<br class="">
Hal Finkel<br class="">
Assistant Computational Scientist<br class="">
Leadership Computing Facility<br class="">
Argonne National Laboratory<br class="">
</font></span></blockquote></div><br class=""></div></div>
</blockquote><br class=""><br class=""><br class="">-- <br class=""><div class=""><span name="x" class=""></span>Hal Finkel<br class="">Assistant Computational Scientist<br class="">Leadership Computing Facility<br class="">Argonne National Laboratory<span name="x" class=""></span><br class=""></div></div></div></div></div></blockquote></div><br class=""></div></div></div>
_______________________________________________<br class="">LLVM Developers mailing list<br class=""><a href="mailto:llvm-dev@lists.llvm.org" class="">llvm-dev@lists.llvm.org</a><br class="">http://lists.llvm.org/cgi-bin/mailman/listinfo/llvm-dev<br class=""></div></blockquote></div><br class=""></body></html>