bgfx/3rdparty/spirv-tools/source/opt/private_to_local_pass.cpp

211 lines
7.1 KiB
C++
Raw Normal View History

2018-04-11 05:44:28 +03:00
// Copyright (c) 2017 Google Inc.
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
2018-09-03 07:14:20 +03:00
#include "source/opt/private_to_local_pass.h"
2018-04-11 05:44:28 +03:00
2018-09-03 07:14:20 +03:00
#include <memory>
#include <utility>
#include <vector>
2018-04-11 05:44:28 +03:00
2018-09-03 07:14:20 +03:00
#include "source/opt/ir_context.h"
2019-05-18 22:27:37 +03:00
#include "source/spirv_constant.h"
2018-09-03 07:14:20 +03:00
namespace spvtools {
namespace opt {
2018-04-11 05:44:28 +03:00
namespace {
2018-09-03 07:14:20 +03:00
2018-04-11 05:44:28 +03:00
const uint32_t kVariableStorageClassInIdx = 0;
const uint32_t kSpvTypePointerTypeIdInIdx = 1;
2018-09-03 07:14:20 +03:00
} // namespace
2018-04-11 05:44:28 +03:00
2018-09-03 07:14:20 +03:00
Pass::Status PrivateToLocalPass::Process() {
2018-04-11 05:44:28 +03:00
bool modified = false;
// Private variables require the shader capability. If this is not a shader,
// there is no work to do.
if (context()->get_feature_mgr()->HasCapability(SpvCapabilityAddresses))
return Status::SuccessWithoutChange;
2018-09-03 07:14:20 +03:00
std::vector<std::pair<Instruction*, Function*>> variables_to_move;
2019-05-18 22:27:37 +03:00
std::unordered_set<uint32_t> localized_variables;
2018-04-11 05:44:28 +03:00
for (auto& inst : context()->types_values()) {
if (inst.opcode() != SpvOpVariable) {
continue;
}
if (inst.GetSingleWordInOperand(kVariableStorageClassInIdx) !=
SpvStorageClassPrivate) {
continue;
}
2018-09-03 07:14:20 +03:00
Function* target_function = FindLocalFunction(inst);
2018-04-11 05:44:28 +03:00
if (target_function != nullptr) {
variables_to_move.push_back({&inst, target_function});
}
}
modified = !variables_to_move.empty();
for (auto p : variables_to_move) {
MoveVariable(p.first, p.second);
2019-05-18 22:27:37 +03:00
localized_variables.insert(p.first->result_id());
}
if (get_module()->version() >= SPV_SPIRV_VERSION_WORD(1, 4)) {
// In SPIR-V 1.4 and later entry points must list private storage class
// variables that are statically used by the entry point. Go through the
// entry points and remove any references to variables that were localized.
for (auto& entry : get_module()->entry_points()) {
std::vector<Operand> new_operands;
for (uint32_t i = 0; i < entry.NumInOperands(); ++i) {
// Execution model, function id and name are always kept.
if (i < 3 ||
!localized_variables.count(entry.GetSingleWordInOperand(i))) {
new_operands.push_back(entry.GetInOperand(i));
}
}
if (new_operands.size() != entry.NumInOperands()) {
entry.SetInOperands(std::move(new_operands));
context()->AnalyzeUses(&entry);
}
}
2018-04-11 05:44:28 +03:00
}
return (modified ? Status::SuccessWithChange : Status::SuccessWithoutChange);
}
2018-09-03 07:14:20 +03:00
Function* PrivateToLocalPass::FindLocalFunction(const Instruction& inst) const {
2018-04-11 05:44:28 +03:00
bool found_first_use = false;
2018-09-03 07:14:20 +03:00
Function* target_function = nullptr;
2018-04-11 05:44:28 +03:00
context()->get_def_use_mgr()->ForEachUser(
inst.result_id(),
2018-09-03 07:14:20 +03:00
[&target_function, &found_first_use, this](Instruction* use) {
BasicBlock* current_block = context()->get_instr_block(use);
2018-04-11 05:44:28 +03:00
if (current_block == nullptr) {
return;
}
if (!IsValidUse(use)) {
found_first_use = true;
target_function = nullptr;
return;
}
2018-09-03 07:14:20 +03:00
Function* current_function = current_block->GetParent();
2018-04-11 05:44:28 +03:00
if (!found_first_use) {
found_first_use = true;
target_function = current_function;
} else if (target_function != current_function) {
target_function = nullptr;
}
});
return target_function;
} // namespace opt
2018-09-03 07:14:20 +03:00
void PrivateToLocalPass::MoveVariable(Instruction* variable,
Function* function) {
2018-04-11 05:44:28 +03:00
// The variable needs to be removed from the global section, and placed in the
// header of the function. First step remove from the global list.
variable->RemoveFromList();
2018-09-03 07:14:20 +03:00
std::unique_ptr<Instruction> var(variable); // Take ownership.
2018-04-11 05:44:28 +03:00
context()->ForgetUses(variable);
// Update the storage class of the variable.
variable->SetInOperand(kVariableStorageClassInIdx, {SpvStorageClassFunction});
// Update the type as well.
uint32_t new_type_id = GetNewType(variable->type_id());
variable->SetResultType(new_type_id);
// Place the variable at the start of the first basic block.
context()->AnalyzeUses(variable);
2018-09-03 07:14:20 +03:00
context()->set_instr_block(variable, &*function->begin());
2018-04-11 05:44:28 +03:00
function->begin()->begin()->InsertBefore(move(var));
// Update uses where the type may have changed.
UpdateUses(variable->result_id());
}
uint32_t PrivateToLocalPass::GetNewType(uint32_t old_type_id) {
auto type_mgr = context()->get_type_mgr();
2018-09-03 07:14:20 +03:00
Instruction* old_type_inst = get_def_use_mgr()->GetDef(old_type_id);
2018-04-11 05:44:28 +03:00
uint32_t pointee_type_id =
old_type_inst->GetSingleWordInOperand(kSpvTypePointerTypeIdInIdx);
uint32_t new_type_id =
type_mgr->FindPointerToType(pointee_type_id, SpvStorageClassFunction);
2018-09-03 07:14:20 +03:00
context()->UpdateDefUse(context()->get_def_use_mgr()->GetDef(new_type_id));
2018-04-11 05:44:28 +03:00
return new_type_id;
}
2018-09-03 07:14:20 +03:00
bool PrivateToLocalPass::IsValidUse(const Instruction* inst) const {
2018-04-11 05:44:28 +03:00
// The cases in this switch have to match the cases in |UpdateUse|.
// If we don't know how to update it, it is not valid.
switch (inst->opcode()) {
case SpvOpLoad:
case SpvOpStore:
case SpvOpImageTexelPointer: // Treat like a load
return true;
case SpvOpAccessChain:
return context()->get_def_use_mgr()->WhileEachUser(
2018-09-03 07:14:20 +03:00
inst, [this](const Instruction* user) {
2018-04-11 05:44:28 +03:00
if (!IsValidUse(user)) return false;
return true;
});
case SpvOpName:
return true;
default:
return spvOpcodeIsDecoration(inst->opcode());
}
}
2018-09-03 07:14:20 +03:00
void PrivateToLocalPass::UpdateUse(Instruction* inst) {
2018-04-11 05:44:28 +03:00
// The cases in this switch have to match the cases in |IsValidUse|. If we
// don't think it is valid, the optimization will not view the variable as a
// candidate, and therefore the use will not be updated.
switch (inst->opcode()) {
case SpvOpLoad:
case SpvOpStore:
case SpvOpImageTexelPointer: // Treat like a load
// The type is fine because it is the type pointed to, and that does not
// change.
break;
case SpvOpAccessChain:
context()->ForgetUses(inst);
inst->SetResultType(GetNewType(inst->type_id()));
context()->AnalyzeUses(inst);
// Update uses where the type may have changed.
UpdateUses(inst->result_id());
break;
case SpvOpName:
2019-05-18 22:27:37 +03:00
case SpvOpEntryPoint: // entry points will be updated separately.
2018-04-11 05:44:28 +03:00
break;
default:
assert(spvOpcodeIsDecoration(inst->opcode()) &&
"Do not know how to update the type for this instruction.");
break;
}
}
void PrivateToLocalPass::UpdateUses(uint32_t id) {
2018-09-03 07:14:20 +03:00
std::vector<Instruction*> uses;
context()->get_def_use_mgr()->ForEachUser(
id, [&uses](Instruction* use) { uses.push_back(use); });
2018-04-11 05:44:28 +03:00
2018-09-03 07:14:20 +03:00
for (Instruction* use : uses) {
2018-04-11 05:44:28 +03:00
UpdateUse(use);
}
}
} // namespace opt
} // namespace spvtools