[llvm] [SPIR-V] Add llvm.loop.unroll metadata lowering (PR #132062)
Dmitry Sidorov via llvm-commits
llvm-commits at lists.llvm.org
Mon Mar 24 10:02:53 PDT 2025
================
@@ -611,6 +611,40 @@ class SPIRVStructurizer : public FunctionPass {
auto MergeAddress = BlockAddress::get(Merge->getParent(), Merge);
auto ContinueAddress = BlockAddress::get(Continue->getParent(), Continue);
SmallVector<Value *, 2> Args = {MergeAddress, ContinueAddress};
+ unsigned LC = SPIRV::LoopControl::None;
+ // Currently used only to store PartialCount value. Later when other
+ // LoopControls are added - this map should be sorted before making
+ // them loop_merge operands to satisfy 3.23. Loop Control requirements.
+ std::vector<std::pair<unsigned, unsigned>> MaskToValueMap;
+ if (getBooleanLoopAttribute(L, "llvm.loop.unroll.disable")) {
+ LC |= SPIRV::LoopControl::DontUnroll;
+ } else {
+ if (getBooleanLoopAttribute(L, "llvm.loop.unroll.enable")) {
+ LC |= SPIRV::LoopControl::Unroll;
+ }
+ std::optional<int> Count =
+ getOptionalIntLoopAttribute(L, "llvm.loop.unroll.count");
+ if (Count && Count != 1) {
+ LC |= SPIRV::LoopControl::PartialCount;
+ MaskToValueMap.emplace_back(
+ std::make_pair(SPIRV::LoopControl::PartialCount, *Count));
+ }
+ if (getBooleanLoopAttribute(L, "llvm.loop.unroll.full")) {
+ // llvm.loop.unroll.full doesn't have a direct counterpart in SPIR-V,
+ // the closest thing we can do is to add Unroll mask and if the trip
+ // count is not known at compile time - either disable unrolling by
+ // setting PartialCount to 1 or reuse already available PartialCount.
+ LC |= SPIRV::LoopControl::Unroll;
+ if ((LC & SPIRV::LoopControl::PartialCount) == 0) {
+ LC |= SPIRV::LoopControl::PartialCount;
+ MaskToValueMap.emplace_back(
+ std::make_pair(SPIRV::LoopControl::PartialCount, 1));
+ }
----------------
MrSidims wrote:
The logic was, that `llvm.loop.unroll.enable` implies, that if the trip count is not known at compile time, then the loop can still be partial unrolled. Meanwhile `llvm.loop.unroll.full` just doesn't hints any unrolling, if trip count is not known at compile time. So (as SPIR-V spec doesn't distinguish between Unroll(enable) and Unroll(full)) `Unroll + PartialCount(1)` would mean to the backend: "please evaluate if it's possible to unroll the loop, and if not - do nothing with it". And I followed this logic long ago in https://github.com/KhronosGroup/SPIRV-LLVM-Translator/pull/1664 .
Now, after some thinking, this logic seem incorrect to me. As if optimizer faces `Unroll + PartialCount(1)` loop controls, it should actually pick conservative approach and don't even try to unroll the loop. So I've removed this logic from the PR.
https://github.com/llvm/llvm-project/pull/132062
More information about the llvm-commits
mailing list