Commit 8e92838b by Jim Stichnoth

Subzero: Manage each Cfg as a std::unique_ptr<Cfg>.

The Cfg::create() method now returns a unique_ptr. Once the parser fully builds the Cfg, it is released onto the work queue, and then acquired and ultimately deleted by the translator thread. BUG= none R=jfb@chromium.org Review URL: https://codereview.chromium.org/892063002
parent 261cae38
...@@ -46,11 +46,7 @@ Cfg::Cfg(GlobalContext *Ctx) ...@@ -46,11 +46,7 @@ Cfg::Cfg(GlobalContext *Ctx)
"Attempt to build cfg when IR generation disabled"); "Attempt to build cfg when IR generation disabled");
} }
Cfg::~Cfg() { Cfg::~Cfg() { assert(ICE_TLS_GET_FIELD(CurrentCfg) == nullptr); }
assert(ICE_TLS_GET_FIELD(CurrentCfg) == this);
// Reset the thread-local CurrentCfg pointer.
ICE_TLS_SET_FIELD(CurrentCfg, nullptr);
}
void Cfg::setError(const IceString &Message) { void Cfg::setError(const IceString &Message) {
HasError = true; HasError = true;
......
...@@ -30,17 +30,14 @@ class Cfg { ...@@ -30,17 +30,14 @@ class Cfg {
public: public:
~Cfg(); ~Cfg();
// TODO(stichnot): Change this to return unique_ptr<Cfg>, and plumb static std::unique_ptr<Cfg> create(GlobalContext *Ctx) {
// it through the callers, to make ownership and lifetime and return std::unique_ptr<Cfg>(new Cfg(Ctx));
// destruction requirements more explicit.
static Cfg *create(GlobalContext *Ctx) {
Cfg *Func = new Cfg(Ctx);
ICE_TLS_SET_FIELD(CurrentCfg, Func);
return Func;
} }
// Gets a pointer to the current thread's Cfg. // Gets a pointer to the current thread's Cfg.
static const Cfg *getCurrentCfg() { return ICE_TLS_GET_FIELD(CurrentCfg); } static const Cfg *getCurrentCfg() { return ICE_TLS_GET_FIELD(CurrentCfg); }
void updateTLS() const { ICE_TLS_SET_FIELD(CurrentCfg, this); } static void setCurrentCfg(const Cfg *Func) {
ICE_TLS_SET_FIELD(CurrentCfg, Func);
}
// Gets a pointer to the current thread's Cfg's allocator. // Gets a pointer to the current thread's Cfg's allocator.
static ArenaAllocator<> *getCurrentCfgAllocator() { static ArenaAllocator<> *getCurrentCfgAllocator() {
assert(ICE_TLS_GET_FIELD(CurrentCfg)); assert(ICE_TLS_GET_FIELD(CurrentCfg));
......
...@@ -48,6 +48,10 @@ template <typename T> static std::string LLVMObjectAsString(const T *O) { ...@@ -48,6 +48,10 @@ template <typename T> static std::string LLVMObjectAsString(const T *O) {
} }
// Base class for converting LLVM to ICE. // Base class for converting LLVM to ICE.
// TODO(stichnot): Redesign Converter, LLVM2ICEConverter,
// LLVM2ICEFunctionConverter, and LLVM2ICEGlobalsConverter with
// respect to Translator. In particular, the unique_ptr ownership
// rules in LLVM2ICEFunctionConverter.
class LLVM2ICEConverter { class LLVM2ICEConverter {
LLVM2ICEConverter(const LLVM2ICEConverter &) = delete; LLVM2ICEConverter(const LLVM2ICEConverter &) = delete;
LLVM2ICEConverter &operator=(const LLVM2ICEConverter &) = delete; LLVM2ICEConverter &operator=(const LLVM2ICEConverter &) = delete;
...@@ -79,15 +83,16 @@ public: ...@@ -79,15 +83,16 @@ public:
LLVM2ICEFunctionConverter(Ice::Converter &Converter) LLVM2ICEFunctionConverter(Ice::Converter &Converter)
: LLVM2ICEConverter(Converter), Func(nullptr) {} : LLVM2ICEConverter(Converter), Func(nullptr) {}
// Caller is expected to delete the returned Ice::Cfg object. void convertFunction(const Function *F) {
Ice::Cfg *convertFunction(const Function *F) { Func = Ice::Cfg::create(Ctx);
Ice::Cfg::setCurrentCfg(Func.get());
VarMap.clear(); VarMap.clear();
NodeMap.clear(); NodeMap.clear();
Func = Ice::Cfg::create(Ctx);
Func->setFunctionName(F->getName()); Func->setFunctionName(F->getName());
Func->setReturnType(convertToIceType(F->getReturnType())); Func->setReturnType(convertToIceType(F->getReturnType()));
Func->setInternal(F->hasInternalLinkage()); Func->setInternal(F->hasInternalLinkage());
Ice::TimerMarker T(Ice::TimerStack::TT_llvmConvert, Func); Ice::TimerMarker T(Ice::TimerStack::TT_llvmConvert, Func.get());
// The initial definition/use of each arg is the entry node. // The initial definition/use of each arg is the entry node.
for (auto ArgI = F->arg_begin(), ArgE = F->arg_end(); ArgI != ArgE; for (auto ArgI = F->arg_begin(), ArgE = F->arg_end(); ArgI != ArgE;
...@@ -106,7 +111,8 @@ public: ...@@ -106,7 +111,8 @@ public:
Func->setEntryNode(mapBasicBlockToNode(&F->getEntryBlock())); Func->setEntryNode(mapBasicBlockToNode(&F->getEntryBlock()));
Func->computePredecessors(); Func->computePredecessors();
return Func; Ice::Cfg::setCurrentCfg(nullptr);
Converter.translateFcn(std::move(Func));
} }
// convertConstant() does not use Func or require it to be a valid // convertConstant() does not use Func or require it to be a valid
...@@ -147,7 +153,7 @@ private: ...@@ -147,7 +153,7 @@ private:
if (VarMap.find(V) == VarMap.end()) { if (VarMap.find(V) == VarMap.end()) {
VarMap[V] = Func->makeVariable(IceTy); VarMap[V] = Func->makeVariable(IceTy);
if (ALLOW_DUMP) if (ALLOW_DUMP)
VarMap[V]->setName(Func, V->getName()); VarMap[V]->setName(Func.get(), V->getName());
} }
return VarMap[V]; return VarMap[V];
} }
...@@ -304,13 +310,13 @@ private: ...@@ -304,13 +310,13 @@ private:
Ice::Inst *convertLoadInstruction(const LoadInst *Inst) { Ice::Inst *convertLoadInstruction(const LoadInst *Inst) {
Ice::Operand *Src = convertOperand(Inst, 0); Ice::Operand *Src = convertOperand(Inst, 0);
Ice::Variable *Dest = mapValueToIceVar(Inst); Ice::Variable *Dest = mapValueToIceVar(Inst);
return Ice::InstLoad::create(Func, Dest, Src); return Ice::InstLoad::create(Func.get(), Dest, Src);
} }
Ice::Inst *convertStoreInstruction(const StoreInst *Inst) { Ice::Inst *convertStoreInstruction(const StoreInst *Inst) {
Ice::Operand *Addr = convertOperand(Inst, 1); Ice::Operand *Addr = convertOperand(Inst, 1);
Ice::Operand *Val = convertOperand(Inst, 0); Ice::Operand *Val = convertOperand(Inst, 0);
return Ice::InstStore::create(Func, Val, Addr); return Ice::InstStore::create(Func.get(), Val, Addr);
} }
Ice::Inst *convertArithInstruction(const Instruction *Inst, Ice::Inst *convertArithInstruction(const Instruction *Inst,
...@@ -319,13 +325,13 @@ private: ...@@ -319,13 +325,13 @@ private:
Ice::Operand *Src0 = convertOperand(Inst, 0); Ice::Operand *Src0 = convertOperand(Inst, 0);
Ice::Operand *Src1 = convertOperand(Inst, 1); Ice::Operand *Src1 = convertOperand(Inst, 1);
Ice::Variable *Dest = mapValueToIceVar(BinOp); Ice::Variable *Dest = mapValueToIceVar(BinOp);
return Ice::InstArithmetic::create(Func, Opcode, Dest, Src0, Src1); return Ice::InstArithmetic::create(Func.get(), Opcode, Dest, Src0, Src1);
} }
Ice::Inst *convertPHINodeInstruction(const PHINode *Inst) { Ice::Inst *convertPHINodeInstruction(const PHINode *Inst) {
unsigned NumValues = Inst->getNumIncomingValues(); unsigned NumValues = Inst->getNumIncomingValues();
Ice::InstPhi *IcePhi = Ice::InstPhi *IcePhi =
Ice::InstPhi::create(Func, NumValues, mapValueToIceVar(Inst)); Ice::InstPhi::create(Func.get(), NumValues, mapValueToIceVar(Inst));
for (unsigned N = 0, E = NumValues; N != E; ++N) { for (unsigned N = 0, E = NumValues; N != E; ++N) {
IcePhi->addArgument(convertOperand(Inst, N), IcePhi->addArgument(convertOperand(Inst, N),
mapBasicBlockToNode(Inst->getIncomingBlock(N))); mapBasicBlockToNode(Inst->getIncomingBlock(N)));
...@@ -340,31 +346,31 @@ private: ...@@ -340,31 +346,31 @@ private:
BasicBlock *BBElse = Inst->getSuccessor(1); BasicBlock *BBElse = Inst->getSuccessor(1);
Ice::CfgNode *NodeThen = mapBasicBlockToNode(BBThen); Ice::CfgNode *NodeThen = mapBasicBlockToNode(BBThen);
Ice::CfgNode *NodeElse = mapBasicBlockToNode(BBElse); Ice::CfgNode *NodeElse = mapBasicBlockToNode(BBElse);
return Ice::InstBr::create(Func, Src, NodeThen, NodeElse); return Ice::InstBr::create(Func.get(), Src, NodeThen, NodeElse);
} else { } else {
BasicBlock *BBSucc = Inst->getSuccessor(0); BasicBlock *BBSucc = Inst->getSuccessor(0);
return Ice::InstBr::create(Func, mapBasicBlockToNode(BBSucc)); return Ice::InstBr::create(Func.get(), mapBasicBlockToNode(BBSucc));
} }
} }
Ice::Inst *convertIntToPtrInstruction(const IntToPtrInst *Inst) { Ice::Inst *convertIntToPtrInstruction(const IntToPtrInst *Inst) {
Ice::Operand *Src = convertOperand(Inst, 0); Ice::Operand *Src = convertOperand(Inst, 0);
Ice::Variable *Dest = mapValueToIceVar(Inst, Ice::getPointerType()); Ice::Variable *Dest = mapValueToIceVar(Inst, Ice::getPointerType());
return Ice::InstAssign::create(Func, Dest, Src); return Ice::InstAssign::create(Func.get(), Dest, Src);
} }
Ice::Inst *convertPtrToIntInstruction(const PtrToIntInst *Inst) { Ice::Inst *convertPtrToIntInstruction(const PtrToIntInst *Inst) {
Ice::Operand *Src = convertOperand(Inst, 0); Ice::Operand *Src = convertOperand(Inst, 0);
Ice::Variable *Dest = mapValueToIceVar(Inst); Ice::Variable *Dest = mapValueToIceVar(Inst);
return Ice::InstAssign::create(Func, Dest, Src); return Ice::InstAssign::create(Func.get(), Dest, Src);
} }
Ice::Inst *convertRetInstruction(const ReturnInst *Inst) { Ice::Inst *convertRetInstruction(const ReturnInst *Inst) {
Ice::Operand *RetOperand = convertOperand(Inst, 0); Ice::Operand *RetOperand = convertOperand(Inst, 0);
if (RetOperand) { if (RetOperand) {
return Ice::InstRet::create(Func, RetOperand); return Ice::InstRet::create(Func.get(), RetOperand);
} else { } else {
return Ice::InstRet::create(Func); return Ice::InstRet::create(Func.get());
} }
} }
...@@ -372,7 +378,7 @@ private: ...@@ -372,7 +378,7 @@ private:
Ice::InstCast::OpKind CastKind) { Ice::InstCast::OpKind CastKind) {
Ice::Operand *Src = convertOperand(Inst, 0); Ice::Operand *Src = convertOperand(Inst, 0);
Ice::Variable *Dest = mapValueToIceVar(Inst); Ice::Variable *Dest = mapValueToIceVar(Inst);
return Ice::InstCast::create(Func, CastKind, Dest, Src); return Ice::InstCast::create(Func.get(), CastKind, Dest, Src);
} }
Ice::Inst *convertICmpInstruction(const ICmpInst *Inst) { Ice::Inst *convertICmpInstruction(const ICmpInst *Inst) {
...@@ -416,7 +422,7 @@ private: ...@@ -416,7 +422,7 @@ private:
break; break;
} }
return Ice::InstIcmp::create(Func, Cond, Dest, Src0, Src1); return Ice::InstIcmp::create(Func.get(), Cond, Dest, Src0, Src1);
} }
Ice::Inst *convertFCmpInstruction(const FCmpInst *Inst) { Ice::Inst *convertFCmpInstruction(const FCmpInst *Inst) {
...@@ -480,14 +486,14 @@ private: ...@@ -480,14 +486,14 @@ private:
break; break;
} }
return Ice::InstFcmp::create(Func, Cond, Dest, Src0, Src1); return Ice::InstFcmp::create(Func.get(), Cond, Dest, Src0, Src1);
} }
Ice::Inst *convertExtractElementInstruction(const ExtractElementInst *Inst) { Ice::Inst *convertExtractElementInstruction(const ExtractElementInst *Inst) {
Ice::Variable *Dest = mapValueToIceVar(Inst); Ice::Variable *Dest = mapValueToIceVar(Inst);
Ice::Operand *Source1 = convertValue(Inst->getOperand(0)); Ice::Operand *Source1 = convertValue(Inst->getOperand(0));
Ice::Operand *Source2 = convertValue(Inst->getOperand(1)); Ice::Operand *Source2 = convertValue(Inst->getOperand(1));
return Ice::InstExtractElement::create(Func, Dest, Source1, Source2); return Ice::InstExtractElement::create(Func.get(), Dest, Source1, Source2);
} }
Ice::Inst *convertInsertElementInstruction(const InsertElementInst *Inst) { Ice::Inst *convertInsertElementInstruction(const InsertElementInst *Inst) {
...@@ -495,7 +501,7 @@ private: ...@@ -495,7 +501,7 @@ private:
Ice::Operand *Source1 = convertValue(Inst->getOperand(0)); Ice::Operand *Source1 = convertValue(Inst->getOperand(0));
Ice::Operand *Source2 = convertValue(Inst->getOperand(1)); Ice::Operand *Source2 = convertValue(Inst->getOperand(1));
Ice::Operand *Source3 = convertValue(Inst->getOperand(2)); Ice::Operand *Source3 = convertValue(Inst->getOperand(2));
return Ice::InstInsertElement::create(Func, Dest, Source1, Source2, return Ice::InstInsertElement::create(Func.get(), Dest, Source1, Source2,
Source3); Source3);
} }
...@@ -504,7 +510,7 @@ private: ...@@ -504,7 +510,7 @@ private:
Ice::Operand *Cond = convertValue(Inst->getCondition()); Ice::Operand *Cond = convertValue(Inst->getCondition());
Ice::Operand *Source1 = convertValue(Inst->getTrueValue()); Ice::Operand *Source1 = convertValue(Inst->getTrueValue());
Ice::Operand *Source2 = convertValue(Inst->getFalseValue()); Ice::Operand *Source2 = convertValue(Inst->getFalseValue());
return Ice::InstSelect::create(Func, Dest, Cond, Source1, Source2); return Ice::InstSelect::create(Func.get(), Dest, Cond, Source1, Source2);
} }
Ice::Inst *convertSwitchInstruction(const SwitchInst *Inst) { Ice::Inst *convertSwitchInstruction(const SwitchInst *Inst) {
...@@ -512,7 +518,7 @@ private: ...@@ -512,7 +518,7 @@ private:
Ice::CfgNode *LabelDefault = mapBasicBlockToNode(Inst->getDefaultDest()); Ice::CfgNode *LabelDefault = mapBasicBlockToNode(Inst->getDefaultDest());
unsigned NumCases = Inst->getNumCases(); unsigned NumCases = Inst->getNumCases();
Ice::InstSwitch *Switch = Ice::InstSwitch *Switch =
Ice::InstSwitch::create(Func, NumCases, Source, LabelDefault); Ice::InstSwitch::create(Func.get(), NumCases, Source, LabelDefault);
unsigned CurrentCase = 0; unsigned CurrentCase = 0;
for (SwitchInst::ConstCaseIt I = Inst->case_begin(), E = Inst->case_end(); for (SwitchInst::ConstCaseIt I = Inst->case_begin(), E = Inst->case_end();
I != E; ++I, ++CurrentCase) { I != E; ++I, ++CurrentCase) {
...@@ -544,14 +550,14 @@ private: ...@@ -544,14 +550,14 @@ private:
report_fatal_error(std::string("Invalid PNaCl intrinsic call: ") + report_fatal_error(std::string("Invalid PNaCl intrinsic call: ") +
LLVMObjectAsString(Inst)); LLVMObjectAsString(Inst));
} }
NewInst = Ice::InstIntrinsicCall::create(Func, NumArgs, Dest, NewInst = Ice::InstIntrinsicCall::create(Func.get(), NumArgs, Dest,
CallTarget, Info->Info); CallTarget, Info->Info);
} }
} }
// Not an intrinsic call. // Not an intrinsic call.
if (NewInst == nullptr) { if (NewInst == nullptr) {
NewInst = Ice::InstCall::create(Func, NumArgs, Dest, CallTarget, NewInst = Ice::InstCall::create(Func.get(), NumArgs, Dest, CallTarget,
Inst->isTailCall()); Inst->isTailCall());
} }
for (unsigned i = 0; i < NumArgs; ++i) { for (unsigned i = 0; i < NumArgs; ++i) {
...@@ -569,11 +575,11 @@ private: ...@@ -569,11 +575,11 @@ private:
uint32_t Align = Inst->getAlignment(); uint32_t Align = Inst->getAlignment();
Ice::Variable *Dest = mapValueToIceVar(Inst, Ice::getPointerType()); Ice::Variable *Dest = mapValueToIceVar(Inst, Ice::getPointerType());
return Ice::InstAlloca::create(Func, ByteCount, Align, Dest); return Ice::InstAlloca::create(Func.get(), ByteCount, Align, Dest);
} }
Ice::Inst *convertUnreachableInstruction(const UnreachableInst * /*Inst*/) { Ice::Inst *convertUnreachableInstruction(const UnreachableInst * /*Inst*/) {
return Ice::InstUnreachable::create(Func); return Ice::InstUnreachable::create(Func.get());
} }
Ice::CfgNode *convertBasicBlock(const BasicBlock *BB) { Ice::CfgNode *convertBasicBlock(const BasicBlock *BB) {
...@@ -621,7 +627,7 @@ private: ...@@ -621,7 +627,7 @@ private:
private: private:
// Data // Data
Ice::Cfg *Func; std::unique_ptr<Ice::Cfg> Func;
std::map<const Value *, Ice::Variable *> VarMap; std::map<const Value *, Ice::Variable *> VarMap;
std::map<const BasicBlock *, Ice::CfgNode *> NodeMap; std::map<const BasicBlock *, Ice::CfgNode *> NodeMap;
}; };
...@@ -872,21 +878,21 @@ void Converter::convertGlobals(Module *Mod) { ...@@ -872,21 +878,21 @@ void Converter::convertGlobals(Module *Mod) {
} }
void Converter::convertFunctions() { void Converter::convertFunctions() {
TimerStackIdT StackID = GlobalContext::TSK_Funcs; const TimerStackIdT StackID = GlobalContext::TSK_Funcs;
for (const Function &I : *Mod) { for (const Function &I : *Mod) {
if (I.empty()) if (I.empty())
continue; continue;
TimerIdT TimerID = 0; TimerIdT TimerID = 0;
if (ALLOW_DUMP && Ctx->getFlags().TimeEachFunction) { const bool TimeThisFunction =
ALLOW_DUMP && Ctx->getFlags().TimeEachFunction;
if (TimeThisFunction) {
TimerID = Ctx->getTimerID(StackID, I.getName()); TimerID = Ctx->getTimerID(StackID, I.getName());
Ctx->pushTimer(TimerID, StackID); Ctx->pushTimer(TimerID, StackID);
} }
LLVM2ICEFunctionConverter FunctionConverter(*this); LLVM2ICEFunctionConverter FunctionConverter(*this);
FunctionConverter.convertFunction(&I);
Cfg *Fcn = FunctionConverter.convertFunction(&I); if (TimeThisFunction)
translateFcn(Fcn);
if (ALLOW_DUMP && Ctx->getFlags().TimeEachFunction)
Ctx->popTimer(TimerID, StackID); Ctx->popTimer(TimerID, StackID);
} }
} }
......
...@@ -23,6 +23,7 @@ ...@@ -23,6 +23,7 @@
#include <limits> #include <limits>
#include <list> #include <list>
#include <map> #include <map>
#include <memory>
#include <mutex> #include <mutex>
#include <string> #include <string>
#include <system_error> #include <system_error>
......
...@@ -162,11 +162,11 @@ GlobalContext::GlobalContext(Ostream *OsDump, Ostream *OsEmit, ...@@ -162,11 +162,11 @@ GlobalContext::GlobalContext(Ostream *OsDump, Ostream *OsEmit,
} }
void GlobalContext::translateFunctions() { void GlobalContext::translateFunctions() {
while (Cfg *Func = cfgQueueBlockingPop()) { while (std::unique_ptr<Cfg> Func = cfgQueueBlockingPop()) {
// Install Func in TLS for Cfg-specific container allocators.
Cfg::setCurrentCfg(Func.get());
// Reset per-function stats being accumulated in TLS. // Reset per-function stats being accumulated in TLS.
resetStats(); resetStats();
// Install Func in TLS for Cfg-specific container allocators.
Func->updateTLS();
// Set verbose level to none if the current function does NOT // Set verbose level to none if the current function does NOT
// match the -verbose-focus command-line option. // match the -verbose-focus command-line option.
if (!matchSymbolName(Func->getFunctionName(), getFlags().VerboseFocusOn)) if (!matchSymbolName(Func->getFunctionName(), getFlags().VerboseFocusOn))
...@@ -191,10 +191,10 @@ void GlobalContext::translateFunctions() { ...@@ -191,10 +191,10 @@ void GlobalContext::translateFunctions() {
Func->emit(); Func->emit();
// TODO(stichnot): actually add to emit queue // TODO(stichnot): actually add to emit queue
} }
// TODO(stichnot): fix multithreaded stats dumping.
dumpStats(Func->getFunctionName()); dumpStats(Func->getFunctionName());
} }
delete Func; Cfg::setCurrentCfg(nullptr);
// The Cfg now gets deleted as Func goes out of scope.
} }
} }
...@@ -548,6 +548,19 @@ void GlobalContext::setTimerName(TimerStackIdT StackID, ...@@ -548,6 +548,19 @@ void GlobalContext::setTimerName(TimerStackIdT StackID,
Timers->at(StackID).setName(NewName); Timers->at(StackID).setName(NewName);
} }
// Note: cfgQueueBlockingPush and cfgQueueBlockingPop use unique_ptr
// at the interface to take and transfer ownership, but they
// internally store the raw Cfg pointer in the work queue. This
// allows e.g. future queue optimizations such as the use of atomics
// to modify queue elements.
void GlobalContext::cfgQueueBlockingPush(std::unique_ptr<Cfg> Func) {
CfgQ.blockingPush(Func.release());
}
std::unique_ptr<Cfg> GlobalContext::cfgQueueBlockingPop() {
return std::unique_ptr<Cfg>(CfgQ.blockingPop());
}
void GlobalContext::dumpStats(const IceString &Name, bool Final) { void GlobalContext::dumpStats(const IceString &Name, bool Final) {
if (!ALLOW_DUMP || !getFlags().DumpStats) if (!ALLOW_DUMP || !getFlags().DumpStats)
return; return;
......
...@@ -280,12 +280,12 @@ public: ...@@ -280,12 +280,12 @@ public:
// queue. Notifies any idle workers that a new function is // queue. Notifies any idle workers that a new function is
// available for translating. May block if the work queue is too // available for translating. May block if the work queue is too
// large, in order to control memory footprint. // large, in order to control memory footprint.
void cfgQueueBlockingPush(Cfg *Func) { CfgQ.blockingPush(Func); } void cfgQueueBlockingPush(std::unique_ptr<Cfg> Func);
// Takes a Cfg from the work queue for translating. May block if // Takes a Cfg from the work queue for translating. May block if
// the work queue is currently empty. Returns nullptr if there is // the work queue is currently empty. Returns nullptr if there is
// no more work - the queue is empty and either end() has been // no more work - the queue is empty and either end() has been
// called or the Sequential flag was set. // called or the Sequential flag was set.
Cfg *cfgQueueBlockingPop() { return CfgQ.blockingPop(); } std::unique_ptr<Cfg> cfgQueueBlockingPop();
// Notifies that no more work will be added to the work queue. // Notifies that no more work will be added to the work queue.
void cfgQueueNotifyEnd() { CfgQ.notifyEnd(); } void cfgQueueNotifyEnd() { CfgQ.notifyEnd(); }
......
...@@ -53,8 +53,8 @@ bool Translator::checkIfUnnamedNameSafe(const IceString &Name, const char *Kind, ...@@ -53,8 +53,8 @@ bool Translator::checkIfUnnamedNameSafe(const IceString &Name, const char *Kind,
return false; return false;
} }
void Translator::translateFcn(Cfg *Func) { void Translator::translateFcn(std::unique_ptr<Cfg> Func) {
Ctx->cfgQueueBlockingPush(Func); Ctx->cfgQueueBlockingPush(std::move(Func));
if (Ctx->getFlags().NumTranslationThreads == 0) { if (Ctx->getFlags().NumTranslationThreads == 0) {
Ctx->translateFunctions(); Ctx->translateFunctions();
} }
......
...@@ -45,7 +45,7 @@ public: ...@@ -45,7 +45,7 @@ public:
/// Translates the constructed ICE function Fcn to machine code. /// Translates the constructed ICE function Fcn to machine code.
/// Takes ownership of Func. /// Takes ownership of Func.
void translateFcn(Cfg *Func); void translateFcn(std::unique_ptr<Cfg> Func);
/// Emits the constant pool. /// Emits the constant pool.
void emitConstants(); void emitConstants();
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment