Simon Pilgrim via llvm-dev
2021-Jul-15 09:05 UTC
[llvm-dev] Not able to vectorize struct members
I've reported this here: https://bugs.llvm.org/show_bug.cgi?id=51103 On 15/07/2021 04:04, Ryan Taylor via llvm-dev wrote:> Given a test case: > > #define N 10 > struct foo { > float a[N][N]; > float b[N]; > }; > > void compute(struct foo *p) > { > for (int i = 0; i < N; i++) { > for (int j = 0; j < N; j++) { > p->a[i][j] += 0.1f * p->b[i]; > } > } > } > > LLVM isn't able to vectorize this code. I'm not sure what the issue is > here as it seems to vectorize this code: > > #define N 10 > struct foo { > float a[N]; > float b[N]; > }; > > void compute(struct foo *p) > { > for (int i = 0; i < N; i++) { > p->a[i] += 0.1f * p->b[i]; > } > } > > Is this a known issue? > > -Ryan > > _______________________________________________ > LLVM Developers mailing list > llvm-dev at lists.llvm.org > https://lists.llvm.org/cgi-bin/mailman/listinfo/llvm-dev-------------- next part -------------- An HTML attachment was scrubbed... URL: <http://lists.llvm.org/pipermail/llvm-dev/attachments/20210715/c7a0bee4/attachment.html>
Ryan Taylor via llvm-dev
2021-Jul-15 13:19 UTC
[llvm-dev] Not able to vectorize struct members
I see Roman has commented on the ticket. The issue is alias analysis it
looks like, we have a partial fix in place in BasicAliasAnalysis.cpp:
unsigned MinNumIndex = std::min(GEP1->getNumIndices(),
GEP2->getNumIndices());
if (MinNumIndex < 2)
return MayAlias;
SmallVector<Value *, 8> IntermediateIndices;
IntermediateIndices.push_back(GEP1->getOperand(1));
// Note,
// 1. The index to struct field can only be literal constant.
// 2. So far all the struct field indexes are the same respectively.
// 3. The array indexes could be different, but they don't matter here.
// 4. The types should be the same, say, different array indexes don't
// matter, and the struct indexes be the same respectively.
// If current type is StructType, and the pair of corresponding indexes
// are not equal, NoAlias
for (unsigned i = 1; i != MinNumIndex - 1; ++i) {
if (isa<StructType>(GetElementPtrInst::getIndexedType(
GEP1->getSourceElementType(), IntermediateIndices))) {
ConstantInt *C1 = dyn_cast<ConstantInt>(GEP1->getOperand(i +
1));
ConstantInt *C2 = dyn_cast<ConstantInt>(GEP2->getOperand(i +
1));
if (C1 && C2 && C1->getSExtValue() !=
C2->getSExtValue())
return NoAlias;
}
IntermediateIndices.push_back(GEP1->getOperand(i + 1));
}
}
// Note that we fall back the original logic here, there are still some
thing
// not covered like p->a[2].x v.s. p->a[1].x.b[2]
but this seems to break some cases and isn't generalized.
-Ryan
On Thu, Jul 15, 2021 at 5:05 AM Simon Pilgrim via llvm-dev <
llvm-dev at lists.llvm.org> wrote:
> I've reported this here: https://bugs.llvm.org/show_bug.cgi?id=51103
> On 15/07/2021 04:04, Ryan Taylor via llvm-dev wrote:
>
> Given a test case:
>
> #define N 10
> struct foo {
> float a[N][N];
> float b[N];
> };
>
> void compute(struct foo *p)
> {
> for (int i = 0; i < N; i++) {
> for (int j = 0; j < N; j++) {
> p->a[i][j] += 0.1f * p->b[i];
> }
> }
> }
>
> LLVM isn't able to vectorize this code. I'm not sure what the issue
is
> here as it seems to vectorize this code:
>
> #define N 10
> struct foo {
> float a[N];
> float b[N];
> };
>
> void compute(struct foo *p)
> {
> for (int i = 0; i < N; i++) {
> p->a[i] += 0.1f * p->b[i];
> }
> }
>
> Is this a known issue?
>
> -Ryan
>
> _______________________________________________
> LLVM Developers mailing listllvm-dev at
lists.llvm.orghttps://lists.llvm.org/cgi-bin/mailman/listinfo/llvm-dev
>
> _______________________________________________
> LLVM Developers mailing list
> llvm-dev at lists.llvm.org
> https://lists.llvm.org/cgi-bin/mailman/listinfo/llvm-dev
>
-------------- next part --------------
An HTML attachment was scrubbed...
URL:
<http://lists.llvm.org/pipermail/llvm-dev/attachments/20210715/2dee84fa/attachment.html>