<div dir="ltr"><div><div>If we added unwind target to every potentially throwing instruction (loads, stores, all binary operations), wouldn't all such instructions have to become BB terminators?   I'd expect that CFG would then end up consisting mostly of single-instruction BBs. This can't be good for compilation performance and optimizations...<br><br></div>Another vague idea: what if lifetime.start() returned some kind of a token, which lifetime.end() has to consume?   That would prevent transformations that don't preserve lifetime scopes (such as the one you've described), wouldn't it?</div><div><br></div>Vadim<br><div><div><div><div><div class="gmail_extra"><br><div class="gmail_quote">On Wed, Dec 3, 2014 at 12:55 PM, Reid Kleckner <span dir="ltr"><<a href="mailto:rnk@google.com" target="_blank">rnk@google.com</a>></span> wrote:<br><blockquote class="gmail_quote" style="margin:0px 0px 0px 0.8ex;border-left:1px solid rgb(204,204,204);padding-left:1ex"><div dir="ltr"><div class="gmail_extra"><div class="gmail_quote"><span class="">On Tue, Dec 2, 2014 at 6:05 PM, Vadim Chugunov <span dir="ltr"><<a href="mailto:vadimcn@gmail.com" target="_blank">vadimcn@gmail.com</a>></span> wrote:<br><blockquote class="gmail_quote" style="margin:0px 0px 0px 0.8ex;border-left:1px solid rgb(204,204,204);padding-left:1ex"><div dir="ltr"><div>Sure, but memory access violations are not the only source of asynchronous exceptions.  There are also stack overflows, integer overflows (on platforms that support that in hardware), signaling NaNs, and so on...<br><br></div><div>I am curious, what do you think of the following idea: what if instead of creating landing pads for running destructors, there were intrinsics for marking start and end of the object's lifetime (along with a pointer to destructor)?  LLVM could then emit a table of live objects for each PC range into LSDA, and the personality routine would interpret that table and invoke destructors.  (catch() would still need a landing pad, of course).<br>/end bike-shedding</div></div></blockquote><div><br></div></span><div>I don't think calls to start / end are good enough, because graphs don't have scope. We are seeing problems with lifetime start / end already. Consider a transformation which turns every branch into a branch to a single basic block which switches over all possible branch targets. This is a valid LLVM transformation, even if it is not an optimization, and it would be impossible to recover the natural scope-like information from start / end call pairs.</div><div><br></div><div>I also think that recovering from async exceptions will always be best effort. The kinds of exceptions you describe are essentially results of undefined behavior in LLVM IR, and can't be handled reliably. Unless we introduce specific constructs with defined behavior (trapping integer divide, trapping FP ops, trapping alloca), it will never work.</div><div><br></div><div>Chris Lattner had a proposal from a long time ago to add 'unwind' labels to every basic block, but it introduces a lot of implicit control flow, which we don't like:</div><div><a href="http://nondot.org/sabre/LLVMNotes/ExceptionHandlingChanges.txt" target="_blank">http://nondot.org/sabre/LLVMNotes/ExceptionHandlingChanges.txt</a><br></div><div><br></div><div>You would do this:</div><div>  %p = call i8* malloc(i32 4)</div><div>  %xp = bitcast i8* %p to i32*</div><div>  ...</div><div><br></div><div>mybb: unwind to label %lpad1</div><div>  %x = load i32* %xp  ; edge to lpad1 here</div><div>  store i32 0, i32* %xp ; edge to lpad1 here</div><div>  call void @f() ; edge to lpad1 here</div><div>  br label %mybb2 ; cannot remove branch due to differing lpads</div><div><br></div><div>mybb2: unwind to label %lpad2</div><div>  ...</div><div><br></div><div>lpad:</div><div>  %xx = load i32* %xp ; we cannot make %xx a phi between %x and 0 due to implicit control flow. Maybe we could split mybb and then make the phi, but, ew.</div><div><br></div><div>This is a mountain. I think you can climb it, but I'm *not* signing up for it. :) Adding and invoking intrinsics for all possibly trapping operations seems much more tractable. Simply outlining try bodies is even easier.</div></div></div></div>
</blockquote></div><br></div></div></div></div></div></div>