21 const std::string& name,
22 const std::shared_ptr<Nodes::GpuSync::GeometryWriterNode>& node,
23 const std::shared_ptr<VKBuffer>& vertex_buffer)
26 error<std::invalid_argument>(
29 std::source_location::current(),
30 "Cannot bind null geometry node '{}'", name);
34 error<std::invalid_argument>(
37 std::source_location::current(),
38 "Cannot bind geometry node '{}' to null vertex buffer", name);
41 size_t vertex_data_size = node->get_vertex_buffer_size_bytes();
43 if (vertex_buffer->get_size_bytes() < vertex_data_size) {
45 "Vertex buffer for '{}' may be too small: {} bytes required, {} available. "
46 "Will upload partial data.",
47 name, vertex_data_size, vertex_buffer->get_size_bytes());
50 std::shared_ptr<VKBuffer> staging =
nullptr;
51 if (!vertex_buffer->is_host_visible()) {
52 size_t staging_size = std::max<size_t>(vertex_buffer->get_size_bytes(), vertex_data_size);
56 "Created staging buffer for device-local geometry '{}' ({} bytes)",
60 "No staging needed for host-visible geometry '{}'", name);
63 std::shared_ptr<VKBuffer> index_buf =
nullptr;
64 std::shared_ptr<VKBuffer> index_staging_buf =
nullptr;
66 if (node->has_indices()) {
67 const size_t index_data_size = node->get_index_count() *
sizeof(uint32_t);
68 index_buf = std::make_shared<VKBuffer>(
77 "Created index staging buffer for device-local geometry '{}' ({} bytes)",
78 name, index_data_size);
83 .gpu_vertex_buffer = vertex_buffer,
84 .staging_buffer = staging,
85 .gpu_index_buffer = index_buf,
86 .index_staging_buffer = index_staging_buf
90 "Bound geometry node '{}' ({} vertices, {} bytes, stride: {}, indices: {})",
91 name, node->get_vertex_count(), vertex_data_size, node->get_vertex_stride(),
92 node->get_index_count());
99 "Attempted to unbind non-existent geometry node '{}'", name);
102 "Unbound geometry node '{}'", name);
113 std::vector<std::string> names;
116 names.push_back(name);
126std::optional<GeometryBindingsProcessor::GeometryBinding>
137 std::shared_ptr<Core::VKImage>
image, std::string binding)
141 .binding = std::move(binding),
150 if (
auto vk = std::dynamic_pointer_cast<VKBuffer>(buffer)) {
151 if (
auto rp = vk->get_render_processor()) {
164 auto vk_buffer = std::dynamic_pointer_cast<VKBuffer>(buffer);
167 "GeometryBindingsProcessor requires VKBuffer, got different buffer type");
172 if (!binding.node->needs_gpu_update()) {
174 "Geometry '{}' unchanged, skipping upload", name);
178 auto vertices = binding.node->get_vertex_data();
180 if (vertices.empty()) {
181 if (binding.gpu_vertex_buffer->is_host_visible()) {
182 binding.gpu_vertex_buffer->clear();
185 if (binding.node->get_vertex_layout()) {
186 binding.gpu_vertex_buffer->set_vertex_layout(
187 binding.node->get_vertex_layout().value());
191 "Geometry '{}' cleared", name);
197 vertices.size_bytes(),
198 binding.gpu_vertex_buffer,
199 binding.staging_buffer);
201 if (binding.node->get_vertex_layout()) {
202 binding.gpu_vertex_buffer->set_vertex_layout(
203 binding.node->get_vertex_layout().value());
206 "Set vertex layout for '{}' ({} vertices, {} attributes)",
208 binding.node->get_vertex_count(),
209 binding.node->get_vertex_layout()->attributes.size());
212 "Geometry node '{}' has no vertex layout. "
213 "RenderProcessor may fail without layout info.",
218 binding.node->clear_gpu_update_flag();
221 bool attached_is_target =
false;
222 for (
const auto& [name, binding] :
m_bindings) {
223 if (binding.gpu_vertex_buffer == vk_buffer) {
224 attached_is_target =
true;
229 if (!attached_is_target && !
m_bindings.empty()) {
230 auto& first_binding =
m_bindings.begin()->second;
231 if (first_binding.node->needs_gpu_update()) {
232 auto vertices = first_binding.node->get_vertex_data();
234 if (!vertices.empty()) {
237 vertices.size_bytes(),
239 first_binding.staging_buffer);
241 if (first_binding.node->get_vertex_layout()) {
242 vk_buffer->set_vertex_layout(
243 first_binding.node->get_vertex_layout().value());
247 first_binding.node->clear_gpu_update_flag();
253 const std::string& name,
256 if (!binding.
node->has_indices()) {
260 const auto indices = binding.
node->get_index_data();
261 const size_t required = indices.size_bytes();
278 "Lazily created index buffer for '{}' ({} bytes)", name, required);
293 "Uploaded {} indices ({} bytes) for '{}'",
294 binding.
node->get_index_count(), required, name);
#define MF_RT_WARN(comp, ctx,...)
#define MF_RT_ERROR(comp, ctx,...)
#define MF_TRACE(comp, ctx,...)
#define MF_RT_TRACE(comp, ctx,...)
#define MF_WARN(comp, ctx,...)
#define MF_DEBUG(comp, ctx,...)
ProcessingToken m_processing_token
std::optional< GeometryBinding > get_binding(const std::string &name) const
Get a specific binding.
void set_texture(std::shared_ptr< Core::VKImage > image, std::string binding)
Supply a texture to bind on the next graphics tick.
std::unordered_map< std::string, GeometryBinding > m_bindings
void upload_index_data(const std::string &name, GeometryBinding &binding)
Upload index data for one binding, creating or growing the GPU index buffer as needed.
std::vector< std::string > get_binding_names() const
Get all binding names.
size_t get_binding_count() const
Get number of active bindings.
std::atomic_flag m_texture_dirty
void processing_function(const std::shared_ptr< Buffer > &buffer) override
BufferProcessor interface - uploads all bound geometries.
void bind_geometry_node(const std::string &name, const std::shared_ptr< Nodes::GpuSync::GeometryWriterNode > &node, const std::shared_ptr< VKBuffer > &vertex_buffer)
Bind a geometry node to a GPU vertex buffer.
bool has_binding(const std::string &name) const
Check if a binding exists.
void unbind_geometry_node(const std::string &name)
Remove a geometry binding.
std::optional< PendingTexture > m_pending_texture
GeometryBindingsProcessor()
void initialize_buffer_service()
void ensure_initialized(const std::shared_ptr< VKBuffer > &buffer)
void upload_resizing(const void *data, size_t size, const std::shared_ptr< VKBuffer > &target, const std::shared_ptr< VKBuffer > &staging, float growth_factor)
Upload size bytes to target, growing both buffers first if needed.
@ GRAPHICS_BACKEND
Standard graphics processing backend configuration.
std::shared_ptr< VKBuffer > create_staging_buffer(size_t size)
Create staging buffer for transfers.
@ BufferProcessing
Buffer processing (Buffers::BufferManager, processing chains)
@ Buffers
Buffers, Managers, processors and processing chains.
@ UNKNOWN
Unknown or undefined modality.
std::shared_ptr< Nodes::GpuSync::GeometryWriterNode > node
std::shared_ptr< VKBuffer > gpu_vertex_buffer
std::shared_ptr< VKBuffer > gpu_index_buffer
std::shared_ptr< VKBuffer > index_staging_buffer
Holds GPU resources for one GeometryWriterNode binding.
std::shared_ptr< Core::VKImage > image