2009-06-04 02:40:21 +08:00
|
|
|
//===--- CGCXXTemp.cpp - Emit LLVM Code for C++ temporaries ---------------===//
|
|
|
|
//
|
|
|
|
// The LLVM Compiler Infrastructure
|
|
|
|
//
|
|
|
|
// This file is distributed under the University of Illinois Open Source
|
|
|
|
// License. See LICENSE.TXT for details.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
//
|
|
|
|
// This contains code dealing with C++ code generation of temporaries
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
|
|
|
#include "CodeGenFunction.h"
|
|
|
|
using namespace clang;
|
|
|
|
using namespace CodeGen;
|
|
|
|
|
|
|
|
void CodeGenFunction::PushCXXTemporary(const CXXTemporary *Temporary,
|
|
|
|
llvm::Value *Ptr) {
|
2009-06-04 03:05:16 +08:00
|
|
|
llvm::BasicBlock *DtorBlock = createBasicBlock("temp.dtor");
|
|
|
|
|
|
|
|
LiveTemporaries.push_back(CXXLiveTemporaryInfo(Temporary, Ptr, DtorBlock, 0));
|
2009-06-04 10:08:08 +08:00
|
|
|
|
|
|
|
PushCleanupBlock(DtorBlock);
|
2009-06-04 03:05:16 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
void CodeGenFunction::PopCXXTemporary() {
|
|
|
|
const CXXLiveTemporaryInfo& Info = LiveTemporaries.back();
|
2009-06-04 02:40:21 +08:00
|
|
|
|
2009-06-04 03:05:16 +08:00
|
|
|
CleanupBlockInfo CleanupInfo = PopCleanupBlock();
|
|
|
|
assert(CleanupInfo.CleanupBlock == Info.DtorBlock &&
|
|
|
|
"Cleanup block mismatch!");
|
|
|
|
assert(!CleanupInfo.SwitchBlock &&
|
|
|
|
"Should not have a switch block for temporary cleanup!");
|
|
|
|
assert(!CleanupInfo.EndBlock &&
|
|
|
|
"Should not have an end block for temporary cleanup!");
|
|
|
|
|
|
|
|
EmitBlock(Info.DtorBlock);
|
|
|
|
|
|
|
|
EmitCXXDestructorCall(Info.Temporary->getDestructor(),
|
|
|
|
Dtor_Complete, Info.ThisPtr);
|
|
|
|
|
|
|
|
LiveTemporaries.pop_back();
|
2009-06-04 02:40:21 +08:00
|
|
|
}
|
|
|
|
|
2009-06-04 03:05:16 +08:00
|
|
|
RValue
|
2009-06-04 02:40:21 +08:00
|
|
|
CodeGenFunction::EmitCXXExprWithTemporaries(const CXXExprWithTemporaries *E,
|
|
|
|
llvm::Value *AggLoc,
|
|
|
|
bool isAggLocVolatile) {
|
|
|
|
// Keep track of the current cleanup stack depth.
|
|
|
|
size_t CleanupStackDepth = CleanupEntries.size();
|
|
|
|
|
|
|
|
unsigned OldNumLiveTemporaries = LiveTemporaries.size();
|
|
|
|
|
|
|
|
RValue RV = EmitAnyExpr(E->getSubExpr(), AggLoc, isAggLocVolatile);
|
|
|
|
|
|
|
|
// Go through the temporaries backwards.
|
|
|
|
for (unsigned i = E->getNumTemporaries(); i != 0; --i) {
|
2009-06-04 02:54:26 +08:00
|
|
|
assert(LiveTemporaries.back().Temporary == E->getTemporary(i - 1));
|
2009-06-04 02:40:21 +08:00
|
|
|
LiveTemporaries.pop_back();
|
|
|
|
}
|
|
|
|
|
|
|
|
assert(OldNumLiveTemporaries == LiveTemporaries.size() &&
|
|
|
|
"Live temporary stack mismatch!");
|
|
|
|
|
|
|
|
EmitCleanupBlocks(CleanupStackDepth);
|
|
|
|
|
|
|
|
return RV;
|
|
|
|
}
|
2009-06-04 10:22:12 +08:00
|
|
|
|
|
|
|
void
|
|
|
|
CodeGenFunction::PushConditionalTempDestruction() {
|
|
|
|
// Store the current number of live temporaries.
|
|
|
|
ConditionalTempDestructionStack.push_back(LiveTemporaries.size());
|
|
|
|
}
|
|
|
|
|
|
|
|
void CodeGenFunction::PopConditionalTempDestruction() {
|
|
|
|
ConditionalTempDestructionStack.pop_back();
|
|
|
|
}
|
|
|
|
|