| // Copyright 2016 The SwiftShader Authors. All Rights Reserved. |
| // |
| // 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 "Optimizer.hpp" |
| |
| #include "src/IceCfg.h" |
| #include "src/IceCfgNode.h" |
| |
| #include <vector> |
| |
| namespace { |
| |
| class Optimizer |
| { |
| public: |
| void run(Ice::Cfg *function); |
| |
| private: |
| void analyzeUses(Ice::Cfg *function); |
| void eliminateDeadCode(); |
| void eliminateUnitializedLoads(); |
| void eliminateLoadsFollowingSingleStore(); |
| void optimizeStoresInSingleBasicBlock(); |
| |
| void replace(Ice::Inst *instruction, Ice::Operand *newValue); |
| void deleteInstruction(Ice::Inst *instruction); |
| bool isDead(Ice::Inst *instruction); |
| |
| static const Ice::InstIntrinsicCall *asLoadSubVector(const Ice::Inst *instruction); |
| static const Ice::InstIntrinsicCall *asStoreSubVector(const Ice::Inst *instruction); |
| static bool isLoad(const Ice::Inst &instruction); |
| static bool isStore(const Ice::Inst &instruction); |
| static Ice::Operand *storeAddress(const Ice::Inst *instruction); |
| static Ice::Operand *loadAddress(const Ice::Inst *instruction); |
| static Ice::Operand *storeData(const Ice::Inst *instruction); |
| static std::size_t storeSize(const Ice::Inst *instruction); |
| static bool loadTypeMatchesStore(const Ice::Inst *load, const Ice::Inst *store); |
| |
| Ice::Cfg *function; |
| Ice::GlobalContext *context; |
| |
| struct Uses : std::vector<Ice::Inst *> |
| { |
| bool areOnlyLoadStore() const; |
| void insert(Ice::Operand *value, Ice::Inst *instruction); |
| void erase(Ice::Inst *instruction); |
| |
| std::vector<Ice::Inst *> loads; |
| std::vector<Ice::Inst *> stores; |
| }; |
| |
| struct LoadStoreInst |
| { |
| LoadStoreInst(Ice::Inst *inst, bool isStore) |
| : inst(inst) |
| , address(isStore ? storeAddress(inst) : loadAddress(inst)) |
| , isStore(isStore) |
| { |
| } |
| |
| Ice::Inst *inst; |
| Ice::Operand *address; |
| bool isStore; |
| }; |
| |
| Optimizer::Uses *getUses(Ice::Operand *); |
| void setUses(Ice::Operand *, Optimizer::Uses *); |
| bool hasUses(Ice::Operand *) const; |
| |
| Ice::CfgNode *getNode(Ice::Inst *); |
| void setNode(Ice::Inst *, Ice::CfgNode *); |
| |
| Ice::Inst *getDefinition(Ice::Variable *); |
| void setDefinition(Ice::Variable *, Ice::Inst *); |
| |
| const std::vector<LoadStoreInst> &getLoadStoreInsts(Ice::CfgNode *); |
| void setLoadStoreInsts(Ice::CfgNode *, std::vector<LoadStoreInst> *); |
| bool hasLoadStoreInsts(Ice::CfgNode *node) const; |
| |
| std::vector<Ice::Operand *> operandsWithUses; |
| }; |
| |
| void Optimizer::run(Ice::Cfg *function) |
| { |
| this->function = function; |
| this->context = function->getContext(); |
| |
| analyzeUses(function); |
| |
| eliminateDeadCode(); |
| eliminateUnitializedLoads(); |
| eliminateLoadsFollowingSingleStore(); |
| optimizeStoresInSingleBasicBlock(); |
| eliminateDeadCode(); |
| |
| for(auto operand : operandsWithUses) |
| { |
| // Deletes the Uses instance on the operand |
| setUses(operand, nullptr); |
| } |
| operandsWithUses.clear(); |
| } |
| |
| void Optimizer::eliminateDeadCode() |
| { |
| bool modified; |
| do |
| { |
| modified = false; |
| for(Ice::CfgNode *basicBlock : function->getNodes()) |
| { |
| for(Ice::Inst &inst : Ice::reverse_range(basicBlock->getInsts())) |
| { |
| if(inst.isDeleted()) |
| { |
| continue; |
| } |
| |
| if(isDead(&inst)) |
| { |
| deleteInstruction(&inst); |
| modified = true; |
| } |
| } |
| } |
| } while(modified); |
| } |
| |
| void Optimizer::eliminateUnitializedLoads() |
| { |
| Ice::CfgNode *entryBlock = function->getEntryNode(); |
| |
| for(Ice::Inst &alloca : entryBlock->getInsts()) |
| { |
| if(alloca.isDeleted()) |
| { |
| continue; |
| } |
| |
| if(!llvm::isa<Ice::InstAlloca>(alloca)) |
| { |
| break; // Allocas are all at the top |
| } |
| |
| Ice::Operand *address = alloca.getDest(); |
| |
| if(!hasUses(address)) |
| { |
| continue; |
| } |
| |
| const auto &addressUses = *getUses(address); |
| |
| if(!addressUses.areOnlyLoadStore()) |
| { |
| continue; |
| } |
| |
| if(addressUses.stores.empty()) |
| { |
| for(Ice::Inst *load : addressUses.loads) |
| { |
| Ice::Variable *loadData = load->getDest(); |
| |
| if(hasUses(loadData)) |
| { |
| for(Ice::Inst *use : *getUses(loadData)) |
| { |
| for(Ice::SizeT i = 0; i < use->getSrcSize(); i++) |
| { |
| if(use->getSrc(i) == loadData) |
| { |
| auto *undef = context->getConstantUndef(loadData->getType()); |
| |
| use->replaceSource(i, undef); |
| } |
| } |
| } |
| |
| setUses(loadData, nullptr); |
| } |
| |
| load->setDeleted(); |
| } |
| |
| alloca.setDeleted(); |
| setUses(address, nullptr); |
| } |
| } |
| } |
| |
| void Optimizer::eliminateLoadsFollowingSingleStore() |
| { |
| Ice::CfgNode *entryBlock = function->getEntryNode(); |
| |
| for(Ice::Inst &alloca : entryBlock->getInsts()) |
| { |
| if(alloca.isDeleted()) |
| { |
| continue; |
| } |
| |
| if(!llvm::isa<Ice::InstAlloca>(alloca)) |
| { |
| break; // Allocas are all at the top |
| } |
| |
| Ice::Operand *address = alloca.getDest(); |
| |
| if(!hasUses(address)) |
| { |
| continue; |
| } |
| |
| auto &addressUses = *getUses(address); |
| |
| if(!addressUses.areOnlyLoadStore()) |
| { |
| continue; |
| } |
| |
| if(addressUses.stores.size() == 1) |
| { |
| Ice::Inst *store = addressUses.stores[0]; |
| Ice::Operand *storeValue = storeData(store); |
| |
| for(Ice::Inst *load = &*++store->getIterator(), *next = nullptr; load != next; next = load, load = &*++store->getIterator()) |
| { |
| if(load->isDeleted() || !isLoad(*load)) |
| { |
| continue; |
| } |
| |
| if(loadAddress(load) != address) |
| { |
| continue; |
| } |
| |
| if(!loadTypeMatchesStore(load, store)) |
| { |
| continue; |
| } |
| |
| // TODO(b/148272103): InstLoad assumes that a constant ptr is an offset, rather than an |
| // absolute address. We need to make sure we don't replace a variable with a constant |
| // on this load. |
| if(llvm::isa<Ice::Constant>(storeValue)) |
| { |
| continue; |
| } |
| |
| replace(load, storeValue); |
| |
| for(size_t i = 0; i < addressUses.loads.size(); i++) |
| { |
| if(addressUses.loads[i] == load) |
| { |
| addressUses.loads[i] = addressUses.loads.back(); |
| addressUses.loads.pop_back(); |
| break; |
| } |
| } |
| |
| for(size_t i = 0; i < addressUses.size(); i++) |
| { |
| if(addressUses[i] == load) |
| { |
| addressUses[i] = addressUses.back(); |
| addressUses.pop_back(); |
| break; |
| } |
| } |
| |
| if(addressUses.size() == 1) |
| { |
| assert(addressUses[0] == store); |
| |
| alloca.setDeleted(); |
| store->setDeleted(); |
| setUses(address, nullptr); |
| |
| if(hasUses(storeValue)) |
| { |
| auto &valueUses = *getUses(storeValue); |
| |
| for(size_t i = 0; i < valueUses.size(); i++) |
| { |
| if(valueUses[i] == store) |
| { |
| valueUses[i] = valueUses.back(); |
| valueUses.pop_back(); |
| break; |
| } |
| } |
| |
| if(valueUses.empty()) |
| { |
| setUses(storeValue, nullptr); |
| } |
| } |
| |
| break; |
| } |
| } |
| } |
| } |
| } |
| |
| void Optimizer::optimizeStoresInSingleBasicBlock() |
| { |
| Ice::CfgNode *entryBlock = function->getEntryNode(); |
| |
| std::vector<std::vector<LoadStoreInst> *> allocatedVectors; |
| |
| for(Ice::Inst &alloca : entryBlock->getInsts()) |
| { |
| if(alloca.isDeleted()) |
| { |
| continue; |
| } |
| |
| if(!llvm::isa<Ice::InstAlloca>(alloca)) |
| { |
| break; // Allocas are all at the top |
| } |
| |
| Ice::Operand *address = alloca.getDest(); |
| |
| if(!hasUses(address)) |
| { |
| continue; |
| } |
| |
| const auto &addressUses = *getUses(address); |
| |
| if(!addressUses.areOnlyLoadStore()) |
| { |
| continue; |
| } |
| |
| Ice::CfgNode *singleBasicBlock = getNode(addressUses.stores[0]); |
| |
| for(size_t i = 1; i < addressUses.stores.size(); i++) |
| { |
| Ice::Inst *store = addressUses.stores[i]; |
| if(getNode(store) != singleBasicBlock) |
| { |
| singleBasicBlock = nullptr; |
| break; |
| } |
| } |
| |
| if(singleBasicBlock) |
| { |
| if(!hasLoadStoreInsts(singleBasicBlock)) |
| { |
| std::vector<LoadStoreInst> *loadStoreInstVector = new std::vector<LoadStoreInst>(); |
| setLoadStoreInsts(singleBasicBlock, loadStoreInstVector); |
| allocatedVectors.push_back(loadStoreInstVector); |
| for(Ice::Inst &inst : singleBasicBlock->getInsts()) |
| { |
| if(inst.isDeleted()) |
| { |
| continue; |
| } |
| |
| bool isStoreInst = isStore(inst); |
| bool isLoadInst = isLoad(inst); |
| |
| if(isStoreInst || isLoadInst) |
| { |
| loadStoreInstVector->push_back(LoadStoreInst(&inst, isStoreInst)); |
| } |
| } |
| } |
| |
| Ice::Inst *store = nullptr; |
| Ice::Operand *storeValue = nullptr; |
| bool unmatchedLoads = false; |
| |
| for(auto &loadStoreInst : getLoadStoreInsts(singleBasicBlock)) |
| { |
| Ice::Inst *inst = loadStoreInst.inst; |
| |
| if((loadStoreInst.address != address) || inst->isDeleted()) |
| { |
| continue; |
| } |
| |
| if(loadStoreInst.isStore) |
| { |
| // New store found. If we had a previous one, try to eliminate it. |
| if(store && !unmatchedLoads) |
| { |
| // If the previous store is wider than the new one, we can't eliminate it |
| // because there could be a wide load reading its non-overwritten data. |
| if(storeSize(inst) >= storeSize(store)) |
| { |
| deleteInstruction(store); |
| } |
| } |
| |
| store = inst; |
| storeValue = storeData(store); |
| unmatchedLoads = false; |
| } |
| else |
| { |
| if(!loadTypeMatchesStore(inst, store)) |
| { |
| unmatchedLoads = true; |
| continue; |
| } |
| |
| // TODO(b/148272103): InstLoad assumes that a constant ptr is an offset, rather than an |
| // absolute address. We need to make sure we don't replace a variable with a constant |
| // on this load. |
| if(llvm::isa<Ice::Constant>(storeValue)) |
| { |
| unmatchedLoads = true; |
| continue; |
| } |
| |
| replace(inst, storeValue); |
| } |
| } |
| } |
| } |
| |
| for(auto loadStoreInstVector : allocatedVectors) |
| { |
| delete loadStoreInstVector; |
| } |
| } |
| |
| void Optimizer::analyzeUses(Ice::Cfg *function) |
| { |
| for(Ice::CfgNode *basicBlock : function->getNodes()) |
| { |
| for(Ice::Inst &instruction : basicBlock->getInsts()) |
| { |
| if(instruction.isDeleted()) |
| { |
| continue; |
| } |
| |
| setNode(&instruction, basicBlock); |
| if(instruction.getDest()) |
| { |
| setDefinition(instruction.getDest(), &instruction); |
| } |
| |
| for(Ice::SizeT i = 0; i < instruction.getSrcSize(); i++) |
| { |
| Ice::SizeT unique = 0; |
| for(; unique < i; unique++) |
| { |
| if(instruction.getSrc(i) == instruction.getSrc(unique)) |
| { |
| break; |
| } |
| } |
| |
| if(i == unique) |
| { |
| Ice::Operand *src = instruction.getSrc(i); |
| getUses(src)->insert(src, &instruction); |
| } |
| } |
| } |
| } |
| } |
| |
| void Optimizer::replace(Ice::Inst *instruction, Ice::Operand *newValue) |
| { |
| Ice::Variable *oldValue = instruction->getDest(); |
| |
| if(!newValue) |
| { |
| newValue = context->getConstantUndef(oldValue->getType()); |
| } |
| |
| if(hasUses(oldValue)) |
| { |
| for(Ice::Inst *use : *getUses(oldValue)) |
| { |
| assert(!use->isDeleted()); // Should have been removed from uses already |
| |
| for(Ice::SizeT i = 0; i < use->getSrcSize(); i++) |
| { |
| if(use->getSrc(i) == oldValue) |
| { |
| use->replaceSource(i, newValue); |
| } |
| } |
| |
| getUses(newValue)->insert(newValue, use); |
| } |
| |
| setUses(oldValue, nullptr); |
| } |
| |
| deleteInstruction(instruction); |
| } |
| |
| void Optimizer::deleteInstruction(Ice::Inst *instruction) |
| { |
| if(!instruction || instruction->isDeleted()) |
| { |
| return; |
| } |
| |
| instruction->setDeleted(); |
| |
| for(Ice::SizeT i = 0; i < instruction->getSrcSize(); i++) |
| { |
| Ice::Operand *src = instruction->getSrc(i); |
| |
| if(hasUses(src)) |
| { |
| auto &srcUses = *getUses(src); |
| |
| srcUses.erase(instruction); |
| |
| if(srcUses.empty()) |
| { |
| setUses(src, nullptr); |
| |
| if(Ice::Variable *var = llvm::dyn_cast<Ice::Variable>(src)) |
| { |
| deleteInstruction(getDefinition(var)); |
| } |
| } |
| } |
| } |
| } |
| |
| bool Optimizer::isDead(Ice::Inst *instruction) |
| { |
| Ice::Variable *dest = instruction->getDest(); |
| |
| if(dest) |
| { |
| return (!hasUses(dest) || getUses(dest)->empty()) && !instruction->hasSideEffects(); |
| } |
| else if(isStore(*instruction)) |
| { |
| if(Ice::Variable *address = llvm::dyn_cast<Ice::Variable>(storeAddress(instruction))) |
| { |
| Ice::Inst *def = getDefinition(address); |
| |
| if(def && llvm::isa<Ice::InstAlloca>(def)) |
| { |
| if(hasUses(address)) |
| { |
| Optimizer::Uses *uses = getUses(address); |
| return uses->size() == uses->stores.size(); // Dead if all uses are stores |
| } |
| else |
| { |
| return true; // No uses |
| } |
| } |
| } |
| } |
| |
| return false; |
| } |
| |
| const Ice::InstIntrinsicCall *Optimizer::asLoadSubVector(const Ice::Inst *instruction) |
| { |
| if(auto *instrinsic = llvm::dyn_cast<Ice::InstIntrinsicCall>(instruction)) |
| { |
| if(instrinsic->getIntrinsicInfo().ID == Ice::Intrinsics::LoadSubVector) |
| { |
| return instrinsic; |
| } |
| } |
| |
| return nullptr; |
| } |
| |
| const Ice::InstIntrinsicCall *Optimizer::asStoreSubVector(const Ice::Inst *instruction) |
| { |
| if(auto *instrinsic = llvm::dyn_cast<Ice::InstIntrinsicCall>(instruction)) |
| { |
| if(instrinsic->getIntrinsicInfo().ID == Ice::Intrinsics::StoreSubVector) |
| { |
| return instrinsic; |
| } |
| } |
| |
| return nullptr; |
| } |
| |
| bool Optimizer::isLoad(const Ice::Inst &instruction) |
| { |
| if(llvm::isa<Ice::InstLoad>(&instruction)) |
| { |
| return true; |
| } |
| |
| return asLoadSubVector(&instruction) != nullptr; |
| } |
| |
| bool Optimizer::isStore(const Ice::Inst &instruction) |
| { |
| if(llvm::isa<Ice::InstStore>(&instruction)) |
| { |
| return true; |
| } |
| |
| return asStoreSubVector(&instruction) != nullptr; |
| } |
| |
| Ice::Operand *Optimizer::storeAddress(const Ice::Inst *instruction) |
| { |
| assert(isStore(*instruction)); |
| |
| if(auto *store = llvm::dyn_cast<Ice::InstStore>(instruction)) |
| { |
| return store->getAddr(); |
| } |
| |
| if(auto *storeSubVector = asStoreSubVector(instruction)) |
| { |
| return storeSubVector->getSrc(2); |
| } |
| |
| return nullptr; |
| } |
| |
| Ice::Operand *Optimizer::loadAddress(const Ice::Inst *instruction) |
| { |
| assert(isLoad(*instruction)); |
| |
| if(auto *load = llvm::dyn_cast<Ice::InstLoad>(instruction)) |
| { |
| return load->getSourceAddress(); |
| } |
| |
| if(auto *loadSubVector = asLoadSubVector(instruction)) |
| { |
| return loadSubVector->getSrc(1); |
| } |
| |
| return nullptr; |
| } |
| |
| Ice::Operand *Optimizer::storeData(const Ice::Inst *instruction) |
| { |
| assert(isStore(*instruction)); |
| |
| if(auto *store = llvm::dyn_cast<Ice::InstStore>(instruction)) |
| { |
| return store->getData(); |
| } |
| |
| if(auto *storeSubVector = asStoreSubVector(instruction)) |
| { |
| return storeSubVector->getSrc(1); |
| } |
| |
| return nullptr; |
| } |
| |
| std::size_t Optimizer::storeSize(const Ice::Inst *store) |
| { |
| assert(isStore(*store)); |
| |
| if(auto *instStore = llvm::dyn_cast<Ice::InstStore>(store)) |
| { |
| return Ice::typeWidthInBytes(instStore->getData()->getType()); |
| } |
| |
| if(auto *storeSubVector = asStoreSubVector(store)) |
| { |
| return llvm::cast<Ice::ConstantInteger32>(storeSubVector->getSrc(3))->getValue(); |
| } |
| |
| return 0; |
| } |
| |
| bool Optimizer::loadTypeMatchesStore(const Ice::Inst *load, const Ice::Inst *store) |
| { |
| if(!load || !store) |
| { |
| return false; |
| } |
| |
| assert(isLoad(*load) && isStore(*store)); |
| assert(loadAddress(load) == storeAddress(store)); |
| |
| if(auto *instStore = llvm::dyn_cast<Ice::InstStore>(store)) |
| { |
| if(auto *instLoad = llvm::dyn_cast<Ice::InstLoad>(load)) |
| { |
| return instStore->getData()->getType() == instLoad->getDest()->getType(); |
| } |
| } |
| |
| if(auto *storeSubVector = asStoreSubVector(store)) |
| { |
| if(auto *loadSubVector = asLoadSubVector(load)) |
| { |
| // Check for matching type and sub-vector width. |
| return storeSubVector->getSrc(1)->getType() == loadSubVector->getDest()->getType() && |
| llvm::cast<Ice::ConstantInteger32>(storeSubVector->getSrc(3))->getValue() == |
| llvm::cast<Ice::ConstantInteger32>(loadSubVector->getSrc(2))->getValue(); |
| } |
| } |
| |
| return false; |
| } |
| |
| Optimizer::Uses *Optimizer::getUses(Ice::Operand *operand) |
| { |
| Optimizer::Uses *uses = (Optimizer::Uses *)operand->Ice::Operand::getExternalData(); |
| if(!uses) |
| { |
| uses = new Optimizer::Uses; |
| setUses(operand, uses); |
| operandsWithUses.push_back(operand); |
| } |
| return uses; |
| } |
| |
| void Optimizer::setUses(Ice::Operand *operand, Optimizer::Uses *uses) |
| { |
| if(auto *oldUses = reinterpret_cast<Optimizer::Uses *>(operand->Ice::Operand::getExternalData())) |
| { |
| delete oldUses; |
| } |
| |
| operand->Ice::Operand::setExternalData(uses); |
| } |
| |
| bool Optimizer::hasUses(Ice::Operand *operand) const |
| { |
| return operand->Ice::Operand::getExternalData() != nullptr; |
| } |
| |
| Ice::CfgNode *Optimizer::getNode(Ice::Inst *inst) |
| { |
| return (Ice::CfgNode *)inst->Ice::Inst::getExternalData(); |
| } |
| |
| void Optimizer::setNode(Ice::Inst *inst, Ice::CfgNode *node) |
| { |
| inst->Ice::Inst::setExternalData(node); |
| } |
| |
| Ice::Inst *Optimizer::getDefinition(Ice::Variable *var) |
| { |
| return (Ice::Inst *)var->Ice::Variable::getExternalData(); |
| } |
| |
| void Optimizer::setDefinition(Ice::Variable *var, Ice::Inst *inst) |
| { |
| var->Ice::Variable::setExternalData(inst); |
| } |
| |
| const std::vector<Optimizer::LoadStoreInst> &Optimizer::getLoadStoreInsts(Ice::CfgNode *node) |
| { |
| return *((const std::vector<LoadStoreInst> *)node->Ice::CfgNode::getExternalData()); |
| } |
| |
| void Optimizer::setLoadStoreInsts(Ice::CfgNode *node, std::vector<LoadStoreInst> *insts) |
| { |
| node->Ice::CfgNode::setExternalData(insts); |
| } |
| |
| bool Optimizer::hasLoadStoreInsts(Ice::CfgNode *node) const |
| { |
| return node->Ice::CfgNode::getExternalData() != nullptr; |
| } |
| |
| bool Optimizer::Uses::areOnlyLoadStore() const |
| { |
| return size() == (loads.size() + stores.size()); |
| } |
| |
| void Optimizer::Uses::insert(Ice::Operand *value, Ice::Inst *instruction) |
| { |
| push_back(instruction); |
| |
| if(isLoad(*instruction)) |
| { |
| if(value == loadAddress(instruction)) |
| { |
| loads.push_back(instruction); |
| } |
| } |
| else if(isStore(*instruction)) |
| { |
| if(value == storeAddress(instruction)) |
| { |
| stores.push_back(instruction); |
| } |
| } |
| } |
| |
| void Optimizer::Uses::erase(Ice::Inst *instruction) |
| { |
| auto &uses = *this; |
| |
| for(size_t i = 0; i < uses.size(); i++) |
| { |
| if(uses[i] == instruction) |
| { |
| uses[i] = back(); |
| pop_back(); |
| |
| for(size_t i = 0; i < loads.size(); i++) |
| { |
| if(loads[i] == instruction) |
| { |
| loads[i] = loads.back(); |
| loads.pop_back(); |
| break; |
| } |
| } |
| |
| for(size_t i = 0; i < stores.size(); i++) |
| { |
| if(stores[i] == instruction) |
| { |
| stores[i] = stores.back(); |
| stores.pop_back(); |
| break; |
| } |
| } |
| |
| break; |
| } |
| } |
| } |
| |
| } // anonymous namespace |
| |
| namespace rr { |
| |
| void optimize(Ice::Cfg *function) |
| { |
| Optimizer optimizer; |
| |
| optimizer.run(function); |
| } |
| |
| } // namespace rr |