2019-02-08 16:20:29 +00:00
|
|
|
// Copyright (c) 2019 Google LLC
|
|
|
|
//
|
|
|
|
// 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.
|
|
|
|
|
|
|
|
#include "eliminate_dead_functions_util.h"
|
|
|
|
|
|
|
|
namespace spvtools {
|
|
|
|
namespace opt {
|
|
|
|
|
|
|
|
namespace eliminatedeadfunctionsutil {
|
|
|
|
|
|
|
|
Module::iterator EliminateFunction(IRContext* context,
|
|
|
|
Module::iterator* func_iter) {
|
2020-08-13 18:54:14 +00:00
|
|
|
bool first_func = *func_iter == context->module()->begin();
|
|
|
|
bool seen_func_end = false;
|
2023-07-11 02:14:11 +00:00
|
|
|
std::unordered_set<Instruction*> to_kill;
|
2019-02-08 16:20:29 +00:00
|
|
|
(*func_iter)
|
2020-08-13 18:54:14 +00:00
|
|
|
->ForEachInst(
|
2023-07-11 02:14:11 +00:00
|
|
|
[context, first_func, func_iter, &seen_func_end,
|
|
|
|
&to_kill](Instruction* inst) {
|
2024-05-21 06:26:42 +00:00
|
|
|
if (inst->opcode() == spv::Op::OpFunctionEnd) {
|
2020-08-13 18:54:14 +00:00
|
|
|
seen_func_end = true;
|
|
|
|
}
|
|
|
|
// Move non-semantic instructions to the previous function or
|
|
|
|
// global values if this is the first function.
|
2024-05-21 06:26:42 +00:00
|
|
|
if (seen_func_end && inst->opcode() == spv::Op::OpExtInst) {
|
2020-08-13 18:54:14 +00:00
|
|
|
assert(inst->IsNonSemanticInstruction());
|
2023-07-11 02:14:11 +00:00
|
|
|
if (to_kill.find(inst) != to_kill.end()) return;
|
2020-08-13 18:54:14 +00:00
|
|
|
std::unique_ptr<Instruction> clone(inst->Clone(context));
|
2024-05-21 06:26:42 +00:00
|
|
|
// Clear uses of "inst" to in case this moves a dependent chain of
|
|
|
|
// instructions.
|
|
|
|
context->get_def_use_mgr()->ClearInst(inst);
|
2020-08-13 18:54:14 +00:00
|
|
|
context->AnalyzeDefUse(clone.get());
|
|
|
|
if (first_func) {
|
|
|
|
context->AddGlobalValue(std::move(clone));
|
|
|
|
} else {
|
|
|
|
auto prev_func_iter = *func_iter;
|
|
|
|
--prev_func_iter;
|
|
|
|
prev_func_iter->AddNonSemanticInstruction(std::move(clone));
|
|
|
|
}
|
|
|
|
inst->ToNop();
|
2023-07-11 02:14:11 +00:00
|
|
|
} else if (to_kill.find(inst) == to_kill.end()) {
|
|
|
|
context->CollectNonSemanticTree(inst, &to_kill);
|
2020-08-13 18:54:14 +00:00
|
|
|
context->KillInst(inst);
|
|
|
|
}
|
|
|
|
},
|
|
|
|
true, true);
|
2023-07-11 02:14:11 +00:00
|
|
|
|
|
|
|
for (auto* dead : to_kill) {
|
|
|
|
context->KillInst(dead);
|
|
|
|
}
|
|
|
|
|
2019-02-08 16:20:29 +00:00
|
|
|
return func_iter->Erase();
|
|
|
|
}
|
|
|
|
|
|
|
|
} // namespace eliminatedeadfunctionsutil
|
|
|
|
} // namespace opt
|
|
|
|
} // namespace spvtools
|