27{
28 buffer_service->initialize_buffer = [this](const std::shared_ptr<void>& vk_buf) -> void {
29 auto buffer = std::static_pointer_cast<Buffers::VKBuffer>(vk_buf);
31 };
32
33 buffer_service->destroy_buffer = [this](const std::shared_ptr<void>& vk_buf) {
34 auto buffer = std::static_pointer_cast<Buffers::VKBuffer>(vk_buf);
36 };
37
38 buffer_service->get_buffer_device_address = [this](const std::shared_ptr<void>& vk_buf) -> uint64_t {
39 auto buffer = std::static_pointer_cast<Buffers::VKBuffer>(vk_buf);
41 return static_cast<uint64_t>(address);
42 };
43
44 buffer_service->execute_immediate = [this](const std::function<void(void*)>& recorder) {
46 recorder(
static_cast<void*
>(
cmd));
47 });
48 };
49
50 buffer_service->record_deferred = [this](const std::function<void(void*)>& recorder) {
52 recorder(
static_cast<void*
>(
cmd));
53 });
54 };
55
56 buffer_service->flush_range = [this](void* memory, size_t offset, size_t size) {
57 vk::DeviceMemory mem(reinterpret_cast<VkDeviceMemory>(memory));
58 vk::MappedMemoryRange
range { mem, offset, size == 0 ? VK_WHOLE_SIZE : size };
59 if (
auto result =
m_context.
get_device().flushMappedMemoryRanges(1, &range); result != vk::Result::eSuccess) {
61 "Failed to flush mapped memory range: {}", vk::to_string(result));
62 }
63 };
64
65 buffer_service->invalidate_range = [this](void* memory, size_t offset, size_t size) {
66 vk::DeviceMemory mem(reinterpret_cast<VkDeviceMemory>(memory));
67 vk::MappedMemoryRange
range { mem, offset, size == 0 ? VK_WHOLE_SIZE : size };
68 if (
auto result =
m_context.
get_device().invalidateMappedMemoryRanges(1, &range); result != vk::Result::eSuccess) {
70 "Failed to invalidate mapped memory range: {}", vk::to_string(result));
71 }
72 };
73
74 buffer_service->map_buffer = [this](void* memory, size_t offset, size_t size) -> void* {
75 vk::DeviceMemory mem(reinterpret_cast<VkDeviceMemory>(memory));
77 };
78
79 buffer_service->unmap_buffer = [this](void* memory) {
80 vk::DeviceMemory mem(reinterpret_cast<VkDeviceMemory>(memory));
82 };
83
84 buffer_service->copy_buffer = [this](void* src, void* dst, size_t size, size_t src_off, size_t dst_off) {
85 vk::Buffer s(static_cast<VkBuffer>(src));
86 vk::Buffer d(static_cast<VkBuffer>(dst));
88 vk::BufferCopy region { src_off, dst_off, size };
89 cmd.copyBuffer(s, d, 1, ®ion);
90 });
91 };
92
93 buffer_service->execute_fenced = [this](const std::function<void(void*)>& recorder)
94 -> std::shared_ptr<void> {
96
97 recorder(
static_cast<void*
>(
cmd));
98
100
102 vk::FenceCreateInfo fence_info {};
103 vk::Fence
fence = device.createFence(fence_info);
104
105 vk::SubmitInfo submit_info {};
106 submit_info.commandBufferCount = 1;
107 submit_info.pCommandBuffers = &
cmd;
108
110 result != vk::Result::eSuccess) {
112 "execute_fenced: queue submit failed: {}", vk::to_string(result));
113 device.destroyFence(
fence);
115 return nullptr;
116 }
117
118 auto handle = std::make_shared<FencedSubmission>();
120 handle->fence =
fence;
121 return handle;
122 };
123
124 buffer_service->wait_fenced = [this](const std::shared_ptr<void>& handle) {
125 if (!handle)
126 return;
127 auto sub = std::static_pointer_cast<FencedSubmission>(handle);
128 if (!sub->fence)
129 return;
130
132 if (auto result = device.waitForFences(1, &sub->fence, VK_TRUE, UINT64_MAX);
133 result != vk::Result::eSuccess) {
135 "wait_fenced: waitForFences failed: {}", vk::to_string(result));
136 }
137 };
138
139 buffer_service->release_fenced = [this](const std::shared_ptr<void>& handle) {
140 if (!handle)
141 return;
142 auto sub = std::static_pointer_cast<FencedSubmission>(handle);
143
145 if (sub->fence) {
146 device.destroyFence(sub->fence);
147 sub->fence = nullptr;
148 }
149 if (sub->cmd) {
151 sub->cmd = nullptr;
152 }
153 };
154
155 buffer_service->copy_buffer_fenced = [this, buffer_service](void* src, void* dst, size_t size, size_t src_off, size_t dst_off)
156 -> std::shared_ptr<void> {
157 vk::Buffer s(static_cast<VkBuffer>(src));
158 vk::Buffer d(static_cast<VkBuffer>(dst));
159 return buffer_service->execute_fenced([&](void* cmd_ptr) {
160 vk::CommandBuffer
cmd(
static_cast<VkCommandBuffer
>(cmd_ptr));
161 vk::BufferCopy region { src_off, dst_off, static_cast<vk::DeviceSize>(size) };
162 cmd.copyBuffer(s, d, 1, ®ion);
163 });
164 };
165}
#define MF_ERROR(comp, ctx,...)
void cleanup_buffer(const std::shared_ptr< Buffers::VKBuffer > &buffer)
Cleanup a buffer and release associated resources.
vk::DeviceAddress get_buffer_device_address(const std::shared_ptr< Buffers::VKBuffer > &buffer) const
Query the Vulkan device address of an initialized BDA-capable buffer.
void execute_immediate_commands(const std::function< void(vk::CommandBuffer)> &recorder)
Execute immediate command recording for buffer operations.
VKCommandManager & m_command_manager
void initialize_buffer(const std::shared_ptr< Buffers::VKBuffer > &buffer)
Initialize a buffer for use with the graphics backend.
void record_deferred_commands(const std::function< void(vk::CommandBuffer)> &recorder)
Record deferred command recording for buffer operations.
void free_command_buffer(vk::CommandBuffer command_buffer)
Free a command buffer back to the pool.
vk::CommandBuffer begin_single_time_commands()
Begin single-time command (for transfers, etc.)
vk::Device get_device() const
Get logical device.
vk::Queue get_graphics_queue() const
Get graphics queue.
@ GraphicsBackend
Graphics/visual rendering backend (Vulkan, OpenGL)
@ Core
Core engine, backend, subsystems.
std::vector< double > range(std::span< const double > data, size_t n_windows, uint32_t hop_size, uint32_t window_size)
Value range (max - min) per window.