22 const std::shared_ptr<Allocator>& allocator,
23 const std::shared_ptr<const IdleWaiter>& idle_waiter) {
24 return std::shared_ptr<HostBuffer>(
new HostBuffer(allocator, idle_waiter));
27 HostBuffer::HostBuffer(
const std::shared_ptr<Allocator>& allocator,
28 const std::shared_ptr<const IdleWaiter>& idle_waiter)
29 : allocator_(allocator), idle_waiter_(idle_waiter) {
34 std::shared_ptr<DeviceBuffer> device_buffer = allocator->CreateBuffer(desc);
35 FML_CHECK(device_buffer) <<
"Failed to allocate device buffer.";
36 device_buffers_[i].push_back(device_buffer);
44 idle_waiter_->WaitIdle();
51 auto [range, device_buffer, raw_device_buffer] =
52 EmplaceInternal(buffer, length, align);
54 return BufferView(std::move(device_buffer), range);
55 }
else if (raw_device_buffer) {
63 auto [range, device_buffer, raw_device_buffer] =
64 EmplaceInternal(buffer, length);
66 return BufferView(std::move(device_buffer), range);
67 }
else if (raw_device_buffer) {
68 return BufferView(raw_device_buffer, range);
77 auto [range, device_buffer, raw_device_buffer] =
78 EmplaceInternal(length, align, cb);
80 return BufferView(std::move(device_buffer), range);
81 }
else if (raw_device_buffer) {
91 .current_buffer = current_buffer_,
92 .total_buffer_count = device_buffers_[frame_index_].size(),
96 bool HostBuffer::MaybeCreateNewBuffer() {
98 if (current_buffer_ >= device_buffers_[frame_index_].size()) {
102 std::shared_ptr<DeviceBuffer> buffer = allocator_->CreateBuffer(desc);
107 device_buffers_[frame_index_].push_back(std::move(buffer));
113 std::tuple<Range, std::shared_ptr<DeviceBuffer>, DeviceBuffer*>
114 HostBuffer::EmplaceInternal(
size_t length,
116 const EmplaceProc& cb) {
124 DeviceBufferDescriptor desc;
127 std::shared_ptr<DeviceBuffer> device_buffer =
128 allocator_->CreateBuffer(desc);
129 if (!device_buffer) {
133 cb(device_buffer->OnGetContents());
134 device_buffer->Flush(Range{0, length});
136 return std::make_tuple(Range{0, length}, std::move(device_buffer),
nullptr);
140 if (align > 0 && offset_ % align) {
141 padding = align - (offset_ % align);
144 if (!MaybeCreateNewBuffer()) {
151 const std::shared_ptr<DeviceBuffer>& current_buffer = GetCurrentBuffer();
152 auto contents = current_buffer->OnGetContents();
153 cb(contents + offset_);
154 Range output_range(offset_, length);
155 current_buffer->Flush(output_range);
158 return std::make_tuple(output_range,
nullptr, current_buffer.get());
161 std::tuple<Range, std::shared_ptr<DeviceBuffer>, DeviceBuffer*>
162 HostBuffer::EmplaceInternal(
const void* buffer,
size_t length) {
166 DeviceBufferDescriptor desc;
169 std::shared_ptr<DeviceBuffer> device_buffer =
170 allocator_->CreateBuffer(desc);
171 if (!device_buffer) {
175 if (!device_buffer->CopyHostBuffer(
static_cast<const uint8_t*
>(buffer),
180 return std::make_tuple(Range{0, length}, std::move(device_buffer),
nullptr);
183 auto old_length = GetLength();
185 if (!MaybeCreateNewBuffer()) {
189 old_length = GetLength();
191 const std::shared_ptr<DeviceBuffer>& current_buffer = GetCurrentBuffer();
192 auto contents = current_buffer->OnGetContents();
194 ::memmove(contents + old_length, buffer, length);
195 current_buffer->Flush(Range{old_length, length});
198 return std::make_tuple(Range{old_length, length},
nullptr,
199 current_buffer.get());
202 std::tuple<Range, std::shared_ptr<DeviceBuffer>, DeviceBuffer*>
203 HostBuffer::EmplaceInternal(
const void* buffer,
size_t length,
size_t align) {
204 if (align == 0 || (GetLength() % align) == 0) {
205 return EmplaceInternal(buffer, length);
209 auto padding = align - (GetLength() % align);
212 }
else if (!MaybeCreateNewBuffer()) {
217 return EmplaceInternal(buffer, length);
220 const std::shared_ptr<DeviceBuffer>& HostBuffer::GetCurrentBuffer()
const {
221 return device_buffers_[frame_index_][current_buffer_];
227 while (device_buffers_[frame_index_].size() > current_buffer_ + 1) {
228 device_buffers_[frame_index_].pop_back();
232 current_buffer_ = 0u;
BufferView Emplace(const BufferType &buffer, size_t alignment=0)
Emplace non-uniform data (like contiguous vertices) onto the host buffer.
TestStateQuery GetStateForTest()
Retrieve internal buffer state for test expectations.
std::function< void(uint8_t *buffer)> EmplaceProc
void Reset()
Resets the contents of the HostBuffer to nothing so it can be reused.
static std::shared_ptr< HostBuffer > Create(const std::shared_ptr< Allocator > &allocator, const std::shared_ptr< const IdleWaiter > &idle_waiter)
Vector2 padding
The halo padding in source space.
static constexpr const size_t kHostBufferArenaSize
Approximately the same size as the max frames in flight.
constexpr size_t kAllocatorBlockSize
Test only internal state.