2023-08-09 07:31:18 +00:00
|
|
|
#include "emulator.h"
|
2023-10-29 21:46:34 +00:00
|
|
|
|
2023-10-31 06:47:58 +00:00
|
|
|
#include <core/PS4/HLE/Graphics/graphics_render.h>
|
2023-10-20 04:25:52 +00:00
|
|
|
#include <Emulator/Host/controller.h>
|
2023-10-29 21:46:34 +00:00
|
|
|
#include <Lib/Timer.h>
|
|
|
|
#include <fmt/core.h>
|
2023-09-25 09:04:40 +00:00
|
|
|
#include <vulkan_util.h>
|
|
|
|
|
2023-10-31 06:47:58 +00:00
|
|
|
#include "core/PS4/HLE/Graphics/video_out.h"
|
2023-10-13 06:40:59 +00:00
|
|
|
#include "Emulator/HLE/Libraries/LibPad/pad.h"
|
2023-10-29 21:46:34 +00:00
|
|
|
#include "Emulator/Util/singleton.h"
|
2023-09-29 05:40:30 +00:00
|
|
|
#include "version.h"
|
2023-08-09 07:31:18 +00:00
|
|
|
|
2023-10-13 06:40:59 +00:00
|
|
|
namespace Emu {
|
2023-09-12 16:39:08 +00:00
|
|
|
|
2023-09-14 15:47:42 +00:00
|
|
|
bool m_emu_needs_exit = false;
|
2023-10-30 12:52:01 +00:00
|
|
|
bool m_game_is_paused = {false};
|
2023-10-29 21:46:34 +00:00
|
|
|
double m_current_time_seconds = {0.0};
|
|
|
|
double m_previous_time_seconds = {0.0};
|
|
|
|
int m_update_num = {0};
|
|
|
|
int m_frame_num = {0};
|
|
|
|
double m_update_time_seconds = {0.0};
|
|
|
|
double m_current_fps = {0.0};
|
|
|
|
int m_max_updates_per_frame = {4};
|
|
|
|
double m_update_fixed_time = 1.0 / 60.0;
|
|
|
|
int m_fps_frames_num = {0};
|
|
|
|
double m_fps_start_time = {0};
|
2023-09-12 16:39:08 +00:00
|
|
|
|
|
|
|
void emuInit(u32 width, u32 height) {
|
2023-10-15 07:03:26 +00:00
|
|
|
auto* window_ctx = singleton<Emu::WindowCtx>::instance();
|
2023-09-12 16:39:08 +00:00
|
|
|
|
2023-09-21 20:48:16 +00:00
|
|
|
window_ctx->m_graphic_ctx.screen_width = width;
|
|
|
|
window_ctx->m_graphic_ctx.screen_height = height;
|
2023-09-12 16:39:08 +00:00
|
|
|
}
|
|
|
|
|
2023-09-15 21:03:11 +00:00
|
|
|
void checkAndWaitForGraphicsInit() {
|
2023-10-15 07:03:26 +00:00
|
|
|
auto* window_ctx = singleton<Emu::WindowCtx>::instance();
|
2023-10-22 14:10:25 +00:00
|
|
|
std::unique_lock lock{window_ctx->m_mutex};
|
2023-09-15 21:03:11 +00:00
|
|
|
|
2023-09-21 20:48:16 +00:00
|
|
|
while (!window_ctx->m_is_graphic_initialized) {
|
2023-10-22 14:10:25 +00:00
|
|
|
window_ctx->m_graphic_initialized_cond.wait(lock);
|
2023-09-15 21:03:11 +00:00
|
|
|
}
|
|
|
|
}
|
2023-09-12 16:39:08 +00:00
|
|
|
static void CreateSdlWindow(WindowCtx* ctx) {
|
|
|
|
int width = static_cast<int>(ctx->m_graphic_ctx.screen_width);
|
|
|
|
int height = static_cast<int>(ctx->m_graphic_ctx.screen_height);
|
|
|
|
|
|
|
|
if (SDL_Init(SDL_INIT_VIDEO) < 0) {
|
2023-10-26 20:29:05 +00:00
|
|
|
fmt::print("{}\n", SDL_GetError());
|
2023-09-15 11:40:03 +00:00
|
|
|
std::exit(0);
|
2023-09-12 16:39:08 +00:00
|
|
|
}
|
2023-09-29 05:40:30 +00:00
|
|
|
std::string title = "shadps4 v" + std::string(Emulator::VERSION);
|
|
|
|
ctx->m_window = SDL_CreateWindowWithPosition(title.c_str(), SDL_WINDOWPOS_CENTERED, SDL_WINDOWPOS_CENTERED, width, height,
|
2023-09-14 15:47:42 +00:00
|
|
|
(static_cast<uint32_t>(SDL_WINDOW_HIDDEN) | static_cast<uint32_t>(SDL_WINDOW_VULKAN)));
|
2023-09-12 16:39:08 +00:00
|
|
|
|
2023-09-14 15:47:42 +00:00
|
|
|
ctx->is_window_hidden = true; // hide window until we need to show something (should draw something in buffers)
|
2023-09-12 16:39:08 +00:00
|
|
|
|
|
|
|
if (ctx->m_window == nullptr) {
|
2023-10-26 20:29:05 +00:00
|
|
|
fmt::print("{}\n", SDL_GetError());
|
2023-09-15 11:40:03 +00:00
|
|
|
std::exit(0);
|
2023-09-12 16:39:08 +00:00
|
|
|
}
|
|
|
|
|
2023-09-14 15:47:42 +00:00
|
|
|
SDL_SetWindowResizable(ctx->m_window, SDL_FALSE); // we don't support resizable atm
|
2023-09-12 16:39:08 +00:00
|
|
|
}
|
2023-10-29 21:46:34 +00:00
|
|
|
static void update() {
|
|
|
|
static double lag = 0.0;
|
|
|
|
|
|
|
|
lag += m_current_time_seconds - m_previous_time_seconds;
|
|
|
|
|
|
|
|
int num = 0;
|
|
|
|
|
|
|
|
while (lag >= m_update_fixed_time) {
|
|
|
|
if (num < m_max_updates_per_frame) {
|
|
|
|
m_update_num++;
|
|
|
|
num++;
|
|
|
|
m_update_time_seconds = m_update_num * m_update_fixed_time;
|
|
|
|
}
|
|
|
|
|
|
|
|
lag -= m_update_fixed_time;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
static void calculateFps(double game_time_s) {
|
|
|
|
m_previous_time_seconds = m_current_time_seconds;
|
|
|
|
m_current_time_seconds = game_time_s;
|
|
|
|
|
|
|
|
m_frame_num++;
|
|
|
|
|
|
|
|
m_fps_frames_num++;
|
|
|
|
if (m_current_time_seconds - m_fps_start_time > 0.25f) {
|
|
|
|
m_current_fps = static_cast<double>(m_fps_frames_num) / (m_current_time_seconds - m_fps_start_time);
|
|
|
|
m_fps_frames_num = 0;
|
|
|
|
m_fps_start_time = m_current_time_seconds;
|
|
|
|
}
|
|
|
|
}
|
2023-08-09 07:31:18 +00:00
|
|
|
void emuRun() {
|
2023-10-29 21:46:34 +00:00
|
|
|
Lib::Timer timer;
|
|
|
|
timer.Start();
|
2023-10-15 07:03:26 +00:00
|
|
|
auto* window_ctx = singleton<Emu::WindowCtx>::instance();
|
2023-09-12 16:39:08 +00:00
|
|
|
{
|
|
|
|
// init window and wait until init finishes
|
2023-10-22 14:10:25 +00:00
|
|
|
std::scoped_lock lock{window_ctx->m_mutex};
|
2023-09-21 20:48:16 +00:00
|
|
|
CreateSdlWindow(window_ctx);
|
|
|
|
Graphics::Vulkan::vulkanCreate(window_ctx);
|
|
|
|
window_ctx->m_is_graphic_initialized = true;
|
2023-10-22 14:10:25 +00:00
|
|
|
window_ctx->m_graphic_initialized_cond.notify_one();
|
2023-10-29 21:46:34 +00:00
|
|
|
calculateFps(timer.GetTimeSec());
|
2023-09-12 16:39:08 +00:00
|
|
|
}
|
2023-09-14 13:49:47 +00:00
|
|
|
|
2023-09-14 15:47:42 +00:00
|
|
|
bool exit_loop = false;
|
2023-09-14 13:49:47 +00:00
|
|
|
|
2023-09-14 15:47:42 +00:00
|
|
|
for (;;) {
|
|
|
|
if (exit_loop) {
|
|
|
|
break;
|
|
|
|
}
|
2023-09-14 13:49:47 +00:00
|
|
|
|
2023-09-14 15:47:42 +00:00
|
|
|
SDL_Event event;
|
|
|
|
if (SDL_PollEvent(&event) != 0) {
|
|
|
|
switch (event.type) {
|
|
|
|
case SDL_EVENT_QUIT: m_emu_needs_exit = true; break;
|
2023-09-12 16:39:08 +00:00
|
|
|
|
2023-09-14 15:47:42 +00:00
|
|
|
case SDL_EVENT_TERMINATING: m_emu_needs_exit = true; break;
|
|
|
|
|
|
|
|
case SDL_EVENT_WILL_ENTER_BACKGROUND: break;
|
|
|
|
|
|
|
|
case SDL_EVENT_DID_ENTER_BACKGROUND: break;
|
|
|
|
|
|
|
|
case SDL_EVENT_WILL_ENTER_FOREGROUND: break;
|
|
|
|
|
|
|
|
case SDL_EVENT_DID_ENTER_FOREGROUND: break;
|
|
|
|
|
|
|
|
case SDL_EVENT_KEY_DOWN:
|
2023-10-30 12:52:01 +00:00
|
|
|
case SDL_EVENT_KEY_UP:
|
|
|
|
if (event.type == SDL_EVENT_KEY_DOWN){
|
|
|
|
if (event.key.keysym.sym == SDLK_p) {
|
|
|
|
m_game_is_paused = !m_game_is_paused;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
keyboardEvent(&event);
|
|
|
|
break;
|
2023-09-14 15:47:42 +00:00
|
|
|
}
|
|
|
|
continue;
|
|
|
|
}
|
2023-10-30 12:52:01 +00:00
|
|
|
if (m_game_is_paused) {
|
|
|
|
if (!timer.IsPaused()) {
|
|
|
|
timer.Pause();
|
|
|
|
}
|
|
|
|
|
|
|
|
SDL_WaitEvent(&event);
|
|
|
|
|
|
|
|
switch (event.type) {
|
|
|
|
case SDL_EVENT_QUIT: m_emu_needs_exit = true; break;
|
|
|
|
|
|
|
|
case SDL_EVENT_TERMINATING: m_emu_needs_exit = true; break;
|
|
|
|
|
|
|
|
case SDL_EVENT_WILL_ENTER_BACKGROUND: break;
|
|
|
|
|
|
|
|
case SDL_EVENT_DID_ENTER_BACKGROUND: break;
|
|
|
|
|
|
|
|
case SDL_EVENT_WILL_ENTER_FOREGROUND: break;
|
|
|
|
|
|
|
|
case SDL_EVENT_DID_ENTER_FOREGROUND: break;
|
|
|
|
|
|
|
|
case SDL_EVENT_KEY_DOWN:
|
|
|
|
case SDL_EVENT_KEY_UP:
|
|
|
|
if (event.type == SDL_EVENT_KEY_DOWN) {
|
|
|
|
if (event.key.keysym.sym == SDLK_p) {
|
|
|
|
m_game_is_paused = !m_game_is_paused;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
keyboardEvent(&event);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
exit_loop = m_emu_needs_exit;
|
|
|
|
continue;
|
2023-10-29 21:46:34 +00:00
|
|
|
}
|
2023-10-30 12:52:01 +00:00
|
|
|
exit_loop = m_emu_needs_exit;
|
|
|
|
if (m_game_is_paused) {
|
|
|
|
if (!timer.IsPaused()) {
|
|
|
|
timer.Pause();
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if (timer.IsPaused()) {
|
|
|
|
timer.Resume();
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!exit_loop) {
|
|
|
|
update();
|
|
|
|
}
|
|
|
|
if (!exit_loop) {
|
|
|
|
if (HLE::Libs::Graphics::VideoOut::videoOutFlip(100000)) { // flip every 0.1 sec
|
|
|
|
calculateFps(timer.GetTimeSec());
|
|
|
|
}
|
2023-10-29 21:46:34 +00:00
|
|
|
}
|
2023-09-14 13:49:47 +00:00
|
|
|
}
|
2023-08-09 07:31:18 +00:00
|
|
|
}
|
2023-09-15 11:40:03 +00:00
|
|
|
std::exit(0);
|
2023-08-09 07:31:18 +00:00
|
|
|
}
|
2023-09-25 09:04:40 +00:00
|
|
|
|
|
|
|
HLE::Libs::Graphics::GraphicCtx* getGraphicCtx() {
|
2023-10-15 07:03:26 +00:00
|
|
|
auto* window_ctx = singleton<Emu::WindowCtx>::instance();
|
2023-10-22 14:10:25 +00:00
|
|
|
std::scoped_lock lock{window_ctx->m_mutex};
|
2023-09-25 09:04:40 +00:00
|
|
|
return &window_ctx->m_graphic_ctx;
|
|
|
|
}
|
|
|
|
|
2023-10-29 21:46:34 +00:00
|
|
|
void updateSDLTitle() {
|
|
|
|
char title[256];
|
|
|
|
sprintf(title, "shadps4 v %s FPS: %f", Emulator::VERSION, m_current_fps);
|
|
|
|
auto* window_ctx = singleton<Emu::WindowCtx>::instance();
|
|
|
|
SDL_SetWindowTitle(window_ctx->m_window, title);
|
|
|
|
}
|
2023-09-27 12:15:18 +00:00
|
|
|
void DrawBuffer(HLE::Libs::Graphics::VideoOutVulkanImage* image) {
|
2023-10-15 07:03:26 +00:00
|
|
|
auto* window_ctx = singleton<Emu::WindowCtx>::instance();
|
2023-09-27 12:15:18 +00:00
|
|
|
if (window_ctx->is_window_hidden) {
|
|
|
|
SDL_ShowWindow(window_ctx->m_window);
|
|
|
|
window_ctx->is_window_hidden = false;
|
|
|
|
}
|
|
|
|
|
2023-10-26 20:38:37 +00:00
|
|
|
window_ctx->swapchain.current_index = static_cast<u32>(-1);
|
2023-09-27 12:15:18 +00:00
|
|
|
|
2023-10-26 20:38:37 +00:00
|
|
|
auto result = vkAcquireNextImageKHR(window_ctx->m_graphic_ctx.m_device, window_ctx->swapchain.swapchain, UINT64_MAX, nullptr,
|
|
|
|
window_ctx->swapchain.present_complete_fence, &window_ctx->swapchain.current_index);
|
2023-09-27 12:15:18 +00:00
|
|
|
|
|
|
|
if (result != VK_SUCCESS) {
|
2023-10-26 20:29:05 +00:00
|
|
|
fmt::print("Can't aquireNextImage\n");
|
2023-09-27 12:15:18 +00:00
|
|
|
std::exit(0);
|
|
|
|
}
|
2023-10-26 20:38:37 +00:00
|
|
|
if (window_ctx->swapchain.current_index == static_cast<u32>(-1)) {
|
2023-10-26 20:29:05 +00:00
|
|
|
fmt::print("Unsupported:swapchain current index is -1\n");
|
2023-09-27 12:15:18 +00:00
|
|
|
std::exit(0);
|
|
|
|
}
|
|
|
|
|
|
|
|
do {
|
2023-10-26 20:38:37 +00:00
|
|
|
result = vkWaitForFences(window_ctx->m_graphic_ctx.m_device, 1, &window_ctx->swapchain.present_complete_fence, VK_TRUE, 100000000);
|
2023-09-27 12:15:18 +00:00
|
|
|
} while (result == VK_TIMEOUT);
|
|
|
|
if (result != VK_SUCCESS) {
|
2023-10-26 20:29:05 +00:00
|
|
|
fmt::print("vkWaitForFences is not success\n");
|
2023-09-27 12:15:18 +00:00
|
|
|
std::exit(0);
|
|
|
|
}
|
|
|
|
|
2023-10-26 20:38:37 +00:00
|
|
|
vkResetFences(window_ctx->m_graphic_ctx.m_device, 1, &window_ctx->swapchain.present_complete_fence);
|
2023-09-27 12:15:18 +00:00
|
|
|
|
2023-10-26 20:38:37 +00:00
|
|
|
auto blt_src_image = image;
|
|
|
|
auto blt_dst_image = window_ctx->swapchain;
|
2023-09-27 12:15:18 +00:00
|
|
|
|
|
|
|
if (blt_src_image == nullptr) {
|
2023-10-26 20:29:05 +00:00
|
|
|
fmt::print("blt_src_image is null\n");
|
2023-09-27 12:15:18 +00:00
|
|
|
std::exit(0);
|
|
|
|
}
|
|
|
|
|
|
|
|
GPU::CommandBuffer buffer(10);
|
|
|
|
|
|
|
|
auto* vk_buffer = buffer.getPool()->buffers[buffer.getIndex()];
|
|
|
|
|
|
|
|
buffer.begin();
|
|
|
|
|
2023-10-26 20:38:37 +00:00
|
|
|
Graphics::Vulkan::vulkanBlitImage(&buffer, blt_src_image, &blt_dst_image);
|
2023-09-27 12:15:18 +00:00
|
|
|
|
|
|
|
VkImageMemoryBarrier pre_present_barrier{};
|
|
|
|
pre_present_barrier.sType = VK_STRUCTURE_TYPE_IMAGE_MEMORY_BARRIER;
|
|
|
|
pre_present_barrier.pNext = nullptr;
|
|
|
|
pre_present_barrier.srcAccessMask = VK_ACCESS_TRANSFER_WRITE_BIT;
|
|
|
|
pre_present_barrier.dstAccessMask = VK_ACCESS_MEMORY_READ_BIT;
|
|
|
|
pre_present_barrier.oldLayout = VK_IMAGE_LAYOUT_TRANSFER_DST_OPTIMAL;
|
|
|
|
pre_present_barrier.newLayout = VK_IMAGE_LAYOUT_PRESENT_SRC_KHR;
|
|
|
|
pre_present_barrier.srcQueueFamilyIndex = VK_QUEUE_FAMILY_IGNORED;
|
|
|
|
pre_present_barrier.dstQueueFamilyIndex = VK_QUEUE_FAMILY_IGNORED;
|
|
|
|
pre_present_barrier.subresourceRange.aspectMask = VK_IMAGE_ASPECT_COLOR_BIT;
|
|
|
|
pre_present_barrier.subresourceRange.baseMipLevel = 0;
|
|
|
|
pre_present_barrier.subresourceRange.levelCount = 1;
|
|
|
|
pre_present_barrier.subresourceRange.baseArrayLayer = 0;
|
|
|
|
pre_present_barrier.subresourceRange.layerCount = 1;
|
2023-10-26 20:38:37 +00:00
|
|
|
pre_present_barrier.image = window_ctx->swapchain.swapchain_images[window_ctx->swapchain.current_index];
|
2023-09-27 12:15:18 +00:00
|
|
|
vkCmdPipelineBarrier(vk_buffer, VK_PIPELINE_STAGE_ALL_COMMANDS_BIT, VK_PIPELINE_STAGE_BOTTOM_OF_PIPE_BIT, 0, 0, nullptr, 0, nullptr, 1,
|
|
|
|
&pre_present_barrier);
|
|
|
|
|
|
|
|
buffer.end();
|
|
|
|
buffer.executeWithSemaphore();
|
|
|
|
|
|
|
|
VkPresentInfoKHR present{};
|
|
|
|
present.sType = VK_STRUCTURE_TYPE_PRESENT_INFO_KHR;
|
|
|
|
present.pNext = nullptr;
|
|
|
|
present.swapchainCount = 1;
|
2023-10-26 20:38:37 +00:00
|
|
|
present.pSwapchains = &window_ctx->swapchain.swapchain;
|
|
|
|
present.pImageIndices = &window_ctx->swapchain.current_index;
|
2023-09-27 12:15:18 +00:00
|
|
|
present.pWaitSemaphores = &buffer.getPool()->semaphores[buffer.getIndex()];
|
|
|
|
present.waitSemaphoreCount = 1;
|
|
|
|
present.pResults = nullptr;
|
|
|
|
|
|
|
|
const auto& queue = window_ctx->m_graphic_ctx.queues[10];
|
|
|
|
|
|
|
|
if (queue.mutex != nullptr) {
|
2023-10-26 20:29:05 +00:00
|
|
|
fmt::print("queue.mutexe is null\n");
|
2023-09-27 12:15:18 +00:00
|
|
|
std::exit(0);
|
|
|
|
}
|
|
|
|
|
|
|
|
result = vkQueuePresentKHR(queue.vk_queue, &present);
|
|
|
|
if (result != VK_SUCCESS) {
|
2023-10-26 20:29:05 +00:00
|
|
|
fmt::print("vkQueuePresentKHR failed\n");
|
2023-09-27 12:15:18 +00:00
|
|
|
std::exit(0);
|
|
|
|
}
|
2023-10-29 21:46:34 +00:00
|
|
|
updateSDLTitle();
|
2023-09-27 12:15:18 +00:00
|
|
|
}
|
2023-09-27 06:26:50 +00:00
|
|
|
|
2023-10-13 06:40:59 +00:00
|
|
|
void keyboardEvent(SDL_Event* event) {
|
|
|
|
using Emulator::HLE::Libraries::LibPad::ScePadButton;
|
|
|
|
|
|
|
|
if (event->type == SDL_EVENT_KEY_DOWN || event->type == SDL_EVENT_KEY_UP) {
|
|
|
|
u32 button = 0;
|
|
|
|
switch (event->key.keysym.sym) {
|
|
|
|
case SDLK_UP: button = ScePadButton::SCE_PAD_BUTTON_UP; break;
|
|
|
|
case SDLK_DOWN: button = ScePadButton::SCE_PAD_BUTTON_DOWN; break;
|
|
|
|
case SDLK_LEFT: button = ScePadButton::SCE_PAD_BUTTON_LEFT; break;
|
|
|
|
case SDLK_RIGHT: button = ScePadButton::SCE_PAD_BUTTON_RIGHT; break;
|
2023-10-13 17:41:03 +00:00
|
|
|
case SDLK_KP_8: button = ScePadButton ::SCE_PAD_BUTTON_TRIANGLE; break;
|
|
|
|
case SDLK_KP_6: button = ScePadButton ::SCE_PAD_BUTTON_CIRCLE; break;
|
|
|
|
case SDLK_KP_2: button = ScePadButton ::SCE_PAD_BUTTON_CROSS; break;
|
|
|
|
case SDLK_KP_4: button = ScePadButton ::SCE_PAD_BUTTON_SQUARE; break;
|
|
|
|
case SDLK_RETURN: button = ScePadButton ::SCE_PAD_BUTTON_OPTIONS; break;
|
|
|
|
|
2023-10-13 06:40:59 +00:00
|
|
|
default: break;
|
|
|
|
}
|
|
|
|
if (button != 0) {
|
2023-10-15 07:03:26 +00:00
|
|
|
auto* controller = singleton<Emulator::Host::Controller::GameController>::instance();
|
2023-10-13 06:40:59 +00:00
|
|
|
controller->checKButton(0, button, event->type == SDL_EVENT_KEY_DOWN);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-10-26 20:38:37 +00:00
|
|
|
} // namespace Emu
|