[asan] add a feature to detect new-delete-size-mismatch (when used with -Xclang -fsized-deallocation). Not yet on Mac. Also, remove some unused code.

llvm-svn: 214296
This commit is contained in:
Kostya Serebryany 2014-07-30 09:48:23 +00:00
parent dda8e784f6
commit 69852a843c
8 changed files with 109 additions and 21 deletions

View File

@ -142,6 +142,8 @@ struct AsanThreadLocalMallocStorage {
void *asan_memalign(uptr alignment, uptr size, StackTrace *stack, void *asan_memalign(uptr alignment, uptr size, StackTrace *stack,
AllocType alloc_type); AllocType alloc_type);
void asan_free(void *ptr, StackTrace *stack, AllocType alloc_type); void asan_free(void *ptr, StackTrace *stack, AllocType alloc_type);
void asan_sized_free(void *ptr, uptr size, StackTrace *stack,
AllocType alloc_type);
void *asan_malloc(uptr size, StackTrace *stack); void *asan_malloc(uptr size, StackTrace *stack);
void *asan_calloc(uptr nmemb, uptr size, StackTrace *stack); void *asan_calloc(uptr nmemb, uptr size, StackTrace *stack);

View File

@ -168,23 +168,6 @@ struct AsanChunk: ChunkBase {
} }
return reinterpret_cast<void*>(Beg() - RZLog2Size(rz_log)); return reinterpret_cast<void*>(Beg() - RZLog2Size(rz_log));
} }
// If we don't use stack depot, we store the alloc/free stack traces
// in the chunk itself.
u32 *AllocStackBeg() {
return (u32*)(Beg() - RZLog2Size(rz_log));
}
uptr AllocStackSize() {
CHECK_LE(RZLog2Size(rz_log), kChunkHeaderSize);
return (RZLog2Size(rz_log) - kChunkHeaderSize) / sizeof(u32);
}
u32 *FreeStackBeg() {
return (u32*)(Beg() + kChunkHeader2Size);
}
uptr FreeStackSize() {
if (user_requested_size < kChunkHeader2Size) return 0;
uptr available = RoundUpTo(user_requested_size, SHADOW_GRANULARITY);
return (available - kChunkHeader2Size) / sizeof(u32);
}
bool AddrIsInside(uptr addr, bool locked_version = false) { bool AddrIsInside(uptr addr, bool locked_version = false) {
return (addr >= Beg()) && (addr < Beg() + UsedSize(locked_version)); return (addr >= Beg()) && (addr < Beg() + UsedSize(locked_version));
} }
@ -464,12 +447,17 @@ static void QuarantineChunk(AsanChunk *m, void *ptr,
} }
} }
static void Deallocate(void *ptr, StackTrace *stack, AllocType alloc_type) { static void Deallocate(void *ptr, uptr delete_size, StackTrace *stack,
AllocType alloc_type) {
uptr p = reinterpret_cast<uptr>(ptr); uptr p = reinterpret_cast<uptr>(ptr);
if (p == 0) return; if (p == 0) return;
uptr chunk_beg = p - kChunkHeaderSize; uptr chunk_beg = p - kChunkHeaderSize;
AsanChunk *m = reinterpret_cast<AsanChunk *>(chunk_beg); AsanChunk *m = reinterpret_cast<AsanChunk *>(chunk_beg);
if (delete_size && flags()->new_delete_size_mismatch &&
delete_size != m->UsedSize()) {
ReportNewDeleteSizeMismatch(p, delete_size, stack);
}
ASAN_FREE_HOOK(ptr); ASAN_FREE_HOOK(ptr);
// Must mark the chunk as quarantined before any changes to its metadata. // Must mark the chunk as quarantined before any changes to its metadata.
AtomicallySetQuarantineFlag(m, ptr, stack); AtomicallySetQuarantineFlag(m, ptr, stack);
@ -496,7 +484,7 @@ static void *Reallocate(void *old_ptr, uptr new_size, StackTrace *stack) {
// If realloc() races with free(), we may start copying freed memory. // If realloc() races with free(), we may start copying freed memory.
// However, we will report racy double-free later anyway. // However, we will report racy double-free later anyway.
REAL(memcpy)(new_ptr, old_ptr, memcpy_size); REAL(memcpy)(new_ptr, old_ptr, memcpy_size);
Deallocate(old_ptr, stack, FROM_MALLOC); Deallocate(old_ptr, 0, stack, FROM_MALLOC);
} }
return new_ptr; return new_ptr;
} }
@ -595,7 +583,12 @@ void *asan_memalign(uptr alignment, uptr size, StackTrace *stack,
} }
void asan_free(void *ptr, StackTrace *stack, AllocType alloc_type) { void asan_free(void *ptr, StackTrace *stack, AllocType alloc_type) {
Deallocate(ptr, stack, alloc_type); Deallocate(ptr, 0, stack, alloc_type);
}
void asan_sized_free(void *ptr, uptr size, StackTrace *stack,
AllocType alloc_type) {
Deallocate(ptr, size, stack, alloc_type);
} }
void *asan_malloc(uptr size, StackTrace *stack) { void *asan_malloc(uptr size, StackTrace *stack) {
@ -617,7 +610,7 @@ void *asan_realloc(void *p, uptr size, StackTrace *stack) {
if (p == 0) if (p == 0)
return Allocate(size, 8, stack, FROM_MALLOC, true); return Allocate(size, 8, stack, FROM_MALLOC, true);
if (size == 0) { if (size == 0) {
Deallocate(p, stack, FROM_MALLOC); Deallocate(p, 0, stack, FROM_MALLOC);
return 0; return 0;
} }
return Reallocate(p, size, stack); return Reallocate(p, size, stack);

View File

@ -58,6 +58,7 @@ struct Flags {
bool poison_heap; bool poison_heap;
bool poison_partial; bool poison_partial;
bool alloc_dealloc_mismatch; bool alloc_dealloc_mismatch;
bool new_delete_size_mismatch;
bool strict_memcmp; bool strict_memcmp;
bool strict_init_order; bool strict_init_order;
bool start_deactivated; bool start_deactivated;

View File

@ -105,6 +105,11 @@ CXX_OPERATOR_ATTRIBUTE
void operator delete[](void *ptr, std::nothrow_t const&) { void operator delete[](void *ptr, std::nothrow_t const&) {
OPERATOR_DELETE_BODY(FROM_NEW_BR); OPERATOR_DELETE_BODY(FROM_NEW_BR);
} }
CXX_OPERATOR_ATTRIBUTE
void operator delete(void *ptr, size_t size) throw() {
GET_STACK_TRACE_FREE;
asan_sized_free(ptr, size, &stack, FROM_NEW);
}
#else // SANITIZER_MAC #else // SANITIZER_MAC
INTERCEPTOR(void, _ZdlPv, void *ptr) { INTERCEPTOR(void, _ZdlPv, void *ptr) {

View File

@ -652,6 +652,28 @@ void ReportDoubleFree(uptr addr, StackTrace *free_stack) {
ReportErrorSummary("double-free", &stack); ReportErrorSummary("double-free", &stack);
} }
void ReportNewDeleteSizeMismatch(uptr addr, uptr delete_size,
StackTrace *free_stack) {
ScopedInErrorReport in_report;
Decorator d;
Printf("%s", d.Warning());
char tname[128];
u32 curr_tid = GetCurrentTidOrInvalid();
Report("ERROR: AddressSanitizer: new-delete-size-mismatch on %p in "
"thread T%d%s:\n",
addr, curr_tid,
ThreadNameWithParenthesis(curr_tid, tname, sizeof(tname)));
Printf("%s sized operator delete called with size %zd\n", d.EndWarning(),
delete_size);
CHECK_GT(free_stack->size, 0);
GET_STACK_TRACE_FATAL(free_stack->trace[0], free_stack->top_frame_bp);
stack.Print();
DescribeHeapAddress(addr, 1);
ReportErrorSummary("new-delete-size-mismatch", &stack);
Report("HINT: if you don't care about these warnings you may set "
"ASAN_OPTIONS=new_delete_size_mismatch=0\n");
}
void ReportFreeNotMalloced(uptr addr, StackTrace *free_stack) { void ReportFreeNotMalloced(uptr addr, StackTrace *free_stack) {
ScopedInErrorReport in_report; ScopedInErrorReport in_report;
Decorator d; Decorator d;

View File

@ -45,6 +45,8 @@ void NORETURN
ReportStackOverflow(uptr pc, uptr sp, uptr bp, void *context, uptr addr); ReportStackOverflow(uptr pc, uptr sp, uptr bp, void *context, uptr addr);
void NORETURN ReportSIGSEGV(const char *description, uptr pc, uptr sp, uptr bp, void NORETURN ReportSIGSEGV(const char *description, uptr pc, uptr sp, uptr bp,
void *context, uptr addr); void *context, uptr addr);
void NORETURN ReportNewDeleteSizeMismatch(uptr addr, uptr delete_size,
StackTrace *free_stack);
void NORETURN ReportDoubleFree(uptr addr, StackTrace *free_stack); void NORETURN ReportDoubleFree(uptr addr, StackTrace *free_stack);
void NORETURN ReportFreeNotMalloced(uptr addr, StackTrace *free_stack); void NORETURN ReportFreeNotMalloced(uptr addr, StackTrace *free_stack);
void NORETURN ReportAllocTypeMismatch(uptr addr, StackTrace *free_stack, void NORETURN ReportAllocTypeMismatch(uptr addr, StackTrace *free_stack,

View File

@ -198,6 +198,10 @@ static void ParseFlagsFromString(Flags *f, const char *str) {
ParseFlag(str, &f->alloc_dealloc_mismatch, "alloc_dealloc_mismatch", ParseFlag(str, &f->alloc_dealloc_mismatch, "alloc_dealloc_mismatch",
"Report errors on malloc/delete, new/free, new/delete[], etc."); "Report errors on malloc/delete, new/free, new/delete[], etc.");
ParseFlag(str, &f->new_delete_size_mismatch, "new_delete_size_mismatch",
"Report errors on mismatch betwen size of new and delete.");
ParseFlag(str, &f->strict_memcmp, "strict_memcmp", ParseFlag(str, &f->strict_memcmp, "strict_memcmp",
"If true, assume that memcmp(p1, p2, n) always reads n bytes before " "If true, assume that memcmp(p1, p2, n) always reads n bytes before "
"comparing p1 and p2."); "comparing p1 and p2.");
@ -274,6 +278,7 @@ void InitializeFlags(Flags *f, const char *env) {
// https://code.google.com/p/address-sanitizer/issues/detail?id=309 // https://code.google.com/p/address-sanitizer/issues/detail?id=309
// TODO(glider,timurrrr): Fix known issues and enable this back. // TODO(glider,timurrrr): Fix known issues and enable this back.
f->alloc_dealloc_mismatch = (SANITIZER_MAC == 0) && (SANITIZER_WINDOWS == 0); f->alloc_dealloc_mismatch = (SANITIZER_MAC == 0) && (SANITIZER_WINDOWS == 0);
f->new_delete_size_mismatch = true;
f->strict_memcmp = true; f->strict_memcmp = true;
f->strict_init_order = false; f->strict_init_order = false;
f->start_deactivated = false; f->start_deactivated = false;

View File

@ -0,0 +1,58 @@
// RUN: %clangxx_asan -Xclang -fsized-deallocation -O0 %s -o %t
// RUN: not %run %t 2>&1 | FileCheck %s
// RUN: ASAN_OPTIONS=new_delete_size_mismatch=1 not %run %t 2>&1 | FileCheck %s
// RUN: ASAN_OPTIONS=new_delete_size_mismatch=0 %run %t
#include <new>
#include <stdio.h>
inline void break_optimization(void *arg) {
__asm__ __volatile__("" : : "r" (arg) : "memory");
}
struct S12 {
int a, b, c;
};
struct S20 {
int a, b, c, d, e;
};
void Del12(S12 *x) {
break_optimization(x);
delete x;
}
void Del12NoThrow(S12 *x) {
break_optimization(x);
operator delete(x, std::nothrow);
}
void Del12Ar(S12 *x) {
break_optimization(x);
delete [] x;
}
void Del12ArNoThrow(S12 *x) {
break_optimization(x);
operator delete[](x, std::nothrow);
}
int main() {
// These are correct.
Del12(new S12);
Del12NoThrow(new S12);
Del12Ar(new S12[100]);
Del12ArNoThrow(new S12[100]);
// Here we pass wrong type of pointer to delete,
// but [] and nothrow variants of delete are not sized.
Del12Ar(reinterpret_cast<S12*>(new S20[100]));
Del12NoThrow(reinterpret_cast<S12*>(new S20));
Del12ArNoThrow(reinterpret_cast<S12*>(new S20[100]));
fprintf(stderr, "OK SO FAR\n");
// CHECK: OK SO FAR
// Here asan should bark as we are passing a wrong type of pointer
// to sized delete.
Del12(reinterpret_cast<S12*>(new S20));
// CHECK: AddressSanitizer: new-delete-size-mismatch
// CHECK: sized operator delete called with size
// CHECK: is located 0 bytes inside of 20-byte region
// CHECK: SUMMARY: AddressSanitizer: new-delete-size-mismatch
}