mirror of
https://github.com/capstone-engine/llvm-capstone.git
synced 2025-02-26 05:18:46 +00:00
As a first step towards fixing PR9641, add a CK_DynamicToNull cast kind which
represents a dynamic cast where we know that the result is always null. For example: struct A { virtual ~A(); }; struct B final : A { }; struct C { }; bool f(B* b) { return dynamic_cast<C*>(b); } llvm-svn: 129256
This commit is contained in:
parent
784ba65787
commit
c602006638
@ -2183,6 +2183,7 @@ private:
|
||||
// These should not have an inheritance path.
|
||||
case CK_BitCast:
|
||||
case CK_Dynamic:
|
||||
case CK_DynamicToNull:
|
||||
case CK_ToUnion:
|
||||
case CK_ArrayToPointerDecay:
|
||||
case CK_FunctionToPointerDecay:
|
||||
|
@ -82,6 +82,10 @@ enum CastKind {
|
||||
/// CK_Dynamic - A C++ dynamic_cast.
|
||||
CK_Dynamic,
|
||||
|
||||
/// CK_DynamicToNull - A C++ dynamic_cast that can be proven to
|
||||
/// always yield a null result.
|
||||
CK_DynamicToNull,
|
||||
|
||||
/// CK_ToUnion - The GCC cast-to-union extension.
|
||||
/// int -> union { int x; float y; }
|
||||
/// float -> union { int x; float y; }
|
||||
|
@ -987,6 +987,8 @@ const char *CastExpr::getCastKindName() const {
|
||||
return "UncheckedDerivedToBase";
|
||||
case CK_Dynamic:
|
||||
return "Dynamic";
|
||||
case CK_DynamicToNull:
|
||||
return "DynamicToNull";
|
||||
case CK_ToUnion:
|
||||
return "ToUnion";
|
||||
case CK_ArrayToPointerDecay:
|
||||
|
@ -1768,6 +1768,7 @@ bool IntExprEvaluator::VisitCastExpr(CastExpr *E) {
|
||||
case CK_DerivedToBase:
|
||||
case CK_UncheckedDerivedToBase:
|
||||
case CK_Dynamic:
|
||||
case CK_DynamicToNull:
|
||||
case CK_ToUnion:
|
||||
case CK_ArrayToPointerDecay:
|
||||
case CK_FunctionToPointerDecay:
|
||||
@ -2315,6 +2316,7 @@ bool ComplexExprEvaluator::VisitCastExpr(CastExpr *E) {
|
||||
case CK_DerivedToBase:
|
||||
case CK_UncheckedDerivedToBase:
|
||||
case CK_Dynamic:
|
||||
case CK_DynamicToNull:
|
||||
case CK_ToUnion:
|
||||
case CK_ArrayToPointerDecay:
|
||||
case CK_FunctionToPointerDecay:
|
||||
|
@ -1867,7 +1867,8 @@ LValue CodeGenFunction::EmitCastLValue(const CastExpr *E) {
|
||||
return MakeAddrLValue(V, E->getType());
|
||||
}
|
||||
|
||||
case CK_Dynamic: {
|
||||
case CK_Dynamic:
|
||||
case CK_DynamicToNull: {
|
||||
LValue LV = EmitLValue(E->getSubExpr());
|
||||
llvm::Value *V = LV.getAddress();
|
||||
const CXXDynamicCastExpr *DCE = cast<CXXDynamicCastExpr>(E);
|
||||
|
@ -255,7 +255,10 @@ void AggExprEmitter::VisitCastExpr(CastExpr *E) {
|
||||
}
|
||||
|
||||
switch (E->getCastKind()) {
|
||||
case CK_Dynamic: {
|
||||
case CK_Dynamic:
|
||||
case CK_DynamicToNull: {
|
||||
|
||||
// FIXME: Actually handle DynamicToNull here.
|
||||
assert(isa<CXXDynamicCastExpr>(E) && "CK_Dynamic without a dynamic_cast?");
|
||||
LValue LV = CGF.EmitCheckedLValue(E->getSubExpr());
|
||||
// FIXME: Do we also need to handle property references here?
|
||||
|
@ -552,6 +552,7 @@ public:
|
||||
case CK_GetObjCProperty:
|
||||
case CK_ToVoid:
|
||||
case CK_Dynamic:
|
||||
case CK_DynamicToNull:
|
||||
case CK_ResolveUnknownAnyType:
|
||||
return 0;
|
||||
|
||||
|
@ -1053,7 +1053,8 @@ Value *ScalarExprEmitter::EmitCastExpr(CastExpr *CE) {
|
||||
CE->path_begin(), CE->path_end(),
|
||||
ShouldNullCheckClassCastValue(CE));
|
||||
}
|
||||
case CK_Dynamic: {
|
||||
case CK_Dynamic:
|
||||
case CK_DynamicToNull: {
|
||||
Value *V = Visit(const_cast<Expr*>(E));
|
||||
const CXXDynamicCastExpr *DCE = cast<CXXDynamicCastExpr>(CE);
|
||||
return CGF.EmitDynamicCast(V, DCE);
|
||||
|
@ -522,6 +522,14 @@ CheckDynamicCast(Sema &Self, ExprResult &SrcExpr, QualType DestType,
|
||||
return;
|
||||
}
|
||||
|
||||
// If the source class is marked 'final', and the destination class does not
|
||||
// derive from the source class, then we know that the result is always null.
|
||||
if (SrcRecord->getDecl()->hasAttr<FinalAttr>() &&
|
||||
!Self.IsDerivedFrom(DestPointee, SrcPointee)) {
|
||||
Kind = CK_DynamicToNull;
|
||||
return;
|
||||
}
|
||||
|
||||
// C++ 5.2.7p5
|
||||
// Upcasts are resolved statically.
|
||||
if (DestRecord && Self.IsDerivedFrom(SrcPointee, DestPointee)) {
|
||||
|
@ -2178,6 +2178,7 @@ void ExprEngine::VisitCast(const CastExpr *CastE, const Expr *Ex,
|
||||
// Various C++ casts that are not handled yet.
|
||||
case CK_ResolveUnknownAnyType:
|
||||
case CK_Dynamic:
|
||||
case CK_DynamicToNull:
|
||||
case CK_ToUnion:
|
||||
case CK_BaseToDerived:
|
||||
case CK_NullToMemberPointer:
|
||||
|
Loading…
x
Reference in New Issue
Block a user