<div dir="ltr"><div class="gmail_quote"><div dir="ltr">On Thu, Jul 16, 2015 at 12:27 AM Hal Finkel <<a href="mailto:hfinkel@anl.gov">hfinkel@anl.gov</a>> wrote:<br></div><blockquote class="gmail_quote" style="margin:0 0 0 .8ex;border-left:1px #ccc solid;padding-left:1ex">----- Original Message -----<br>
> From: "Chandler Carruth" <<a href="mailto:chandlerc@google.com" target="_blank">chandlerc@google.com</a>><br>
> To: "Hal Finkel" <<a href="mailto:hfinkel@anl.gov" target="_blank">hfinkel@anl.gov</a>>, "LLVM Dev" <<a href="mailto:llvmdev@cs.uiuc.edu" target="_blank">llvmdev@cs.uiuc.edu</a>><br>
> Sent: Thursday, July 16, 2015 1:00:05 AM<br>
> Subject: Re: [LLVMdev] [RFC] Defining Infinite Loops<br>
><br>
><br>
> FWIW, I'm very much in favor of having a firm and clear answer to<br>
> these questions.<br>
><br>
> I also agree that it is an absolute requirement that LLVM have *some*<br>
> mechanism for supporting both languages with defined behavior for<br>
> infinite loops and a language requirement that all loops terminate.<br>
><br>
><br>
> However, I'd like to float an alternative approach. I've not spent a<br>
> lot of time thinking about it, so I'm not sure its actually better.<br>
> I'm wondering if you've already thought about it.<br>
><br>
><br>
> What if we have an @llvm.noop.sideeffect() or some such which doesn't<br>
> read or write memory in any way, but which a frontend can place<br>
> inside a loop body to mark that its execution (perhaps infinitely)<br>
> is in-and-of-itself a side effect of the program. We could then<br>
> teach loop unrolling or the few other things that would care to<br>
> collapse multiple ones into a single one, and not count them towards<br>
> anything.<br>
><br>
><br>
> I know an intrinsic is kind of awkward for this, but it seems like<br>
> the least bad way we have to annotate a loop in a fully generic way.<br>
> I'd somewhat like this to be a property of the *loop* and not of the<br>
> function. And it really needs to be truly generic, handling<br>
> unnatural CFGs and even recursion-loops. My only idea for how to<br>
> accomplish that is an intrinsic to mark the dynamic path which if<br>
> executed infinitely can't be eliminated.<br>
<br>
My largest concern is that the frontend would need to add these things all over the place, not just before the loop backedges. For one thing, if the language has gotos, where should they be inserted?</blockquote><div><br></div><div>The target of every goto.</div><div><br></div><div>For computed goto, very label whose address is taken.</div><div><br></div><div>This at least doesn't seem that bad to me.</div><div> </div><blockquote class="gmail_quote" style="margin:0 0 0 .8ex;border-left:1px #ccc solid;padding-left:1ex"> Before every branch will be conservatively correct, but I'm worried that will unnecessarily block optimizations. They'd also be needed at the entry to every function.</blockquote><div><br></div><div>Only external, address taken, or internal-and-recursively-called functions. All of which we already have some blockers to optimization, so this part i'm not worried about.</div><div> </div><blockquote class="gmail_quote" style="margin:0 0 0 .8ex;border-left:1px #ccc solid;padding-left:1ex"> On the other hand, maybe if we added an optimization that removed these things along any control-flow path that already had any other side effect, it might not be too bad?<br></blockquote><div><br></div><div>Absolutely, much like lower-expect, we'd need to make sure that easy cases were folded quickly in the optimizer so this didn't get out of hand.</div><div> </div><blockquote class="gmail_quote" style="margin:0 0 0 .8ex;border-left:1px #ccc solid;padding-left:1ex">
<br>
><br>
><br>
> As for why I'm particularly interested in this being a property of<br>
> the loop, consider if you were to have a mixture of Java and C++<br>
> code, all compiled to LLVM. How do you inline between them?<br>
><br>
<br>
You add the attribute to the caller.</blockquote><div><br></div><div>This has the really unfortunate side effect of pessimizing code during cross language optimizations.</div><div> </div><blockquote class="gmail_quote" style="margin:0 0 0 .8ex;border-left:1px #ccc solid;padding-left:1ex"> FWIW, I suspect I might care a lot about this particular case (because I believe that Fortran has defined behavior for infinite loops).<br></blockquote><div><br></div><div>Yea, you could argue that C does too, which is one reason why I'm so interested in this being done really well even in an LTO situation.</div><div><br></div><div>I think it would be really useful to not have this cross between adjacent loops after inlining when they come from different source languages, and it would be nice for it to not apply to nested loops when those nested loops were inlined from a language without this guarantee.</div><div><br></div><div>But I'm still not convinced that the noise of the intrinsic is *definitely* worth it. I come from the background of the C++ rules' rationale, and so I naturally see the languages that define this as giving up optimizations and so wanting to localize the impact of that... Not sure that's actually the right perspective though. ;]</div><div> </div><blockquote class="gmail_quote" style="margin:0 0 0 .8ex;border-left:1px #ccc solid;padding-left:1ex">
<br>
 -Hal<br>
