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
|
|
|
|
|
|
|
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;
|
|
|
|
|
|
|
|
context = std::make_unique<Vulkan::Context>();
|
2023-01-31 23:18:07 +00:00
|
|
|
if (!context->init_win32(0, hWnd))
|
|
|
|
return false;
|
|
|
|
|
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());
|
2023-02-01 20:47:42 +00:00
|
|
|
std::string shaderstring = _tToChar(GUI.OGLshaderFileName);
|
|
|
|
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)
|
|
|
|
context->wait_idle();
|
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;
|
|
|
|
|
|
|
|
if (GUI.ReduceInputLag)
|
2023-02-10 22:03:05 +00:00
|
|
|
context->wait_idle();
|
2023-01-31 23:18:07 +00:00
|
|
|
|
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);
|
|
|
|
|
|
|
|
if (!Settings.AutoDisplayMessages) {
|
|
|
|
WinSetCustomDisplaySurface((void*)Dst.Surface, Dst.Pitch / 2, dstRect.right - dstRect.left, dstRect.bottom - dstRect.top, GetFilterScale(GUI.Scale));
|
|
|
|
S9xDisplayMessages((uint16*)Dst.Surface, Dst.Pitch / 2, dstRect.right - dstRect.left, dstRect.bottom - dstRect.top, GetFilterScale(GUI.Scale));
|
|
|
|
}
|
|
|
|
|
|
|
|
RECT windowSize, displayRect;
|
|
|
|
GetClientRect(hWnd, &windowSize);
|
|
|
|
//Get maximum rect respecting AR setting
|
|
|
|
displayRect = CalculateDisplayRect(Dst.Width, Dst.Height, windowSize.right, windowSize.bottom);
|
|
|
|
|
2023-01-31 23:38:02 +00:00
|
|
|
if (shaderchain)
|
|
|
|
{
|
|
|
|
shaderchain->do_frame(Dst.Surface, Dst.Width, Dst.Height, Dst.Pitch, vk::Format::eR5G6B5UnormPack16, displayRect.left, displayRect.top, displayRect.right - displayRect.left, displayRect.bottom - displayRect.top);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2023-02-10 22:03:05 +00:00
|
|
|
simple_output->set_filter(Settings.BilinearFilter);
|
|
|
|
simple_output->do_frame(Dst.Surface, Dst.Width, Dst.Height, Dst.Pitch, displayRect.left, displayRect.top, displayRect.right - displayRect.left, displayRect.bottom - displayRect.top);
|
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;
|
|
|
|
|
|
|
|
bool vsync_changed = context->swapchain->set_vsync(GUI.Vsync);
|
|
|
|
|
|
|
|
if (newWidth != current_width || newHeight != current_height || vsync_changed)
|
|
|
|
{
|
|
|
|
context->recreate_swapchain(newWidth, newHeight);
|
|
|
|
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;
|
|
|
|
}
|
|
|
|
|
2023-02-10 22:03:05 +00:00
|
|
|
if (context->swapchain->set_vsync(GUI.Vsync))
|
2023-02-01 20:47:42 +00:00
|
|
|
{
|
2023-02-10 22:03:05 +00:00
|
|
|
context->recreate_swapchain();
|
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
|
|
|
|
2023-02-01 20:47:42 +00:00
|
|
|
std::vector<SlangShader::Parameter> *CVulkan::GetShaderParameters()
|
|
|
|
{
|
|
|
|
if (shaderchain)
|
|
|
|
{
|
|
|
|
return &shaderchain->preset->parameters;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
std::function<void(const char *)> CVulkan::GetShaderParametersSaveFunction()
|
|
|
|
{
|
|
|
|
return [&](const char *filename) {
|
|
|
|
if (shaderchain)
|
|
|
|
shaderchain->preset->save_to_file(filename);
|
|
|
|
};
|
2023-02-10 22:03:05 +00:00
|
|
|
}
|