2017-11-29 21:46:09 +01:00
|
|
|
#include "dxgi_presenter.h"
|
|
|
|
|
|
|
|
#include "../spirv/spirv_module.h"
|
|
|
|
|
2018-04-10 18:46:39 +02:00
|
|
|
#include <dxgi_presenter_frag.h>
|
|
|
|
#include <dxgi_presenter_vert.h>
|
|
|
|
|
2017-11-29 21:46:09 +01:00
|
|
|
namespace dxvk {
|
|
|
|
|
|
|
|
DxgiPresenter::DxgiPresenter(
|
2018-04-10 20:44:55 +02:00
|
|
|
const Rc<DxvkDevice>& device,
|
|
|
|
HWND window)
|
2017-12-01 10:08:49 +01:00
|
|
|
: m_device (device),
|
|
|
|
m_context (device->createContext()) {
|
2017-11-29 21:46:09 +01:00
|
|
|
|
|
|
|
// Create Vulkan surface for the window
|
|
|
|
HINSTANCE instance = reinterpret_cast<HINSTANCE>(
|
|
|
|
GetWindowLongPtr(window, GWLP_HINSTANCE));
|
|
|
|
|
|
|
|
m_surface = m_device->adapter()->createSurface(instance, window);
|
|
|
|
|
2017-12-31 00:23:34 +01:00
|
|
|
// Reset options for the swap chain itself. We will
|
|
|
|
// create a swap chain object before presentation.
|
|
|
|
m_options.preferredSurfaceFormat = { VK_FORMAT_UNDEFINED, VK_COLOR_SPACE_SRGB_NONLINEAR_KHR };
|
|
|
|
m_options.preferredPresentMode = VK_PRESENT_MODE_FIFO_KHR;
|
|
|
|
m_options.preferredBufferSize = { 0u, 0u };
|
2017-11-29 21:46:09 +01:00
|
|
|
|
2018-04-13 13:47:15 +02:00
|
|
|
// Samplers for presentation. We'll create one with point sampling that will
|
|
|
|
// be used when the back buffer resolution matches the output resolution, and
|
|
|
|
// one with linar sampling that will be used when the image will be scaled.
|
|
|
|
m_samplerFitting = this->createSampler(VK_FILTER_NEAREST, VK_SAMPLER_ADDRESS_MODE_CLAMP_TO_BORDER);
|
|
|
|
m_samplerScaling = this->createSampler(VK_FILTER_LINEAR, VK_SAMPLER_ADDRESS_MODE_CLAMP_TO_BORDER);
|
|
|
|
|
|
|
|
// Create objects required for the gamma ramp. This is implemented partially
|
|
|
|
// with an UBO, which stores global parameters, and a lookup texture, which
|
|
|
|
// stores the actual gamma ramp and can be sampled with a linear filter.
|
|
|
|
m_gammaUbo = this->createGammaUbo();
|
|
|
|
m_gammaSampler = this->createSampler(VK_FILTER_LINEAR, VK_SAMPLER_ADDRESS_MODE_CLAMP_TO_EDGE);
|
|
|
|
m_gammaTexture = this->createGammaTexture();
|
|
|
|
m_gammaTextureView = this->createGammaTextureView();
|
2017-12-03 20:23:26 +01:00
|
|
|
|
2017-11-29 21:46:09 +01:00
|
|
|
// Set up context state. The shader bindings and the
|
|
|
|
// constant state objects will never be modified.
|
2017-12-08 00:02:43 +01:00
|
|
|
DxvkInputAssemblyState iaState;
|
|
|
|
iaState.primitiveTopology = VK_PRIMITIVE_TOPOLOGY_TRIANGLE_STRIP;
|
|
|
|
iaState.primitiveRestart = VK_FALSE;
|
2018-02-23 12:55:23 +01:00
|
|
|
iaState.patchVertexCount = 0;
|
2017-12-08 00:02:43 +01:00
|
|
|
m_context->setInputAssemblyState(iaState);
|
2017-11-29 21:46:09 +01:00
|
|
|
|
|
|
|
m_context->setInputLayout(
|
2017-12-08 00:44:58 +01:00
|
|
|
0, nullptr, 0, nullptr);
|
2017-11-29 21:46:09 +01:00
|
|
|
|
2017-12-08 00:02:43 +01:00
|
|
|
DxvkRasterizerState rsState;
|
|
|
|
rsState.enableDepthClamp = VK_FALSE;
|
|
|
|
rsState.enableDiscard = VK_FALSE;
|
|
|
|
rsState.polygonMode = VK_POLYGON_MODE_FILL;
|
|
|
|
rsState.cullMode = VK_CULL_MODE_BACK_BIT;
|
|
|
|
rsState.frontFace = VK_FRONT_FACE_COUNTER_CLOCKWISE;
|
|
|
|
rsState.depthBiasEnable = VK_FALSE;
|
|
|
|
rsState.depthBiasConstant = 0.0f;
|
|
|
|
rsState.depthBiasClamp = 0.0f;
|
|
|
|
rsState.depthBiasSlope = 0.0f;
|
|
|
|
m_context->setRasterizerState(rsState);
|
2017-11-29 21:46:09 +01:00
|
|
|
|
2017-12-08 00:02:43 +01:00
|
|
|
DxvkMultisampleState msState;
|
2017-12-11 13:03:07 +01:00
|
|
|
msState.sampleMask = 0xffffffff;
|
2017-12-08 00:02:43 +01:00
|
|
|
msState.enableAlphaToCoverage = VK_FALSE;
|
|
|
|
msState.enableAlphaToOne = VK_FALSE;
|
|
|
|
m_context->setMultisampleState(msState);
|
2017-11-29 21:46:09 +01:00
|
|
|
|
|
|
|
VkStencilOpState stencilOp;
|
|
|
|
stencilOp.failOp = VK_STENCIL_OP_KEEP;
|
|
|
|
stencilOp.passOp = VK_STENCIL_OP_KEEP;
|
|
|
|
stencilOp.depthFailOp = VK_STENCIL_OP_KEEP;
|
|
|
|
stencilOp.compareOp = VK_COMPARE_OP_ALWAYS;
|
|
|
|
stencilOp.compareMask = 0xFFFFFFFF;
|
|
|
|
stencilOp.writeMask = 0xFFFFFFFF;
|
|
|
|
stencilOp.reference = 0;
|
|
|
|
|
2017-12-08 00:02:43 +01:00
|
|
|
DxvkDepthStencilState dsState;
|
2017-12-08 01:06:48 +01:00
|
|
|
dsState.enableDepthTest = VK_FALSE;
|
|
|
|
dsState.enableDepthWrite = VK_FALSE;
|
|
|
|
dsState.enableDepthBounds = VK_FALSE;
|
|
|
|
dsState.enableStencilTest = VK_FALSE;
|
|
|
|
dsState.depthCompareOp = VK_COMPARE_OP_ALWAYS;
|
|
|
|
dsState.stencilOpFront = stencilOp;
|
|
|
|
dsState.stencilOpBack = stencilOp;
|
|
|
|
dsState.depthBoundsMin = 0.0f;
|
|
|
|
dsState.depthBoundsMax = 1.0f;
|
2018-01-01 23:00:07 +01:00
|
|
|
m_context->setDepthStencilState(dsState);
|
2017-12-08 01:06:48 +01:00
|
|
|
|
|
|
|
DxvkLogicOpState loState;
|
|
|
|
loState.enableLogicOp = VK_FALSE;
|
|
|
|
loState.logicOp = VK_LOGIC_OP_NO_OP;
|
|
|
|
m_context->setLogicOpState(loState);
|
|
|
|
|
2018-01-13 03:53:33 +01:00
|
|
|
m_blendMode.enableBlending = VK_FALSE;
|
|
|
|
m_blendMode.colorSrcFactor = VK_BLEND_FACTOR_ONE;
|
|
|
|
m_blendMode.colorDstFactor = VK_BLEND_FACTOR_ONE_MINUS_SRC_ALPHA;
|
|
|
|
m_blendMode.colorBlendOp = VK_BLEND_OP_ADD;
|
|
|
|
m_blendMode.alphaSrcFactor = VK_BLEND_FACTOR_ONE;
|
|
|
|
m_blendMode.alphaDstFactor = VK_BLEND_FACTOR_ONE_MINUS_SRC_ALPHA;
|
|
|
|
m_blendMode.alphaBlendOp = VK_BLEND_OP_ADD;
|
|
|
|
m_blendMode.writeMask = VK_COLOR_COMPONENT_R_BIT
|
|
|
|
| VK_COLOR_COMPONENT_G_BIT
|
|
|
|
| VK_COLOR_COMPONENT_B_BIT
|
|
|
|
| VK_COLOR_COMPONENT_A_BIT;
|
2017-12-07 09:38:31 +01:00
|
|
|
|
|
|
|
m_context->bindShader(
|
|
|
|
VK_SHADER_STAGE_VERTEX_BIT,
|
|
|
|
this->createVertexShader());
|
|
|
|
|
|
|
|
m_context->bindShader(
|
|
|
|
VK_SHADER_STAGE_FRAGMENT_BIT,
|
|
|
|
this->createFragmentShader());
|
2018-01-13 03:53:33 +01:00
|
|
|
|
|
|
|
m_hud = hud::Hud::createHud(m_device);
|
2017-11-29 21:46:09 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
DxgiPresenter::~DxgiPresenter() {
|
2017-12-27 15:55:46 +01:00
|
|
|
m_device->waitForIdle();
|
2017-11-29 21:46:09 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2017-12-02 11:46:25 +01:00
|
|
|
void DxgiPresenter::initBackBuffer(const Rc<DxvkImage>& image) {
|
2018-04-10 20:44:55 +02:00
|
|
|
m_context->beginRecording(
|
|
|
|
m_device->createCommandList());
|
|
|
|
|
2017-12-05 13:00:06 +01:00
|
|
|
VkImageSubresourceRange sr;
|
|
|
|
sr.aspectMask = VK_IMAGE_ASPECT_COLOR_BIT;
|
|
|
|
sr.baseMipLevel = 0;
|
|
|
|
sr.levelCount = image->info().mipLevels;
|
|
|
|
sr.baseArrayLayer = 0;
|
|
|
|
sr.layerCount = image->info().numLayers;
|
|
|
|
|
|
|
|
m_context->initImage(image, sr);
|
2018-04-10 20:44:55 +02:00
|
|
|
|
2017-12-02 11:46:25 +01:00
|
|
|
m_device->submitCommandList(
|
|
|
|
m_context->endRecording(),
|
|
|
|
nullptr, nullptr);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2017-12-12 00:27:49 +01:00
|
|
|
void DxgiPresenter::presentImage() {
|
2018-01-13 03:53:33 +01:00
|
|
|
if (m_hud != nullptr) {
|
|
|
|
m_hud->render({
|
|
|
|
m_options.preferredBufferSize.width,
|
|
|
|
m_options.preferredBufferSize.height,
|
|
|
|
});
|
2018-01-10 13:43:23 +01:00
|
|
|
}
|
|
|
|
|
2017-12-31 00:23:34 +01:00
|
|
|
const bool fitSize =
|
|
|
|
m_backBuffer->info().extent.width == m_options.preferredBufferSize.width
|
|
|
|
&& m_backBuffer->info().extent.height == m_options.preferredBufferSize.height;
|
|
|
|
|
2017-12-01 10:08:49 +01:00
|
|
|
m_context->beginRecording(
|
|
|
|
m_device->createCommandList());
|
|
|
|
|
2017-12-12 00:27:49 +01:00
|
|
|
VkImageSubresourceLayers resolveSubresources;
|
|
|
|
resolveSubresources.aspectMask = VK_IMAGE_ASPECT_COLOR_BIT;
|
|
|
|
resolveSubresources.mipLevel = 0;
|
|
|
|
resolveSubresources.baseArrayLayer = 0;
|
|
|
|
resolveSubresources.layerCount = 1;
|
|
|
|
|
|
|
|
if (m_backBufferResolve != nullptr) {
|
|
|
|
m_context->resolveImage(
|
|
|
|
m_backBufferResolve, resolveSubresources,
|
2018-02-20 22:26:23 +01:00
|
|
|
m_backBuffer, resolveSubresources,
|
|
|
|
VK_FORMAT_UNDEFINED);
|
2017-12-12 00:27:49 +01:00
|
|
|
}
|
|
|
|
|
2017-12-16 21:30:48 +01:00
|
|
|
const DxvkSwapSemaphores sem = m_swapchain->getSemaphorePair();
|
|
|
|
|
|
|
|
auto framebuffer = m_swapchain->getFramebuffer(sem.acquireSync);
|
2017-11-29 21:46:09 +01:00
|
|
|
auto framebufferSize = framebuffer->size();
|
2017-12-05 13:00:06 +01:00
|
|
|
|
2017-11-29 21:46:09 +01:00
|
|
|
m_context->bindFramebuffer(framebuffer);
|
|
|
|
|
|
|
|
VkViewport viewport;
|
|
|
|
viewport.x = 0.0f;
|
|
|
|
viewport.y = 0.0f;
|
|
|
|
viewport.width = static_cast<float>(framebufferSize.width);
|
|
|
|
viewport.height = static_cast<float>(framebufferSize.height);
|
|
|
|
viewport.minDepth = 0.0f;
|
|
|
|
viewport.maxDepth = 1.0f;
|
|
|
|
|
|
|
|
VkRect2D scissor;
|
|
|
|
scissor.offset.x = 0;
|
|
|
|
scissor.offset.y = 0;
|
|
|
|
scissor.extent.width = framebufferSize.width;
|
|
|
|
scissor.extent.height = framebufferSize.height;
|
|
|
|
|
|
|
|
m_context->setViewports(1, &viewport, &scissor);
|
|
|
|
|
2017-12-31 00:23:34 +01:00
|
|
|
m_context->bindResourceSampler(BindingIds::Sampler,
|
|
|
|
fitSize ? m_samplerFitting : m_samplerScaling);
|
|
|
|
|
2018-01-13 03:53:33 +01:00
|
|
|
m_blendMode.enableBlending = VK_FALSE;
|
|
|
|
m_context->setBlendMode(0, m_blendMode);
|
|
|
|
|
2018-03-10 11:16:52 +01:00
|
|
|
m_context->bindResourceView(BindingIds::Texture, m_backBufferView, nullptr);
|
2017-11-29 21:46:09 +01:00
|
|
|
m_context->draw(4, 1, 0, 0);
|
|
|
|
|
2018-04-13 13:47:15 +02:00
|
|
|
m_context->bindResourceSampler(BindingIds::GammaSmp, m_gammaSampler);
|
|
|
|
m_context->bindResourceView (BindingIds::GammaTex, m_gammaTextureView, nullptr);
|
|
|
|
m_context->bindResourceBuffer (BindingIds::GammaUbo, DxvkBufferSlice(m_gammaUbo));
|
2018-04-10 20:44:55 +02:00
|
|
|
|
2018-01-13 03:53:33 +01:00
|
|
|
if (m_hud != nullptr) {
|
|
|
|
m_blendMode.enableBlending = VK_TRUE;
|
|
|
|
m_context->setBlendMode(0, m_blendMode);
|
|
|
|
|
2018-03-10 11:16:52 +01:00
|
|
|
m_context->bindResourceView(BindingIds::Texture, m_hud->texture(), nullptr);
|
2018-01-13 03:53:33 +01:00
|
|
|
m_context->draw(4, 1, 0, 0);
|
|
|
|
}
|
|
|
|
|
2017-12-01 10:08:49 +01:00
|
|
|
m_device->submitCommandList(
|
|
|
|
m_context->endRecording(),
|
2017-12-16 21:30:48 +01:00
|
|
|
sem.acquireSync, sem.presentSync);
|
2017-11-29 21:46:09 +01:00
|
|
|
|
2017-12-16 21:30:48 +01:00
|
|
|
m_swapchain->present(sem.presentSync);
|
2017-11-29 21:46:09 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2017-12-19 16:01:50 +01:00
|
|
|
void DxgiPresenter::updateBackBuffer(const Rc<DxvkImage>& image) {
|
2017-12-12 00:27:49 +01:00
|
|
|
// Explicitly destroy the old stuff
|
2017-12-19 16:01:50 +01:00
|
|
|
m_backBuffer = image;
|
2017-12-12 00:27:49 +01:00
|
|
|
m_backBufferResolve = nullptr;
|
|
|
|
m_backBufferView = nullptr;
|
|
|
|
|
|
|
|
// If a multisampled back buffer was requested, we also need to
|
|
|
|
// create a resolve image with otherwise identical properties.
|
|
|
|
// Multisample images cannot be sampled from.
|
2017-12-19 16:01:50 +01:00
|
|
|
if (image->info().sampleCount != VK_SAMPLE_COUNT_1_BIT) {
|
2017-12-12 00:27:49 +01:00
|
|
|
DxvkImageCreateInfo resolveInfo;
|
|
|
|
resolveInfo.type = VK_IMAGE_TYPE_2D;
|
2017-12-19 16:01:50 +01:00
|
|
|
resolveInfo.format = image->info().format;
|
2017-12-12 00:27:49 +01:00
|
|
|
resolveInfo.flags = 0;
|
|
|
|
resolveInfo.sampleCount = VK_SAMPLE_COUNT_1_BIT;
|
2017-12-19 16:01:50 +01:00
|
|
|
resolveInfo.extent = image->info().extent;
|
2017-12-12 00:27:49 +01:00
|
|
|
resolveInfo.numLayers = 1;
|
|
|
|
resolveInfo.mipLevels = 1;
|
|
|
|
resolveInfo.usage = VK_IMAGE_USAGE_SAMPLED_BIT
|
|
|
|
| VK_IMAGE_USAGE_TRANSFER_DST_BIT;
|
|
|
|
resolveInfo.stages = VK_PIPELINE_STAGE_FRAGMENT_SHADER_BIT
|
|
|
|
| VK_PIPELINE_STAGE_TRANSFER_BIT;
|
|
|
|
resolveInfo.access = VK_ACCESS_SHADER_READ_BIT
|
|
|
|
| VK_ACCESS_TRANSFER_WRITE_BIT;
|
|
|
|
resolveInfo.tiling = VK_IMAGE_TILING_OPTIMAL;
|
|
|
|
resolveInfo.layout = VK_IMAGE_LAYOUT_SHADER_READ_ONLY_OPTIMAL;
|
|
|
|
|
|
|
|
m_backBufferResolve = m_device->createImage(
|
|
|
|
resolveInfo, VK_MEMORY_PROPERTY_DEVICE_LOCAL_BIT);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Create an image view that allows the
|
|
|
|
// image to be bound as a shader resource.
|
|
|
|
DxvkImageViewCreateInfo viewInfo;
|
|
|
|
viewInfo.type = VK_IMAGE_VIEW_TYPE_2D;
|
2017-12-19 16:01:50 +01:00
|
|
|
viewInfo.format = image->info().format;
|
2017-12-12 00:27:49 +01:00
|
|
|
viewInfo.aspect = VK_IMAGE_ASPECT_COLOR_BIT;
|
|
|
|
viewInfo.minLevel = 0;
|
|
|
|
viewInfo.numLevels = 1;
|
|
|
|
viewInfo.minLayer = 0;
|
|
|
|
viewInfo.numLayers = 1;
|
|
|
|
|
|
|
|
m_backBufferView = m_device->createImageView(
|
|
|
|
m_backBufferResolve != nullptr
|
|
|
|
? m_backBufferResolve
|
|
|
|
: m_backBuffer,
|
|
|
|
viewInfo);
|
|
|
|
|
|
|
|
this->initBackBuffer(m_backBuffer);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2017-12-31 00:23:34 +01:00
|
|
|
void DxgiPresenter::recreateSwapchain(const DxvkSwapchainProperties& options) {
|
|
|
|
const bool doRecreate =
|
|
|
|
options.preferredSurfaceFormat.format != m_options.preferredSurfaceFormat.format
|
|
|
|
|| options.preferredSurfaceFormat.colorSpace != m_options.preferredSurfaceFormat.colorSpace
|
|
|
|
|| options.preferredPresentMode != m_options.preferredPresentMode
|
|
|
|
|| options.preferredBufferSize.width != m_options.preferredBufferSize.width
|
|
|
|
|| options.preferredBufferSize.height != m_options.preferredBufferSize.height;
|
|
|
|
|
|
|
|
if (doRecreate) {
|
|
|
|
Logger::info(str::format(
|
|
|
|
"DxgiPresenter: Recreating swap chain: ",
|
|
|
|
"\n Format: ", options.preferredSurfaceFormat.format,
|
|
|
|
"\n Present mode: ", options.preferredPresentMode,
|
|
|
|
"\n Buffer size: ", options.preferredBufferSize.width, "x", options.preferredBufferSize.height));
|
|
|
|
|
|
|
|
m_options = options;
|
|
|
|
|
|
|
|
if (m_swapchain == nullptr) {
|
|
|
|
m_swapchain = m_device->createSwapchain(
|
|
|
|
m_surface, options);
|
|
|
|
} else {
|
|
|
|
m_swapchain->changeProperties(options);
|
|
|
|
}
|
|
|
|
}
|
2017-12-04 22:21:02 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2017-12-31 00:23:34 +01:00
|
|
|
VkSurfaceFormatKHR DxgiPresenter::pickSurfaceFormat(DXGI_FORMAT fmt) const {
|
2017-12-04 22:21:02 +01:00
|
|
|
std::vector<VkSurfaceFormatKHR> formats;
|
|
|
|
|
|
|
|
switch (fmt) {
|
|
|
|
case DXGI_FORMAT_R8G8B8A8_UNORM:
|
|
|
|
case DXGI_FORMAT_B8G8R8A8_UNORM: {
|
|
|
|
formats.push_back({ VK_FORMAT_R8G8B8A8_UNORM, VK_COLOR_SPACE_SRGB_NONLINEAR_KHR });
|
|
|
|
formats.push_back({ VK_FORMAT_B8G8R8A8_UNORM, VK_COLOR_SPACE_SRGB_NONLINEAR_KHR });
|
|
|
|
} break;
|
|
|
|
|
|
|
|
case DXGI_FORMAT_R8G8B8A8_UNORM_SRGB:
|
|
|
|
case DXGI_FORMAT_B8G8R8A8_UNORM_SRGB: {
|
|
|
|
formats.push_back({ VK_FORMAT_R8G8B8A8_SRGB, VK_COLOR_SPACE_SRGB_NONLINEAR_KHR });
|
|
|
|
formats.push_back({ VK_FORMAT_B8G8R8A8_SRGB, VK_COLOR_SPACE_SRGB_NONLINEAR_KHR });
|
|
|
|
} break;
|
|
|
|
|
2018-03-17 09:20:06 +01:00
|
|
|
case DXGI_FORMAT_R10G10B10A2_UNORM: {
|
|
|
|
formats.push_back({ VK_FORMAT_A2B10G10R10_UNORM_PACK32, VK_COLOR_SPACE_SRGB_NONLINEAR_KHR });
|
|
|
|
formats.push_back({ VK_FORMAT_A2R10G10B10_UNORM_PACK32, VK_COLOR_SPACE_SRGB_NONLINEAR_KHR });
|
|
|
|
} break;
|
|
|
|
|
|
|
|
case DXGI_FORMAT_R16G16B16A16_FLOAT: {
|
|
|
|
formats.push_back({ VK_FORMAT_R16G16B16A16_SFLOAT, VK_COLOR_SPACE_SRGB_NONLINEAR_KHR });
|
|
|
|
} break;
|
|
|
|
|
2017-12-04 22:21:02 +01:00
|
|
|
default:
|
|
|
|
Logger::warn(str::format("DxgiPresenter: Unknown format: ", fmt));
|
|
|
|
}
|
|
|
|
|
|
|
|
return m_surface->pickSurfaceFormat(
|
|
|
|
formats.size(), formats.data());
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2017-12-31 00:23:34 +01:00
|
|
|
VkPresentModeKHR DxgiPresenter::pickPresentMode(VkPresentModeKHR preferred) const {
|
|
|
|
return m_surface->pickPresentMode(1, &preferred);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2018-04-13 13:47:15 +02:00
|
|
|
void DxgiPresenter::setGammaControl(
|
|
|
|
const DXGI_VK_GAMMA_INPUT_CONTROL* pGammaControl,
|
|
|
|
const DXGI_VK_GAMMA_CURVE* pGammaCurve) {
|
2018-04-10 20:44:55 +02:00
|
|
|
m_context->beginRecording(
|
|
|
|
m_device->createCommandList());
|
|
|
|
|
2018-04-13 13:47:15 +02:00
|
|
|
m_context->updateBuffer(m_gammaUbo,
|
|
|
|
0, sizeof(DXGI_VK_GAMMA_INPUT_CONTROL),
|
|
|
|
pGammaControl);
|
|
|
|
|
|
|
|
m_context->updateImage(m_gammaTexture,
|
|
|
|
VkImageSubresourceLayers { VK_IMAGE_ASPECT_COLOR_BIT, 0, 0, 1 },
|
|
|
|
VkOffset3D { 0, 0, 0 },
|
|
|
|
VkExtent3D { DXGI_VK_GAMMA_CP_COUNT, 1, 1 },
|
|
|
|
pGammaCurve, 0, 0);
|
2018-04-10 20:44:55 +02:00
|
|
|
|
|
|
|
m_device->submitCommandList(
|
|
|
|
m_context->endRecording(),
|
|
|
|
nullptr, nullptr);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2018-04-13 13:47:15 +02:00
|
|
|
Rc<DxvkSampler> DxgiPresenter::createSampler(
|
|
|
|
VkFilter filter,
|
|
|
|
VkSamplerAddressMode addressMode) {
|
|
|
|
DxvkSamplerCreateInfo samplerInfo;
|
|
|
|
samplerInfo.magFilter = filter;
|
|
|
|
samplerInfo.minFilter = filter;
|
|
|
|
samplerInfo.mipmapMode = VK_SAMPLER_MIPMAP_MODE_NEAREST;
|
|
|
|
samplerInfo.mipmapLodBias = 0.0f;
|
|
|
|
samplerInfo.mipmapLodMin = 0.0f;
|
|
|
|
samplerInfo.mipmapLodMax = 0.0f;
|
|
|
|
samplerInfo.useAnisotropy = VK_FALSE;
|
|
|
|
samplerInfo.maxAnisotropy = 1.0f;
|
|
|
|
samplerInfo.addressModeU = addressMode;
|
|
|
|
samplerInfo.addressModeV = addressMode;
|
|
|
|
samplerInfo.addressModeW = addressMode;
|
|
|
|
samplerInfo.compareToDepth = VK_FALSE;
|
|
|
|
samplerInfo.compareOp = VK_COMPARE_OP_ALWAYS;
|
|
|
|
samplerInfo.borderColor = VK_BORDER_COLOR_FLOAT_TRANSPARENT_BLACK;
|
|
|
|
samplerInfo.usePixelCoord = VK_FALSE;
|
|
|
|
return m_device->createSampler(samplerInfo);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Rc<DxvkBuffer> DxgiPresenter::createGammaUbo() {
|
|
|
|
DxvkBufferCreateInfo info;
|
|
|
|
info.size = sizeof(DXGI_VK_GAMMA_INPUT_CONTROL);
|
|
|
|
info.usage = VK_BUFFER_USAGE_TRANSFER_DST_BIT
|
|
|
|
| VK_BUFFER_USAGE_UNIFORM_BUFFER_BIT;
|
|
|
|
info.stages = VK_PIPELINE_STAGE_TRANSFER_BIT
|
|
|
|
| VK_PIPELINE_STAGE_FRAGMENT_SHADER_BIT;
|
|
|
|
info.access = VK_ACCESS_TRANSFER_WRITE_BIT
|
|
|
|
| VK_ACCESS_SHADER_READ_BIT;
|
|
|
|
return m_device->createBuffer(info, VK_MEMORY_PROPERTY_DEVICE_LOCAL_BIT);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Rc<DxvkImage> DxgiPresenter::createGammaTexture() {
|
|
|
|
DxvkImageCreateInfo info;
|
|
|
|
info.type = VK_IMAGE_TYPE_1D;
|
|
|
|
info.format = VK_FORMAT_R16G16B16A16_UNORM;
|
|
|
|
info.flags = 0;
|
|
|
|
info.sampleCount = VK_SAMPLE_COUNT_1_BIT;
|
|
|
|
info.extent = { DXGI_VK_GAMMA_CP_COUNT, 1, 1 };
|
|
|
|
info.numLayers = 1;
|
|
|
|
info.mipLevels = 1;
|
|
|
|
info.usage = VK_IMAGE_USAGE_TRANSFER_DST_BIT
|
|
|
|
| VK_IMAGE_USAGE_SAMPLED_BIT;
|
|
|
|
info.stages = VK_PIPELINE_STAGE_TRANSFER_BIT
|
|
|
|
| VK_PIPELINE_STAGE_FRAGMENT_SHADER_BIT;
|
|
|
|
info.access = VK_ACCESS_TRANSFER_WRITE_BIT
|
|
|
|
| VK_ACCESS_SHADER_READ_BIT;
|
|
|
|
info.tiling = VK_IMAGE_TILING_OPTIMAL;
|
|
|
|
info.layout = VK_IMAGE_LAYOUT_SHADER_READ_ONLY_OPTIMAL;
|
|
|
|
return m_device->createImage(info, VK_MEMORY_PROPERTY_DEVICE_LOCAL_BIT);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Rc<DxvkImageView> DxgiPresenter::createGammaTextureView() {
|
|
|
|
DxvkImageViewCreateInfo info;
|
|
|
|
info.type = VK_IMAGE_VIEW_TYPE_1D;
|
|
|
|
info.format = VK_FORMAT_R16G16B16A16_UNORM;
|
|
|
|
info.aspect = VK_IMAGE_ASPECT_COLOR_BIT;
|
|
|
|
info.minLevel = 0;
|
|
|
|
info.numLevels = 1;
|
|
|
|
info.minLayer = 0;
|
|
|
|
info.numLayers = 1;
|
|
|
|
return m_device->createImageView(m_gammaTexture, info);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2017-11-29 21:46:09 +01:00
|
|
|
Rc<DxvkShader> DxgiPresenter::createVertexShader() {
|
2018-04-10 18:46:39 +02:00
|
|
|
const SpirvCodeBuffer codeBuffer(dxgi_presenter_vert);
|
2017-11-29 21:46:09 +01:00
|
|
|
|
|
|
|
return m_device->createShader(
|
2017-12-07 09:38:31 +01:00
|
|
|
VK_SHADER_STAGE_VERTEX_BIT,
|
2018-01-12 14:25:26 +01:00
|
|
|
0, nullptr, { 0u, 1u },
|
2018-04-10 18:46:39 +02:00
|
|
|
codeBuffer);
|
2017-11-29 21:46:09 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Rc<DxvkShader> DxgiPresenter::createFragmentShader() {
|
2018-04-10 18:46:39 +02:00
|
|
|
const SpirvCodeBuffer codeBuffer(dxgi_presenter_frag);
|
2017-11-29 21:46:09 +01:00
|
|
|
|
2017-12-07 09:38:31 +01:00
|
|
|
// Shader resource slots
|
2018-04-13 13:47:15 +02:00
|
|
|
const std::array<DxvkResourceSlot, 5> resourceSlots = {{
|
2018-04-10 20:44:55 +02:00
|
|
|
{ BindingIds::Sampler, VK_DESCRIPTOR_TYPE_SAMPLER, VK_IMAGE_VIEW_TYPE_MAX_ENUM },
|
|
|
|
{ BindingIds::Texture, VK_DESCRIPTOR_TYPE_SAMPLED_IMAGE, VK_IMAGE_VIEW_TYPE_2D },
|
2018-04-13 13:47:15 +02:00
|
|
|
{ BindingIds::GammaSmp, VK_DESCRIPTOR_TYPE_SAMPLER, VK_IMAGE_VIEW_TYPE_MAX_ENUM },
|
|
|
|
{ BindingIds::GammaTex, VK_DESCRIPTOR_TYPE_SAMPLED_IMAGE, VK_IMAGE_VIEW_TYPE_1D },
|
2018-04-10 20:44:55 +02:00
|
|
|
{ BindingIds::GammaUbo, VK_DESCRIPTOR_TYPE_UNIFORM_BUFFER, VK_IMAGE_VIEW_TYPE_MAX_ENUM },
|
2017-12-07 09:38:31 +01:00
|
|
|
}};
|
2017-11-29 21:46:09 +01:00
|
|
|
|
|
|
|
// Create the actual shader module
|
|
|
|
return m_device->createShader(
|
2017-12-07 09:38:31 +01:00
|
|
|
VK_SHADER_STAGE_FRAGMENT_BIT,
|
|
|
|
resourceSlots.size(),
|
|
|
|
resourceSlots.data(),
|
2018-01-12 14:25:26 +01:00
|
|
|
{ 1u, 1u },
|
2018-04-10 18:46:39 +02:00
|
|
|
codeBuffer);
|
2017-12-03 00:40:58 +01:00
|
|
|
}
|
|
|
|
|
2017-11-29 21:46:09 +01:00
|
|
|
}
|