blob: bfbb9a89b7717e18fd79a0df3ce2a5566330878d [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: Sanjay Ghemawat
#include <stdlib.h> // for rand()
#include <vector>
#include <set>
#include <algorithm>
#include <utility>
#include "addressmap-inl.h"
#include "base/logging.h"
#include "base/commandlineflags.h"
DEFINE_int32(iters, 20, "Number of test iterations");
DEFINE_int32(N, 100000, "Number of elements to test per iteration");
using std::pair;
using std::make_pair;
using std::vector;
using std::set;
using std::random_shuffle;
struct UniformRandomNumberGenerator {
size_t Uniform(size_t max_size) {
if (max_size == 0)
return 0;
return rand() % max_size; // not a great random-number fn, but portable
}
};
static UniformRandomNumberGenerator rnd;
// pair of associated value and object size
typedef pair<int, size_t> ValueT;
struct PtrAndSize {
char* ptr;
size_t size;
PtrAndSize(char* p, size_t s) : ptr(p), size(s) {}
};
size_t SizeFunc(const ValueT& v) { return v.second; }
static void SetCheckCallback(const void* ptr, ValueT* val,
set<pair<const void*, int> >* check_set) {
check_set->insert(make_pair(ptr, val->first));
}
int main(int argc, char** argv) {
// Get a bunch of pointers
const int N = FLAGS_N;
static const int kMaxRealSize = 49;
// 100Mb to stress not finding previous object (AddressMap's cluster is 1Mb):
static const size_t kMaxSize = 100*1000*1000;
vector<PtrAndSize> ptrs_and_sizes;
for (int i = 0; i < N; ++i) {
size_t s = rnd.Uniform(kMaxRealSize);
ptrs_and_sizes.push_back(PtrAndSize(new char[s], s));
}
for (int x = 0; x < FLAGS_iters; ++x) {
RAW_LOG(INFO, "Iteration %d/%d...\n", x, FLAGS_iters);
// Permute pointers to get rid of allocation order issues
random_shuffle(ptrs_and_sizes.begin(), ptrs_and_sizes.end());
AddressMap<ValueT> map(malloc, free);
const ValueT* result;
const void* res_p;
// Insert a bunch of entries
for (int i = 0; i < N; ++i) {
char* p = ptrs_and_sizes[i].ptr;
CHECK(!map.Find(p));
int offs = rnd.Uniform(ptrs_and_sizes[i].size);
CHECK(!map.FindInside(&SizeFunc, kMaxSize, p + offs, &res_p));
map.Insert(p, make_pair(i, ptrs_and_sizes[i].size));
CHECK(result = map.Find(p));
CHECK_EQ(result->first, i);
CHECK(result = map.FindInside(&SizeFunc, kMaxRealSize, p + offs, &res_p));
CHECK_EQ(res_p, p);
CHECK_EQ(result->first, i);
map.Insert(p, make_pair(i + N, ptrs_and_sizes[i].size));
CHECK(result = map.Find(p));
CHECK_EQ(result->first, i + N);
}
// Delete the even entries
for (int i = 0; i < N; i += 2) {
void* p = ptrs_and_sizes[i].ptr;
ValueT removed;
CHECK(map.FindAndRemove(p, &removed));
CHECK_EQ(removed.first, i + N);
}
// Lookup the odd entries and adjust them
for (int i = 1; i < N; i += 2) {
char* p = ptrs_and_sizes[i].ptr;
CHECK(result = map.Find(p));
CHECK_EQ(result->first, i + N);
int offs = rnd.Uniform(ptrs_and_sizes[i].size);
CHECK(result = map.FindInside(&SizeFunc, kMaxRealSize, p + offs, &res_p));
CHECK_EQ(res_p, p);
CHECK_EQ(result->first, i + N);
map.Insert(p, make_pair(i + 2*N, ptrs_and_sizes[i].size));
CHECK(result = map.Find(p));
CHECK_EQ(result->first, i + 2*N);
}
// Insert even entries back
for (int i = 0; i < N; i += 2) {
char* p = ptrs_and_sizes[i].ptr;
int offs = rnd.Uniform(ptrs_and_sizes[i].size);
CHECK(!map.FindInside(&SizeFunc, kMaxSize, p + offs, &res_p));
map.Insert(p, make_pair(i + 2*N, ptrs_and_sizes[i].size));
CHECK(result = map.Find(p));
CHECK_EQ(result->first, i + 2*N);
CHECK(result = map.FindInside(&SizeFunc, kMaxRealSize, p + offs, &res_p));
CHECK_EQ(res_p, p);
CHECK_EQ(result->first, i + 2*N);
}
// Check all entries
set<pair<const void*, int> > check_set;
map.Iterate(SetCheckCallback, &check_set);
CHECK_EQ(check_set.size(), N);
for (int i = 0; i < N; ++i) {
void* p = ptrs_and_sizes[i].ptr;
check_set.erase(make_pair(p, i + 2*N));
CHECK(result = map.Find(p));
CHECK_EQ(result->first, i + 2*N);
}
CHECK_EQ(check_set.size(), 0);
}
for (int i = 0; i < N; ++i) {
delete[] ptrs_and_sizes[i].ptr;
}
printf("PASS\n");
return 0;
}