<html><head><meta http-equiv="Content-Type" content="text/html charset=windows-1252"></head><body style="word-wrap: break-word; -webkit-nbsp-mode: space; -webkit-line-break: after-white-space;"><br><div><div>On Apr 23, 2014, at 11:02 AM, suyog sarda <<a href="mailto:sardask01@gmail.com">sardask01@gmail.com</a>> wrote:</div><br class="Apple-interchange-newline"><blockquote type="cite"><div dir="ltr"><div><div><div><div><div><div><div><div><div><div>Hi all,<br><br></div>Finally !! Some interesting findings from my side on why this issue is happening. I will try to explain my findings with pseudo code :<br>


<br></div>Sample pseudo code and how clang process it :<br><br></div>Consider our code having three class a,b and c<br><br></div><i>main()<br>{<br></i></div><i>    try {<br></i></div><i>           b tmp1();<br></i></div>

<i>           c tmp2(b(a));<br>
     }<br><br></i></div><i>   catch {...}<br>}<br></i></div><br></div><div>A piece of code in clang to focus :<br>
<br><i>void CodeGenFunction::EmitCXXTryStmt(const CXXTryStmt &S) {<br>  EnterCXXTryStmt(S);<br>  EmitStmt(S.getTryBlock());<br>  ExitCXXTryStmt(S);<br>}</i></div><div><i><br></i></div><div>There is an EHStack, which will contain entries which will be popped out to emit cleanup code (destructor) in case of exception happening. </div>

<div><br></div><div>When try-catch block is encountered, we visit <i>'EmitStmt'</i> function, after some more function call we visit following code for emitting code for local variables <i>tmp1</i> and <i>tmp2:</i><i><br>

</i></div><div>
<br><i>void CodeGenFunction::EmitAutoVarDecl(const VarDecl &D) {<br>  AutoVarEmission emission = EmitAutoVarAlloca(D);<br>  EmitAutoVarInit(emission);<br>  EmitAutoVarCleanups(emission);<br>}</i><br></div><div>


<div><div><br></div><div>First we visit <i>EmitAutoVarInit </i>where some initialization happens by visiting inner expression. For <i>tmp1</i> object inner expression is empty, so next function <i>EmitAutoVarCleanup</i> will be called and entry of this object <i>tmp1</i> will be put into EHStack. Since the scope of<i> tmp1</i> is till end of try block, it will remain in EHStack till try block exits.</div>

<div><br></div><div>Next, We visit second object <i>tmp2</i> construction, where inner expression consist of temporaries. We visit function <i>EmitAutoVarInit </i>where we process inner expression. Entries of Inner temporary are added into EHStack till all temporary entries are processed, but as soon as the processing of all temporaries gets over i.e. their scope gets over, these entries are popped out of EHstack and their destructor is called as a part of cleanup code. </div>

<div>Please note we are yet to add entry of <i>tmp2 </i>in EHStack as we are yet to visit <i>EmitAutoVarCleanups</i> function.</div><div><br>Now, The main problem starts - if any of the destructor of temporaries throw, we visit EHStack, pop out each entry from EHStack and emit cleanup code (destructor) for each entry. Note that, the entry of outermost local variable is still not present in the EHStack as we are yet to visit function <i>EmitAutoVarCleanups, </i>as a result of this, its cleanup code (destructor) is not emitted and we land into resource leak!!</div>

<div>
<br></div><div>I just altered the order of function calls - first call <i>EmitAutoVarCleanups </i>and then <i>EmitAutoVarInit - </i>which ensures that entry of auto variables are added in EHStack<i> </i>before we process inner expression<i>. </i>This resolves our original issue 'init-temp1.c', however it introduces 10 regressions all related to location of destructors of temporaries. </div></div></div></div></blockquote><blockquote type="cite"><div dir="ltr"><div><div>

<div><br></div><div>This is a resource leak issue and hence i am pressing more on it (Order in which destructors should be called is not clear, but we should eliminate atleast the resource leak). I am not sure although if my approach to resolve this issue is ok. Basically, we somehow need to ensure that auto object entry gets added in EHStack (may be a call to <i>EmitAutoVarCleanups</i> function) before we pop out entries for temporaries.</div></div></div></div></blockquote><div><br></div>Your proposed change causes the destructor for the auto object to be run if the initializer throws, even if the auto var constructor hasn’t been run yet.</div><div><br></div><div>John.</div></body></html>