2016-07-27 21:37:04 +00:00
|
|
|
// Copyright (c) 2016 Google Inc.
|
|
|
|
//
|
|
|
|
// Permission is hereby granted, free of charge, to any person obtaining a
|
|
|
|
// copy of this software and/or associated documentation files (the
|
|
|
|
// "Materials"), to deal in the Materials without restriction, including
|
|
|
|
// without limitation the rights to use, copy, modify, merge, publish,
|
|
|
|
// distribute, sublicense, and/or sell copies of the Materials, and to
|
|
|
|
// permit persons to whom the Materials are furnished to do so, subject to
|
|
|
|
// the following conditions:
|
|
|
|
//
|
|
|
|
// The above copyright notice and this permission notice shall be included
|
|
|
|
// in all copies or substantial portions of the Materials.
|
|
|
|
//
|
|
|
|
// MODIFICATIONS TO THIS FILE MAY MEAN IT NO LONGER ACCURATELY REFLECTS
|
|
|
|
// KHRONOS STANDARDS. THE UNMODIFIED, NORMATIVE VERSIONS OF KHRONOS
|
|
|
|
// SPECIFICATIONS AND HEADER INFORMATION ARE LOCATED AT
|
|
|
|
// https://www.khronos.org/registry/
|
|
|
|
//
|
|
|
|
// THE MATERIALS ARE PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
|
|
|
|
// EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
|
|
|
// MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.
|
|
|
|
// IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY
|
|
|
|
// CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT,
|
|
|
|
// TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE
|
|
|
|
// MATERIALS OR THE USE OR OTHER DEALINGS IN THE MATERIALS.
|
|
|
|
|
|
|
|
#include "def_use_manager.h"
|
|
|
|
|
|
|
|
#include <cassert>
|
|
|
|
#include <functional>
|
|
|
|
|
|
|
|
#include "instruction.h"
|
|
|
|
#include "module.h"
|
|
|
|
|
|
|
|
namespace spvtools {
|
|
|
|
namespace opt {
|
|
|
|
namespace analysis {
|
|
|
|
|
|
|
|
void DefUseManager::AnalyzeDefUse(ir::Module* module) {
|
|
|
|
module->ForEachInst(std::bind(&DefUseManager::AnalyzeInstDefUse, this,
|
|
|
|
std::placeholders::_1));
|
|
|
|
}
|
|
|
|
|
2016-08-12 14:05:58 +00:00
|
|
|
void DefUseManager::AnalyzeInstDefUse(ir::Instruction* inst) {
|
|
|
|
const uint32_t def_id = inst->result_id();
|
2016-08-12 14:43:26 +00:00
|
|
|
// Clear the records of def_id first if it has been recorded before.
|
|
|
|
ClearDef(def_id);
|
|
|
|
|
2016-08-12 14:05:58 +00:00
|
|
|
if (def_id != 0) id_to_def_[def_id] = inst;
|
|
|
|
|
|
|
|
for (uint32_t i = 0; i < inst->NumOperands(); ++i) {
|
|
|
|
switch (inst->GetOperand(i).type) {
|
|
|
|
// For any id type but result id type
|
|
|
|
case SPV_OPERAND_TYPE_ID:
|
|
|
|
case SPV_OPERAND_TYPE_TYPE_ID:
|
|
|
|
case SPV_OPERAND_TYPE_MEMORY_SEMANTICS_ID:
|
|
|
|
case SPV_OPERAND_TYPE_SCOPE_ID: {
|
|
|
|
uint32_t use_id = inst->GetSingleWordOperand(i);
|
|
|
|
// use_id is used by the instruction generating def_id.
|
|
|
|
id_to_uses_[use_id].push_back({inst, i});
|
|
|
|
if (def_id != 0) result_id_to_used_ids_[def_id].push_back(use_id);
|
|
|
|
} break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-07-27 21:37:04 +00:00
|
|
|
ir::Instruction* DefUseManager::GetDef(uint32_t id) {
|
|
|
|
if (id_to_def_.count(id) == 0) return nullptr;
|
|
|
|
return id_to_def_.at(id);
|
|
|
|
}
|
|
|
|
|
|
|
|
UseList* DefUseManager::GetUses(uint32_t id) {
|
|
|
|
if (id_to_uses_.count(id) == 0) return nullptr;
|
|
|
|
return &id_to_uses_.at(id);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool DefUseManager::KillDef(uint32_t id) {
|
|
|
|
if (id_to_def_.count(id) == 0) return false;
|
|
|
|
|
2016-08-12 14:43:26 +00:00
|
|
|
ir::Instruction* defining_inst = id_to_def_.at(id);
|
|
|
|
ClearDef(id);
|
|
|
|
defining_inst->ToNop();
|
2016-07-27 21:37:04 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool DefUseManager::ReplaceAllUsesWith(uint32_t before, uint32_t after) {
|
|
|
|
if (before == after) return false;
|
|
|
|
if (id_to_uses_.count(before) == 0) return false;
|
|
|
|
|
|
|
|
for (auto it = id_to_uses_[before].cbegin(); it != id_to_uses_[before].cend();
|
|
|
|
++it) {
|
|
|
|
// Make the modification in the instruction.
|
|
|
|
it->inst->SetOperand(it->operand_index, {after});
|
|
|
|
// Register the use of |after| id into id_to_uses_.
|
|
|
|
// TODO(antiagainst): de-duplication.
|
|
|
|
id_to_uses_[after].push_back({it->inst, it->operand_index});
|
|
|
|
}
|
|
|
|
id_to_uses_.erase(before);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2016-08-12 14:43:26 +00:00
|
|
|
void DefUseManager::ClearDef(uint32_t def_id) {
|
|
|
|
if (id_to_def_.count(def_id) == 0) return;
|
|
|
|
|
|
|
|
// Go through all ids used by this instruction, remove this instruction's
|
|
|
|
// uses of them.
|
|
|
|
for (const auto use_id : result_id_to_used_ids_[def_id]) {
|
|
|
|
if (id_to_uses_.count(use_id) == 0) continue;
|
|
|
|
auto& uses = id_to_uses_[use_id];
|
|
|
|
for (auto it = uses.begin(); it != uses.end();) {
|
|
|
|
if (it->inst->result_id() == def_id) {
|
|
|
|
it = uses.erase(it);
|
|
|
|
} else {
|
|
|
|
++it;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (uses.empty()) id_to_uses_.erase(use_id);
|
|
|
|
}
|
|
|
|
result_id_to_used_ids_.erase(def_id);
|
|
|
|
id_to_uses_.erase(def_id); // Remove all uses of this id.
|
|
|
|
id_to_def_.erase(def_id);
|
|
|
|
}
|
|
|
|
|
2016-07-27 21:37:04 +00:00
|
|
|
} // namespace analysis
|
|
|
|
} // namespace opt
|
|
|
|
} // namespace spvtools
|