<div dir="ltr"><div dir="ltr"><div></div><div>Hi Philip,</div><div><br></div><div>could you explain the downsides of Option 2 more, perhaps with examples? They seem pretty non-obvious to me at a first glance.<br></div></div><div><br></div><div>Naively, I'd argue that programming language semantics tend to always be understood under "as-if" rules, which seems to imply that Option 2 is the right one. If a callee allocates and immediately frees memory, how can the caller even tell?</div><div><br></div><div>Cheers,</div><div>Nicolai<br></div><div><br></div><div class="gmail_quote"><div dir="ltr" class="gmail_attr">On Fri, Apr 9, 2021 at 9:05 PM Philip Reames via llvm-dev <<a href="mailto:llvm-dev@lists.llvm.org">llvm-dev@lists.llvm.org</a>> wrote:<br></div><blockquote class="gmail_quote" style="margin:0px 0px 0px 0.8ex;border-left:1px solid rgb(204,204,204);padding-left:1ex">
<div>
<p>I've stumbled across a case related to the nofree attribute where
we seem to have inconsistent interpretations of the attribute
semantic in tree. I'd like some input from others as to what the
"right" semantic should be.</p>
<p>The basic question is does the presence of nofree prevent the
callee from allocating and freeing memory entirely within it's
dynamic scope? At first, it seems obvious that it does, but that
turns out to be a bit inconsistent with other attributes and leads
to some surprising results. <br>
</p>
<p>For reference in the following discussion, here is the current
wording for the nofree function attribute in LangRef:</p>
<blockquote>
<p>"This function attribute indicates that the function does not,
directly or
indirectly, call a memory-deallocation function (free, for
example). As a
result, uncaptured pointers that are known to be dereferenceable
prior to a
call to a function with the <code class="gmail-notranslate"><span>nofree</span></code> attribute
are still known to be
dereferenceable after the call (the capturing condition is
necessary in
environments where the function might communicate the pointer to
another thread
which then deallocates the memory)."</p>
</blockquote>
<p>For discussion purposes, please assume the concurrency case has
been separately proven. That's not the point I'm getting at here.<br>
</p>
<p>The two possible semantics as I see them are:</p>
<p><b>Option 1</b> - nofree implies no call to free, period</p>
<p>This is the one that to me seems most consistent with the current
wording, but it prevents the callee from allocating storage and
freeing it entirely within it's scope. This is, for instance, a
reasonable thing a target might want to do when lowering large
allocs. This requires transforms to be careful in stripping the
attribute, but isn't entirely horrible.<br>
</p>
<p>The more surprising bit is that it means we can not infer nofree
from readonly or readnone. Why? Because both are specified only
in terms of memory effects visible to the caller. As a result, a
readnone function can allocate storage, write to it, and still be
readonly. Our current inference rules for readnone and readonly
do exploit this flexibility.</p>
<p>The optimizer does currently assume that readonly implies
nofree. (See the accessor on Function) Removing this
substantially weakens our ability to infer nofree when faced with
a function declaration which hasn't been explicitly annotated for
nofree. We can get most of this back by adding appropriate
annotations to intrinsics, but not all. <br>
</p>
<p><b>Option 2</b> - nofree applies to memory visible to the caller</p>
<p>In this case, we'd add wording to the nofree definition analogous
to that in the readonly/readnone specification. (There's a
subtlety about the precise definition of visible here, but for the
moment, let's hand wave in the same way we do for the other
attributes.)</p>
<p>This allows us to infer nofree from readonly, but essentially
cripples our ability to drive transformations within an annotated
function. We'd have to restrict all transforms and inference to
cases where we can prove that the object being affected is visible
to the caller. <br>
</p>
<p>The benefit is that this makes it slightly easier to infer nofree
in some cases. The main impact of this is improving ability to
reason about dereferenceability for uncaptured objects over calls
to functions for which we inferred nofree. <br>
</p>
<p>The downside of this is that we essentially loose all ability to
reason about nofree in a context free manner. For a specific
example of the impact of this, it means we can't infer
dereferenceability for an object allocated in F, and returned
(e.g. not freed), in the scope of F.</p>
<p>This breaks hoisting and vectorization improvements (e.g.
unconditional loads instead of predicated ones) I've checked in
over the last few months, and makes the ongoing deref redefinition
work substantially harder. <a href="https://reviews.llvm.org/D100141" target="_blank">https://reviews.llvm.org/D100141</a> shows
what this looks like code wise.<br>
</p>
<p><b>My Take</b><br>
</p>
<p>At first, I was strongly convinced that option 1 was the right
choice. So much so in fact that I nearly didn't bother to post
this question. However, after giving it more thought, I've come
to distrust my own response a bit. I definitely have a conflict
of interest here. Option 2 requires me to effectively cripple
several recent optimizer enhancements, and maybe even revert some
code which becomes effectively useless. It also makes a project
I'm currently working on (deref redef) substantially harder. <br>
</p>
<p>On the other hand, the inconsistency with readonly and readnone
is surprising. I can see an argument for that being the right
overall approach long term.</p>
<p>So essentially, this email is me asking for a sanity check. Do
folks think option 1 is the right option? Or am I forcing it to
be the right option because it makes things easier for me?</p>
<p>Philip<br>
</p>
</div>
_______________________________________________<br>
LLVM Developers mailing list<br>
<a href="mailto:llvm-dev@lists.llvm.org" target="_blank">llvm-dev@lists.llvm.org</a><br>
<a href="https://lists.llvm.org/cgi-bin/mailman/listinfo/llvm-dev" rel="noreferrer" target="_blank">https://lists.llvm.org/cgi-bin/mailman/listinfo/llvm-dev</a><br>
</blockquote></div><br clear="all"><br>-- <br><div dir="ltr" class="gmail_signature">Lerne, wie die Welt wirklich ist,<br>aber vergiss niemals, wie sie sein sollte.</div></div>