<html><head></head><body style="word-wrap: break-word; -webkit-nbsp-mode: space; -webkit-line-break: after-white-space; ">Hi Paul,<div><br></div><div>On 2009-12-04, at 09:34, Paul Melis wrote:</div><div><div><br class="Apple-interchange-newline"><blockquote type="cite"><div>I hope you don't mind me sending this mail directly to you (instead of to llvm-dev), but you seem to be the expert on LLVM's GC support :) If you'd rather have me send to llvm-dev, please say so.<br></div></blockquote><div><br></div><div>You'll reach a wider audience with the list, though I haven't been able to keep up with it lately.</div><div><br></div><blockquote type="cite"><div>I'm trying to get a very simple copying collector to work with LLVM, basically your standard semi-space collector with Cheney scan, using the included shadow stack support to track roots. I'm writing everything in C, using llvm-gcc's gcroot attribute in a test app to mark GC roots where appropriate, which seems to work fine when translating to LLVM IR (and finally an executable).<br></div></blockquote><div><br></div><div>Be forewarned that this attribute should be considered ‘experimental’ at best.</div><div><br></div><blockquote type="cite"><div>Implementing a GC is actually quite new to me, so perhaps that's where I'm getting confused. Anyways, suppose I have a function that has a pointer to a GC-tracked object as a parameter and a collection can be triggered from this function, e.g.<br><br>object*<br>f(object *some_obj)<br>{<br> object __attribute__((gcroot)) *new_obj;<br> new_obj = allocate_object();<br> // A collection could have been executed at this point,<br> // and so some_obj might have become invalid<br> new_obj->child = some_obj;<br> return new_obj;<br>}<br></div></blockquote><div><br></div><div>To ensure that *some_obj isn't collected prematurely, your code will need to copy the value of some_obj into a gcroot local at the top of the function body.</div><div><br></div><div>You also need to beware of temporaries:</div><div><br></div></div></div><blockquote class="webkit-indent-blockquote" style="margin: 0 0 0 40px; border: none; padding: 0px;"><font class="Apple-style-span" face="'Courier New'" size="3"><span class="Apple-style-span" style="font-size: 12px;">object *f();<br>object *g();<br>void h(object *x, object *y);<br><br></span></font></blockquote><blockquote class="webkit-indent-blockquote" style="margin: 0 0 0 40px; border: none; padding: 0px;"><font class="Apple-style-span" face="'Courier New'" size="3"><span class="Apple-style-span" style="font-size: 12px;">void func() {</span></font></blockquote><blockquote class="webkit-indent-blockquote" style="margin: 0 0 0 40px; border: none; padding: 0px;"><font class="Apple-style-span" face="'Courier New'" size="3"><span class="Apple-style-span" style="font-size: 12px;"> // dangerous<br> object __attribute((gcroot)) *result1 = h(f(), <span class="Apple-style-span" style="font-size: medium; ">g());</span></span></font></blockquote><blockquote class="webkit-indent-blockquote" style="margin: 0 0 0 40px; border: none; padding: 0px;"><font class="Apple-style-span" face="'Courier New'"><br></font></blockquote><blockquote class="webkit-indent-blockquote" style="margin-top: 0px; margin-right: 0px; margin-bottom: 0px; margin-left: 40px; border-top-style: none; border-right-style: none; border-bottom-style: none; border-left-style: none; border-width: initial; border-color: initial; padding-top: 0px; padding-right: 0px; padding-bottom: 0px; padding-left: 0px; "><font class="Apple-style-span" face="'Courier New'" size="3"><span class="Apple-style-span" style="font-size: 12px; "> // safer</span></font></blockquote><blockquote class="webkit-indent-blockquote" style="margin-top: 0px; margin-right: 0px; margin-bottom: 0px; margin-left: 40px; border-top-style: none; border-right-style: none; border-bottom-style: none; border-left-style: none; border-width: initial; border-color: initial; padding-top: 0px; padding-right: 0px; padding-bottom: 0px; padding-left: 0px; "><font class="Apple-style-span" face="'Courier New'" size="3"><span class="Apple-style-span" style="font-size: 12px; "> object __attribute((gcroot)) *temp1 = f();<br> object __attribute((gcroot)) *temp2 = g();</span></font></blockquote><blockquote class="webkit-indent-blockquote" style="margin-top: 0px; margin-right: 0px; margin-bottom: 0px; margin-left: 40px; border-top-style: none; border-right-style: none; border-bottom-style: none; border-left-style: none; border-width: initial; border-color: initial; padding-top: 0px; padding-right: 0px; padding-bottom: 0px; padding-left: 0px; "><font class="Apple-style-span" face="'Courier New'" size="3"><span class="Apple-style-span" style="font-size: 12px; "> object __attribute((gcroot)) *result2 = h(temp1, temp2<span class="Apple-style-span" style="font-size: medium; ">);</span></span></font></blockquote><blockquote class="webkit-indent-blockquote" style="margin: 0 0 0 40px; border: none; padding: 0px;"><font class="Apple-style-span" face="'Courier New'">}</font></blockquote><div><div><div><div><br></div></div><blockquote type="cite"><div>How can I ensure that the pointer parameter (or any other root) is updated during the collection phase? It seems that LLVM's shadow stack data structures only provide the root objects pointed to and not the memory locations of the pointer variables holding the roots.<br></div></blockquote></div><br></div><div><div>The shadow stack structure is itself the storage location for the variables; the LLVM IR is rearranged to read the stack roots directly from the shadow stack.</div><div><br></div></div><div><div>— Gordon</div><div><br></div></div></body></html>