2019-12-16 03:28:01 +00:00
|
|
|
#include <cstring>
|
|
|
|
|
|
|
|
#include "d3d9_initializer.h"
|
|
|
|
|
|
|
|
namespace dxvk {
|
|
|
|
|
|
|
|
D3D9Initializer::D3D9Initializer(
|
|
|
|
const Rc<DxvkDevice>& Device)
|
2022-06-21 23:17:26 +02:00
|
|
|
: m_device(Device), m_context(m_device->createContext(DxvkContextType::Supplementary)) {
|
2019-12-16 03:28:01 +00:00
|
|
|
m_context->beginRecording(
|
|
|
|
m_device->createCommandList());
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
D3D9Initializer::~D3D9Initializer() {
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void D3D9Initializer::Flush() {
|
2021-06-28 19:19:29 +02:00
|
|
|
std::lock_guard<dxvk::mutex> lock(m_mutex);
|
2019-12-16 03:28:01 +00:00
|
|
|
|
|
|
|
if (m_transferCommands != 0)
|
|
|
|
FlushInternal();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void D3D9Initializer::InitBuffer(
|
|
|
|
D3D9CommonBuffer* pBuffer) {
|
|
|
|
VkMemoryPropertyFlags memFlags = pBuffer->GetBuffer<D3D9_COMMON_BUFFER_TYPE_REAL>()->memFlags();
|
|
|
|
|
|
|
|
(memFlags & VK_MEMORY_PROPERTY_HOST_VISIBLE_BIT)
|
|
|
|
? InitHostVisibleBuffer(pBuffer->GetBufferSlice<D3D9_COMMON_BUFFER_TYPE_REAL>())
|
|
|
|
: InitDeviceLocalBuffer(pBuffer->GetBufferSlice<D3D9_COMMON_BUFFER_TYPE_REAL>());
|
|
|
|
|
|
|
|
if (pBuffer->GetMapMode() == D3D9_COMMON_BUFFER_MAP_MODE_BUFFER)
|
|
|
|
InitHostVisibleBuffer(pBuffer->GetBufferSlice<D3D9_COMMON_BUFFER_TYPE_STAGING>());
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void D3D9Initializer::InitTexture(
|
|
|
|
D3D9CommonTexture* pTexture,
|
2022-03-16 02:28:22 +01:00
|
|
|
void* pInitialData) {
|
2019-12-16 03:28:01 +00:00
|
|
|
if (pTexture->GetMapMode() == D3D9_COMMON_TEXTURE_MAP_MODE_NONE)
|
|
|
|
return;
|
|
|
|
|
2022-03-16 02:28:22 +01:00
|
|
|
if (pTexture->GetImage() != nullptr)
|
|
|
|
InitDeviceLocalTexture(pTexture);
|
|
|
|
|
|
|
|
if (pTexture->Desc()->Pool != D3DPOOL_DEFAULT)
|
|
|
|
InitHostVisibleTexture(pTexture, pInitialData);
|
2019-12-16 03:28:01 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void D3D9Initializer::InitDeviceLocalBuffer(
|
|
|
|
DxvkBufferSlice Slice) {
|
2021-06-28 19:19:29 +02:00
|
|
|
std::lock_guard<dxvk::mutex> lock(m_mutex);
|
2019-12-16 03:28:01 +00:00
|
|
|
|
|
|
|
m_transferCommands += 1;
|
|
|
|
|
2022-02-17 15:33:24 +01:00
|
|
|
m_context->initBuffer(
|
|
|
|
Slice.buffer());
|
2019-12-16 03:28:01 +00:00
|
|
|
|
|
|
|
FlushImplicit();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void D3D9Initializer::InitHostVisibleBuffer(
|
|
|
|
DxvkBufferSlice Slice) {
|
|
|
|
// If the buffer is mapped, we can write data directly
|
|
|
|
// to the mapped memory region instead of doing it on
|
|
|
|
// the GPU. Same goes for zero-initialization.
|
|
|
|
std::memset(
|
|
|
|
Slice.mapPtr(0), 0,
|
|
|
|
Slice.length());
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void D3D9Initializer::InitDeviceLocalTexture(
|
|
|
|
D3D9CommonTexture* pTexture) {
|
2021-06-28 19:19:29 +02:00
|
|
|
std::lock_guard<dxvk::mutex> lock(m_mutex);
|
2019-12-16 03:28:01 +00:00
|
|
|
|
|
|
|
auto InitImage = [&](Rc<DxvkImage> image) {
|
|
|
|
if (image == nullptr)
|
|
|
|
return;
|
|
|
|
|
2022-07-15 17:23:54 +02:00
|
|
|
auto formatInfo = lookupFormatInfo(image->info().format);
|
2019-12-16 03:28:01 +00:00
|
|
|
|
|
|
|
m_transferCommands += 1;
|
|
|
|
|
|
|
|
// While the Microsoft docs state that resource contents are
|
|
|
|
// undefined if no initial data is provided, some applications
|
2022-02-17 15:20:09 +01:00
|
|
|
// expect a resource to be pre-cleared.
|
2019-12-16 03:28:01 +00:00
|
|
|
VkImageSubresourceRange subresources;
|
|
|
|
subresources.aspectMask = formatInfo->aspectMask;
|
|
|
|
subresources.baseMipLevel = 0;
|
|
|
|
subresources.levelCount = image->info().mipLevels;
|
|
|
|
subresources.baseArrayLayer = 0;
|
|
|
|
subresources.layerCount = image->info().numLayers;
|
|
|
|
|
2022-02-17 15:20:09 +01:00
|
|
|
m_context->initImage(image, subresources, VK_IMAGE_LAYOUT_UNDEFINED);
|
2019-12-16 03:28:01 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
InitImage(pTexture->GetImage());
|
|
|
|
|
|
|
|
FlushImplicit();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void D3D9Initializer::InitHostVisibleTexture(
|
|
|
|
D3D9CommonTexture* pTexture,
|
|
|
|
void* pInitialData) {
|
|
|
|
// If the buffer is mapped, we can write data directly
|
|
|
|
// to the mapped memory region instead of doing it on
|
|
|
|
// the GPU. Same goes for zero-initialization.
|
2022-08-18 19:20:06 +02:00
|
|
|
void* mapPtr = pTexture->GetData(0);
|
|
|
|
if (pInitialData) {
|
|
|
|
// Initial data is only supported for textures with 1 subresource
|
|
|
|
VkExtent3D mipExtent = pTexture->GetExtentMip(0);
|
|
|
|
const DxvkFormatInfo* formatInfo = lookupFormatInfo(pTexture->GetFormatMapping().FormatColor);
|
|
|
|
VkExtent3D blockCount = util::computeBlockCount(mipExtent, formatInfo->blockSize);
|
|
|
|
uint32_t pitch = blockCount.width * formatInfo->elementSize;
|
|
|
|
uint32_t alignedPitch = align(pitch, 4);
|
|
|
|
|
|
|
|
util::packImageData(
|
|
|
|
mapPtr,
|
|
|
|
pInitialData,
|
|
|
|
pitch,
|
|
|
|
pitch * blockCount.height,
|
|
|
|
alignedPitch,
|
|
|
|
alignedPitch * blockCount.height,
|
|
|
|
D3D9CommonTexture::GetImageTypeFromResourceType(pTexture->GetType()),
|
|
|
|
mipExtent,
|
|
|
|
pTexture->Desc()->ArraySize,
|
|
|
|
formatInfo,
|
|
|
|
VK_IMAGE_ASPECT_COLOR_BIT);
|
|
|
|
} else {
|
|
|
|
// All subresources are allocated in one chunk of memory.
|
|
|
|
// So we can just get the pointer for subresource 0 and memset all of them at once.
|
|
|
|
std::memset(
|
|
|
|
mapPtr, 0,
|
|
|
|
pTexture->GetTotalSize());
|
2019-12-16 03:28:01 +00:00
|
|
|
}
|
2022-08-18 19:20:06 +02:00
|
|
|
pTexture->UnmapData();
|
2019-12-16 03:28:01 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void D3D9Initializer::FlushImplicit() {
|
|
|
|
if (m_transferCommands > MaxTransferCommands
|
|
|
|
|| m_transferMemory > MaxTransferMemory)
|
|
|
|
FlushInternal();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void D3D9Initializer::FlushInternal() {
|
2023-03-16 12:44:58 +01:00
|
|
|
m_context->flushCommandList(nullptr);
|
2019-12-16 03:28:01 +00:00
|
|
|
|
|
|
|
m_transferCommands = 0;
|
|
|
|
m_transferMemory = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|