зеркало из https://github.com/microsoft/clang-1.git
Use function pointers, rather than references, to pass Destroyers
around, in the process cleaning up the various gcc/msvc compiler workarounds. git-svn-id: https://llvm.org/svn/llvm-project/cfe/trunk@149036 91177308-0d34-0410-b5e6-96231b3b80d8
This commit is contained in:
Родитель
24466d8a90
Коммит
516bbd42e6
|
@ -535,9 +535,9 @@ static void enterBlockScope(CodeGenFunction &CGF, BlockDecl *block) {
|
|||
// Block captures count as local values and have imprecise semantics.
|
||||
// They also can't be arrays, so need to worry about that.
|
||||
if (dtorKind == QualType::DK_objc_strong_lifetime) {
|
||||
destroyer = &CodeGenFunction::destroyARCStrongImprecise;
|
||||
destroyer = CodeGenFunction::destroyARCStrongImprecise;
|
||||
} else {
|
||||
destroyer = &CGF.getDestroyer(dtorKind);
|
||||
destroyer = CGF.getDestroyer(dtorKind);
|
||||
}
|
||||
|
||||
// GEP down to the address.
|
||||
|
@ -554,7 +554,7 @@ static void enterBlockScope(CodeGenFunction &CGF, BlockDecl *block) {
|
|||
cleanupKind = InactiveNormalAndEHCleanup;
|
||||
|
||||
CGF.pushDestroy(cleanupKind, addr, variable->getType(),
|
||||
*destroyer, useArrayEHCleanup);
|
||||
destroyer, useArrayEHCleanup);
|
||||
|
||||
// Remember where that cleanup was.
|
||||
capture.setCleanup(CGF.EHStack.stable_begin());
|
||||
|
|
|
@ -945,13 +945,13 @@ namespace {
|
|||
|
||||
class DestroyField : public EHScopeStack::Cleanup {
|
||||
const FieldDecl *field;
|
||||
CodeGenFunction::Destroyer &destroyer;
|
||||
CodeGenFunction::Destroyer *destroyer;
|
||||
bool useEHCleanupForArray;
|
||||
|
||||
public:
|
||||
DestroyField(const FieldDecl *field, CodeGenFunction::Destroyer *destroyer,
|
||||
bool useEHCleanupForArray)
|
||||
: field(field), destroyer(*destroyer),
|
||||
: field(field), destroyer(destroyer),
|
||||
useEHCleanupForArray(useEHCleanupForArray) {}
|
||||
|
||||
void Emit(CodeGenFunction &CGF, Flags flags) {
|
||||
|
|
|
@ -1098,8 +1098,6 @@ llvm::Value *CodeGenFunction::getNormalCleanupDestSlot() {
|
|||
void CodeGenFunction::EmitCXXTemporary(const CXXTemporary *Temporary,
|
||||
QualType TempType,
|
||||
llvm::Value *Ptr) {
|
||||
// This local is a GCC and MSVC compiler workaround.
|
||||
Destroyer *destroyer = &destroyCXXObject;
|
||||
pushDestroy(NormalAndEHCleanup, Ptr, TempType, *destroyer,
|
||||
pushDestroy(NormalAndEHCleanup, Ptr, TempType, destroyCXXObject,
|
||||
/*useEHCleanup*/ true);
|
||||
}
|
||||
|
|
|
@ -310,12 +310,12 @@ namespace {
|
|||
DestroyObject(llvm::Value *addr, QualType type,
|
||||
CodeGenFunction::Destroyer *destroyer,
|
||||
bool useEHCleanupForArray)
|
||||
: addr(addr), type(type), destroyer(*destroyer),
|
||||
: addr(addr), type(type), destroyer(destroyer),
|
||||
useEHCleanupForArray(useEHCleanupForArray) {}
|
||||
|
||||
llvm::Value *addr;
|
||||
QualType type;
|
||||
CodeGenFunction::Destroyer &destroyer;
|
||||
CodeGenFunction::Destroyer *destroyer;
|
||||
bool useEHCleanupForArray;
|
||||
|
||||
void Emit(CodeGenFunction &CGF, Flags flags) {
|
||||
|
@ -430,7 +430,7 @@ static void EmitAutoVarWithLifetime(CodeGenFunction &CGF, const VarDecl &var,
|
|||
break;
|
||||
|
||||
case Qualifiers::OCL_Strong: {
|
||||
CodeGenFunction::Destroyer &destroyer =
|
||||
CodeGenFunction::Destroyer *destroyer =
|
||||
(var.hasAttr<ObjCPreciseLifetimeAttr>()
|
||||
? CodeGenFunction::destroyARCStrongPrecise
|
||||
: CodeGenFunction::destroyARCStrongImprecise);
|
||||
|
@ -1111,7 +1111,7 @@ void CodeGenFunction::emitAutoVarTypeCleanup(
|
|||
}
|
||||
|
||||
// If we haven't chosen a more specific destroyer, use the default.
|
||||
if (!destroyer) destroyer = &getDestroyer(dtorKind);
|
||||
if (!destroyer) destroyer = getDestroyer(dtorKind);
|
||||
|
||||
// Use an EH cleanup in array destructors iff the destructor itself
|
||||
// is being pushed as an EH cleanup.
|
||||
|
@ -1155,25 +1155,17 @@ void CodeGenFunction::EmitAutoVarCleanups(const AutoVarEmission &emission) {
|
|||
enterByrefCleanup(emission);
|
||||
}
|
||||
|
||||
CodeGenFunction::Destroyer &
|
||||
CodeGenFunction::Destroyer *
|
||||
CodeGenFunction::getDestroyer(QualType::DestructionKind kind) {
|
||||
// This is surprisingly compiler-dependent. GCC 4.2 can't bind
|
||||
// references to functions directly in returns, and using '*&foo'
|
||||
// confuses MSVC. Luckily, the following code pattern works in both.
|
||||
Destroyer *destroyer = 0;
|
||||
switch (kind) {
|
||||
case QualType::DK_none: llvm_unreachable("no destroyer for trivial dtor");
|
||||
case QualType::DK_cxx_destructor:
|
||||
destroyer = &destroyCXXObject;
|
||||
break;
|
||||
return destroyCXXObject;
|
||||
case QualType::DK_objc_strong_lifetime:
|
||||
destroyer = &destroyARCStrongPrecise;
|
||||
break;
|
||||
return destroyARCStrongPrecise;
|
||||
case QualType::DK_objc_weak_lifetime:
|
||||
destroyer = &destroyARCWeak;
|
||||
break;
|
||||
return destroyARCWeak;
|
||||
}
|
||||
return *destroyer;
|
||||
}
|
||||
|
||||
/// pushDestroy - Push the standard destructor for the given type.
|
||||
|
@ -1187,7 +1179,7 @@ void CodeGenFunction::pushDestroy(QualType::DestructionKind dtorKind,
|
|||
}
|
||||
|
||||
void CodeGenFunction::pushDestroy(CleanupKind cleanupKind, llvm::Value *addr,
|
||||
QualType type, Destroyer &destroyer,
|
||||
QualType type, Destroyer *destroyer,
|
||||
bool useEHCleanupForArray) {
|
||||
pushFullExprCleanup<DestroyObject>(cleanupKind, addr, type,
|
||||
destroyer, useEHCleanupForArray);
|
||||
|
@ -1205,7 +1197,7 @@ void CodeGenFunction::pushDestroy(CleanupKind cleanupKind, llvm::Value *addr,
|
|||
/// used when destroying array elements, in case one of the
|
||||
/// destructions throws an exception
|
||||
void CodeGenFunction::emitDestroy(llvm::Value *addr, QualType type,
|
||||
Destroyer &destroyer,
|
||||
Destroyer *destroyer,
|
||||
bool useEHCleanupForArray) {
|
||||
const ArrayType *arrayType = getContext().getAsArrayType(type);
|
||||
if (!arrayType)
|
||||
|
@ -1242,7 +1234,7 @@ void CodeGenFunction::emitDestroy(llvm::Value *addr, QualType type,
|
|||
void CodeGenFunction::emitArrayDestroy(llvm::Value *begin,
|
||||
llvm::Value *end,
|
||||
QualType type,
|
||||
Destroyer &destroyer,
|
||||
Destroyer *destroyer,
|
||||
bool checkZeroLength,
|
||||
bool useEHCleanup) {
|
||||
assert(!type->isArrayType());
|
||||
|
@ -1293,7 +1285,7 @@ void CodeGenFunction::emitArrayDestroy(llvm::Value *begin,
|
|||
static void emitPartialArrayDestroy(CodeGenFunction &CGF,
|
||||
llvm::Value *begin, llvm::Value *end,
|
||||
QualType type,
|
||||
CodeGenFunction::Destroyer &destroyer) {
|
||||
CodeGenFunction::Destroyer *destroyer) {
|
||||
// If the element type is itself an array, drill down.
|
||||
unsigned arrayDepth = 0;
|
||||
while (const ArrayType *arrayType = CGF.getContext().getAsArrayType(type)) {
|
||||
|
@ -1326,13 +1318,13 @@ namespace {
|
|||
llvm::Value *ArrayBegin;
|
||||
llvm::Value *ArrayEnd;
|
||||
QualType ElementType;
|
||||
CodeGenFunction::Destroyer &Destroyer;
|
||||
CodeGenFunction::Destroyer *Destroyer;
|
||||
public:
|
||||
RegularPartialArrayDestroy(llvm::Value *arrayBegin, llvm::Value *arrayEnd,
|
||||
QualType elementType,
|
||||
CodeGenFunction::Destroyer *destroyer)
|
||||
: ArrayBegin(arrayBegin), ArrayEnd(arrayEnd),
|
||||
ElementType(elementType), Destroyer(*destroyer) {}
|
||||
ElementType(elementType), Destroyer(destroyer) {}
|
||||
|
||||
void Emit(CodeGenFunction &CGF, Flags flags) {
|
||||
emitPartialArrayDestroy(CGF, ArrayBegin, ArrayEnd,
|
||||
|
@ -1347,14 +1339,14 @@ namespace {
|
|||
llvm::Value *ArrayBegin;
|
||||
llvm::Value *ArrayEndPointer;
|
||||
QualType ElementType;
|
||||
CodeGenFunction::Destroyer &Destroyer;
|
||||
CodeGenFunction::Destroyer *Destroyer;
|
||||
public:
|
||||
IrregularPartialArrayDestroy(llvm::Value *arrayBegin,
|
||||
llvm::Value *arrayEndPointer,
|
||||
QualType elementType,
|
||||
CodeGenFunction::Destroyer *destroyer)
|
||||
: ArrayBegin(arrayBegin), ArrayEndPointer(arrayEndPointer),
|
||||
ElementType(elementType), Destroyer(*destroyer) {}
|
||||
ElementType(elementType), Destroyer(destroyer) {}
|
||||
|
||||
void Emit(CodeGenFunction &CGF, Flags flags) {
|
||||
llvm::Value *arrayEnd = CGF.Builder.CreateLoad(ArrayEndPointer);
|
||||
|
@ -1377,10 +1369,10 @@ namespace {
|
|||
void CodeGenFunction::pushIrregularPartialArrayCleanup(llvm::Value *arrayBegin,
|
||||
llvm::Value *arrayEndPointer,
|
||||
QualType elementType,
|
||||
Destroyer &destroyer) {
|
||||
Destroyer *destroyer) {
|
||||
pushFullExprCleanup<IrregularPartialArrayDestroy>(EHCleanup,
|
||||
arrayBegin, arrayEndPointer,
|
||||
elementType, &destroyer);
|
||||
elementType, destroyer);
|
||||
}
|
||||
|
||||
/// pushRegularPartialArrayCleanup - Push an EH cleanup to destroy
|
||||
|
@ -1396,10 +1388,10 @@ void CodeGenFunction::pushIrregularPartialArrayCleanup(llvm::Value *arrayBegin,
|
|||
void CodeGenFunction::pushRegularPartialArrayCleanup(llvm::Value *arrayBegin,
|
||||
llvm::Value *arrayEnd,
|
||||
QualType elementType,
|
||||
Destroyer &destroyer) {
|
||||
Destroyer *destroyer) {
|
||||
pushFullExprCleanup<RegularPartialArrayDestroy>(EHCleanup,
|
||||
arrayBegin, arrayEnd,
|
||||
elementType, &destroyer);
|
||||
elementType, destroyer);
|
||||
}
|
||||
|
||||
namespace {
|
||||
|
|
|
@ -343,7 +343,7 @@ void CodeGenFunction::GenerateCXXGlobalDtorFunc(llvm::Function *Fn,
|
|||
llvm::Function *
|
||||
CodeGenFunction::generateDestroyHelper(llvm::Constant *addr,
|
||||
QualType type,
|
||||
Destroyer &destroyer,
|
||||
Destroyer *destroyer,
|
||||
bool useEHCleanupForArray) {
|
||||
FunctionArgList args;
|
||||
ImplicitParamDecl dst(0, SourceLocation(), 0, getContext().VoidPtrTy);
|
||||
|
|
|
@ -487,21 +487,17 @@ CodeGenFunction::EmitReferenceBindingToExpr(const Expr *E,
|
|||
case Qualifiers::OCL_Strong: {
|
||||
bool precise = VD && VD->hasAttr<ObjCPreciseLifetimeAttr>();
|
||||
CleanupKind cleanupKind = getARCCleanupKind();
|
||||
// This local is a GCC and MSVC compiler workaround.
|
||||
Destroyer *destroyer = precise ? &destroyARCStrongPrecise :
|
||||
&destroyARCStrongImprecise;
|
||||
pushDestroy(cleanupKind, ReferenceTemporary, ObjCARCReferenceLifetimeType,
|
||||
*destroyer, cleanupKind & EHCleanup);
|
||||
precise ? destroyARCStrongPrecise : destroyARCStrongImprecise,
|
||||
cleanupKind & EHCleanup);
|
||||
break;
|
||||
}
|
||||
|
||||
case Qualifiers::OCL_Weak: {
|
||||
// This local is a GCC and MSVC compiler workaround.
|
||||
Destroyer *destroyer = &destroyARCWeak;
|
||||
// __weak objects always get EH cleanups; otherwise, exceptions
|
||||
// could cause really nasty crashes instead of mere leaks.
|
||||
pushDestroy(NormalAndEHCleanup, ReferenceTemporary,
|
||||
ObjCARCReferenceLifetimeType, *destroyer, true);
|
||||
ObjCARCReferenceLifetimeType, destroyARCWeak, true);
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
|
|
@ -1047,13 +1047,13 @@ namespace {
|
|||
private:
|
||||
llvm::Value *addr;
|
||||
const ObjCIvarDecl *ivar;
|
||||
CodeGenFunction::Destroyer &destroyer;
|
||||
CodeGenFunction::Destroyer *destroyer;
|
||||
bool useEHCleanupForArray;
|
||||
public:
|
||||
DestroyIvar(llvm::Value *addr, const ObjCIvarDecl *ivar,
|
||||
CodeGenFunction::Destroyer *destroyer,
|
||||
bool useEHCleanupForArray)
|
||||
: addr(addr), ivar(ivar), destroyer(*destroyer),
|
||||
: addr(addr), ivar(ivar), destroyer(destroyer),
|
||||
useEHCleanupForArray(useEHCleanupForArray) {}
|
||||
|
||||
void Emit(CodeGenFunction &CGF, Flags flags) {
|
||||
|
@ -1093,11 +1093,11 @@ static void emitCXXDestructMethod(CodeGenFunction &CGF,
|
|||
// Use a call to objc_storeStrong to destroy strong ivars, for the
|
||||
// general benefit of the tools.
|
||||
if (dtorKind == QualType::DK_objc_strong_lifetime) {
|
||||
destroyer = &destroyARCStrongWithStore;
|
||||
destroyer = destroyARCStrongWithStore;
|
||||
|
||||
// Otherwise use the default for the destruction kind.
|
||||
} else {
|
||||
destroyer = &CGF.getDestroyer(dtorKind);
|
||||
destroyer = CGF.getDestroyer(dtorKind);
|
||||
}
|
||||
|
||||
CleanupKind cleanupKind = CGF.getCleanupKind(dtorKind);
|
||||
|
|
|
@ -1244,27 +1244,27 @@ public:
|
|||
void pushIrregularPartialArrayCleanup(llvm::Value *arrayBegin,
|
||||
llvm::Value *arrayEndPointer,
|
||||
QualType elementType,
|
||||
Destroyer &destroyer);
|
||||
Destroyer *destroyer);
|
||||
void pushRegularPartialArrayCleanup(llvm::Value *arrayBegin,
|
||||
llvm::Value *arrayEnd,
|
||||
QualType elementType,
|
||||
Destroyer &destroyer);
|
||||
Destroyer *destroyer);
|
||||
|
||||
void pushDestroy(QualType::DestructionKind dtorKind,
|
||||
llvm::Value *addr, QualType type);
|
||||
void pushDestroy(CleanupKind kind, llvm::Value *addr, QualType type,
|
||||
Destroyer &destroyer, bool useEHCleanupForArray);
|
||||
void emitDestroy(llvm::Value *addr, QualType type, Destroyer &destroyer,
|
||||
Destroyer *destroyer, bool useEHCleanupForArray);
|
||||
void emitDestroy(llvm::Value *addr, QualType type, Destroyer *destroyer,
|
||||
bool useEHCleanupForArray);
|
||||
llvm::Function *generateDestroyHelper(llvm::Constant *addr,
|
||||
QualType type,
|
||||
Destroyer &destroyer,
|
||||
Destroyer *destroyer,
|
||||
bool useEHCleanupForArray);
|
||||
void emitArrayDestroy(llvm::Value *begin, llvm::Value *end,
|
||||
QualType type, Destroyer &destroyer,
|
||||
QualType type, Destroyer *destroyer,
|
||||
bool checkZeroLength, bool useEHCleanup);
|
||||
|
||||
Destroyer &getDestroyer(QualType::DestructionKind destructionKind);
|
||||
Destroyer *getDestroyer(QualType::DestructionKind destructionKind);
|
||||
|
||||
/// Determines whether an EH cleanup is required to destroy a type
|
||||
/// with the given destruction kind.
|
||||
|
|
Загрузка…
Ссылка в новой задаче