<br>
><br>
> Anyways, I've not spent a lot of time thinking about what this might<br>
> break for languages that allow infinite loops. Maybe it doesn't work<br>
> as well as I'd hope.<br>
><br>
><br>
> -Chandler<br>
><br>
><br>
> On Wed, Jul 15, 2015 at 9:16 PM Hal Finkel < <a href="mailto:hfinkel@anl.gov" target="_blank">hfinkel@anl.gov</a> > wrote:<br>
><br>
><br>
> Hello everyone,<br>
><br>
> The topic of whether or not LLVM allows for infinite loops has come<br>
> up a lot recently (several times this week already). Regarding<br>
> motivation, there are two important facts:<br>
><br>
> 1. Some languages, such as Java, have well-defined infinite loops.<br>
> See:<br>
><br>
> <a href="https://urldefense.proofpoint.com/v2/url?u=http-3A__docs.oracle.com_javase_specs_jls_se7_html_jls-2D17.html-23jls-2D17.4.9&d=AwMFaQ&c=8hUWFZcy2Z-Za5rBPlktOQ&r=Mfk2qtn1LTDThVkh6-oGglNfMADXfJdty4_bhmuhMHA&m=eLViOW4bjssEF_xOHiKx6sFEUc6i4VKNWuF0OVDPXOI&s=Kb6aDGhW4iPMrrrPbvDmW2qHt3Tq5vojsDvW4iOlFl0&e=" rel="noreferrer" target="_blank">http://docs.oracle.com/javase/specs/jls/se7/html/jls-17.html#jls-17.4.9</a><br>
><br>
> and:<br>
><br>
> <a href="https://urldefense.proofpoint.com/v2/url?u=http-3A__docs.oracle.com_javase_specs_jls_se7_html_jls-2D17.html-23jls-2D17.4.2&d=AwMFaQ&c=8hUWFZcy2Z-Za5rBPlktOQ&r=Mfk2qtn1LTDThVkh6-oGglNfMADXfJdty4_bhmuhMHA&m=eLViOW4bjssEF_xOHiKx6sFEUc6i4VKNWuF0OVDPXOI&s=RGEOoGCFTcs_zYp4sRk_pkHy1_Z5V_hROu6a3oasEF8&e=" rel="noreferrer" target="_blank">http://docs.oracle.com/javase/specs/jls/se7/html/jls-17.html#jls-17.4.2</a><br>
><br>
> and, as a community, it seems to be important for us to support such<br>
> languages. That means that we must have a way, at the IR level, to<br>
> support and model infinite loops.<br>
><br>
> 2. Other languages, such a C and C++, allow us to assume that<br>
> otherwise-side-effect-free loops terminate, specifically, for C++,<br>
> 1.10p27 says:<br>
><br>
> The implementation may assume that any thread will eventually do one<br>
> of the following:<br>
> - terminate<br>
> - make a call to a library I/O function<br>
> - access or modify a volatile object, or<br>
> - perform a synchronization operation or an atomic operation<br>
><br>
> [Note: This is intended to allow compiler transformations such as<br>
> removal of empty loops, even<br>
> when termination cannot be proven. — end note ]<br>
><br>
> and taking advantage of these guarantees is part of providing a<br>
> high-quality optimizer for C/C++ programs.<br>
><br>
> And this leaves us somewhat in a bind. To provide a high-quality<br>
> C/C++ optimizer, we want to take advantage of this guarantee, but we<br>
> can't do so in a generic sense without harming our ability to serve<br>
> as a compiler for other languages.<br>
><br>
> In 2010, Nick proposed to add a 'halting' attribute that could be<br>
> added to functions to indicate that they would not execute<br>
> indefinitely (<br>
> <a href="http://lists.cs.uiuc.edu/pipermail/llvm-commits/Week-of-Mon-20100705/103670.html" rel="noreferrer" target="_blank">http://lists.cs.uiuc.edu/pipermail/llvm-commits/Week-of-Mon-20100705/103670.html</a><br>
> ). At the time that the patch was proposed, there were<br>
> infrastructure problems with inferring the attribute for functions<br>
> with loops (related to using function-level analysis passes from a<br>
> CGSCC pass), but hopefully those will be fixed with the new pass<br>
> manager. Regardless, however, such inference is much more powerful<br>
> if it can take advantage of the guarantees that C/C++ provide.<br>
><br>
> Thus, while I would eventually like a 'halting' attribute, or some<br>
> variant of that (including, for example, the lack of calls to<br>
> longjmp), I think that a first step is to provide an attribute that<br>
> Clang, and other frontends, can add when producing IR from sources<br>
> where the language provides C/C++-like guarantees on loop<br>
> termination. This attribute would indicate that the function will<br>
> not execute indefinitely without producing some<br>
> externally-observable side effect (calling an external function or<br>
> executing a volatile/atomic memory access). I could name this<br>
> attribute 'finite', but bikeshedding is welcome.<br>
><br>
> With such an attribute in place, we would be able to clarify our<br>
> overall position on infinite loops, be in a stronger position to<br>
> infer more specific function properties (like halting), and can put<br>
> in place generally-correct fixes to outstanding bugs (PR24078, for<br>
> example). I know there are some Clang users who want it to optimize<br>
> while honoring infinite loops, and I think adding this attribute<br>
> helps them as well (assuming we'd provide some non-default option to<br>
> prevent Clang from adding it). Thoughts?<br>
><br>
> Thanks again,<br>
> Hal<br>
><br>
> --<br>
> Hal Finkel<br>
> Assistant Computational Scientist<br>
> Leadership Computing Facility<br>
> Argonne National Laboratory<br>
><br>
> _______________________________________________<br>
> LLVM Developers mailing list<br>
> <a href="mailto:LLVMdev@cs.uiuc.edu" target="_blank">LLVMdev@cs.uiuc.edu</a> <a href="http://llvm.cs.uiuc.edu" rel="noreferrer" target="_blank">http://llvm.cs.uiuc.edu</a><br>
> <a href="http://lists.cs.uiuc.edu/mailman/listinfo/llvmdev" rel="noreferrer" target="_blank">http://lists.cs.uiuc.edu/mailman/listinfo/llvmdev</a><br>
><br>
<br>
--<br>
Hal Finkel<br>
Assistant Computational Scientist<br>
Leadership Computing Facility<br>
Argonne National Laboratory<br>
</blockquote></div></div>