|  | // 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 <unordered_map> | 
|  | #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; | 
|  | }; | 
|  |  | 
|  | std::unordered_map<Ice::Operand*, Uses> uses; | 
|  | std::unordered_map<Ice::Inst*, Ice::CfgNode*> node; | 
|  | std::unordered_map<Ice::Variable*, Ice::Inst*> definition; | 
|  | }; | 
|  |  | 
|  | void Optimizer::run(Ice::Cfg *function) | 
|  | { | 
|  | this->function = function; | 
|  | this->context = function->getContext(); | 
|  |  | 
|  | analyzeUses(function); | 
|  |  | 
|  | eliminateDeadCode(); | 
|  | eliminateUnitializedLoads(); | 
|  | eliminateLoadsFollowingSingleStore(); | 
|  | optimizeStoresInSingleBasicBlock(); | 
|  | eliminateDeadCode(); | 
|  | } | 
|  |  | 
|  | 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)) | 
|  | { | 
|  | return;   // Allocas are all at the top | 
|  | } | 
|  |  | 
|  | Ice::Operand *address = alloca.getDest(); | 
|  | const auto &addressEntry = uses.find(address); | 
|  | const auto &addressUses = addressEntry->second; | 
|  |  | 
|  | if(!addressUses.areOnlyLoadStore()) | 
|  | { | 
|  | continue; | 
|  | } | 
|  |  | 
|  | if(addressUses.stores.empty()) | 
|  | { | 
|  | for(Ice::Inst *load : addressUses.loads) | 
|  | { | 
|  | Ice::Variable *loadData = load->getDest(); | 
|  |  | 
|  | for(Ice::Inst *use : uses[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); | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | uses.erase(loadData); | 
|  |  | 
|  | load->setDeleted(); | 
|  | } | 
|  |  | 
|  | alloca.setDeleted(); | 
|  | uses.erase(addressEntry); | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | void Optimizer::eliminateLoadsFollowingSingleStore() | 
|  | { | 
|  | Ice::CfgNode *entryBlock = function->getEntryNode(); | 
|  |  | 
|  | for(Ice::Inst &alloca : entryBlock->getInsts()) | 
|  | { | 
|  | if(alloca.isDeleted()) | 
|  | { | 
|  | continue; | 
|  | } | 
|  |  | 
|  | if(!llvm::isa<Ice::InstAlloca>(alloca)) | 
|  | { | 
|  | return;   // Allocas are all at the top | 
|  | } | 
|  |  | 
|  | Ice::Operand *address = alloca.getDest(); | 
|  | const auto &addressEntry = uses.find(address); | 
|  | auto &addressUses = addressEntry->second; | 
|  |  | 
|  | 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; | 
|  | } | 
|  |  | 
|  | 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(); | 
|  | uses.erase(address); | 
|  |  | 
|  | auto &valueUses = uses[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()) | 
|  | { | 
|  | uses.erase(storeValue); | 
|  | } | 
|  |  | 
|  | break; | 
|  | } | 
|  | } | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | void Optimizer::optimizeStoresInSingleBasicBlock() | 
|  | { | 
|  | Ice::CfgNode *entryBlock = function->getEntryNode(); | 
|  |  | 
|  | for(Ice::Inst &alloca : entryBlock->getInsts()) | 
|  | { | 
|  | if(alloca.isDeleted()) | 
|  | { | 
|  | continue; | 
|  | } | 
|  |  | 
|  | if(!llvm::isa<Ice::InstAlloca>(alloca)) | 
|  | { | 
|  | return;   // Allocas are all at the top | 
|  | } | 
|  |  | 
|  | Ice::Operand *address = alloca.getDest(); | 
|  | const auto &addressEntry = uses.find(address); | 
|  | const auto &addressUses = addressEntry->second; | 
|  |  | 
|  | if(!addressUses.areOnlyLoadStore()) | 
|  | { | 
|  | continue; | 
|  | } | 
|  |  | 
|  | Ice::CfgNode *singleBasicBlock = node[addressUses.stores[0]]; | 
|  |  | 
|  | for(size_t i = 1; i < addressUses.stores.size(); i++) | 
|  | { | 
|  | Ice::Inst *store = addressUses.stores[i]; | 
|  | if(node[store] != singleBasicBlock) | 
|  | { | 
|  | singleBasicBlock = nullptr; | 
|  | break; | 
|  | } | 
|  | } | 
|  |  | 
|  | if(singleBasicBlock) | 
|  | { | 
|  | auto &insts = singleBasicBlock->getInsts(); | 
|  | Ice::Inst *store = nullptr; | 
|  | Ice::Operand *storeValue = nullptr; | 
|  | bool unmatchedLoads = false; | 
|  |  | 
|  | for(Ice::Inst &inst : insts) | 
|  | { | 
|  | if(inst.isDeleted()) | 
|  | { | 
|  | continue; | 
|  | } | 
|  |  | 
|  | if(isStore(inst)) | 
|  | { | 
|  | if(storeAddress(&inst) != address) | 
|  | { | 
|  | continue; | 
|  | } | 
|  |  | 
|  | // 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(isLoad(inst)) | 
|  | { | 
|  | Ice::Inst *load = &inst; | 
|  |  | 
|  | if(loadAddress(load) != address) | 
|  | { | 
|  | continue; | 
|  | } | 
|  |  | 
|  | if(!loadTypeMatchesStore(load, store)) | 
|  | { | 
|  | unmatchedLoads = true; | 
|  | continue; | 
|  | } | 
|  |  | 
|  | replace(load, storeValue); | 
|  | } | 
|  | } | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | void Optimizer::analyzeUses(Ice::Cfg *function) | 
|  | { | 
|  | uses.clear(); | 
|  | node.clear(); | 
|  | definition.clear(); | 
|  |  | 
|  | for(Ice::CfgNode *basicBlock : function->getNodes()) | 
|  | { | 
|  | for(Ice::Inst &instruction : basicBlock->getInsts()) | 
|  | { | 
|  | if(instruction.isDeleted()) | 
|  | { | 
|  | continue; | 
|  | } | 
|  |  | 
|  | node[&instruction] = basicBlock; | 
|  | definition[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); | 
|  | uses[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()); | 
|  | } | 
|  |  | 
|  | for(Ice::Inst *use : uses[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); | 
|  | } | 
|  | } | 
|  |  | 
|  | uses[newValue].insert(newValue, use); | 
|  | } | 
|  |  | 
|  | uses.erase(oldValue); | 
|  |  | 
|  | 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); | 
|  |  | 
|  | const auto &srcEntry = uses.find(src); | 
|  |  | 
|  | if(srcEntry != uses.end()) | 
|  | { | 
|  | auto &srcUses = srcEntry->second; | 
|  |  | 
|  | srcUses.erase(instruction); | 
|  |  | 
|  | if(srcUses.empty()) | 
|  | { | 
|  | uses.erase(srcEntry); | 
|  |  | 
|  | if(Ice::Variable *var = llvm::dyn_cast<Ice::Variable>(src)) | 
|  | { | 
|  | deleteInstruction(definition[var]); | 
|  | } | 
|  | } | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | bool Optimizer::isDead(Ice::Inst *instruction) | 
|  | { | 
|  | Ice::Variable *dest = instruction->getDest(); | 
|  |  | 
|  | if(dest) | 
|  | { | 
|  | return uses[dest].empty() && !instruction->hasSideEffects(); | 
|  | } | 
|  | else if(isStore(*instruction)) | 
|  | { | 
|  | if(Ice::Variable *address = llvm::dyn_cast<Ice::Variable>(storeAddress(instruction))) | 
|  | { | 
|  | Ice::Inst *def = definition[address]; | 
|  |  | 
|  | if(def && llvm::isa<Ice::InstAlloca>(def)) | 
|  | { | 
|  | return uses[address].size() == uses[address].stores.size();   // Dead if all uses are stores | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | 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; | 
|  | } | 
|  |  | 
|  | 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; | 
|  | } | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | namespace sw | 
|  | { | 
|  | void optimize(Ice::Cfg *function) | 
|  | { | 
|  | Optimizer optimizer; | 
|  |  | 
|  | optimizer.run(function); | 
|  | } | 
|  | } |