Neil Henning via llvm-dev
2020-Jul-16 13:25 UTC
[llvm-dev] LLVM 11 and trunk selecting 4 wide instead of 8 wide loop vectorization for AVX-enabled target
Hey list, I've recently done the first test run of bumping our Burst compiler from LLVM 10 -> 11 now that the branch has been cut, and have noticed an apparent loop vectorization codegen regression for X86 with AVX or AVX2 enabled. The following IR example is vectorized to 4 wide with LLVM 11 and trunk whereas in LLVM 10 it (correctly as per what we want) vectorized it 8 wide matching the ymm registers. ; ModuleID = '../test.ll' source_filename = "main" target datalayout "e-m:w-p270:32:32-p271:32:32-p272:64:64-i64:64-f80:128-n8:16:32:64-S128" target triple = "x86_64-pc-windows-msvc-coff" %"Burst.Compiler.IL.Tests.VectorsMaths/FloatPointer.0" = type { float*, i32, [4 x i8] } ; Function Attrs: nofree define dllexport void @func(float* noalias nocapture %output, %"Burst.Compiler.IL.Tests.VectorsMaths/FloatPointer.0"* nocapture nonnull readonly dereferenceable(16) %a, %"Burst.Compiler.IL.Tests.VectorsMaths/FloatPointer.0"* nocapture nonnull readonly dereferenceable(16) %b) local_unnamed_addr #0 !ubaa. !1 { entry: %0 = getelementptr %"Burst.Compiler.IL.Tests.VectorsMaths/FloatPointer.0", %"Burst.Compiler.IL.Tests.VectorsMaths/FloatPointer.0"* %a, i64 0, i32 1 %1 = load i32, i32* %0, align 1 %.not = icmp eq i32 %1, 0 br i1 %.not, label %BL.0042, label %BL.0005.lr.ph BL.0005.lr.ph: ; preds = %entry %2 = bitcast %"Burst.Compiler.IL.Tests.VectorsMaths/FloatPointer.0"* %a to i8** %3 = load i8*, i8** %2, align 1 %4 = bitcast %"Burst.Compiler.IL.Tests.VectorsMaths/FloatPointer.0"* %b to i8** %5 = load i8*, i8** %4, align 1 %wide.trip.count = zext i32 %1 to i64 br label %BL.0005 BL.0005: ; preds = %BL.0005, % BL.0005.lr.ph %indvars.iv = phi i64 [ 0, %BL.0005.lr.ph ], [ %indvars.iv.next, %BL.0005 ] %6 = shl nuw nsw i64 %indvars.iv, 2 %7 = getelementptr float, float* %output, i64 %indvars.iv %8 = getelementptr i8, i8* %3, i64 %6 %9 = bitcast i8* %8 to float* %10 = load float, float* %9, align 4 %11 = getelementptr i8, i8* %5, i64 %6 %12 = bitcast i8* %11 to float* %13 = load float, float* %12, align 4 %14 = tail call float @llvm.pow.f32(float %10, float %13) store float %14, float* %7, align 4 %indvars.iv.next = add nuw nsw i64 %indvars.iv, 1 %exitcond.not = icmp eq i64 %indvars.iv.next, %wide.trip.count br i1 %exitcond.not, label %BL.0042, label %BL.0005 BL.0042: ; preds = %BL.0005, %entry ret void } ; Function Attrs: norecurse readnone define dllexport void @burst.initialize(i8* (i8*)* nocapture readnone %callback) local_unnamed_addr #1 !ubaa. !0 { entry: ret void } ; Function Attrs: nounwind readnone speculatable willreturn declare float @llvm.pow.f32(float, float) #2 attributes #0 = { nofree } attributes #1 = { norecurse readnone } attributes #2 = { nounwind readnone speculatable willreturn } !ubaa.Burst.Compiler.IL.Tests.VectorsMaths\2FFloatPointer.0 = !{!0, !0, !0, !0} !0 = !{i1 false} !1 = !{i1 true, i1 false, i1 false} If I run this with ../llvm-project/llvm/build/bin/opt.exe -o - -S -O3 ../avx_sad_4.ll -mattr=avx -debug, I can see that the loop vectorizer correctly considers using 8-wide ymm registers for this, but has decided that the 4-wide variant is cheaper based on some cost modelling I don't understand. So is this expected behaviour? I know there was some cost model changes in the 10->11 timeframe. Thanks for any help, Cheers, -Neil. -- Neil Henning Senior Software Engineer Compiler unity.com -------------- next part -------------- An HTML attachment was scrubbed... URL: <http://lists.llvm.org/pipermail/llvm-dev/attachments/20200716/337f6f39/attachment.html>
Roman Lebedev via llvm-dev
2020-Jul-16 13:38 UTC
[llvm-dev] LLVM 11 and trunk selecting 4 wide instead of 8 wide loop vectorization for AVX-enabled target
Did you specify the target CPU the code should be optimized for? For clang that is -march=native/znver2/... / -mtune=<same> For opt/llc that is --mcpu=<same> I would expect that by default, some generic baseline is picked. On Thu, Jul 16, 2020 at 4:25 PM Neil Henning via llvm-dev < llvm-dev at lists.llvm.org> wrote:> Hey list, > > I've recently done the first test run of bumping our Burst compiler from > LLVM 10 -> 11 now that the branch has been cut, and have noticed an > apparent loop vectorization codegen regression for X86 with AVX or AVX2 > enabled. The following IR example is vectorized to 4 wide with LLVM 11 and > trunk whereas in LLVM 10 it (correctly as per what we want) vectorized it 8 > wide matching the ymm registers. > > ; ModuleID = '../test.ll' > source_filename = "main" > target datalayout > "e-m:w-p270:32:32-p271:32:32-p272:64:64-i64:64-f80:128-n8:16:32:64-S128" > target triple = "x86_64-pc-windows-msvc-coff" > > %"Burst.Compiler.IL.Tests.VectorsMaths/FloatPointer.0" = type { float*, > i32, [4 x i8] } > > ; Function Attrs: nofree > define dllexport void @func(float* noalias nocapture %output, > %"Burst.Compiler.IL.Tests.VectorsMaths/FloatPointer.0"* nocapture nonnull > readonly dereferenceable(16) %a, > %"Burst.Compiler.IL.Tests.VectorsMaths/FloatPointer.0"* nocapture nonnull > readonly dereferenceable(16) %b) local_unnamed_addr #0 !ubaa. !1 { > entry: > %0 = getelementptr > %"Burst.Compiler.IL.Tests.VectorsMaths/FloatPointer.0", > %"Burst.Compiler.IL.Tests.VectorsMaths/FloatPointer.0"* %a, i64 0, i32 1 > %1 = load i32, i32* %0, align 1 > %.not = icmp eq i32 %1, 0 > br i1 %.not, label %BL.0042, label %BL.0005.lr.ph > > BL.0005.lr.ph: ; preds = %entry > %2 = bitcast %"Burst.Compiler.IL.Tests.VectorsMaths/FloatPointer.0"* %a > to i8** > %3 = load i8*, i8** %2, align 1 > %4 = bitcast %"Burst.Compiler.IL.Tests.VectorsMaths/FloatPointer.0"* %b > to i8** > %5 = load i8*, i8** %4, align 1 > %wide.trip.count = zext i32 %1 to i64 > br label %BL.0005 > > BL.0005: ; preds = %BL.0005, % > BL.0005.lr.ph > %indvars.iv = phi i64 [ 0, %BL.0005.lr.ph ], [ %indvars.iv.next, > %BL.0005 ] > %6 = shl nuw nsw i64 %indvars.iv, 2 > %7 = getelementptr float, float* %output, i64 %indvars.iv > %8 = getelementptr i8, i8* %3, i64 %6 > %9 = bitcast i8* %8 to float* > %10 = load float, float* %9, align 4 > %11 = getelementptr i8, i8* %5, i64 %6 > %12 = bitcast i8* %11 to float* > %13 = load float, float* %12, align 4 > %14 = tail call float @llvm.pow.f32(float %10, float %13) > store float %14, float* %7, align 4 > %indvars.iv.next = add nuw nsw i64 %indvars.iv, 1 > %exitcond.not = icmp eq i64 %indvars.iv.next, %wide.trip.count > br i1 %exitcond.not, label %BL.0042, label %BL.0005 > > BL.0042: ; preds = %BL.0005, > %entry > ret void > } > > ; Function Attrs: norecurse readnone > define dllexport void @burst.initialize(i8* (i8*)* nocapture readnone > %callback) local_unnamed_addr #1 !ubaa. !0 { > entry: > ret void > } > > ; Function Attrs: nounwind readnone speculatable willreturn > declare float @llvm.pow.f32(float, float) #2 > > attributes #0 = { nofree } > attributes #1 = { norecurse readnone } > attributes #2 = { nounwind readnone speculatable willreturn } > > !ubaa.Burst.Compiler.IL.Tests.VectorsMaths\2FFloatPointer.0 = !{!0, !0, > !0, !0} > > !0 = !{i1 false} > !1 = !{i1 true, i1 false, i1 false} > > If I run this with ../llvm-project/llvm/build/bin/opt.exe -o - -S -O3 > ../avx_sad_4.ll -mattr=avx -debug, I can see that the loop vectorizer > correctly considers using 8-wide ymm registers for this, but has decided > that the 4-wide variant is cheaper based on some cost modelling I don't > understand. > > So is this expected behaviour? I know there was some cost model changes in > the 10->11 timeframe. > > Thanks for any help, > > Cheers, > -Neil. >Roman> -- > Neil Henning > Senior Software Engineer Compiler > unity.com > _______________________________________________ > 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/20200716/bb590ea5/attachment.html>
Neil Henning via llvm-dev
2020-Jul-16 14:20 UTC
[llvm-dev] LLVM 11 and trunk selecting 4 wide instead of 8 wide loop vectorization for AVX-enabled target
Tried a bunch of them there (x86-64, haswell, znver2) and they all defaulted to 4-wide - haswell additionally caused some extra loop unrolling but still with 8-wide pows. Cheers, -Neil. On Thu, Jul 16, 2020 at 2:39 PM Roman Lebedev <lebedev.ri at gmail.com> wrote:> Did you specify the target CPU the code should be optimized for? > For clang that is -march=native/znver2/... / -mtune=<same> > For opt/llc that is --mcpu=<same> > I would expect that by default, some generic baseline is picked. > > On Thu, Jul 16, 2020 at 4:25 PM Neil Henning via llvm-dev < > llvm-dev at lists.llvm.org> wrote: > >> Hey list, >> >> I've recently done the first test run of bumping our Burst compiler from >> LLVM 10 -> 11 now that the branch has been cut, and have noticed an >> apparent loop vectorization codegen regression for X86 with AVX or AVX2 >> enabled. The following IR example is vectorized to 4 wide with LLVM 11 and >> trunk whereas in LLVM 10 it (correctly as per what we want) vectorized it 8 >> wide matching the ymm registers. >> >> ; ModuleID = '../test.ll' >> source_filename = "main" >> target datalayout >> "e-m:w-p270:32:32-p271:32:32-p272:64:64-i64:64-f80:128-n8:16:32:64-S128" >> target triple = "x86_64-pc-windows-msvc-coff" >> >> %"Burst.Compiler.IL.Tests.VectorsMaths/FloatPointer.0" = type { float*, >> i32, [4 x i8] } >> >> ; Function Attrs: nofree >> define dllexport void @func(float* noalias nocapture %output, >> %"Burst.Compiler.IL.Tests.VectorsMaths/FloatPointer.0"* nocapture nonnull >> readonly dereferenceable(16) %a, >> %"Burst.Compiler.IL.Tests.VectorsMaths/FloatPointer.0"* nocapture nonnull >> readonly dereferenceable(16) %b) local_unnamed_addr #0 !ubaa. !1 { >> entry: >> %0 = getelementptr >> %"Burst.Compiler.IL.Tests.VectorsMaths/FloatPointer.0", >> %"Burst.Compiler.IL.Tests.VectorsMaths/FloatPointer.0"* %a, i64 0, i32 1 >> %1 = load i32, i32* %0, align 1 >> %.not = icmp eq i32 %1, 0 >> br i1 %.not, label %BL.0042, label %BL.0005.lr.ph >> >> BL.0005.lr.ph: ; preds = %entry >> %2 = bitcast %"Burst.Compiler.IL.Tests.VectorsMaths/FloatPointer.0"* %a >> to i8** >> %3 = load i8*, i8** %2, align 1 >> %4 = bitcast %"Burst.Compiler.IL.Tests.VectorsMaths/FloatPointer.0"* %b >> to i8** >> %5 = load i8*, i8** %4, align 1 >> %wide.trip.count = zext i32 %1 to i64 >> br label %BL.0005 >> >> BL.0005: ; preds = %BL.0005, % >> BL.0005.lr.ph >> %indvars.iv = phi i64 [ 0, %BL.0005.lr.ph ], [ %indvars.iv.next, >> %BL.0005 ] >> %6 = shl nuw nsw i64 %indvars.iv, 2 >> %7 = getelementptr float, float* %output, i64 %indvars.iv >> %8 = getelementptr i8, i8* %3, i64 %6 >> %9 = bitcast i8* %8 to float* >> %10 = load float, float* %9, align 4 >> %11 = getelementptr i8, i8* %5, i64 %6 >> %12 = bitcast i8* %11 to float* >> %13 = load float, float* %12, align 4 >> %14 = tail call float @llvm.pow.f32(float %10, float %13) >> store float %14, float* %7, align 4 >> %indvars.iv.next = add nuw nsw i64 %indvars.iv, 1 >> %exitcond.not = icmp eq i64 %indvars.iv.next, %wide.trip.count >> br i1 %exitcond.not, label %BL.0042, label %BL.0005 >> >> BL.0042: ; preds = %BL.0005, >> %entry >> ret void >> } >> >> ; Function Attrs: norecurse readnone >> define dllexport void @burst.initialize(i8* (i8*)* nocapture readnone >> %callback) local_unnamed_addr #1 !ubaa. !0 { >> entry: >> ret void >> } >> >> ; Function Attrs: nounwind readnone speculatable willreturn >> declare float @llvm.pow.f32(float, float) #2 >> >> attributes #0 = { nofree } >> attributes #1 = { norecurse readnone } >> attributes #2 = { nounwind readnone speculatable willreturn } >> >> !ubaa.Burst.Compiler.IL.Tests.VectorsMaths\2FFloatPointer.0 = !{!0, !0, >> !0, !0} >> >> !0 = !{i1 false} >> !1 = !{i1 true, i1 false, i1 false} >> >> If I run this with ../llvm-project/llvm/build/bin/opt.exe -o - -S -O3 >> ../avx_sad_4.ll -mattr=avx -debug, I can see that the loop vectorizer >> correctly considers using 8-wide ymm registers for this, but has decided >> that the 4-wide variant is cheaper based on some cost modelling I don't >> understand. >> >> So is this expected behaviour? I know there was some cost model changes >> in the 10->11 timeframe. >> >> Thanks for any help, >> >> Cheers, >> -Neil. >> > Roman > > >> -- >> Neil Henning >> Senior Software Engineer Compiler >> unity.com >> _______________________________________________ >> LLVM Developers mailing list >> llvm-dev at lists.llvm.org >> https://lists.llvm.org/cgi-bin/mailman/listinfo/llvm-dev >> >-- Neil Henning Senior Software Engineer Compiler unity.com -------------- next part -------------- An HTML attachment was scrubbed... URL: <http://lists.llvm.org/pipermail/llvm-dev/attachments/20200716/ed9e83d1/attachment.html>
Apparently Analagous Threads
- LLVM 11 and trunk selecting 4 wide instead of 8 wide loop vectorization for AVX-enabled target
- LLVM 11 and trunk selecting 4 wide instead of 8 wide loop vectorization for AVX-enabled target
- Help with SROA throwing away no-alias information
- Question on target-features
- Question on target-features