2019-10-05 09:50:14 +00:00
|
|
|
/*
|
|
|
|
* Created on: Oct 3, 2019
|
|
|
|
|
|
|
|
Copyright 2019 flyinghead
|
|
|
|
|
|
|
|
This file is part of Flycast.
|
|
|
|
|
|
|
|
Flycast is free software: you can redistribute it and/or modify
|
|
|
|
it under the terms of the GNU General Public License as published by
|
|
|
|
the Free Software Foundation, either version 2 of the License, or
|
|
|
|
(at your option) any later version.
|
|
|
|
|
|
|
|
Flycast is distributed in the hope that it will be useful,
|
|
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
GNU General Public License for more details.
|
|
|
|
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
|
|
along with Flycast. If not, see <https://www.gnu.org/licenses/>.
|
|
|
|
*/
|
|
|
|
#pragma once
|
|
|
|
#include "vulkan.h"
|
2019-11-29 18:28:22 +00:00
|
|
|
#include "vmallocator.h"
|
2022-03-12 16:56:46 +00:00
|
|
|
#include "utils.h"
|
2019-10-05 09:50:14 +00:00
|
|
|
|
|
|
|
struct BufferData
|
|
|
|
{
|
2021-03-13 13:13:39 +00:00
|
|
|
BufferData(vk::DeviceSize size, const vk::BufferUsageFlags& usage,
|
2021-11-01 12:05:22 +00:00
|
|
|
const vk::MemoryPropertyFlags& propertyFlags =
|
|
|
|
vk::MemoryPropertyFlagBits::eHostVisible
|
|
|
|
#ifndef __APPLE__
|
|
|
|
// host coherent memory not supported on apple platforms
|
|
|
|
| vk::MemoryPropertyFlagBits::eHostCoherent
|
|
|
|
#endif
|
|
|
|
);
|
2019-10-12 11:47:25 +00:00
|
|
|
~BufferData()
|
|
|
|
{
|
|
|
|
buffer.reset();
|
|
|
|
}
|
2019-10-05 09:50:14 +00:00
|
|
|
|
2019-11-10 09:16:18 +00:00
|
|
|
void upload(u32 size, const void *data, u32 bufOffset = 0) const
|
2019-10-05 09:50:14 +00:00
|
|
|
{
|
2021-11-01 12:05:22 +00:00
|
|
|
verify((bool)(m_propertyFlags & vk::MemoryPropertyFlagBits::eHostVisible));
|
2019-10-21 14:39:16 +00:00
|
|
|
verify(bufOffset + size <= bufferSize);
|
2019-10-05 09:50:14 +00:00
|
|
|
|
2019-11-26 09:42:44 +00:00
|
|
|
void* dataPtr = (u8 *)allocation.MapMemory() + bufOffset;
|
2019-10-05 09:50:14 +00:00
|
|
|
memcpy(dataPtr, data, size);
|
2021-11-01 12:05:22 +00:00
|
|
|
allocation.UnmapMemory();
|
2019-10-05 09:50:14 +00:00
|
|
|
}
|
|
|
|
|
2021-01-23 14:59:57 +00:00
|
|
|
void upload(size_t count, const u32 *sizes, const void * const *data, u32 bufOffset = 0) const
|
2019-10-06 19:21:31 +00:00
|
|
|
{
|
2021-11-01 12:05:22 +00:00
|
|
|
verify((bool)(m_propertyFlags & vk::MemoryPropertyFlagBits::eHostVisible));
|
2019-10-06 19:21:31 +00:00
|
|
|
|
|
|
|
u32 totalSize = 0;
|
2020-03-29 18:58:49 +00:00
|
|
|
for (size_t i = 0; i < count; i++)
|
2019-10-06 19:21:31 +00:00
|
|
|
totalSize += sizes[i];
|
2019-10-21 14:39:16 +00:00
|
|
|
verify(bufOffset + totalSize <= bufferSize);
|
2019-11-26 09:42:44 +00:00
|
|
|
void* dataPtr = (u8 *)allocation.MapMemory() + bufOffset;
|
2020-03-29 18:58:49 +00:00
|
|
|
for (size_t i = 0; i < count; i++)
|
2019-10-06 19:21:31 +00:00
|
|
|
{
|
2019-10-15 14:49:20 +00:00
|
|
|
if (data[i] != nullptr)
|
|
|
|
memcpy(dataPtr, data[i], sizes[i]);
|
2019-10-06 19:21:31 +00:00
|
|
|
dataPtr = (u8 *)dataPtr + sizes[i];
|
|
|
|
}
|
2021-11-01 12:05:22 +00:00
|
|
|
allocation.UnmapMemory();
|
2019-10-06 19:21:31 +00:00
|
|
|
}
|
|
|
|
|
2019-11-12 18:39:58 +00:00
|
|
|
void download(u32 size, void *data, u32 bufOffset = 0) const
|
2019-10-09 19:16:12 +00:00
|
|
|
{
|
2021-11-01 12:05:22 +00:00
|
|
|
verify((bool)(m_propertyFlags & vk::MemoryPropertyFlagBits::eHostVisible));
|
2019-10-21 14:39:16 +00:00
|
|
|
verify(bufOffset + size <= bufferSize);
|
2019-10-09 19:16:12 +00:00
|
|
|
|
2019-11-26 09:42:44 +00:00
|
|
|
void* dataPtr = (u8 *)allocation.MapMemory() + bufOffset;
|
2019-10-09 19:16:12 +00:00
|
|
|
memcpy(data, dataPtr, size);
|
2021-11-01 12:05:22 +00:00
|
|
|
allocation.UnmapMemory();
|
2019-10-12 11:47:25 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void *MapMemory()
|
|
|
|
{
|
2019-11-26 09:42:44 +00:00
|
|
|
return allocation.MapMemory();
|
2019-10-12 11:47:25 +00:00
|
|
|
}
|
|
|
|
void UnmapMemory()
|
|
|
|
{
|
2021-11-01 12:05:22 +00:00
|
|
|
allocation.UnmapMemory();
|
2019-10-09 19:16:12 +00:00
|
|
|
}
|
|
|
|
|
2019-10-12 11:47:25 +00:00
|
|
|
vk::UniqueBuffer buffer;
|
|
|
|
vk::DeviceSize bufferSize;
|
2019-11-26 09:42:44 +00:00
|
|
|
Allocation allocation;
|
2019-10-05 09:50:14 +00:00
|
|
|
|
|
|
|
private:
|
|
|
|
vk::BufferUsageFlags m_usage;
|
|
|
|
vk::MemoryPropertyFlags m_propertyFlags;
|
|
|
|
};
|
2022-03-12 16:56:46 +00:00
|
|
|
|
|
|
|
class BufferPacker
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
BufferPacker();
|
|
|
|
|
|
|
|
vk::DeviceSize addUniform(const void *p, size_t size) {
|
|
|
|
return add(p, size, uniformAlignment);
|
|
|
|
}
|
|
|
|
|
|
|
|
vk::DeviceSize addStorage(const void *p, size_t size) {
|
|
|
|
return add(p, size, storageAlignment);
|
|
|
|
}
|
|
|
|
|
|
|
|
vk::DeviceSize add(const void *p, size_t size, u32 alignment = 4)
|
|
|
|
{
|
|
|
|
u32 padding = align(offset, std::max(4u, alignment));
|
|
|
|
if (padding != 0)
|
|
|
|
{
|
|
|
|
chunks.push_back(nullptr);
|
|
|
|
chunkSizes.push_back(padding);
|
|
|
|
offset += padding;
|
|
|
|
}
|
|
|
|
vk::DeviceSize start = offset;
|
|
|
|
chunks.push_back(p);
|
|
|
|
chunkSizes.push_back(size);
|
|
|
|
offset += size;
|
|
|
|
|
|
|
|
return start;
|
|
|
|
}
|
|
|
|
|
|
|
|
void upload(BufferData& bufferData, u32 bufOffset = 0)
|
|
|
|
{
|
|
|
|
bufferData.upload(chunks.size(), &chunkSizes[0], &chunks[0], bufOffset);
|
|
|
|
}
|
|
|
|
|
|
|
|
vk::DeviceSize size() const {
|
|
|
|
return offset;
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
std::vector<const void *> chunks;
|
|
|
|
std::vector<u32> chunkSizes;
|
|
|
|
vk::DeviceSize offset = 0;
|
|
|
|
vk::DeviceSize uniformAlignment;
|
|
|
|
vk::DeviceSize storageAlignment;
|
|
|
|
};
|