-
Notifications
You must be signed in to change notification settings - Fork 18
/
Copy pathvma_handle.hpp
222 lines (192 loc) · 7.87 KB
/
vma_handle.hpp
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
#pragma once
#include "avk/avk.hpp"
namespace avk
{
/** Class handling the lifetime of one VMA allocation.
* Also provides some convenience methods.
*/
template <typename T>
struct vma_handle
{
/** Construct emptyness */
vma_handle() : mAllocator{nullptr}, mCreateInfo{}, mAllocation{nullptr}, mAllocationInfo{}, mResource{nullptr}
{ }
/** Initialize with VMA structs and the already created resource. */
vma_handle(VmaAllocator aAllocator, VmaAllocationCreateInfo aAllocInfo, VmaAllocation aAlloc, T aResource)
: mAllocator{ aAllocator }
, mCreateInfo{ std::move(aAllocInfo) }
, mAllocation{ std::move(aAlloc) }
, mResource{ std::move(aResource) }
{
vmaGetAllocationInfo(mAllocator, mAllocation, &mAllocationInfo);
}
/** Create VmaAllocator, VmaAllocationCreateInfo, and VmaAllocation internally.
* This is only implemented for certain types via template specialization: vk::Buffer, vk::Image
*/
template <typename C>
vma_handle(VmaAllocator aAllocator, vk::MemoryPropertyFlags aMemPropFlags, const C& aResourceCreateInfo);
/** Move-construct a vma_handle */
vma_handle(vma_handle&& aOther) noexcept : mAllocator{nullptr}, mCreateInfo{}, mAllocation{nullptr}, mAllocationInfo{}, mResource{nullptr}
{
std::swap(mAllocator, aOther.mAllocator);
std::swap(mCreateInfo, aOther.mCreateInfo);
std::swap(mAllocation, aOther.mAllocation);
std::swap(mAllocationInfo, aOther.mAllocationInfo);
std::swap(mResource, aOther.mResource);
}
vma_handle(const vma_handle& aOther) = delete;
/** Move-assign a vma_handle */
vma_handle& operator=(vma_handle&& aOther) noexcept
{
std::swap(mAllocator, aOther.mAllocator);
std::swap(mCreateInfo, aOther.mCreateInfo);
std::swap(mAllocation, aOther.mAllocation);
std::swap(mAllocationInfo, aOther.mAllocationInfo);
std::swap(mResource, aOther.mResource);
return *this;
}
vma_handle& operator=(const vma_handle& aOther) = delete;
/** Destroy the resource and free the allocation
* This is only implemented for certain types via template specialization: vk::Buffer, vk::Image
* That also means that this type is only usable with certain resource types.
*/
~vma_handle();
/** Get the allocator that was used to allocate this resource */
auto allocator() const
{
return mAllocator;
}
/** Get the config a.k.a. create-info that was used to allocate this resource. */
auto create_info() const
{
return mCreateInfo;
}
/** Get the VMA-handle of this resource's allocation. */
auto allocation() const
{
return mAllocation;
}
/** Get the resource handle. */
T resource() const
{
return mResource;
}
/** Get the memory properties from the allocation */
vk::MemoryPropertyFlags memory_properties() const
{
VkMemoryPropertyFlags result;
vmaGetMemoryTypeProperties(mAllocator, mAllocationInfo.memoryType, &result);
return vk::MemoryPropertyFlags{ result };
}
/** Map the memory in order to write data into, or read data from it.
* If data shall be read from it and the memory is not host coherent, an invalidate-instruction will be issued.
*
* Hint: Consider using avk::scoped_mapping instead of calling this method directly.
*
* @param aAccess Specify your intent: Are you going to read from the memory, or write into it, or both?
* @return Pointer to the mapped memory.
*/
void* map_memory(mapping_access aAccess) const
{
const auto memProps = memory_properties();
assert(has_flag(memProps, vk::MemoryPropertyFlagBits::eHostVisible)); // => Allocation ended up in mappable memory. You can map it and access it directly.
void* mappedData;
auto result = vmaMapMemory(mAllocator, mAllocation, &mappedData);
assert(result >= 0);
if (has_flag(aAccess, mapping_access::read) && !has_flag(memProps, vk::MemoryPropertyFlagBits::eHostCoherent)) {
result = vmaInvalidateAllocation(mAllocator, mAllocation, 0, VK_WHOLE_SIZE);
assert(result >= 0);
}
return mappedData;
}
/** Unmap memory that has been mapped before via mem_handle::map_memory.
* If data shall be written to it and the memory is not host coherent, a flush-instruction will be issued.
*
* Hint: Consider using avk::scoped_mapping instead of calling this method directly.
*
* @param aAccess Specify your intent: Are you going to read from the memory, or write into it, or both?
*/
void unmap_memory(mapping_access aAccess) const
{
const auto memProps = memory_properties();
assert(has_flag(memProps, vk::MemoryPropertyFlagBits::eHostVisible)); // => Allocation ended up in mappable memory. You can map it and access it directly.
if (has_flag(aAccess, mapping_access::write) && !has_flag(memProps, vk::MemoryPropertyFlagBits::eHostCoherent)) {
VkResult result = vmaFlushAllocation(mAllocator, mAllocation, 0, VK_WHOLE_SIZE);
assert(result >= 0);
}
vmaUnmapMemory(mAllocator, mAllocation);
}
VmaAllocator mAllocator;
VmaAllocationCreateInfo mCreateInfo;
VmaAllocation mAllocation;
VmaAllocationInfo mAllocationInfo;
T mResource;
};
// Fail if not used with either vk::Buffer or vk::Image
template <typename T>
template <typename C>
vma_handle<T>::vma_handle(VmaAllocator aAllocator, vk::MemoryPropertyFlags aMemPropFlags, const C& aResourceCreateInfo)
{
throw avk::runtime_error(std::string("VMA allocation not implemented for type ") + typeid(T).name());
}
// Constructor's template specialization for vk::Buffer
template <>
template <>
inline vma_handle<vk::Buffer>::vma_handle(VmaAllocator aAllocator, vk::MemoryPropertyFlags aMemPropFlags, const vk::BufferCreateInfo& aResourceCreateInfo)
: mAllocator{ aAllocator }
, mCreateInfo{}, mAllocation{nullptr}, mAllocationInfo{}
{
mCreateInfo.requiredFlags = static_cast<VkMemoryPropertyFlags>(aMemPropFlags);
mCreateInfo.usage = VMA_MEMORY_USAGE_UNKNOWN;
VkBuffer buffer;
auto result = vmaCreateBuffer(aAllocator, &static_cast<const VkBufferCreateInfo&>(aResourceCreateInfo), &mCreateInfo, &buffer, &mAllocation, &mAllocationInfo);
assert(result >= 0);
mResource = buffer;
}
// Constructor's template specialization for vk::Image
template <>
template <>
inline vma_handle<vk::Image>::vma_handle(VmaAllocator aAllocator, vk::MemoryPropertyFlags aMemPropFlags, const vk::ImageCreateInfo& aResourceCreateInfo)
: mAllocator{ aAllocator }
, mCreateInfo{}, mAllocation{nullptr}, mAllocationInfo{}
{
mCreateInfo.requiredFlags = static_cast<VkMemoryPropertyFlags>(aMemPropFlags);
mCreateInfo.usage = VMA_MEMORY_USAGE_UNKNOWN;
VkImage image;
auto result = vmaCreateImage(aAllocator, &static_cast<const VkImageCreateInfo&>(aResourceCreateInfo), &mCreateInfo, &image, &mAllocation, &mAllocationInfo);
assert(result >= 0);
mResource = image;
}
// Fail if not used with either vk::Buffer or vk::Image
template <typename T>
vma_handle<T>::~vma_handle()
{
throw avk::runtime_error(std::string("VMA allocation not implemented for type ") + typeid(T).name());
}
// Destructor's template specialization for vk::Buffer
template <>
inline vma_handle<vk::Buffer>::~vma_handle()
{
if (static_cast<bool>(mResource)) {
vmaDestroyBuffer(mAllocator, static_cast<VkBuffer>(mResource), mAllocation);
mAllocator = nullptr;
mCreateInfo = {};
mAllocation = nullptr;
mAllocationInfo = {};
mResource = nullptr;
}
}
// Destructor's template specialzation for vk::Image:
template <>
inline vma_handle<vk::Image>::~vma_handle()
{
if (static_cast<bool>(mResource)) {
vmaDestroyImage(mAllocator, static_cast<VkImage>(mResource), mAllocation);
mAllocator = nullptr;
mCreateInfo = {};
mAllocation = nullptr;
mAllocationInfo = {};
mResource = nullptr;
}
}
}