blob: 811c9dd8200a3e5d7918fb57c415a323740a75ae [file] [log] [blame]
// Protocol Buffers - Google's data interchange format
// Copyright 2008 Google Inc. All rights reserved.
//
// Use of this source code is governed by a BSD-style
// license that can be found in the LICENSE file or at
// https://developers.google.com/open-source/licenses/bsd
#ifndef GOOGLE_PROTOBUF_ARENA_TEST_UTIL_H__
#define GOOGLE_PROTOBUF_ARENA_TEST_UTIL_H__
#include <cstddef>
#include "absl/container/flat_hash_set.h"
#include "absl/log/absl_check.h"
#include "google/protobuf/arena.h"
#include "google/protobuf/io/coded_stream.h"
#include "google/protobuf/io/zero_copy_stream_impl_lite.h"
// Must be included last.
#include "google/protobuf/port_def.inc"
namespace google {
namespace protobuf {
namespace internal {
struct ArenaTestPeer {
static void ReturnArrayMemory(Arena* arena, void* p, size_t size) {
arena->ReturnArrayMemory(p, size);
}
static auto PeekCleanupListForTesting(Arena* arena) {
return arena->PeekCleanupListForTesting();
}
template <typename T, typename... U>
static constexpr auto GetConstructType() {
return Arena::GetConstructType<T, U...>();
}
using ConstructType = Arena::ConstructType;
};
struct CleanupGrowthInfo {
size_t space_used;
absl::flat_hash_set<void*> cleanups;
};
template <typename Func>
CleanupGrowthInfo CleanupGrowth(Arena& arena, Func f) {
auto old_space_used = arena.SpaceUsed();
auto old_cleanups = ArenaTestPeer::PeekCleanupListForTesting(&arena);
f();
auto new_space_used = arena.SpaceUsed();
auto new_cleanups = ArenaTestPeer::PeekCleanupListForTesting(&arena);
CleanupGrowthInfo res;
res.space_used = new_space_used - old_space_used;
res.cleanups.insert(new_cleanups.begin(), new_cleanups.end());
for (auto p : old_cleanups) res.cleanups.erase(p);
return res;
}
class NoHeapChecker {
public:
NoHeapChecker() { capture_alloc.Hook(); }
~NoHeapChecker();
private:
class NewDeleteCapture {
public:
// TODO: Implement this for opensource protobuf.
void Hook() {}
void Unhook() {}
int alloc_count() { return 0; }
int free_count() { return 0; }
} capture_alloc;
};
// Owns the internal T only if it's not owned by an arena.
// T needs to be arena constructible and destructor skippable.
template <typename T>
class ArenaHolder {
public:
explicit ArenaHolder(Arena* arena)
: field_(Arena::Create<T>(arena)), owned_by_arena_(arena != nullptr) {
ABSL_DCHECK(google::protobuf::Arena::is_arena_constructable<T>::value);
ABSL_DCHECK(google::protobuf::Arena::is_destructor_skippable<T>::value);
}
~ArenaHolder() {
if (!owned_by_arena_) {
delete field_;
}
}
T* get() { return field_; }
T* operator->() { return field_; }
T& operator*() { return *field_; }
private:
T* field_;
bool owned_by_arena_;
};
} // namespace internal
} // namespace protobuf
} // namespace google
#include "google/protobuf/port_undef.inc"
#endif // GOOGLE_PROTOBUF_ARENA_TEST_UTIL_H__