2019-08-01 21:48:31 +08:00
|
|
|
//===-- asan_oob_test.cpp -------------------------------------------------===//
|
2013-01-22 14:39:20 +08:00
|
|
|
//
|
2019-01-19 16:50:56 +08:00
|
|
|
// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
|
|
|
|
// See https://llvm.org/LICENSE.txt for license information.
|
|
|
|
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
|
2013-01-22 14:39:20 +08:00
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
//
|
|
|
|
// This file is a part of AddressSanitizer, an address sanity checker.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
#include "asan_test_utils.h"
|
|
|
|
|
|
|
|
NOINLINE void asan_write_sized_aligned(uint8_t *p, size_t size) {
|
|
|
|
EXPECT_EQ(0U, ((uintptr_t)p % size));
|
|
|
|
if (size == 1) asan_write((uint8_t*)p);
|
|
|
|
else if (size == 2) asan_write((uint16_t*)p);
|
|
|
|
else if (size == 4) asan_write((uint32_t*)p);
|
|
|
|
else if (size == 8) asan_write((uint64_t*)p);
|
|
|
|
}
|
|
|
|
|
|
|
|
template<typename T>
|
|
|
|
NOINLINE void oob_test(int size, int off) {
|
|
|
|
char *p = (char*)malloc_aaa(size);
|
|
|
|
// fprintf(stderr, "writing %d byte(s) into [%p,%p) with offset %d\n",
|
|
|
|
// sizeof(T), p, p + size, off);
|
|
|
|
asan_write((T*)(p + off));
|
|
|
|
free_aaa(p);
|
|
|
|
}
|
|
|
|
|
2019-08-02 10:27:04 +08:00
|
|
|
static std::string GetLeftOOBMessage(int off) {
|
|
|
|
char str[100];
|
|
|
|
sprintf(str, "is located.*%d byte.*to the left", off);
|
|
|
|
return str;
|
|
|
|
}
|
|
|
|
|
|
|
|
static std::string GetRightOOBMessage(int off) {
|
|
|
|
char str[100];
|
2019-08-05 23:10:15 +08:00
|
|
|
#if !defined(_WIN32)
|
2019-08-02 10:27:04 +08:00
|
|
|
// FIXME: Fix PR42868 and remove SEGV match.
|
2019-08-05 23:04:20 +08:00
|
|
|
sprintf(str, "is located.*%d byte.*to the right|SEGV", off);
|
2019-08-05 23:10:15 +08:00
|
|
|
#else
|
|
|
|
// `|` doesn't work in googletest's regexes on Windows,
|
|
|
|
// see googletest/docs/advanced.md#regular-expression-syntax
|
|
|
|
// But it's not needed on Windows anyways.
|
|
|
|
sprintf(str, "is located.*%d byte.*to the right", off);
|
|
|
|
#endif
|
2019-08-02 10:27:04 +08:00
|
|
|
return str;
|
|
|
|
}
|
|
|
|
|
2013-01-22 14:39:20 +08:00
|
|
|
template<typename T>
|
|
|
|
void OOBTest() {
|
|
|
|
for (int size = sizeof(T); size < 20; size += 5) {
|
2019-08-02 10:27:04 +08:00
|
|
|
for (int i = -5; i < 0; i++)
|
|
|
|
EXPECT_DEATH(oob_test<T>(size, i), GetLeftOOBMessage(-i));
|
2013-01-22 14:39:20 +08:00
|
|
|
|
|
|
|
for (int i = 0; i < (int)(size - sizeof(T) + 1); i++)
|
|
|
|
oob_test<T>(size, i);
|
|
|
|
|
|
|
|
for (int i = size - sizeof(T) + 1; i <= (int)(size + 2 * sizeof(T)); i++) {
|
|
|
|
// we don't catch unaligned partially OOB accesses.
|
|
|
|
if (i % sizeof(T)) continue;
|
2019-08-02 10:27:04 +08:00
|
|
|
int off = i >= size ? (i - size) : 0;
|
|
|
|
EXPECT_DEATH(oob_test<T>(size, i), GetRightOOBMessage(off));
|
2013-01-22 14:39:20 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-08-02 10:27:04 +08:00
|
|
|
EXPECT_DEATH(oob_test<T>(kLargeMalloc, -1), GetLeftOOBMessage(1));
|
|
|
|
EXPECT_DEATH(oob_test<T>(kLargeMalloc, kLargeMalloc), GetRightOOBMessage(0));
|
2013-01-22 14:39:20 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
// TODO(glider): the following tests are EXTREMELY slow on Darwin:
|
|
|
|
// AddressSanitizer.OOB_char (125503 ms)
|
|
|
|
// AddressSanitizer.OOB_int (126890 ms)
|
|
|
|
// AddressSanitizer.OOBRightTest (315605 ms)
|
|
|
|
// AddressSanitizer.SimpleStackTest (366559 ms)
|
|
|
|
|
|
|
|
TEST(AddressSanitizer, OOB_char) {
|
|
|
|
OOBTest<U1>();
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST(AddressSanitizer, OOB_int) {
|
|
|
|
OOBTest<U4>();
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST(AddressSanitizer, OOBRightTest) {
|
2014-05-23 19:52:38 +08:00
|
|
|
size_t max_access_size = SANITIZER_WORDSIZE == 64 ? 8 : 4;
|
|
|
|
for (size_t access_size = 1; access_size <= max_access_size;
|
|
|
|
access_size *= 2) {
|
2013-01-22 14:39:20 +08:00
|
|
|
for (size_t alloc_size = 1; alloc_size <= 8; alloc_size++) {
|
|
|
|
for (size_t offset = 0; offset <= 8; offset += access_size) {
|
|
|
|
void *p = malloc(alloc_size);
|
|
|
|
// allocated: [p, p + alloc_size)
|
|
|
|
// accessed: [p + offset, p + offset + access_size)
|
|
|
|
uint8_t *addr = (uint8_t*)p + offset;
|
|
|
|
if (offset + access_size <= alloc_size) {
|
|
|
|
asan_write_sized_aligned(addr, access_size);
|
|
|
|
} else {
|
|
|
|
int outside_bytes = offset > alloc_size ? (offset - alloc_size) : 0;
|
|
|
|
EXPECT_DEATH(asan_write_sized_aligned(addr, access_size),
|
2019-08-02 10:27:04 +08:00
|
|
|
GetRightOOBMessage(outside_bytes));
|
2013-01-22 14:39:20 +08:00
|
|
|
}
|
|
|
|
free(p);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST(AddressSanitizer, LargeOOBRightTest) {
|
|
|
|
size_t large_power_of_two = 1 << 19;
|
|
|
|
for (size_t i = 16; i <= 256; i *= 2) {
|
|
|
|
size_t size = large_power_of_two - i;
|
|
|
|
char *p = Ident(new char[size]);
|
2019-08-02 10:27:04 +08:00
|
|
|
EXPECT_DEATH(p[size] = 0, GetRightOOBMessage(0));
|
2013-01-22 14:39:20 +08:00
|
|
|
delete [] p;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST(AddressSanitizer, DISABLED_DemoOOBLeftLow) {
|
|
|
|
oob_test<U1>(10, -1);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST(AddressSanitizer, DISABLED_DemoOOBLeftHigh) {
|
|
|
|
oob_test<U1>(kLargeMalloc, -1);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST(AddressSanitizer, DISABLED_DemoOOBRightLow) {
|
|
|
|
oob_test<U1>(10, 10);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST(AddressSanitizer, DISABLED_DemoOOBRightHigh) {
|
|
|
|
oob_test<U1>(kLargeMalloc, kLargeMalloc);
|
|
|
|
}
|