diff options
author | Ted Kremenek <kremenek@apple.com> | 2009-07-10 21:11:16 +0000 |
---|---|---|
committer | Ted Kremenek <kremenek@apple.com> | 2009-07-10 21:11:16 +0000 |
commit | 3f9811b46abcbb34c76d0e742dd31f899312d2bf (patch) | |
tree | 41216e3090a49904fe5399288b0ba6bcb5778086 | |
parent | 77a1fe945b3d0f1efa22e9d36a8ea7e272dd93c2 (diff) |
Fix crash in StoreManager::NewCastRegion regarding handling casts to void*,
void**, void***, etc. Such casts should just pass the region through.
git-svn-id: https://llvm.org/svn/llvm-project/cfe/trunk@75281 91177308-0d34-0410-b5e6-96231b3b80d8
-rw-r--r-- | lib/Analysis/Store.cpp | 27 | ||||
-rw-r--r-- | test/Analysis/misc-ps.m | 14 |
2 files changed, 38 insertions, 3 deletions
diff --git a/lib/Analysis/Store.cpp b/lib/Analysis/Store.cpp index d7633a46ac..ad6bd7e4f6 100644 --- a/lib/Analysis/Store.cpp +++ b/lib/Analysis/Store.cpp @@ -45,6 +45,24 @@ static bool IsCompleteType(ASTContext &Ctx, QualType Ty) { return true; } +static bool isVoidOrHigherOrderVoidPtr(ASTContext &Ctx, QualType Ty) { + while (true) { + Ty = Ctx.getCanonicalType(Ty); + + if (Ty->isVoidType()) + return true; + + if (const PointerType *PT = Ty->getAsPointerType()) { + Ty = PT->getPointeeType(); + continue; + } + + break; + } + + return false; +} + StoreManager::CastResult StoreManager::NewCastRegion(const GRState *state, const MemRegion* R, QualType CastToTy) { @@ -64,6 +82,10 @@ StoreManager::NewCastRegion(const GRState *state, const MemRegion* R, // already be handled. QualType PointeeTy = CastToTy->getAsPointerType()->getPointeeType(); + // Casts to 'void*', 'void**', 'void***', etc., should just pass through. + if (isVoidOrHigherOrderVoidPtr(Ctx, PointeeTy)) + return CastResult(state, R); + // Process region cast according to the kind of the region being cast. switch (R->getKind()) { case MemRegion::BEG_TYPED_REGIONS: @@ -78,9 +100,8 @@ StoreManager::NewCastRegion(const GRState *state, const MemRegion* R, case MemRegion::CodeTextRegionKind: { // CodeTextRegion should be cast to only function pointer type. - assert(CastToTy->isFunctionPointerType() || CastToTy->isBlockPointerType() - || (CastToTy->isPointerType() && - CastToTy->getAsPointerType()->getPointeeType()->isVoidType())); + assert(CastToTy->isFunctionPointerType() || + CastToTy->isBlockPointerType()); break; } diff --git a/test/Analysis/misc-ps.m b/test/Analysis/misc-ps.m index 5e063d07fd..19d7622805 100644 --- a/test/Analysis/misc-ps.m +++ b/test/Analysis/misc-ps.m @@ -324,3 +324,17 @@ void test_rdar_7034511(NSArray *y) { if (x == ((void*) 0)) {} } +// Handle arbitrary void*^n -> void*^m casts. This was previously causing +// a crash in CastRegion. +void handle_higher_order_voidptr_casts() { + void **ptr; + typedef void *PVOID; + typedef long INT_PTR, *PINT_PTR; + typedef INT_PTR (*FARPROC)(); + FARPROC handle_higher_order_voidptr_casts_aux(); + PVOID handle_higher_order_voidptr_casts_aux_2(PVOID volatile *x); + + ptr = (void**) handle_higher_order_voidptr_casts_aux(); + handle_higher_order_voidptr_casts_aux_2(ptr); +} + |