mirror of
https://github.com/ClickHouse/ClickHouse.git
synced 2024-12-14 02:12:21 +00:00
58 lines
1.7 KiB
C++
58 lines
1.7 KiB
C++
#pragma once
|
|
|
|
#include <Common/Arena.h>
|
|
#include <common/StringRef.h>
|
|
#include <IO/WriteBuffer.h>
|
|
|
|
|
|
namespace DB
|
|
{
|
|
|
|
/** Writes data contiguously into Arena.
|
|
* As it will be located in contiguous memory segment, it can be read back with ReadBufferFromMemory.
|
|
*
|
|
* While using this object, no other allocations in arena are possible.
|
|
*/
|
|
class WriteBufferFromArena : public WriteBuffer
|
|
{
|
|
private:
|
|
Arena & arena;
|
|
const char *& begin;
|
|
|
|
void nextImpl() override
|
|
{
|
|
/// Allocate more memory. At least same size as used before (this gives 2x growth ratio),
|
|
/// and at most grab all remaining size in current chunk of arena.
|
|
size_t continuation_size = std::max(count(), arena.remainingSpaceInCurrentChunk());
|
|
|
|
/// allocContinue method will possibly move memory region to new place and modify "begin" pointer.
|
|
|
|
char * continuation = arena.allocContinue(continuation_size, begin);
|
|
char * end = continuation + continuation_size;
|
|
|
|
/// internal buffer points to whole memory segment and working buffer - to free space for writing.
|
|
internalBuffer() = Buffer(const_cast<char *>(begin), end);
|
|
buffer() = Buffer(continuation, end);
|
|
}
|
|
|
|
public:
|
|
/// begin_ - start of previously used contiguous memory segment or nullptr (see Arena::allocContinue method).
|
|
WriteBufferFromArena(Arena & arena_, const char *& begin_)
|
|
: WriteBuffer(nullptr, 0), arena(arena_), begin(begin_)
|
|
{
|
|
nextImpl();
|
|
pos = working_buffer.begin();
|
|
}
|
|
|
|
StringRef finish()
|
|
{
|
|
/// Return over-allocated memory back into arena.
|
|
arena.rollback(buffer().end() - position());
|
|
/// Reference to written data.
|
|
return { position() - count(), count() };
|
|
}
|
|
};
|
|
|
|
}
|
|
|