[PATCH] D53137: Scalable vector core instruction support + size queries

Sander de Smalen via Phabricator via llvm-commits llvm-commits at lists.llvm.org
Fri Aug 9 06:17:54 PDT 2019


sdesmalen added a comment.

In D53137#1622713 <https://reviews.llvm.org/D53137#1622713>, @huntergr wrote:

> See some of the inline comments -- there are a few places where we'd just end up duplicating code if used that way. The names can certainly be improved for clarity, though, and we could state that the (scalable|fixed)-only interfaces should be used in preference to a joint one.


Yes, adding some words to state that in the description of the method would be good. There are indeed cases where `MinSize` is needed directly (like in DAGCombiner.cpp where it uses `MaximumLegalStoreInBits = MinSize`), but such a case is very explicit (it needs to specifically query the `known` part of the size), which I think is not very common. We should strive to reduce those uses as much as possible. For Alignment for example, I think we can use a separate method  (see comment <https://reviews.llvm.org/D53137#inline-590826>).

>> I realise that the entire code-base currently expects 'unsigned' or 'int', but this can easily be fixed by adding a (hopefully temporary) overloaded cast operator to the struct that produces a scalar value, like:
>> 
>>   /// Casts to a uint64_t if this is a fixed-width size.
>>   ///
>>   /// NOTE: This interface is obsolescent and will be removed
>>   /// in a future version of LLVM in favour of getFixedSize().
>>   operator uint64_t() const {
>>     assert(isFixed() && "Expected fixed size data type");
>>     return getFixedSize();
>>   } 
>>    
>> 
>> We can then update the codebase piece by piece, incrementally making use of `ScalableSize` or its interfaces. When all that is done, the codebase should compile without errors when building for `LLVM_TARGETS_TO_BUILD=AArch64` after we remove the overloaded operator.
> 
> I'm fine with that approach if others approve; I was trying to minimize the overall impact on the codebase though.

Great! Looking forward to hear how others feel about it.



================
Comment at: llvm/include/llvm/Support/ScalableSize.h:83
+
+  friend bool operator>(const ScalableSize &LHS, const ScalableSize &RHS) {
+    return RHS < LHS;
----------------
huntergr wrote:
> sdesmalen wrote:
> > All the comparison operators assert that the types are both fixed-width or both scalable.
> > Is there value in also adding the following interfaces?
> > 
> >   // Returns true if A is known to be at least as big as B, e.g.
> >   // If A is scalable, and B is not, returns A.MinSize >= B.MinSize
> >   // If A is not scalable, and B is, returns false regardless of MinSize.
> >   bool knownGreaterOrEqual(const ScalableSize &B) const { ... }
> >   // or alternatively: "atLeastAsBigAs"?
> > and
> >   // Returns true if A is known to always be larger than B, e.g.
> >   // If A is scalable, and B is not, returns A.MinSize > B.MinSize
> >   // If A is not scalable, and B is, returns false regardless of MinSize.
> >   bool knownGreater(const ScalableSize &B) const { ... }
> If you can think of an immediate use case, sure.
> 
> Otherwise I'd leave it to another patch which requires that information.
One use-case is in DAGCombiner.cpp, where it currently has:

  VT.getMinSizeInBits() >= MaximumLegalStoreInBits

which would become:

  VT.getSizeInBits().isKnownGreaterOrEqual(MaximumLegalStoreInBits)

This removes another use-case for using `MinSize` directly.


================
Comment at: llvm/include/llvm/Support/ScalableSize.h:120
+  // isScalable method below.
+  uint64_t getMinSize() const {
+    return MinSize;
----------------
huntergr wrote:
> sdesmalen wrote:
> > I think this only really makes sense in the context of scalable vectors, and don't think we want to expose it as a property for both fixed/scalable vectors, e.g.:
> > 
> >   uint64_t getMinSize() const {
> >     assert (isScalable() && "MinSize only makes sense in the context of a scalable vector,"
> >                             " use getFixedSize() instead");
> >     return MinSize;
> >   }
> > 
> > For cases where we need to use it in a comparison (to know if it is at least a given number of bytes) we can do this in a separate comparison interface.
> This is used for the alignment checks at the moment, where we just need to know the minimum for scalable and exact for fixed.
> 
> I'd rather not have code that looks like
> 
> ```
> unsigned Align;
> if (VTy->isScalable())
>   Align = getMinSize(VTy);
> else
>   Align = getFixedSize(VTy);
> ```
> 
> in several places, when it could be done with one.
> 
> We can certainly bikeshed the names, though, and come up with a more explicit one which acknowledges it can represent a known quantity from fixed or scalable vectors.
> 
> I would also need to convert a larger part of the codebase to use scalable types in order to use this right now, and I've already had pushback on the size of the changes.
For Alignment, because it is so common, I think it is worth adding a separate method:

  unsigned getNaturalAlignment() const { return (unsigned) getKnownSize(); };


================
Comment at: llvm/lib/CodeGen/SelectionDAG/SelectionDAGBuilder.cpp:9177
                                i < CLI.NumFixedArgs,
-                               i, j*Parts[j].getValueType().getStoreSize());
+                               i, j*Parts[j].getValueType().getMinStoreSize());
         if (NumParts > 1 && j == 0)
----------------
huntergr wrote:
> huntergr wrote:
> > sdesmalen wrote:
> > > `OutputArg::PartOffset` is defined explicitly as a byte offset, so until we change that, this should be `getFixedSize()`.
> > Then we would assert when running unit tests.
> > 
> > I'm trying to juggle between an enormous patch that fixes everything 'properly', and something which gets us partway and keeps us running but needs fixes elsewhere later (which seemed to be what other reviewers would prefer).
> > 
> > I could certainly add a comment here to explain that changes are needed, though.
> A followup thought on this; would it be preferable to split the EVT/MVT side of this patch out into a separate one, and fix this properly there?
Right, I see. At some point this class should probably use [[https://reviews.llvm.org/D61435 | StackOffset]] to represent such offsets. For now, I'd say this line warrants a "FIXME" comment.


================
Comment at: llvm/lib/CodeGen/SelectionDAG/SelectionDAGBuilder.cpp:9177
                                i < CLI.NumFixedArgs,
-                               i, j*Parts[j].getValueType().getStoreSize());
+                               i, j*Parts[j].getValueType().getMinStoreSize());
         if (NumParts > 1 && j == 0)
----------------
sdesmalen wrote:
> huntergr wrote:
> > huntergr wrote:
> > > sdesmalen wrote:
> > > > `OutputArg::PartOffset` is defined explicitly as a byte offset, so until we change that, this should be `getFixedSize()`.
> > > Then we would assert when running unit tests.
> > > 
> > > I'm trying to juggle between an enormous patch that fixes everything 'properly', and something which gets us partway and keeps us running but needs fixes elsewhere later (which seemed to be what other reviewers would prefer).
> > > 
> > > I could certainly add a comment here to explain that changes are needed, though.
> > A followup thought on this; would it be preferable to split the EVT/MVT side of this patch out into a separate one, and fix this properly there?
> Right, I see. At some point this class should probably use [[https://reviews.llvm.org/D61435 | StackOffset]] to represent such offsets. For now, I'd say this line warrants a "FIXME" comment.
> A followup thought on this; would it be preferable to split the EVT/MVT side of this patch out into a separate one, and fix this properly there?
I think so.


================
Comment at: llvm/lib/IR/Instructions.cpp:3039
   // match
-  if (SrcBits == 0 || DestBits == 0)
+  if (SrcBits.getMinSize() == 0 || DestBits.getMinSize() == 0)
     return false;
----------------
huntergr wrote:
> sdesmalen wrote:
> > MinSize suggests that it could be larger at runtime, so `SrcBits.getMinSize() == 0` would always be true. If you instead overload `operator bool()` which checks for Fixed size and Scalable size to be 0, you can rewrite this as `(!SrcBits || !DestBits)` .
> No. `0 * vscale` is still 0 for any value of vscale.
> 
> The ScalableSize class says the Scalable flag indicates the total size is an integer multiple of the known minimum size.
> 
> I suspect a better way of doing this test would be to explicitly check for the elements of one being pointer type and not the other, instead of relying on a hack with the size.
Perhaps I am a little pedantic with how I read this, because I expect the minimum size of every object to be 0, always :)

So perhaps instead of using the name `MinSize`, a name like `KnownSize` would be better suited. (which would also nicely match with the `isKnownGreaterOrEqual()` suggestion for DAGCombiner.cpp).


================
Comment at: llvm/lib/Target/X86/X86ISelLowering.cpp:721
+    // Scalable vectors aren't supported on this backend.
+    if (VT.isScalableVector())
+      continue;
----------------
huntergr wrote:
> sdesmalen wrote:
> > Adding `MVT::fixedwidth_[integer_|fp_]vector_valuetypes()` seems like a more natural interface than skipping based on `isScalableVector()`. There already seem to be iterators for `MVT::(integer_|fp_)_scalable_vector_valuetypes`.
> > 
> > You can do that in a separate patch, so this patch can focus solely on scalable size queries.
> Yeah, I thought of that after I'd submitted the patch. Will do.
Thanks!


Repository:
  rG LLVM Github Monorepo

CHANGES SINCE LAST ACTION
  https://reviews.llvm.org/D53137/new/

https://reviews.llvm.org/D53137





More information about the llvm-commits mailing list