mirror of
https://github.com/KhronosGroup/SPIRV-Tools
synced 2024-11-23 20:20:06 +00:00
This reverts commit d4fac3451b
.
This commit is contained in:
parent
3118276370
commit
bd0a2da946
@ -20,6 +20,7 @@
|
|||||||
#include <utility>
|
#include <utility>
|
||||||
|
|
||||||
#include "source/cfa.h"
|
#include "source/cfa.h"
|
||||||
|
#include "source/opt/reflect.h"
|
||||||
#include "source/util/make_unique.h"
|
#include "source/util/make_unique.h"
|
||||||
|
|
||||||
// Indices of operands in SPIR-V instructions
|
// Indices of operands in SPIR-V instructions
|
||||||
@ -232,6 +233,220 @@ bool InlinePass::CloneSameBlockOps(
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void InlinePass::MoveInstsBeforeEntryBlock(
|
||||||
|
std::unordered_map<uint32_t, Instruction*>* preCallSB,
|
||||||
|
BasicBlock* new_blk_ptr, BasicBlock::iterator call_inst_itr,
|
||||||
|
UptrVectorIterator<BasicBlock> call_block_itr) {
|
||||||
|
for (auto cii = call_block_itr->begin(); cii != call_inst_itr;
|
||||||
|
cii = call_block_itr->begin()) {
|
||||||
|
Instruction* inst = &*cii;
|
||||||
|
inst->RemoveFromList();
|
||||||
|
std::unique_ptr<Instruction> cp_inst(inst);
|
||||||
|
// Remember same-block ops for possible regeneration.
|
||||||
|
if (IsSameBlockOp(&*cp_inst)) {
|
||||||
|
auto* sb_inst_ptr = cp_inst.get();
|
||||||
|
(*preCallSB)[cp_inst->result_id()] = sb_inst_ptr;
|
||||||
|
}
|
||||||
|
new_blk_ptr->AddInstruction(std::move(cp_inst));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
std::unique_ptr<BasicBlock> InlinePass::AddGuardBlock(
|
||||||
|
std::vector<std::unique_ptr<BasicBlock>>* new_blocks,
|
||||||
|
std::unordered_map<uint32_t, uint32_t>* callee2caller,
|
||||||
|
std::unique_ptr<BasicBlock> new_blk_ptr, uint32_t entry_blk_label_id) {
|
||||||
|
const auto guard_block_id = context()->TakeNextId();
|
||||||
|
if (guard_block_id == 0) {
|
||||||
|
return nullptr;
|
||||||
|
}
|
||||||
|
AddBranch(guard_block_id, &new_blk_ptr);
|
||||||
|
new_blocks->push_back(std::move(new_blk_ptr));
|
||||||
|
// Start the next block.
|
||||||
|
new_blk_ptr = MakeUnique<BasicBlock>(NewLabel(guard_block_id));
|
||||||
|
// Reset the mapping of the callee's entry block to point to
|
||||||
|
// the guard block. Do this so we can fix up phis later on to
|
||||||
|
// satisfy dominance.
|
||||||
|
(*callee2caller)[entry_blk_label_id] = guard_block_id;
|
||||||
|
return new_blk_ptr;
|
||||||
|
}
|
||||||
|
|
||||||
|
InstructionList::iterator InlinePass::AddStoresForVariableInitializers(
|
||||||
|
const std::unordered_map<uint32_t, uint32_t>& callee2caller,
|
||||||
|
std::unique_ptr<BasicBlock>* new_blk_ptr,
|
||||||
|
UptrVectorIterator<BasicBlock> callee_first_block_itr) {
|
||||||
|
auto callee_var_itr = callee_first_block_itr->begin();
|
||||||
|
while (callee_var_itr->opcode() == SpvOp::SpvOpVariable) {
|
||||||
|
if (callee_var_itr->NumInOperands() == 2) {
|
||||||
|
assert(callee2caller.count(callee_var_itr->result_id()) &&
|
||||||
|
"Expected the variable to have already been mapped.");
|
||||||
|
uint32_t new_var_id = callee2caller.at(callee_var_itr->result_id());
|
||||||
|
|
||||||
|
// The initializer must be a constant or global value. No mapped
|
||||||
|
// should be used.
|
||||||
|
uint32_t val_id = callee_var_itr->GetSingleWordInOperand(1);
|
||||||
|
AddStore(new_var_id, val_id, new_blk_ptr);
|
||||||
|
}
|
||||||
|
++callee_var_itr;
|
||||||
|
}
|
||||||
|
return callee_var_itr;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool InlinePass::InlineInstructionInBB(
|
||||||
|
const std::unordered_map<uint32_t, uint32_t>& callee2caller,
|
||||||
|
BasicBlock* new_blk_ptr, const Instruction* inst) {
|
||||||
|
// If we have return, it must be at the end of the callee. We will handle
|
||||||
|
// it at the end.
|
||||||
|
if (inst->opcode() == SpvOpReturnValue || inst->opcode() == SpvOpReturn)
|
||||||
|
return true;
|
||||||
|
|
||||||
|
// Copy callee instruction and remap all input Ids.
|
||||||
|
std::unique_ptr<Instruction> cp_inst(inst->Clone(context()));
|
||||||
|
cp_inst->ForEachInId([&callee2caller](uint32_t* iid) {
|
||||||
|
const auto mapItr = callee2caller.find(*iid);
|
||||||
|
if (mapItr != callee2caller.end()) {
|
||||||
|
*iid = mapItr->second;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
// If result id is non-zero, remap it.
|
||||||
|
const uint32_t rid = cp_inst->result_id();
|
||||||
|
if (rid != 0) {
|
||||||
|
const auto mapItr = callee2caller.find(rid);
|
||||||
|
if (mapItr == callee2caller.end()) return false;
|
||||||
|
uint32_t nid = mapItr->second;
|
||||||
|
cp_inst->SetResultId(nid);
|
||||||
|
get_decoration_mgr()->CloneDecorations(rid, nid);
|
||||||
|
}
|
||||||
|
new_blk_ptr->AddInstruction(std::move(cp_inst));
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
std::unique_ptr<BasicBlock> InlinePass::InlineReturn(
|
||||||
|
const std::unordered_map<uint32_t, uint32_t>& callee2caller,
|
||||||
|
std::vector<std::unique_ptr<BasicBlock>>* new_blocks,
|
||||||
|
std::unique_ptr<BasicBlock> new_blk_ptr, Function* calleeFn,
|
||||||
|
const Instruction* inst, uint32_t returnVarId) {
|
||||||
|
// Store return value to return variable.
|
||||||
|
if (inst->opcode() == SpvOpReturnValue) {
|
||||||
|
assert(returnVarId != 0);
|
||||||
|
uint32_t valId = inst->GetInOperand(kSpvReturnValueId).words[0];
|
||||||
|
const auto mapItr = callee2caller.find(valId);
|
||||||
|
if (mapItr != callee2caller.end()) {
|
||||||
|
valId = mapItr->second;
|
||||||
|
}
|
||||||
|
AddStore(returnVarId, valId, &new_blk_ptr);
|
||||||
|
}
|
||||||
|
|
||||||
|
uint32_t returnLabelId = 0;
|
||||||
|
for (auto callee_block_itr = calleeFn->begin();
|
||||||
|
callee_block_itr != calleeFn->end(); ++callee_block_itr) {
|
||||||
|
if (callee_block_itr->tail()->opcode() == SpvOpUnreachable ||
|
||||||
|
callee_block_itr->tail()->opcode() == SpvOpKill) {
|
||||||
|
returnLabelId = context()->TakeNextId();
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (returnLabelId == 0) return new_blk_ptr;
|
||||||
|
|
||||||
|
if (inst->opcode() == SpvOpReturn || inst->opcode() == SpvOpReturnValue)
|
||||||
|
AddBranch(returnLabelId, &new_blk_ptr);
|
||||||
|
new_blocks->push_back(std::move(new_blk_ptr));
|
||||||
|
return MakeUnique<BasicBlock>(NewLabel(returnLabelId));
|
||||||
|
}
|
||||||
|
|
||||||
|
bool InlinePass::InlineEntryBlock(
|
||||||
|
const std::unordered_map<uint32_t, uint32_t>& callee2caller,
|
||||||
|
std::unique_ptr<BasicBlock>* new_blk_ptr,
|
||||||
|
UptrVectorIterator<BasicBlock> callee_first_block) {
|
||||||
|
auto callee_inst_itr = AddStoresForVariableInitializers(
|
||||||
|
callee2caller, new_blk_ptr, callee_first_block);
|
||||||
|
|
||||||
|
while (callee_inst_itr != callee_first_block->end()) {
|
||||||
|
if (!InlineInstructionInBB(callee2caller, new_blk_ptr->get(),
|
||||||
|
&*callee_inst_itr)) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
++callee_inst_itr;
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
std::unique_ptr<BasicBlock> InlinePass::InlineBasicBlocks(
|
||||||
|
std::vector<std::unique_ptr<BasicBlock>>* new_blocks,
|
||||||
|
const std::unordered_map<uint32_t, uint32_t>& callee2caller,
|
||||||
|
std::unique_ptr<BasicBlock> new_blk_ptr, Function* calleeFn) {
|
||||||
|
auto callee_block_itr = calleeFn->begin();
|
||||||
|
++callee_block_itr;
|
||||||
|
|
||||||
|
while (callee_block_itr != calleeFn->end()) {
|
||||||
|
new_blocks->push_back(std::move(new_blk_ptr));
|
||||||
|
const auto mapItr =
|
||||||
|
callee2caller.find(callee_block_itr->GetLabelInst()->result_id());
|
||||||
|
if (mapItr == callee2caller.end()) return nullptr;
|
||||||
|
new_blk_ptr = MakeUnique<BasicBlock>(NewLabel(mapItr->second));
|
||||||
|
|
||||||
|
auto tail_inst_itr = callee_block_itr->end();
|
||||||
|
for (auto inst_itr = callee_block_itr->begin(); inst_itr != tail_inst_itr;
|
||||||
|
++inst_itr) {
|
||||||
|
if (!InlineInstructionInBB(callee2caller, new_blk_ptr.get(),
|
||||||
|
&*inst_itr)) {
|
||||||
|
return nullptr;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
++callee_block_itr;
|
||||||
|
}
|
||||||
|
return new_blk_ptr;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool InlinePass::MoveCallerInstsAfterFunctionCall(
|
||||||
|
std::unordered_map<uint32_t, Instruction*>* preCallSB,
|
||||||
|
std::unordered_map<uint32_t, uint32_t>* postCallSB,
|
||||||
|
std::unique_ptr<BasicBlock>* new_blk_ptr,
|
||||||
|
BasicBlock::iterator call_inst_itr, bool multiBlocks) {
|
||||||
|
// Copy remaining instructions from caller block.
|
||||||
|
for (Instruction* inst = call_inst_itr->NextNode(); inst;
|
||||||
|
inst = call_inst_itr->NextNode()) {
|
||||||
|
inst->RemoveFromList();
|
||||||
|
std::unique_ptr<Instruction> cp_inst(inst);
|
||||||
|
// If multiple blocks generated, regenerate any same-block
|
||||||
|
// instruction that has not been seen in this last block.
|
||||||
|
if (multiBlocks) {
|
||||||
|
if (!CloneSameBlockOps(&cp_inst, postCallSB, preCallSB, new_blk_ptr)) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Remember same-block ops in this block.
|
||||||
|
if (IsSameBlockOp(&*cp_inst)) {
|
||||||
|
const uint32_t rid = cp_inst->result_id();
|
||||||
|
(*postCallSB)[rid] = rid;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
new_blk_ptr->get()->AddInstruction(std::move(cp_inst));
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
void InlinePass::MoveLoopMergeInstToFirstBlock(
|
||||||
|
std::vector<std::unique_ptr<BasicBlock>>* new_blocks) {
|
||||||
|
// Move the OpLoopMerge from the last block back to the first, where
|
||||||
|
// it belongs.
|
||||||
|
auto& first = new_blocks->front();
|
||||||
|
auto& last = new_blocks->back();
|
||||||
|
assert(first != last);
|
||||||
|
|
||||||
|
// Insert a modified copy of the loop merge into the first block.
|
||||||
|
auto loop_merge_itr = last->tail();
|
||||||
|
--loop_merge_itr;
|
||||||
|
assert(loop_merge_itr->opcode() == SpvOpLoopMerge);
|
||||||
|
std::unique_ptr<Instruction> cp_inst(loop_merge_itr->Clone(context()));
|
||||||
|
first->tail().InsertBefore(std::move(cp_inst));
|
||||||
|
|
||||||
|
// Remove the loop merge from the last block.
|
||||||
|
loop_merge_itr->RemoveFromList();
|
||||||
|
delete &*loop_merge_itr;
|
||||||
|
}
|
||||||
|
|
||||||
bool InlinePass::GenInlineCode(
|
bool InlinePass::GenInlineCode(
|
||||||
std::vector<std::unique_ptr<BasicBlock>>* new_blocks,
|
std::vector<std::unique_ptr<BasicBlock>>* new_blocks,
|
||||||
std::vector<std::unique_ptr<Instruction>>* new_vars,
|
std::vector<std::unique_ptr<Instruction>>* new_vars,
|
||||||
@ -250,13 +465,19 @@ bool InlinePass::GenInlineCode(
|
|||||||
// valid. These operations can fail.
|
// valid. These operations can fail.
|
||||||
context()->InvalidateAnalyses(IRContext::kAnalysisDefUse);
|
context()->InvalidateAnalyses(IRContext::kAnalysisDefUse);
|
||||||
|
|
||||||
|
// If the caller is a loop header and the callee has multiple blocks, then the
|
||||||
|
// normal inlining logic will place the OpLoopMerge in the last of several
|
||||||
|
// blocks in the loop. Instead, it should be placed at the end of the first
|
||||||
|
// block. We'll wait to move the OpLoopMerge until the end of the regular
|
||||||
|
// inlining logic, and only if necessary.
|
||||||
|
bool caller_is_loop_header = call_block_itr->GetLoopMergeInst() != nullptr;
|
||||||
|
|
||||||
|
// Single-trip loop continue block
|
||||||
|
std::unique_ptr<BasicBlock> single_trip_loop_cont_blk;
|
||||||
|
|
||||||
Function* calleeFn = id2function_[call_inst_itr->GetSingleWordOperand(
|
Function* calleeFn = id2function_[call_inst_itr->GetSingleWordOperand(
|
||||||
kSpvFunctionCallFunctionId)];
|
kSpvFunctionCallFunctionId)];
|
||||||
|
|
||||||
// Check for multiple returns in the callee.
|
|
||||||
auto fi = early_return_funcs_.find(calleeFn->result_id());
|
|
||||||
const bool earlyReturn = fi != early_return_funcs_.end();
|
|
||||||
|
|
||||||
// Map parameters to actual arguments.
|
// Map parameters to actual arguments.
|
||||||
MapParams(calleeFn, call_inst_itr, &callee2caller);
|
MapParams(calleeFn, call_inst_itr, &callee2caller);
|
||||||
|
|
||||||
@ -266,6 +487,31 @@ bool InlinePass::GenInlineCode(
|
|||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// First block needs to use label of original block
|
||||||
|
// but map callee label in case of phi reference.
|
||||||
|
uint32_t entry_blk_label_id = calleeFn->begin()->GetLabelInst()->result_id();
|
||||||
|
callee2caller[entry_blk_label_id] = call_block_itr->id();
|
||||||
|
std::unique_ptr<BasicBlock> new_blk_ptr =
|
||||||
|
MakeUnique<BasicBlock>(NewLabel(call_block_itr->id()));
|
||||||
|
|
||||||
|
// Move instructions of original caller block up to call instruction.
|
||||||
|
MoveInstsBeforeEntryBlock(&preCallSB, new_blk_ptr.get(), call_inst_itr,
|
||||||
|
call_block_itr);
|
||||||
|
|
||||||
|
if (caller_is_loop_header &&
|
||||||
|
(*(calleeFn->begin())).GetMergeInst() != nullptr) {
|
||||||
|
// We can't place both the caller's merge instruction and
|
||||||
|
// another merge instruction in the same block. So split the
|
||||||
|
// calling block. Insert an unconditional branch to a new guard
|
||||||
|
// block. Later, once we know the ID of the last block, we
|
||||||
|
// will move the caller's OpLoopMerge from the last generated
|
||||||
|
// block into the first block. We also wait to avoid
|
||||||
|
// invalidating various iterators.
|
||||||
|
new_blk_ptr = AddGuardBlock(new_blocks, &callee2caller,
|
||||||
|
std::move(new_blk_ptr), entry_blk_label_id);
|
||||||
|
if (new_blk_ptr == nullptr) return false;
|
||||||
|
}
|
||||||
|
|
||||||
// Create return var if needed.
|
// Create return var if needed.
|
||||||
const uint32_t calleeTypeId = calleeFn->type_id();
|
const uint32_t calleeTypeId = calleeFn->type_id();
|
||||||
uint32_t returnVarId = 0;
|
uint32_t returnVarId = 0;
|
||||||
@ -277,341 +523,50 @@ bool InlinePass::GenInlineCode(
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Create set of callee result ids. Used to detect forward references
|
calleeFn->WhileEachInst([&callee2caller, this](const Instruction* cpi) {
|
||||||
std::unordered_set<uint32_t> callee_result_ids;
|
// Create set of callee result ids. Used to detect forward references
|
||||||
calleeFn->ForEachInst([&callee_result_ids](const Instruction* cpi) {
|
|
||||||
const uint32_t rid = cpi->result_id();
|
const uint32_t rid = cpi->result_id();
|
||||||
if (rid != 0) callee_result_ids.insert(rid);
|
if (rid != 0 && callee2caller.find(rid) == callee2caller.end()) {
|
||||||
|
const uint32_t nid = context()->TakeNextId();
|
||||||
|
if (nid == 0) return false;
|
||||||
|
callee2caller[rid] = nid;
|
||||||
|
}
|
||||||
|
return true;
|
||||||
});
|
});
|
||||||
|
|
||||||
// If the caller is a loop header and the callee has multiple blocks, then the
|
// Inline the entry block of the callee function.
|
||||||
// normal inlining logic will place the OpLoopMerge in the last of several
|
if (!InlineEntryBlock(callee2caller, &new_blk_ptr, calleeFn->begin())) {
|
||||||
// blocks in the loop. Instead, it should be placed at the end of the first
|
|
||||||
// block. We'll wait to move the OpLoopMerge until the end of the regular
|
|
||||||
// inlining logic, and only if necessary.
|
|
||||||
bool caller_is_loop_header = false;
|
|
||||||
if (call_block_itr->GetLoopMergeInst()) {
|
|
||||||
caller_is_loop_header = true;
|
|
||||||
}
|
|
||||||
|
|
||||||
bool callee_begins_with_structured_header =
|
|
||||||
(*(calleeFn->begin())).GetMergeInst() != nullptr;
|
|
||||||
|
|
||||||
// Clone and map callee code. Copy caller block code to beginning of
|
|
||||||
// first block and end of last block.
|
|
||||||
bool prevInstWasReturn = false;
|
|
||||||
uint32_t singleTripLoopHeaderId = 0;
|
|
||||||
uint32_t singleTripLoopContinueId = 0;
|
|
||||||
uint32_t returnLabelId = 0;
|
|
||||||
bool multiBlocks = false;
|
|
||||||
// new_blk_ptr is a new basic block in the caller. New instructions are
|
|
||||||
// written to it. It is created when we encounter the OpLabel
|
|
||||||
// of the first callee block. It is appended to new_blocks only when
|
|
||||||
// it is complete.
|
|
||||||
std::unique_ptr<BasicBlock> new_blk_ptr;
|
|
||||||
bool successful = calleeFn->WhileEachInst(
|
|
||||||
[&new_blocks, &callee2caller, &call_block_itr, &call_inst_itr,
|
|
||||||
&new_blk_ptr, &prevInstWasReturn, &returnLabelId, &returnVarId,
|
|
||||||
caller_is_loop_header, callee_begins_with_structured_header,
|
|
||||||
&calleeTypeId, &multiBlocks, &postCallSB, &preCallSB, earlyReturn,
|
|
||||||
&singleTripLoopHeaderId, &singleTripLoopContinueId, &callee_result_ids,
|
|
||||||
this](const Instruction* cpi) {
|
|
||||||
switch (cpi->opcode()) {
|
|
||||||
case SpvOpFunction:
|
|
||||||
case SpvOpFunctionParameter:
|
|
||||||
// Already processed
|
|
||||||
break;
|
|
||||||
case SpvOpVariable:
|
|
||||||
if (cpi->NumInOperands() == 2) {
|
|
||||||
assert(callee2caller.count(cpi->result_id()) &&
|
|
||||||
"Expected the variable to have already been mapped.");
|
|
||||||
uint32_t new_var_id = callee2caller.at(cpi->result_id());
|
|
||||||
|
|
||||||
// The initializer must be a constant or global value. No mapped
|
|
||||||
// should be used.
|
|
||||||
uint32_t val_id = cpi->GetSingleWordInOperand(1);
|
|
||||||
AddStore(new_var_id, val_id, &new_blk_ptr);
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
case SpvOpUnreachable:
|
|
||||||
case SpvOpKill: {
|
|
||||||
// Generate a return label so that we split the block with the
|
|
||||||
// function call. Copy the terminator into the new block.
|
|
||||||
if (returnLabelId == 0) {
|
|
||||||
returnLabelId = context()->TakeNextId();
|
|
||||||
if (returnLabelId == 0) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
std::unique_ptr<Instruction> terminator(
|
|
||||||
new Instruction(context(), cpi->opcode(), 0, 0, {}));
|
|
||||||
new_blk_ptr->AddInstruction(std::move(terminator));
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
case SpvOpLabel: {
|
|
||||||
// If previous instruction was early return, insert branch
|
|
||||||
// instruction to return block.
|
|
||||||
if (prevInstWasReturn) {
|
|
||||||
if (returnLabelId == 0) {
|
|
||||||
returnLabelId = context()->TakeNextId();
|
|
||||||
if (returnLabelId == 0) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
AddBranch(returnLabelId, &new_blk_ptr);
|
|
||||||
prevInstWasReturn = false;
|
|
||||||
}
|
|
||||||
// Finish current block (if it exists) and get label for next block.
|
|
||||||
uint32_t labelId;
|
|
||||||
bool firstBlock = false;
|
|
||||||
if (new_blk_ptr != nullptr) {
|
|
||||||
new_blocks->push_back(std::move(new_blk_ptr));
|
|
||||||
// If result id is already mapped, use it, otherwise get a new
|
|
||||||
// one.
|
|
||||||
const uint32_t rid = cpi->result_id();
|
|
||||||
const auto mapItr = callee2caller.find(rid);
|
|
||||||
labelId = (mapItr != callee2caller.end())
|
|
||||||
? mapItr->second
|
|
||||||
: context()->TakeNextId();
|
|
||||||
if (labelId == 0) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
// First block needs to use label of original block
|
|
||||||
// but map callee label in case of phi reference.
|
|
||||||
labelId = call_block_itr->id();
|
|
||||||
callee2caller[cpi->result_id()] = labelId;
|
|
||||||
firstBlock = true;
|
|
||||||
}
|
|
||||||
// Create first/next block.
|
|
||||||
new_blk_ptr = MakeUnique<BasicBlock>(NewLabel(labelId));
|
|
||||||
if (firstBlock) {
|
|
||||||
// Copy contents of original caller block up to call instruction.
|
|
||||||
for (auto cii = call_block_itr->begin(); cii != call_inst_itr;
|
|
||||||
cii = call_block_itr->begin()) {
|
|
||||||
Instruction* inst = &*cii;
|
|
||||||
inst->RemoveFromList();
|
|
||||||
std::unique_ptr<Instruction> cp_inst(inst);
|
|
||||||
// Remember same-block ops for possible regeneration.
|
|
||||||
if (IsSameBlockOp(&*cp_inst)) {
|
|
||||||
auto* sb_inst_ptr = cp_inst.get();
|
|
||||||
preCallSB[cp_inst->result_id()] = sb_inst_ptr;
|
|
||||||
}
|
|
||||||
new_blk_ptr->AddInstruction(std::move(cp_inst));
|
|
||||||
}
|
|
||||||
if (caller_is_loop_header &&
|
|
||||||
callee_begins_with_structured_header) {
|
|
||||||
// We can't place both the caller's merge instruction and
|
|
||||||
// another merge instruction in the same block. So split the
|
|
||||||
// calling block. Insert an unconditional branch to a new guard
|
|
||||||
// block. Later, once we know the ID of the last block, we
|
|
||||||
// will move the caller's OpLoopMerge from the last generated
|
|
||||||
// block into the first block. We also wait to avoid
|
|
||||||
// invalidating various iterators.
|
|
||||||
const auto guard_block_id = context()->TakeNextId();
|
|
||||||
if (guard_block_id == 0) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
AddBranch(guard_block_id, &new_blk_ptr);
|
|
||||||
new_blocks->push_back(std::move(new_blk_ptr));
|
|
||||||
// Start the next block.
|
|
||||||
new_blk_ptr = MakeUnique<BasicBlock>(NewLabel(guard_block_id));
|
|
||||||
// Reset the mapping of the callee's entry block to point to
|
|
||||||
// the guard block. Do this so we can fix up phis later on to
|
|
||||||
// satisfy dominance.
|
|
||||||
callee2caller[cpi->result_id()] = guard_block_id;
|
|
||||||
}
|
|
||||||
// If callee has early return, insert a header block for
|
|
||||||
// single-trip loop that will encompass callee code. Start
|
|
||||||
// postheader block.
|
|
||||||
//
|
|
||||||
// Note: Consider the following combination:
|
|
||||||
// - the caller is a single block loop
|
|
||||||
// - the callee does not begin with a structure header
|
|
||||||
// - the callee has multiple returns.
|
|
||||||
// We still need to split the caller block and insert a guard
|
|
||||||
// block. But we only need to do it once. We haven't done it yet,
|
|
||||||
// but the single-trip loop header will serve the same purpose.
|
|
||||||
if (earlyReturn) {
|
|
||||||
singleTripLoopHeaderId = context()->TakeNextId();
|
|
||||||
if (singleTripLoopHeaderId == 0) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
AddBranch(singleTripLoopHeaderId, &new_blk_ptr);
|
|
||||||
new_blocks->push_back(std::move(new_blk_ptr));
|
|
||||||
new_blk_ptr =
|
|
||||||
MakeUnique<BasicBlock>(NewLabel(singleTripLoopHeaderId));
|
|
||||||
returnLabelId = context()->TakeNextId();
|
|
||||||
singleTripLoopContinueId = context()->TakeNextId();
|
|
||||||
if (returnLabelId == 0 || singleTripLoopContinueId == 0) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
AddLoopMerge(returnLabelId, singleTripLoopContinueId,
|
|
||||||
&new_blk_ptr);
|
|
||||||
uint32_t postHeaderId = context()->TakeNextId();
|
|
||||||
if (postHeaderId == 0) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
AddBranch(postHeaderId, &new_blk_ptr);
|
|
||||||
new_blocks->push_back(std::move(new_blk_ptr));
|
|
||||||
new_blk_ptr = MakeUnique<BasicBlock>(NewLabel(postHeaderId));
|
|
||||||
multiBlocks = true;
|
|
||||||
// Reset the mapping of the callee's entry block to point to
|
|
||||||
// the post-header block. Do this so we can fix up phis later
|
|
||||||
// on to satisfy dominance.
|
|
||||||
callee2caller[cpi->result_id()] = postHeaderId;
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
multiBlocks = true;
|
|
||||||
}
|
|
||||||
} break;
|
|
||||||
case SpvOpReturnValue: {
|
|
||||||
// Store return value to return variable.
|
|
||||||
assert(returnVarId != 0);
|
|
||||||
uint32_t valId = cpi->GetInOperand(kSpvReturnValueId).words[0];
|
|
||||||
const auto mapItr = callee2caller.find(valId);
|
|
||||||
if (mapItr != callee2caller.end()) {
|
|
||||||
valId = mapItr->second;
|
|
||||||
}
|
|
||||||
AddStore(returnVarId, valId, &new_blk_ptr);
|
|
||||||
|
|
||||||
// Remember we saw a return; if followed by a label, will need to
|
|
||||||
// insert branch.
|
|
||||||
prevInstWasReturn = true;
|
|
||||||
} break;
|
|
||||||
case SpvOpReturn: {
|
|
||||||
// Remember we saw a return; if followed by a label, will need to
|
|
||||||
// insert branch.
|
|
||||||
prevInstWasReturn = true;
|
|
||||||
} break;
|
|
||||||
case SpvOpFunctionEnd: {
|
|
||||||
// If there was an early return, we generated a return label id
|
|
||||||
// for it. Now we have to generate the return block with that Id.
|
|
||||||
if (returnLabelId != 0) {
|
|
||||||
// If previous instruction was return, insert branch instruction
|
|
||||||
// to return block.
|
|
||||||
if (prevInstWasReturn) AddBranch(returnLabelId, &new_blk_ptr);
|
|
||||||
if (earlyReturn) {
|
|
||||||
// If we generated a loop header for the single-trip loop
|
|
||||||
// to accommodate early returns, insert the continue
|
|
||||||
// target block now, with a false branch back to the loop
|
|
||||||
// header.
|
|
||||||
new_blocks->push_back(std::move(new_blk_ptr));
|
|
||||||
new_blk_ptr =
|
|
||||||
MakeUnique<BasicBlock>(NewLabel(singleTripLoopContinueId));
|
|
||||||
uint32_t false_id = GetFalseId();
|
|
||||||
if (false_id == 0) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
AddBranchCond(false_id, singleTripLoopHeaderId, returnLabelId,
|
|
||||||
&new_blk_ptr);
|
|
||||||
}
|
|
||||||
// Generate the return block.
|
|
||||||
new_blocks->push_back(std::move(new_blk_ptr));
|
|
||||||
new_blk_ptr = MakeUnique<BasicBlock>(NewLabel(returnLabelId));
|
|
||||||
multiBlocks = true;
|
|
||||||
}
|
|
||||||
// Load return value into result id of call, if it exists.
|
|
||||||
if (returnVarId != 0) {
|
|
||||||
const uint32_t resId = call_inst_itr->result_id();
|
|
||||||
assert(resId != 0);
|
|
||||||
AddLoad(calleeTypeId, resId, returnVarId, &new_blk_ptr);
|
|
||||||
}
|
|
||||||
// Copy remaining instructions from caller block.
|
|
||||||
for (Instruction* inst = call_inst_itr->NextNode(); inst;
|
|
||||||
inst = call_inst_itr->NextNode()) {
|
|
||||||
inst->RemoveFromList();
|
|
||||||
std::unique_ptr<Instruction> cp_inst(inst);
|
|
||||||
// If multiple blocks generated, regenerate any same-block
|
|
||||||
// instruction that has not been seen in this last block.
|
|
||||||
if (multiBlocks) {
|
|
||||||
if (!CloneSameBlockOps(&cp_inst, &postCallSB, &preCallSB,
|
|
||||||
&new_blk_ptr)) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Remember same-block ops in this block.
|
|
||||||
if (IsSameBlockOp(&*cp_inst)) {
|
|
||||||
const uint32_t rid = cp_inst->result_id();
|
|
||||||
postCallSB[rid] = rid;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
new_blk_ptr->AddInstruction(std::move(cp_inst));
|
|
||||||
}
|
|
||||||
// Finalize inline code.
|
|
||||||
new_blocks->push_back(std::move(new_blk_ptr));
|
|
||||||
} break;
|
|
||||||
default: {
|
|
||||||
// Copy callee instruction and remap all input Ids.
|
|
||||||
std::unique_ptr<Instruction> cp_inst(cpi->Clone(context()));
|
|
||||||
bool succeeded = cp_inst->WhileEachInId(
|
|
||||||
[&callee2caller, &callee_result_ids, this](uint32_t* iid) {
|
|
||||||
const auto mapItr = callee2caller.find(*iid);
|
|
||||||
if (mapItr != callee2caller.end()) {
|
|
||||||
*iid = mapItr->second;
|
|
||||||
} else if (callee_result_ids.find(*iid) !=
|
|
||||||
callee_result_ids.end()) {
|
|
||||||
// Forward reference. Allocate a new id, map it,
|
|
||||||
// use it and check for it when remapping result ids
|
|
||||||
const uint32_t nid = context()->TakeNextId();
|
|
||||||
if (nid == 0) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
callee2caller[*iid] = nid;
|
|
||||||
*iid = nid;
|
|
||||||
}
|
|
||||||
return true;
|
|
||||||
});
|
|
||||||
if (!succeeded) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
// If result id is non-zero, remap it. If already mapped, use mapped
|
|
||||||
// value, else use next id.
|
|
||||||
const uint32_t rid = cp_inst->result_id();
|
|
||||||
if (rid != 0) {
|
|
||||||
const auto mapItr = callee2caller.find(rid);
|
|
||||||
uint32_t nid;
|
|
||||||
if (mapItr != callee2caller.end()) {
|
|
||||||
nid = mapItr->second;
|
|
||||||
} else {
|
|
||||||
nid = context()->TakeNextId();
|
|
||||||
if (nid == 0) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
callee2caller[rid] = nid;
|
|
||||||
}
|
|
||||||
cp_inst->SetResultId(nid);
|
|
||||||
get_decoration_mgr()->CloneDecorations(rid, nid);
|
|
||||||
}
|
|
||||||
new_blk_ptr->AddInstruction(std::move(cp_inst));
|
|
||||||
} break;
|
|
||||||
}
|
|
||||||
return true;
|
|
||||||
});
|
|
||||||
|
|
||||||
if (!successful) {
|
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (caller_is_loop_header && (new_blocks->size() > 1)) {
|
// Inline blocks of the callee function other than the entry block.
|
||||||
// Move the OpLoopMerge from the last block back to the first, where
|
new_blk_ptr = InlineBasicBlocks(new_blocks, callee2caller,
|
||||||
// it belongs.
|
std::move(new_blk_ptr), calleeFn);
|
||||||
auto& first = new_blocks->front();
|
if (new_blk_ptr == nullptr) return false;
|
||||||
auto& last = new_blocks->back();
|
|
||||||
assert(first != last);
|
|
||||||
|
|
||||||
// Insert a modified copy of the loop merge into the first block.
|
new_blk_ptr =
|
||||||
auto loop_merge_itr = last->tail();
|
InlineReturn(callee2caller, new_blocks, std::move(new_blk_ptr), calleeFn,
|
||||||
--loop_merge_itr;
|
&*(calleeFn->tail()->tail()), returnVarId);
|
||||||
assert(loop_merge_itr->opcode() == SpvOpLoopMerge);
|
|
||||||
std::unique_ptr<Instruction> cp_inst(loop_merge_itr->Clone(context()));
|
|
||||||
first->tail().InsertBefore(std::move(cp_inst));
|
|
||||||
|
|
||||||
// Remove the loop merge from the last block.
|
// Load return value into result id of call, if it exists.
|
||||||
loop_merge_itr->RemoveFromList();
|
if (returnVarId != 0) {
|
||||||
delete &*loop_merge_itr;
|
const uint32_t resId = call_inst_itr->result_id();
|
||||||
|
assert(resId != 0);
|
||||||
|
AddLoad(calleeTypeId, resId, returnVarId, &new_blk_ptr);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Move instructions of original caller block after call instruction.
|
||||||
|
if (!MoveCallerInstsAfterFunctionCall(&preCallSB, &postCallSB, &new_blk_ptr,
|
||||||
|
call_inst_itr,
|
||||||
|
calleeFn->begin() != calleeFn->end()))
|
||||||
|
return false;
|
||||||
|
|
||||||
|
// Finalize inline code.
|
||||||
|
new_blocks->push_back(std::move(new_blk_ptr));
|
||||||
|
|
||||||
|
if (caller_is_loop_header && (new_blocks->size() > 1))
|
||||||
|
MoveLoopMergeInstToFirstBlock(new_blocks);
|
||||||
|
|
||||||
// Update block map given replacement blocks.
|
// Update block map given replacement blocks.
|
||||||
for (auto& blk : *new_blocks) {
|
for (auto& blk : *new_blocks) {
|
||||||
id2block_[blk->id()] = &*blk;
|
id2block_[blk->id()] = &*blk;
|
||||||
@ -624,7 +579,21 @@ bool InlinePass::IsInlinableFunctionCall(const Instruction* inst) {
|
|||||||
const uint32_t calleeFnId =
|
const uint32_t calleeFnId =
|
||||||
inst->GetSingleWordOperand(kSpvFunctionCallFunctionId);
|
inst->GetSingleWordOperand(kSpvFunctionCallFunctionId);
|
||||||
const auto ci = inlinable_.find(calleeFnId);
|
const auto ci = inlinable_.find(calleeFnId);
|
||||||
return ci != inlinable_.cend();
|
if (ci == inlinable_.cend()) return false;
|
||||||
|
|
||||||
|
if (early_return_funcs_.find(calleeFnId) != early_return_funcs_.end()) {
|
||||||
|
// We rely on the merge-return pass to handle the early return case
|
||||||
|
// in advance.
|
||||||
|
std::string message =
|
||||||
|
"The function '" + id2function_[calleeFnId]->DefInst().PrettyPrint() +
|
||||||
|
"' could not be inlined because the return instruction "
|
||||||
|
"is not at the end of the function. This could be fixed by "
|
||||||
|
"running merge-return before inlining.";
|
||||||
|
consumer()(SPV_MSG_WARNING, "", {0, 0, 0}, message.c_str());
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
void InlinePass::UpdateSucceedingPhis(
|
void InlinePass::UpdateSucceedingPhis(
|
||||||
@ -645,26 +614,6 @@ void InlinePass::UpdateSucceedingPhis(
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
bool InlinePass::HasNoReturnInStructuredConstruct(Function* func) {
|
|
||||||
// If control not structured, do not do loop/return analysis
|
|
||||||
// TODO: Analyze returns in non-structured control flow
|
|
||||||
if (!context()->get_feature_mgr()->HasCapability(SpvCapabilityShader))
|
|
||||||
return false;
|
|
||||||
const auto structured_analysis = context()->GetStructuredCFGAnalysis();
|
|
||||||
// Search for returns in structured construct.
|
|
||||||
bool return_in_construct = false;
|
|
||||||
for (auto& blk : *func) {
|
|
||||||
auto terminal_ii = blk.cend();
|
|
||||||
--terminal_ii;
|
|
||||||
if (spvOpcodeIsReturn(terminal_ii->opcode()) &&
|
|
||||||
structured_analysis->ContainingConstruct(blk.id()) != 0) {
|
|
||||||
return_in_construct = true;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return !return_in_construct;
|
|
||||||
}
|
|
||||||
|
|
||||||
bool InlinePass::HasNoReturnInLoop(Function* func) {
|
bool InlinePass::HasNoReturnInLoop(Function* func) {
|
||||||
// If control not structured, do not do loop/return analysis
|
// If control not structured, do not do loop/return analysis
|
||||||
// TODO: Analyze returns in non-structured control flow
|
// TODO: Analyze returns in non-structured control flow
|
||||||
@ -686,10 +635,18 @@ bool InlinePass::HasNoReturnInLoop(Function* func) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
void InlinePass::AnalyzeReturns(Function* func) {
|
void InlinePass::AnalyzeReturns(Function* func) {
|
||||||
|
// Analyze functions without a return in loop.
|
||||||
if (HasNoReturnInLoop(func)) {
|
if (HasNoReturnInLoop(func)) {
|
||||||
no_return_in_loop_.insert(func->result_id());
|
no_return_in_loop_.insert(func->result_id());
|
||||||
if (!HasNoReturnInStructuredConstruct(func))
|
}
|
||||||
|
// Analyze functions with a return before its tail basic block.
|
||||||
|
for (auto& blk : *func) {
|
||||||
|
auto terminal_ii = blk.cend();
|
||||||
|
--terminal_ii;
|
||||||
|
if (spvOpcodeIsReturn(terminal_ii->opcode()) && &blk != func->tail()) {
|
||||||
early_return_funcs_.insert(func->result_id());
|
early_return_funcs_.insert(func->result_id());
|
||||||
|
break;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -124,10 +124,6 @@ class InlinePass : public Pass {
|
|||||||
// Return true if |inst| is a function call that can be inlined.
|
// Return true if |inst| is a function call that can be inlined.
|
||||||
bool IsInlinableFunctionCall(const Instruction* inst);
|
bool IsInlinableFunctionCall(const Instruction* inst);
|
||||||
|
|
||||||
// Return true if |func| does not have a return that is
|
|
||||||
// nested in a structured if, switch or loop.
|
|
||||||
bool HasNoReturnInStructuredConstruct(Function* func);
|
|
||||||
|
|
||||||
// Return true if |func| has no return in a loop. The current analysis
|
// Return true if |func| has no return in a loop. The current analysis
|
||||||
// requires structured control flow, so return false if control flow not
|
// requires structured control flow, so return false if control flow not
|
||||||
// structured ie. module is not a shader.
|
// structured ie. module is not a shader.
|
||||||
@ -171,6 +167,64 @@ class InlinePass : public Pass {
|
|||||||
// Set of functions that are originally called directly or indirectly from a
|
// Set of functions that are originally called directly or indirectly from a
|
||||||
// continue construct.
|
// continue construct.
|
||||||
std::unordered_set<uint32_t> funcs_called_from_continue_;
|
std::unordered_set<uint32_t> funcs_called_from_continue_;
|
||||||
|
|
||||||
|
private:
|
||||||
|
// Moves instructions of the caller function up to the call instruction
|
||||||
|
// to |new_blk_ptr|.
|
||||||
|
void MoveInstsBeforeEntryBlock(
|
||||||
|
std::unordered_map<uint32_t, Instruction*>* preCallSB,
|
||||||
|
BasicBlock* new_blk_ptr, BasicBlock::iterator call_inst_itr,
|
||||||
|
UptrVectorIterator<BasicBlock> call_block_itr);
|
||||||
|
|
||||||
|
// Returns a new guard block after adding a branch to the end of
|
||||||
|
// |new_blocks|.
|
||||||
|
std::unique_ptr<BasicBlock> AddGuardBlock(
|
||||||
|
std::vector<std::unique_ptr<BasicBlock>>* new_blocks,
|
||||||
|
std::unordered_map<uint32_t, uint32_t>* callee2caller,
|
||||||
|
std::unique_ptr<BasicBlock> new_blk_ptr, uint32_t entry_blk_label_id);
|
||||||
|
|
||||||
|
// Add store instructions for initializers of variables.
|
||||||
|
InstructionList::iterator AddStoresForVariableInitializers(
|
||||||
|
const std::unordered_map<uint32_t, uint32_t>& callee2caller,
|
||||||
|
std::unique_ptr<BasicBlock>* new_blk_ptr,
|
||||||
|
UptrVectorIterator<BasicBlock> callee_block_itr);
|
||||||
|
|
||||||
|
// Inlines a single instruction of the callee function.
|
||||||
|
bool InlineInstructionInBB(
|
||||||
|
const std::unordered_map<uint32_t, uint32_t>& callee2caller,
|
||||||
|
BasicBlock* new_blk_ptr, const Instruction* inst);
|
||||||
|
|
||||||
|
// Inlines the return instruction of the callee function.
|
||||||
|
std::unique_ptr<BasicBlock> InlineReturn(
|
||||||
|
const std::unordered_map<uint32_t, uint32_t>& callee2caller,
|
||||||
|
std::vector<std::unique_ptr<BasicBlock>>* new_blocks,
|
||||||
|
std::unique_ptr<BasicBlock> new_blk_ptr, Function* calleeFn,
|
||||||
|
const Instruction* inst, uint32_t returnVarId);
|
||||||
|
|
||||||
|
// Inlines the entry block of the callee function.
|
||||||
|
bool InlineEntryBlock(
|
||||||
|
const std::unordered_map<uint32_t, uint32_t>& callee2caller,
|
||||||
|
std::unique_ptr<BasicBlock>* new_blk_ptr,
|
||||||
|
UptrVectorIterator<BasicBlock> callee_first_block);
|
||||||
|
|
||||||
|
// Inlines basic blocks of the callee function other than the entry basic
|
||||||
|
// block.
|
||||||
|
std::unique_ptr<BasicBlock> InlineBasicBlocks(
|
||||||
|
std::vector<std::unique_ptr<BasicBlock>>* new_blocks,
|
||||||
|
const std::unordered_map<uint32_t, uint32_t>& callee2caller,
|
||||||
|
std::unique_ptr<BasicBlock> new_blk_ptr, Function* calleeFn);
|
||||||
|
|
||||||
|
// Moves instructions of the caller function after the call instruction
|
||||||
|
// to |new_blk_ptr|.
|
||||||
|
bool MoveCallerInstsAfterFunctionCall(
|
||||||
|
std::unordered_map<uint32_t, Instruction*>* preCallSB,
|
||||||
|
std::unordered_map<uint32_t, uint32_t>* postCallSB,
|
||||||
|
std::unique_ptr<BasicBlock>* new_blk_ptr,
|
||||||
|
BasicBlock::iterator call_inst_itr, bool multiBlocks);
|
||||||
|
|
||||||
|
// Move the OpLoopMerge from the last block back to the first.
|
||||||
|
void MoveLoopMergeInstToFirstBlock(
|
||||||
|
std::vector<std::unique_ptr<BasicBlock>>* new_blocks);
|
||||||
};
|
};
|
||||||
|
|
||||||
} // namespace opt
|
} // namespace opt
|
||||||
|
@ -102,12 +102,12 @@ OpStore %30 %29
|
|||||||
OpStore %32 %31
|
OpStore %32 %31
|
||||||
%33 = OpLoad %S_t %s0
|
%33 = OpLoad %S_t %s0
|
||||||
OpStore %param %33
|
OpStore %param %33
|
||||||
%41 = OpAccessChain %_ptr_Function_18 %param %int_2
|
%42 = OpAccessChain %_ptr_Function_18 %param %int_2
|
||||||
%42 = OpLoad %18 %41
|
%43 = OpLoad %18 %42
|
||||||
%43 = OpAccessChain %_ptr_Function_v2float %param %int_0
|
%44 = OpAccessChain %_ptr_Function_v2float %param %int_0
|
||||||
%44 = OpLoad %v2float %43
|
%45 = OpLoad %v2float %44
|
||||||
%45 = OpImageSampleImplicitLod %v4float %42 %44
|
%46 = OpImageSampleImplicitLod %v4float %43 %45
|
||||||
OpStore %outColor %45
|
OpStore %outColor %46
|
||||||
OpReturn
|
OpReturn
|
||||||
OpFunctionEnd
|
OpFunctionEnd
|
||||||
)";
|
)";
|
||||||
@ -191,10 +191,10 @@ OpFunctionEnd
|
|||||||
%34 = OpVariable %_ptr_Function_20 Function
|
%34 = OpVariable %_ptr_Function_20 Function
|
||||||
%35 = OpVariable %_ptr_Function_20 Function
|
%35 = OpVariable %_ptr_Function_20 Function
|
||||||
%25 = OpVariable %_ptr_Function_20 Function
|
%25 = OpVariable %_ptr_Function_20 Function
|
||||||
%36 = OpLoad %20 %sampler16
|
%37 = OpLoad %20 %sampler16
|
||||||
OpStore %34 %36
|
OpStore %34 %37
|
||||||
%37 = OpLoad %20 %34
|
%38 = OpLoad %20 %34
|
||||||
OpStore %35 %37
|
OpStore %35 %38
|
||||||
%26 = OpLoad %20 %35
|
%26 = OpLoad %20 %35
|
||||||
OpStore %25 %26
|
OpStore %25 %26
|
||||||
%27 = OpLoad %20 %25
|
%27 = OpLoad %20 %25
|
||||||
@ -301,12 +301,12 @@ OpStore %31 %30
|
|||||||
OpStore %33 %32
|
OpStore %33 %32
|
||||||
%34 = OpLoad %S_t %s0
|
%34 = OpLoad %S_t %s0
|
||||||
OpStore %param %34
|
OpStore %param %34
|
||||||
%44 = OpAccessChain %_ptr_Function_19 %param %int_2
|
%45 = OpAccessChain %_ptr_Function_19 %param %int_2
|
||||||
%45 = OpLoad %19 %44
|
%46 = OpLoad %19 %45
|
||||||
%46 = OpAccessChain %_ptr_Function_v2float %param %int_0
|
%47 = OpAccessChain %_ptr_Function_v2float %param %int_0
|
||||||
%47 = OpLoad %v2float %46
|
%48 = OpLoad %v2float %47
|
||||||
%48 = OpImageSampleImplicitLod %v4float %45 %47
|
%49 = OpImageSampleImplicitLod %v4float %46 %48
|
||||||
OpStore %outColor %48
|
OpStore %outColor %49
|
||||||
OpReturn
|
OpReturn
|
||||||
OpFunctionEnd
|
OpFunctionEnd
|
||||||
)";
|
)";
|
||||||
|
File diff suppressed because it is too large
Load Diff
Loading…
Reference in New Issue
Block a user