]> git.tdb.fi Git - libs/gl.git/blob - source/backends/vulkan/transferqueue.h
58b1bc2d6fa5f4743964c72d1e15e01305a97af0
[libs/gl.git] / source / backends / vulkan / transferqueue.h
1 #ifndef MSP_GL_VULKAN_TRANSFERQUEUE_H_
2 #define MSP_GL_VULKAN_TRANSFERQUEUE_H_
3
4 #include <functional>
5 #include <vector>
6 #include "handles.h"
7
8 namespace Msp {
9 namespace GL {
10
11 class Buffer;
12 class Device;
13
14 class TransferQueue
15 {
16 private:
17         struct StagingBuffer
18         {
19                 Device &device;
20                 VkBuffer buffer = 0;
21                 unsigned memory_id = 0;
22                 std::size_t size = 0;
23                 std::size_t used = 0;
24                 void *mapped_address = 0;
25
26                 StagingBuffer(Device &, std::size_t);
27                 StagingBuffer(StagingBuffer &&);
28                 ~StagingBuffer();
29         };
30
31         struct PendingTransfer
32         {
33                 unsigned order = 0;
34                 int buffer_index = -1;
35                 std::size_t offset = 0;
36                 std::size_t size = 0;
37                 std::function<void()> synchronize;
38                 std::function<void(VkCommandBuffer, VkBuffer, std::size_t)> transfer;
39         };
40
41         Device &device;
42         std::size_t default_buffer_size = 16*1048576;
43         std::vector<StagingBuffer> buffers;
44         std::vector<PendingTransfer> transfers;
45         std::map<const void *, unsigned> next_orders;
46
47 public:
48         TransferQueue(Device &);
49
50         template<typename S, typename T>
51         void *prepare_transfer(const void *, bool, std::size_t, S &&, T &&);
52
53 private:
54         PendingTransfer &prepare_transfer(const void *, bool, std::size_t);
55
56 public:
57         void dispatch_transfers(VkCommandBuffer);
58 };
59
60 template<typename S, typename T>
61 void *TransferQueue::prepare_transfer(const void *object, bool ordered, std::size_t size, S &&synchronize, T &&transfer)
62 {
63         PendingTransfer &pt = prepare_transfer(object, ordered, size);
64         pt.synchronize = std::forward<S>(synchronize);
65         pt.transfer = std::forward<T>(transfer);
66         return (pt.buffer_index<0 ? 0 : static_cast<char *>(buffers[pt.buffer_index].mapped_address)+pt.offset);
67 }
68
69 } // namespace GL
70 } // namespace Msp
71
72 #endif