2011-03-18 22:37:00 +00:00
|
|
|
// Copyright (c) 2011 The LevelDB Authors. All rights reserved.
|
|
|
|
// Use of this source code is governed by a BSD-style license that can be
|
|
|
|
// found in the LICENSE file. See the AUTHORS file for names of contributors.
|
|
|
|
|
|
|
|
#ifndef STORAGE_LEVELDB_UTIL_ARENA_H_
|
|
|
|
#define STORAGE_LEVELDB_UTIL_ARENA_H_
|
|
|
|
|
2019-03-11 13:04:53 -07:00
|
|
|
#include <atomic>
|
|
|
|
#include <cassert>
|
|
|
|
#include <cstddef>
|
|
|
|
#include <cstdint>
|
2011-03-18 22:37:00 +00:00
|
|
|
#include <vector>
|
|
|
|
|
|
|
|
namespace leveldb {
|
|
|
|
|
|
|
|
class Arena {
|
|
|
|
public:
|
|
|
|
Arena();
|
2019-05-03 09:31:18 -07:00
|
|
|
|
|
|
|
Arena(const Arena&) = delete;
|
|
|
|
Arena& operator=(const Arena&) = delete;
|
|
|
|
|
2011-03-18 22:37:00 +00:00
|
|
|
~Arena();
|
|
|
|
|
|
|
|
// Return a pointer to a newly allocated memory block of "bytes" bytes.
|
|
|
|
char* Allocate(size_t bytes);
|
|
|
|
|
2019-03-11 13:04:53 -07:00
|
|
|
// Allocate memory with the normal alignment guarantees provided by malloc.
|
2011-03-18 22:37:00 +00:00
|
|
|
char* AllocateAligned(size_t bytes);
|
|
|
|
|
|
|
|
// Returns an estimate of the total memory usage of data allocated
|
2015-11-11 03:54:18 -08:00
|
|
|
// by the arena.
|
2011-03-18 22:37:00 +00:00
|
|
|
size_t MemoryUsage() const {
|
2019-03-11 13:04:53 -07:00
|
|
|
return memory_usage_.load(std::memory_order_relaxed);
|
2011-03-18 22:37:00 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
char* AllocateFallback(size_t bytes);
|
|
|
|
char* AllocateNewBlock(size_t block_bytes);
|
|
|
|
|
|
|
|
// Allocation state
|
|
|
|
char* alloc_ptr_;
|
|
|
|
size_t alloc_bytes_remaining_;
|
|
|
|
|
|
|
|
// Array of new[] allocated memory blocks
|
|
|
|
std::vector<char*> blocks_;
|
|
|
|
|
2015-11-11 03:54:18 -08:00
|
|
|
// Total memory usage of the arena.
|
2019-03-11 13:04:53 -07:00
|
|
|
//
|
|
|
|
// TODO(costan): This member is accessed via atomics, but the others are
|
|
|
|
// accessed without any locking. Is this OK?
|
|
|
|
std::atomic<size_t> memory_usage_;
|
2011-03-18 22:37:00 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
inline char* Arena::Allocate(size_t bytes) {
|
|
|
|
// The semantics of what to return are a bit messy if we allow
|
|
|
|
// 0-byte allocations, so we disallow them here (we don't need
|
|
|
|
// them for our internal use).
|
|
|
|
assert(bytes > 0);
|
|
|
|
if (bytes <= alloc_bytes_remaining_) {
|
|
|
|
char* result = alloc_ptr_;
|
|
|
|
alloc_ptr_ += bytes;
|
|
|
|
alloc_bytes_remaining_ -= bytes;
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
return AllocateFallback(bytes);
|
|
|
|
}
|
|
|
|
|
2011-10-31 17:22:06 +00:00
|
|
|
} // namespace leveldb
|
2011-03-18 22:37:00 +00:00
|
|
|
|
|
|
|
#endif // STORAGE_LEVELDB_UTIL_ARENA_H_
|