<div dir="ltr"><div>Hi,</div><div><br></div><div>thanks Hanna for pointing at the contradiction under this modeling.<br></div><div><br></div><div>I wonder if HwModes can help us here. I feel in some way ELEN is playing a similar role to RISC-V's XLEN. This way could assign different value types to the register classes associated to the different LMUL values.<br></div><div><br></div><div>E.g.</div><div>ELEN=32 the register class for base registers (i.e. LMUL=1) could include nxv1i32, nxv1f32, nxv2i16, nxv2i8, etc.</div><div>ELEN=64 the register class for base registers could include nxv1i64, nxv1f64, nxv2i32, nxv2f32, ... but does not have to include nxv1f32, nxv1i32, nxv2i16, etc. (my understanding is that there is an ongoing proposal to efficiently allow manipulating such values as if they were subregisters of the base registers, but I'm ignoring that for now).<br></div><div><br></div><div>Kind regards,<br></div></div><br><div class="gmail_quote"><div dir="ltr" class="gmail_attr">Missatge de Hanna Kruppe via llvm-dev <<a href="mailto:llvm-dev@lists.llvm.org">llvm-dev@lists.llvm.org</a>> del dia dt., 7 d’abr. 2020 a les 13:52:<br></div><blockquote class="gmail_quote" style="margin:0px 0px 0px 0.8ex;border-left:1px solid rgb(204,204,204);padding-left:1ex">Hi all,<br>
<br>
On Tue, 7 Apr 2020 at 11:04, Renato Golin via llvm-dev<br>
<<a href="mailto:llvm-dev@lists.llvm.org" target="_blank">llvm-dev@lists.llvm.org</a>> wrote:<br>
><br>
> On Tue, 7 Apr 2020 at 09:30, Kai Wang via llvm-dev<br>
> <<a href="mailto:llvm-dev@lists.llvm.org" target="_blank">llvm-dev@lists.llvm.org</a>> wrote:<br>
> > LMUL = 1 LMUL = 2 LMUL = 4 LMUL = 8<br>
> > int64_t | vscale x 1 x i64 | vscale x 2 x i64 | vscale x 4 x i64 | vscale x 8 x i64<br>
> > int32_t | vscale x 2 x i32 | vscale x 4 x i32 | vscale x 8 x i32 | vscale x 16 x i32<br>
> > int16_t | vscale x 4 x i16 | vscale x 8 x i16 | vscale x 16 x i16 | vscale x 32 x i16<br>
> > int8_t | vscale x 8 x i8 | vscale x 16 x i8 | vscale x 32 x i8 | vscale x 64 x i8<br>
> ><br>
> > We have another architecture parameter, ELEN, which means the maximum size of a single vector element in bits.<br>
><br>
> Hi,<br>
><br>
> For my own education, some quick questions:<br>
><br>
> 1. is LMUL always a multiple of ELEN?<br>
<br>
<br>
This happens to be true (at least in the current spec, disregarding<br>
some in-progress proposals) just because both are powers of two and<br>
the largest possible LMUL equals the smallest possible ELEN (8), but I<br>
don't think there is any meaning to be found in this observation. The<br>
two values govern unrelated aspects of the vector unit.<br>
<br>
> 2. Is this fixed on the hardware, depending on the actual lengths, or<br>
> is this dynamically set by software (on a register or status flag)?<br>
> 2a. If dynamic, can it change from program to program? Function to function?<br>
<br>
<br>
It's not clear whether by "this" you mean ELEN, LMUL, or something<br>
else. ELEN is fixed in hardware. LMUL is a property of each individual<br>
instruction. Most instructions take it from a control register, a few<br>
encode it in the instruction as an immediate, but in any case it needs<br>
to be statically determined (on a per-instruction basis) to be able to<br>
allocate registers. This is not just a constraint for<br>
compiler-generated code, but also for all hand-written assembly code<br>
I've seen or can imagine.<br>
<br>
><br>
> > We hope the type system could be consistent under ELEN = 32 and ELEN = 64. However, vscale may be a fractional value under ELEN = 32 in the above type system. When ELEN = 32, i64 is an invalid type (we could ignore the first row for ELEN = 32) and vscale may become 1/2 on run time to fit the architecture (if the vector register only has 32 bits).<br>
><br>
> Do you mean ELEN=32 like this?<br>
> int32_t | vscale x 1 x i32 | vscale x 2 x i32 | vscale x 4 x i32 |<br>
> vscale x 8 x i32<br>
> int16_t | vscale x 2 x i16 | vscale x 4 x i16 | vscale x 8 x i16 |<br>
> vscale x 16 x i16<br>
> int8_t | vscale x 4 x i8 | vscale x 8 x i8 | vscale x 16 x i8 |<br>
> vscale x 32 x i8<br>
><br>
> If the type is invalid, you would need to legalise it, and in that<br>
> case create some cluttered accessors (via insert/extract element) and<br>
> possibly use intrinsics to expose underlying instructions that can<br>
> deal with it.<br>
><br>
> Perhaps I'm not clear on what you need, but vscale is supposed to be<br>
> the number of valid elements (lanes), and given i64 is invalid, vscale<br>
> wouldn't apply?<br>
<br>
<br>
I don't know what "vscale wouldn't apply" is supposed to mean. Whether<br>
it's legal or not, you can write LLVM IR using (for example) the type<br>
<vscale x 1 x i64> even if the target doesn't natively support it. The<br>
purpose of legalization is to make sure that results in the behavior<br>
the type is supposed to have. For <vscale x 1 x i32>, this means among<br>
other things:<br>
<br>
- it has the same number of elements as <vscale x 1 x i32>, but each<br>
element is twice as big<br>
- it has half as many elements (each of the same size) as <vscale x 2 x i64><br>
- its total size in bits is the same as <vscale x 2 x i32><br>
<br>
I think that focusing on the completely illegal i64 might obscure the<br>
real problem I see with the fractional vscale concept. Let's look at<br>
<vscale x 1 x i32> instead. The elements are clearly legal in this<br>
context, even in some vector types, but the <vscale x 1 x i32> type is<br>
absent from Kai's table. This makes sense: the same vector register<br>
fits 2x as many i32 elements as i64 elements, so if you start with<br>
<vscale x 1 x i64> mapping to a single register, then <vscale x 2 x<br>
i32> is the same size and fits in the same register class, while<br>
<vscale x 1 x i32> is too small and must be legalized somehow.<br>
<br>
But how? If we take Kai's table as gospel and look at a VLEN = ELEN =<br>
32 machine, the vector type <vscale x 2 x i32> is supposed to map to a<br>
single vector register, which is 32b small, and thus <vscale x 2 x<br>
i32> would have just one element in this context (matching the "vscale<br>
= 1/2" intuition). To be consistent with this, <vscale x 1 x i32><br>
would have be contain just *half* an element. This is not something<br>
any legalization strategy can achieve, because it is a fundamentally<br>
impossible notion. So we end up in a situation where some types are<br>
not just illegal and have to be legalized, but are contradictory and<br>
can't be legalized in any meaningful way.<br>
<br>
I don't think LLVM can/should support this kind of contradiction. Some<br>
types have to be legalized, sometimes the legalization is not<br>
efficient, sometimes it's not even implemented, that's all fine. But<br>
letting some targets decide that <vscale x 1 x i32> is a fundamentally<br>
impossible type to even assign a meaning to... that seems<br>
unprecedented and contrary to the philosophy of LLVM IR as reasonably<br>
target-independent IR.<br>
<br>
The obvious solution is to use a different set of legal vector types<br>
(and thus, a different interpretation of vscale) depending on the<br>
largest legal element type (ELEN in RISC-V jargon). This corresponds<br>
to the table for ELEN=32 that Renato gave above. Kai's proposal is<br>
intended to avoid this, and I can understand the desire for that, but<br>
it really seems like the lesser evil to me.<br>
<br>
Best regards<br>
Hanna<br>
<br>
<br>
> > Is there any problem to assume vscale to be fractional under some circumstances? vscale should be an unknown value when compiling. So, it should have no impact on code generation and optimization. The relationship between types is correct regardless vscale’s value. Is there anything I missed?<br>
><br>
> I believe the assumption was always that vscale is an integer.<br>
> Representing it as a fraction would need code change for sure, but<br>
> also reevaluate the assumptions.<br>
><br>
> I'm copying some SVE and LV people to give a more informed opinion.<br>
><br>
> cheers,<br>
> --renato<br>
> _______________________________________________<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>
_______________________________________________<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">Roger Ferrer Ibáñez<br></div>