blob: ad3482ed26fb1dee39c08f7514c4fb9cf3ebd1f9 [file] [log] [blame]
// Copyright (c) 2005, Google Inc.
// All rights reserved.
//
// Redistribution and use in source and binary forms, with or without
// modification, are permitted provided that the following conditions are
// met:
//
// * Redistributions of source code must retain the above copyright
// notice, this list of conditions and the following disclaimer.
// * Redistributions in binary form must reproduce the above
// copyright notice, this list of conditions and the following disclaimer
// in the documentation and/or other materials provided with the
// distribution.
// * Neither the name of Google Inc. nor the names of its
// contributors may be used to endorse or promote products derived from
// this software without specific prior written permission.
//
// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
// ---
// Author: Michael Chastain
//
// This is a unit test for large allocations in malloc and friends.
// "Large" means "so large that they overflow the address space".
// For 32 bits, this means allocations near 2^32 bytes and 2^31 bytes.
// For 64 bits, this means allocations near 2^64 bytes and 2^63 bytes.
#include <stddef.h> // for size_t, NULL
#include <stdlib.h> // for malloc, free, realloc
#include <stdio.h>
#include <set> // for set, etc
#include "base/logging.h" // for operator<<, CHECK, etc
using std::set;
// Alloc a size that should always fail.
void TryAllocExpectFail(size_t size) {
void* p1 = malloc(size);
CHECK(p1 == NULL);
void* p2 = malloc(1);
CHECK(p2 != NULL);
void* p3 = realloc(p2, size);
CHECK(p3 == NULL);
free(p2);
}
// Alloc a size that might work and might fail.
// If it does work, touch some pages.
void TryAllocMightFail(size_t size) {
unsigned char* p = static_cast<unsigned char*>(malloc(size));
if ( p != NULL ) {
unsigned char volatile* vp = p; // prevent optimizations
static const size_t kPoints = 1024;
for ( size_t i = 0; i < kPoints; ++i ) {
vp[i * (size / kPoints)] = static_cast<unsigned char>(i);
}
for ( size_t i = 0; i < kPoints; ++i ) {
CHECK(vp[i * (size / kPoints)] == static_cast<unsigned char>(i));
}
vp[size-1] = 'M';
CHECK(vp[size-1] == 'M');
}
free(p);
}
int main (int argc, char** argv) {
// Allocate some 0-byte objects. They better be unique.
// 0 bytes is not large but it exercises some paths related to
// large-allocation code.
{
static const int kZeroTimes = 1024;
printf("Test malloc(0) x %d\n", kZeroTimes);
set<char*> p_set;
for ( int i = 0; i < kZeroTimes; ++i ) {
char* p = new char;
CHECK(p != NULL);
CHECK(p_set.find(p) == p_set.end());
p_set.insert(p_set.end(), p);
}
// Just leak the memory.
}
// Grab some memory so that some later allocations are guaranteed to fail.
printf("Test small malloc\n");
void* p_small = malloc(4*1048576);
CHECK(p_small != NULL);
// Test sizes up near the maximum size_t.
// These allocations test the wrap-around code.
printf("Test malloc(0 - N)\n");
const size_t zero = 0;
static const size_t kMinusNTimes = 16384;
for ( size_t i = 1; i < kMinusNTimes; ++i ) {
TryAllocExpectFail(zero - i);
}
// Test sizes a bit smaller.
// The small malloc above guarantees that all these return NULL.
printf("Test malloc(0 - 1048576 - N)\n");
static const size_t kMinusMBMinusNTimes = 16384;
for ( size_t i = 0; i < kMinusMBMinusNTimes; ++i) {
TryAllocExpectFail(zero - 1048576 - i);
}
// Test sizes at half of size_t.
// These might or might not fail to allocate.
printf("Test malloc(max/2 +- N)\n");
static const size_t kHalfPlusMinusTimes = 64;
const size_t half = (zero - 2) / 2 + 1;
for ( size_t i = 0; i < kHalfPlusMinusTimes; ++i) {
TryAllocMightFail(half - i);
TryAllocMightFail(half + i);
}
printf("PASS\n");
return 0;
}