snes9x/win32/CVulkan.cpp

291 lines
8.2 KiB
C++
Raw Permalink Normal View History

2023-01-30 23:54:23 +00:00
#include "CVulkan.h"
2023-01-31 23:18:07 +00:00
#include "snes9x.h"
#include <win32_display.h>
#include <gfx.h>
2023-01-30 23:54:23 +00:00
2023-01-31 23:18:07 +00:00
#include "../filter/hq2x.h"
#include "../filter/2xsai.h"
2023-01-30 23:54:23 +00:00
#include "snes9x_imgui.h"
#include "imgui_impl_vulkan.h"
bool CVulkan::InitImGui()
{
auto defaults = S9xImGuiGetDefaults();
defaults.font_size = GUI.OSDSize;
2023-05-08 17:02:50 +00:00
defaults.spacing = (int)(defaults.font_size / 2.4);
S9xImGuiInit(&defaults);
ImGui_ImplVulkan_LoadFunctions([](const char* function, void* instance) {
return VULKAN_HPP_DEFAULT_DISPATCHER.vkGetInstanceProcAddr(*((VkInstance*)instance), function);
}, &context->instance.get());
vk::DescriptorPoolSize pool_sizes[] =
{
{ vk::DescriptorType::eCombinedImageSampler, 1000 },
{ vk::DescriptorType::eUniformBuffer, 1000 }
};
auto descriptor_pool_create_info = vk::DescriptorPoolCreateInfo{}
.setPoolSizes(pool_sizes)
.setMaxSets(1000)
.setFlags(vk::DescriptorPoolCreateFlagBits::eFreeDescriptorSet);
2024-04-12 23:47:16 +00:00
imgui_descriptor_pool = context->device.createDescriptorPoolUnique(descriptor_pool_create_info).value;
ImGui_ImplVulkan_InitInfo init_info{};
init_info.Instance = context->instance.get();
init_info.PhysicalDevice = context->physical_device;
init_info.Device = context->device;;
init_info.QueueFamily = context->graphics_queue_family_index;
init_info.Queue = context->queue;
init_info.DescriptorPool = (VkDescriptorPool)imgui_descriptor_pool.get();
init_info.Subpass = 0;
init_info.MinImageCount = context->swapchain->get_num_frames();
init_info.ImageCount = context->swapchain->get_num_frames();
init_info.MSAASamples = VK_SAMPLE_COUNT_1_BIT;
ImGui_ImplVulkan_Init(&init_info, (VkRenderPass)context->swapchain->get_render_pass());
auto cmd = context->begin_cmd_buffer();
ImGui_ImplVulkan_CreateFontsTexture(cmd);
context->end_cmd_buffer();
context->wait_idle();
return true;
}
2023-01-30 23:54:23 +00:00
bool CVulkan::Initialize(HWND hWnd)
{
this->hWnd = hWnd;
RECT window_rect;
GetWindowRect(hWnd, &window_rect);
current_width = window_rect.right - window_rect.left;
current_height = window_rect.bottom - window_rect.top;
try {
context = std::make_unique<Vulkan::Context>();
2024-10-03 16:27:07 +00:00
if (!context->init_win32())
return false;
if (!context->create_win32_surface(0, hWnd))
return false;
if (!context->create_swapchain())
return false;
}
catch (std::exception& e)
{
2023-01-31 23:18:07 +00:00
return false;
}
2023-01-31 23:18:07 +00:00
context->swapchain->set_vsync(GUI.Vsync);
if (!Settings.AutoDisplayMessages)
{
Settings.DisplayIndicators = true;
InitImGui();
}
2023-02-06 17:47:56 +00:00
if (GUI.shaderEnabled && GUI.OGLshaderFileName && GUI.OGLshaderFileName[0])
2023-01-31 23:38:02 +00:00
{
shaderchain = std::make_unique<Vulkan::ShaderChain>(context.get());
std::string shaderstring = std::string(_tToChar(GUI.OGLshaderFileName));
2023-02-01 20:47:42 +00:00
if (!shaderchain->load_shader_preset(shaderstring))
2023-01-31 23:38:02 +00:00
{
2023-02-06 17:47:56 +00:00
shaderchain.reset();
}
else
{
current_shadername = shaderstring;
return true;
2023-01-31 23:38:02 +00:00
}
}
2023-02-10 22:03:05 +00:00
simple_output = std::make_unique<Vulkan::SimpleOutput>(context.get(), vk::Format::eR5G6B5UnormPack16);
2023-01-30 23:54:23 +00:00
return true;
}
void CVulkan::DeInitialize()
{
2023-02-01 20:47:42 +00:00
current_shadername = "";
2023-02-06 17:47:56 +00:00
if (context)
{
2023-02-06 17:47:56 +00:00
context->wait_idle();
if (ImGui::GetCurrentContext())
{
ImGui_ImplVulkan_Shutdown();
imgui_descriptor_pool.reset();
S9xImGuiDeinit();
}
}
2023-01-31 23:38:02 +00:00
shaderchain.reset();
2023-02-10 22:03:05 +00:00
simple_output.reset();
2023-01-31 23:18:07 +00:00
filtered_image.clear();
context.reset();
2023-01-30 23:54:23 +00:00
}
void CVulkan::Render(SSurface Src)
{
2023-01-31 23:18:07 +00:00
if (!context)
return;
2023-02-01 20:47:42 +00:00
SSurface Dst{};
2023-01-31 23:18:07 +00:00
RECT dstRect = GetFilterOutputSize(Src);
Dst.Width = dstRect.right - dstRect.left;
Dst.Height = dstRect.bottom - dstRect.top;
Dst.Pitch = Dst.Width << 1;
size_t requiredSize = Dst.Width * Dst.Height;
if (filtered_image.size() < requiredSize)
filtered_image.resize(requiredSize * 2);
Dst.Surface = (unsigned char *)filtered_image.data();
RenderMethod(Src, Dst, &dstRect);
RECT windowSize, displayRect;
GetClientRect(hWnd, &windowSize);
//Get maximum rect respecting AR setting
displayRect = CalculateDisplayRect(Dst.Width, Dst.Height, windowSize.right, windowSize.bottom);
bool result;
if (S9xImGuiDraw(windowSize.right, windowSize.bottom))
{
ImDrawData* draw_data = ImGui::GetDrawData();
context->swapchain->on_render_pass_end([&, draw_data] {
ImGui_ImplVulkan_RenderDrawData(draw_data, context->swapchain->get_cmd());
});
}
2023-01-31 23:38:02 +00:00
if (shaderchain)
{
result = shaderchain->do_frame_without_swap(Dst.Surface, Dst.Width, Dst.Height, Dst.Pitch, vk::Format::eR5G6B5UnormPack16, displayRect.left, displayRect.top, displayRect.right - displayRect.left, displayRect.bottom - displayRect.top);
}
else if (simple_output)
{
simple_output->set_filter(Settings.BilinearFilter);
result = simple_output->do_frame_without_swap(Dst.Surface, Dst.Width, Dst.Height, Dst.Pitch, displayRect.left, displayRect.top, displayRect.right - displayRect.left, displayRect.bottom - displayRect.top);
2023-01-31 23:38:02 +00:00
}
WinThrottleFramerate();
if (result)
{
context->swapchain->swap();
if (GUI.ReduceInputLag)
context->wait_idle();
}
2023-01-30 23:54:23 +00:00
}
bool CVulkan::ChangeRenderSize(unsigned int newWidth, unsigned int newHeight)
{
2023-01-31 23:18:07 +00:00
if (!context)
return false;
2024-06-13 21:50:12 +00:00
if (newWidth != current_width || newHeight != current_height)
2023-01-31 23:18:07 +00:00
{
2024-10-03 16:27:07 +00:00
context->recreate_swapchain();
2023-01-31 23:18:07 +00:00
context->wait_idle();
current_width = newWidth;
current_height = newHeight;
}
return true;
2023-01-30 23:54:23 +00:00
}
bool CVulkan::ApplyDisplayChanges(void)
{
2023-02-01 20:47:42 +00:00
if ((!GUI.shaderEnabled && shaderchain) || (GUI.shaderEnabled && !shaderchain))
{
DeInitialize();
Initialize(hWnd);
return true;
}
std::string shadername = std::string(_tToChar(GUI.OGLshaderFileName));
if (GUI.shaderEnabled && shaderchain && (shadername != current_shadername))
{
shaderchain.reset();
shaderchain = std::make_unique<Vulkan::ShaderChain>(context.get());
if (!shaderchain->load_shader_preset(shadername))
{
DeInitialize();
GUI.shaderEnabled = false;
Initialize(hWnd);
return false;
}
current_shadername = shadername;
}
context->swapchain->set_vsync(GUI.Vsync);
2023-02-01 20:47:42 +00:00
return true;
2023-01-30 23:54:23 +00:00
}
bool CVulkan::SetFullscreen(bool fullscreen)
{
return false;
}
void CVulkan::SetSnes9xColorFormat()
{
2023-01-31 23:18:07 +00:00
GUI.ScreenDepth = 16;
GUI.BlueShift = 0;
GUI.GreenShift = 6;
GUI.RedShift = 11;
S9xBlit2xSaIFilterInit();
S9xBlitHQ2xFilterInit();
GUI.NeedDepthConvert = FALSE;
GUI.DepthConverted = TRUE;
2023-01-30 23:54:23 +00:00
}
void CVulkan::EnumModes(std::vector<dMode>* modeVector)
{
2023-01-31 23:18:07 +00:00
DISPLAY_DEVICE dd;
dd.cb = sizeof(dd);
DWORD dev = 0;
int iMode = 0;
dMode mode;
while (EnumDisplayDevices(0, dev, &dd, 0))
{
if (!(dd.StateFlags & DISPLAY_DEVICE_MIRRORING_DRIVER) && (dd.StateFlags & DISPLAY_DEVICE_PRIMARY_DEVICE))
{
DEVMODE dm;
memset(&dm, 0, sizeof(dm));
dm.dmSize = sizeof(dm);
iMode = 0;
while (EnumDisplaySettings(dd.DeviceName, iMode, &dm)) {
if (dm.dmBitsPerPel >= 32) {
mode.width = dm.dmPelsWidth;
mode.height = dm.dmPelsHeight;
mode.rate = dm.dmDisplayFrequency;
mode.depth = dm.dmBitsPerPel;
modeVector->push_back(mode);
}
iMode++;
}
}
dev++;
}
2023-01-30 23:54:23 +00:00
}
2023-01-31 23:18:07 +00:00
std::vector<ShaderParam> *CVulkan::GetShaderParameters()
2023-02-01 20:47:42 +00:00
{
if (shaderchain)
{
// SlangShader::Parameter currently equal ShaderParam, so no conversion is neccessary
return (std::vector<ShaderParam>*)&shaderchain->preset->parameters;
2023-02-01 20:47:42 +00:00
}
else
return nullptr;
}
std::function<void(const char *)> CVulkan::GetShaderParametersSaveFunction()
{
return [&](const char *filename) {
if (shaderchain)
shaderchain->preset->save_to_file(filename);
};
2024-04-12 23:47:16 +00:00
}