<div dir="ltr"><div class="gmail_extra"><div class="gmail_quote">On Fri, Jun 8, 2018 at 4:10 AM, Graham Hunter via llvm-dev <span dir="ltr"><<a href="mailto:llvm-dev@lists.llvm.org" target="_blank">llvm-dev@lists.llvm.org</a>></span> wrote:<br><blockquote class="gmail_quote" style="margin:0 0 0 .8ex;border-left:1px #ccc solid;padding-left:1ex">Hi,<br>
<span class=""><br>
> On 6 Jun 2018, at 17:36, David A. Greene <<a href="mailto:dag@cray.com">dag@cray.com</a>> wrote:<br>
> <br>
> Graham Hunter via llvm-dev <<a href="mailto:llvm-dev@lists.llvm.org">llvm-dev@lists.llvm.org</a>> writes:<br>
> <br>
>>> Ok, now I understand what you're getting at. A ConstantExpr would<br>
>>> encapsulate this computation. We alreay have "non-static-constant"<br>
>>> values for ConstantExpr like sizeof and offsetof. I would see<br>
>>> VScaleConstant in that same tradition. In your struct example,<br>
>>> getSizeExpressionInBits would return:<br>
>>> <br>
>>> add(mul(256, vscale), 64)<br>
>>> <br>
>>> Does that satisfy your needs?<br>
>> <br>
>> Ah, I think the use of 'expression' in the name definitely confuses the issue then. This<br>
>> isn't for expressing the size in IR, where you would indeed just multiply by vscale and<br>
>> add any fixed-length size.<br>
> <br>
> Ok, thanks for clarifying. The use of "expression" is confusing.<br>
> <br>
>> This is for the analysis code around the IR -- lots of code asks for the size of a Type in<br>
>> bits to determine what it can do to a Value with that type. Some of them are specific to<br>
>> scalar Types, like determining whether a sign/zero extend is needed. Others would<br>
>> apply to vector types (including scalable vectors), such as checking whether two<br>
>> Types have the exact same size so that a bitcast can be used instead of a more<br>
>> expensive operation like copying to memory and back to convert.<br>
> <br>
> If this method returns two integers, how does LLVM interpret the<br>
> comparison? If the return value is { <unscaled>, <scaled> } then how<br>
> do, say { 1024, 0 } and { 0, 128 } compare? Doesn't it depend on the<br>
> vscale? They could be the same size or not, depending on the target<br>
> characteristics.<br>
<br>
</span>I did have a paragraph on that in the RFC, but perhaps a list would be<br>
a better format (assuming X,Y,etc are non-zero):<br>
<br>
{ X, 0 } <cmp> { Y, 0 }: Normal unscaled comparison.<br>
<br>
{ 0, X } <cmp> { 0, Y }: Normal comparison within a function, or across<br>
functions that inherit vector length. Cannot be<br>
compared across non-inheriting functions.<br>
<br>
{ X, 0 } > { 0, Y }: Cannot return true.<br>
<br>
{ X, 0 } = { 0, Y }: Cannot return true.<br>
<br>
{ X, 0 } < { 0, Y }: Can return true.<br>
<br>
{ Xu, Xs } <cmp> { Yu, Ys }: Gets complicated, need to subtract common<br>
terms and try the above comparisons; it<br>
may not be possible to get a good answer.<br>
<br>
I don't know if we need a 'maybe' result for cases comparing scaled<br>
vs. unscaled; I believe the gcc implementation of SVE allows for such<br>
results, but that supports a generic polynomial length representation.<br>
<br>
I think in code, we'd have an inline function to deal with the first case<br>
and an likely-not-taken call to a separate function to handle all the<br>
scalable cases.<br>
<span class=""><br>
> Are bitcasts between scaled types and non-scaled types disallowed? I<br>
> could certainly see an argument for disallowing it. I could argue that<br>
> for bitcasting purposes that the unscaled and scaled parts would have to<br>
> exactly match in order to do a legal bitcast. Is that the intent?<br>
<br>
</span>I would propose disallowing bitcasts, but allowing extracting a subvector<br>
if the minimum number of scaled bits matches the number of unscaled bits.<br>
<div><div class="h5"><br>
> <br>
>>> Is there anything about vscale or a scalable vector that requires a<br>
>>> minimum bit width? For example, is this legal?<br>
>>> <br>
>>> <scalable 1 x double><br>
>>> <br>
>>> I know it won't map to an SVE type. I'm simply curious because<br>
>>> traditionally Cray machines defined vectors in terms of<br>
>>> machine-dependent "maxvl" with an element type, so with the above vscale<br>
>>> would == maxvl. Not that we make any such things anymore. But maybe<br>
>>> someone else does?<br>
>> <br>
>> That's legal in IR, yes, and we believe it should be usable to represent the vectors for<br>
>> RISC-V's 'V' extension. The main problem there is that they have a dynamic vector<br>
>> length within the loop so that they can perform the last iterations of a loop within vector<br>
>> registers when there's less than a full register worth of data remaining. SVE uses<br>
>> predication (masking) to achieve the same effect.<br>
>> <br>
>> For the 'V' extension, vscale would indeed correspond to 'maxvl', and I'm hoping that a<br>
>> 'setvl' intrinsic that provides a predicate will avoid the need for modelling a change in<br>
>> dynamic vector length -- reducing the vector length is effectively equivalent to an implied<br>
>> predicate on all operations. This avoids needing to add a token operand to all existing<br>
>> instructions that work on vector types.<br>
> <br>
> Right. In that way the RISC V method is very much like what the old<br>
> Cray machines did with the Vector Length register.<br>
> <br>
> So in LLVM IR you would have "setvl" return a predicate and then apply<br>
> that predicate to operations using the current select method? How does<br>
> instruction selection map that back onto a simple setvl + unpredicated<br>
> vector instructions?<br>
> <br>
> For conditional code both vector length and masking must be taken into<br>
> account. If "setvl" returns a predicate then that predicate would have<br>
> to be combined in some way with the conditional predicate (typically via<br>
> an AND operation in an IR that directly supports predicates). Since<br>
> LLVM IR doesn't have predicates _per_se_, would it turn into nested<br>
> selects or something? Untangling that in instruction selection seems<br>
> difficult but perhaps I'm missing something.<br>
<br>
</div></div>My idea is for the RISC-V backend to recognize when a setvl intrinsic has<br>
been used, and replace the use of its value in AND operations with an<br>
all-true value (with constant folding to remove unnecessary ANDs) then<br>
replace any masked instructions (generally loads, stores, anything else<br>
that might generate an exception or modify state that it shouldn't) with<br>
target-specific nodes that understand the dynamic vlen.<br>
<br>
This could be part of lowering, or maybe a separate IR pass, rather than ISel.<br>
I *think* this will work, but if someone can come up with some IR where it<br>
wouldn't work then please let me know (e.g. global-state-changing instructions<br>
that could move out of blocks where one setvl predicate is used and into one<br>
where another is used).<br>
<br>
Unfortunately, I can't find a description of the instructions included in<br>
the 'V' extension in the online manual (other than setvl or configuring<br>
registers), so I can't tell if there's something I'm missing.<br></blockquote><div><br></div><div>RVV is a little bit behind SVE in the process :-) On the whole it's following the style of vector processor that has had several implementations at Berkeley, dating back a decade or more. The set of operations is pretty much nailed down now, but things such as the exact instruction encodings are still in flux. There is an intention to get some experience with compilers and FPGA (at least) implementations of the proposal before ratifying it as part of the RISC-V standard. So details could well change during that period.</div><div><br></div></div></div></div>