// Copyright (c) 2012 The Chromium Authors. All rights reserved. // Use of this source code is governed by a BSD-style license that can be // found in the LICENSE file. #include #include #include "base/logging.h" #include "base/process/process_metrics.h" #include "build/build_config.h" #include "testing/gtest/include/gtest/gtest.h" #if defined(USE_TCMALLOC) extern "C" { void* tc_malloc(size_t size); void tc_free(void*); } namespace { using std::min; #ifdef NDEBUG void* TCMallocDoMallocForTest(size_t size) { return tc_malloc(size); } void TCMallocDoFreeForTest(void* ptr) { tc_free(ptr); } #endif // Fill a buffer of the specified size with a predetermined pattern static void Fill(unsigned char* buffer, int n) { for (int i = 0; i < n; i++) { buffer[i] = (i & 0xff); } } // Check that the specified buffer has the predetermined pattern // generated by Fill() static bool Valid(unsigned char* buffer, int n) { for (int i = 0; i < n; i++) { if (buffer[i] != (i & 0xff)) { return false; } } return true; } // Return the next interesting size/delta to check. Returns -1 if no more. static int NextSize(int size) { if (size < 100) return size + 1; if (size < 100000) { // Find next power of two int power = 1; while (power < size) power <<= 1; // Yield (power-1, power, power+1) if (size < power - 1) return power - 1; if (size == power - 1) return power; CHECK_EQ(size, power); return power + 1; } else { return -1; } } static void TestCalloc(size_t n, size_t s, bool ok) { char* p = reinterpret_cast(calloc(n, s)); if (!ok) { EXPECT_EQ(NULL, p) << "calloc(n, s) should not succeed"; } else { EXPECT_NE(reinterpret_cast(NULL), p) << "calloc(n, s) should succeed"; for (size_t i = 0; i < n * s; i++) { EXPECT_EQ('\0', p[i]); } free(p); } } } // namespace TEST(TCMallocTest, Malloc) { // Try allocating data with a bunch of alignments and sizes for (int size = 1; size < 1048576; size *= 2) { unsigned char* ptr = reinterpret_cast(malloc(size)); // Should be 2 byte aligned EXPECT_EQ(0u, reinterpret_cast(ptr) & 1); Fill(ptr, size); EXPECT_TRUE(Valid(ptr, size)); free(ptr); } } TEST(TCMallocTest, Calloc) { TestCalloc(0, 0, true); TestCalloc(0, 1, true); TestCalloc(1, 1, true); TestCalloc(1 << 10, 0, true); TestCalloc(1 << 20, 0, true); TestCalloc(0, 1 << 10, true); TestCalloc(0, 1 << 20, true); TestCalloc(1 << 20, 2, true); TestCalloc(2, 1 << 20, true); TestCalloc(1000, 1000, true); } #ifdef NDEBUG // This makes sure that reallocing a small number of bytes in either // direction doesn't cause us to allocate new memory. Tcmalloc in debug mode // does not follow this. TEST(TCMallocTest, ReallocSmallDelta) { int start_sizes[] = {100, 1000, 10000, 100000}; int deltas[] = {1, -2, 4, -8, 16, -32, 64, -128}; for (unsigned s = 0; s < sizeof(start_sizes) / sizeof(*start_sizes); ++s) { void* p = malloc(start_sizes[s]); ASSERT_TRUE(p); // The larger the start-size, the larger the non-reallocing delta. for (unsigned d = 0; d < s * 2; ++d) { void* new_p = realloc(p, start_sizes[s] + deltas[d]); ASSERT_EQ(p, new_p); // realloc should not allocate new memory } // Test again, but this time reallocing smaller first. for (unsigned d = 0; d < s * 2; ++d) { void* new_p = realloc(p, start_sizes[s] - deltas[d]); ASSERT_EQ(p, new_p); // realloc should not allocate new memory } free(p); } } #endif TEST(TCMallocTest, Realloc) { for (int src_size = 0; src_size >= 0; src_size = NextSize(src_size)) { for (int dst_size = 0; dst_size >= 0; dst_size = NextSize(dst_size)) { unsigned char* src = reinterpret_cast(malloc(src_size)); Fill(src, src_size); unsigned char* dst = reinterpret_cast(realloc(src, dst_size)); EXPECT_TRUE(Valid(dst, min(src_size, dst_size))); Fill(dst, dst_size); EXPECT_TRUE(Valid(dst, dst_size)); if (dst != NULL) free(dst); } } // Now make sure realloc works correctly even when we overflow the // packed cache, so some entries are evicted from the cache. // The cache has 2^12 entries, keyed by page number. const int kNumEntries = 1 << 14; int** p = reinterpret_cast(malloc(sizeof(*p) * kNumEntries)); int sum = 0; for (int i = 0; i < kNumEntries; i++) { // no page size is likely to be bigger than 8192? p[i] = reinterpret_cast(malloc(8192)); p[i][1000] = i; // use memory deep in the heart of p } for (int i = 0; i < kNumEntries; i++) { p[i] = reinterpret_cast(realloc(p[i], 9000)); } for (int i = 0; i < kNumEntries; i++) { sum += p[i][1000]; free(p[i]); } EXPECT_EQ(kNumEntries / 2 * (kNumEntries - 1), sum); // assume kNE is even free(p); } #ifdef NDEBUG TEST(TCMallocFreeTest, BadPointerInFirstPageOfTheLargeObject) { const size_t kPageSize = base::GetPageSize(); char* p = reinterpret_cast(TCMallocDoMallocForTest(10 * kPageSize + 1)); for (unsigned offset = 1; offset < kPageSize; offset <<= 1) { ASSERT_DEATH(TCMallocDoFreeForTest(p + offset), "Pointer is not pointing to the start of a span"); } TCMallocDoFreeForTest(p); } // TODO(ssid): Fix flakiness and enable the test, crbug.com/571549. TEST(TCMallocFreeTest, DISABLED_BadPageAlignedPointerInsideLargeObject) { const size_t kPageSize = base::GetPageSize(); const size_t kMaxSize = 10 * kPageSize; char* p = reinterpret_cast(TCMallocDoMallocForTest(kMaxSize + 1)); for (unsigned offset = kPageSize; offset < kMaxSize; offset += kPageSize) { // Only the first and last page of a span are in heap map. So for others // tcmalloc will give a general error of invalid pointer. ASSERT_DEATH(TCMallocDoFreeForTest(p + offset), ""); } ASSERT_DEATH(TCMallocDoFreeForTest(p + kMaxSize), "Pointer is not pointing to the start of a span"); TCMallocDoFreeForTest(p); } TEST(TCMallocFreeTest, DoubleFreeLargeObject) { const size_t kMaxSize = 10 * base::GetPageSize(); char* p = reinterpret_cast(TCMallocDoMallocForTest(kMaxSize + 1)); ASSERT_DEATH(TCMallocDoFreeForTest(p); TCMallocDoFreeForTest(p), "Object was not in-use"); } TEST(TCMallocFreeTest, DoubleFreeSmallObject) { const size_t kPageSize = base::GetPageSize(); for (size_t size = 1; size <= kPageSize; size <<= 1) { char* p = reinterpret_cast(TCMallocDoMallocForTest(size)); ASSERT_DEATH(TCMallocDoFreeForTest(p); TCMallocDoFreeForTest(p), "Circular loop in list detected"); } } #endif // NDEBUG #endif