2024-06-10 12:18:42 +00:00
|
|
|
// SPDX-FileCopyrightText: Copyright 2024 shadPS4 Emulator Project
|
|
|
|
// SPDX-License-Identifier: GPL-2.0-or-later
|
|
|
|
|
2024-07-31 10:25:55 +00:00
|
|
|
#include <fmt/core.h>
|
|
|
|
|
2024-06-10 12:18:42 +00:00
|
|
|
#include "common/config.h"
|
2024-06-11 10:14:33 +00:00
|
|
|
#include "common/debug.h"
|
2024-06-10 12:18:42 +00:00
|
|
|
#include "common/logging/backend.h"
|
2024-07-28 13:54:09 +00:00
|
|
|
#include "common/logging/log.h"
|
2024-07-11 12:35:58 +00:00
|
|
|
#include "common/ntapi.h"
|
2024-06-10 12:18:42 +00:00
|
|
|
#include "common/path_util.h"
|
2024-07-09 09:18:34 +00:00
|
|
|
#include "common/polyfill_thread.h"
|
2024-08-15 09:33:10 +00:00
|
|
|
#include "common/scm_rev.h"
|
2024-06-10 12:18:42 +00:00
|
|
|
#include "common/singleton.h"
|
2024-07-04 11:30:34 +00:00
|
|
|
#include "common/version.h"
|
2024-07-31 10:25:55 +00:00
|
|
|
#include "core/file_format/playgo_chunk.h"
|
2024-07-28 13:54:09 +00:00
|
|
|
#include "core/file_format/psf.h"
|
|
|
|
#include "core/file_format/splash.h"
|
2024-06-10 12:18:42 +00:00
|
|
|
#include "core/file_sys/fs.h"
|
2024-07-28 13:54:09 +00:00
|
|
|
#include "core/libraries/disc_map/disc_map.h"
|
2024-06-10 12:18:42 +00:00
|
|
|
#include "core/libraries/kernel/thread_management.h"
|
2024-07-28 13:54:09 +00:00
|
|
|
#include "core/libraries/libc/libc.h"
|
|
|
|
#include "core/libraries/libc_internal/libc_internal.h"
|
2024-06-10 12:18:42 +00:00
|
|
|
#include "core/libraries/libs.h"
|
2024-07-28 13:54:09 +00:00
|
|
|
#include "core/libraries/rtc/rtc.h"
|
2024-07-31 10:25:55 +00:00
|
|
|
#include "core/libraries/videoout/video_out.h"
|
2024-06-10 12:18:42 +00:00
|
|
|
#include "core/linker.h"
|
|
|
|
#include "core/memory.h"
|
|
|
|
#include "emulator.h"
|
2024-07-31 22:11:58 +00:00
|
|
|
#include "video_core/renderdoc.h"
|
2024-07-28 13:54:09 +00:00
|
|
|
|
2024-06-10 12:18:42 +00:00
|
|
|
Frontend::WindowSDL* g_window = nullptr;
|
|
|
|
|
|
|
|
namespace Core {
|
|
|
|
|
|
|
|
static constexpr s32 WindowWidth = 1280;
|
|
|
|
static constexpr s32 WindowHeight = 720;
|
|
|
|
|
2024-07-15 21:22:47 +00:00
|
|
|
Emulator::Emulator() {
|
2024-06-10 12:18:42 +00:00
|
|
|
// Read configuration file.
|
|
|
|
const auto config_dir = Common::FS::GetUserPath(Common::FS::PathType::UserDir);
|
|
|
|
Config::load(config_dir / "config.toml");
|
|
|
|
|
2024-07-11 12:35:58 +00:00
|
|
|
// Initialize NT API functions
|
2024-07-11 13:29:36 +00:00
|
|
|
#ifdef _WIN32
|
2024-07-11 12:35:58 +00:00
|
|
|
Common::NtApi::Initialize();
|
2024-07-11 13:29:36 +00:00
|
|
|
#endif
|
2024-07-11 12:35:58 +00:00
|
|
|
|
2024-06-10 12:18:42 +00:00
|
|
|
// Start logger.
|
|
|
|
Common::Log::Initialize();
|
|
|
|
Common::Log::Start();
|
2024-07-04 11:30:34 +00:00
|
|
|
LOG_INFO(Loader, "Starting shadps4 emulator v{} ", Common::VERSION);
|
2024-07-31 22:11:58 +00:00
|
|
|
LOG_INFO(Loader, "Revision {}", Common::g_scm_rev);
|
|
|
|
LOG_INFO(Loader, "Branch {}", Common::g_scm_branch);
|
|
|
|
LOG_INFO(Loader, "Description {}", Common::g_scm_desc);
|
2024-07-15 21:22:47 +00:00
|
|
|
|
|
|
|
// Defer until after logging is initialized.
|
|
|
|
memory = Core::Memory::Instance();
|
|
|
|
controller = Common::Singleton<Input::GameController>::Instance();
|
|
|
|
linker = Common::Singleton<Core::Linker>::Instance();
|
2024-07-28 13:54:09 +00:00
|
|
|
|
|
|
|
// Load renderdoc module.
|
|
|
|
VideoCore::LoadRenderDoc();
|
2024-06-10 12:18:42 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
Emulator::~Emulator() {
|
|
|
|
const auto config_dir = Common::FS::GetUserPath(Common::FS::PathType::UserDir);
|
|
|
|
Config::save(config_dir / "config.toml");
|
|
|
|
}
|
|
|
|
|
|
|
|
void Emulator::Run(const std::filesystem::path& file) {
|
|
|
|
// Applications expect to be run from /app0 so mount the file's parent path as app0.
|
|
|
|
auto* mnt = Common::Singleton<Core::FileSys::MntPoints>::Instance();
|
|
|
|
mnt->Mount(file.parent_path(), "/app0");
|
|
|
|
|
|
|
|
// Loading param.sfo file if exists
|
2024-06-15 14:51:51 +00:00
|
|
|
std::string id;
|
2024-07-24 06:12:53 +00:00
|
|
|
std::string title;
|
|
|
|
std::string app_version;
|
2024-06-10 12:18:42 +00:00
|
|
|
std::filesystem::path sce_sys_folder = file.parent_path() / "sce_sys";
|
|
|
|
if (std::filesystem::is_directory(sce_sys_folder)) {
|
|
|
|
for (const auto& entry : std::filesystem::directory_iterator(sce_sys_folder)) {
|
|
|
|
if (entry.path().filename() == "param.sfo") {
|
|
|
|
auto* param_sfo = Common::Singleton<PSF>::Instance();
|
2024-06-11 02:42:21 +00:00
|
|
|
param_sfo->open(sce_sys_folder.string() + "/param.sfo", {});
|
2024-06-15 14:51:51 +00:00
|
|
|
id = std::string(param_sfo->GetString("CONTENT_ID"), 7, 9);
|
2024-07-24 06:12:53 +00:00
|
|
|
title = param_sfo->GetString("TITLE");
|
2024-06-10 12:18:42 +00:00
|
|
|
LOG_INFO(Loader, "Game id: {} Title: {}", id, title);
|
|
|
|
u32 fw_version = param_sfo->GetInteger("SYSTEM_VER");
|
2024-07-24 06:12:53 +00:00
|
|
|
app_version = param_sfo->GetString("APP_VER");
|
2024-06-10 12:18:42 +00:00
|
|
|
LOG_INFO(Loader, "Fw: {:#x} App Version: {}", fw_version, app_version);
|
2024-07-24 16:02:38 +00:00
|
|
|
} else if (entry.path().filename() == "playgo-chunk.dat") {
|
|
|
|
auto* playgo = Common::Singleton<PlaygoChunk>::Instance();
|
|
|
|
playgo->Open(sce_sys_folder.string() + "/playgo-chunk.dat");
|
2024-06-10 12:18:42 +00:00
|
|
|
} else if (entry.path().filename() == "pic0.png" ||
|
|
|
|
entry.path().filename() == "pic1.png") {
|
|
|
|
auto* splash = Common::Singleton<Splash>::Instance();
|
|
|
|
if (splash->IsLoaded()) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
if (!splash->Open(entry.path().string())) {
|
|
|
|
LOG_ERROR(Loader, "Game splash: unable to open file");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2024-07-31 22:11:58 +00:00
|
|
|
|
2024-07-25 07:13:14 +00:00
|
|
|
std::string game_title = fmt::format("{} - {} <{}>", id, title, app_version);
|
2024-07-31 22:11:58 +00:00
|
|
|
std::string window_title = "";
|
|
|
|
if (Common::isRelease) {
|
|
|
|
window_title = fmt::format("shadPS4 v{} | {}", Common::VERSION, game_title);
|
|
|
|
} else {
|
|
|
|
window_title =
|
|
|
|
fmt::format("shadPS4 v{} {} | {}", Common::VERSION, Common::g_scm_desc, game_title);
|
|
|
|
}
|
2024-07-24 06:12:53 +00:00
|
|
|
window =
|
2024-07-31 21:56:10 +00:00
|
|
|
std::make_unique<Frontend::WindowSDL>(WindowWidth, WindowHeight, controller, window_title);
|
2024-07-24 06:12:53 +00:00
|
|
|
|
|
|
|
g_window = window.get();
|
2024-06-10 12:18:42 +00:00
|
|
|
|
2024-06-15 14:51:51 +00:00
|
|
|
const auto& mount_data_dir = Common::FS::GetUserPath(Common::FS::PathType::GameDataDir) / id;
|
|
|
|
if (!std::filesystem::exists(mount_data_dir)) {
|
|
|
|
std::filesystem::create_directory(mount_data_dir);
|
|
|
|
}
|
|
|
|
mnt->Mount(mount_data_dir, "/data"); // should just exist, manually create with game serial
|
|
|
|
const auto& mount_temp_dir = Common::FS::GetUserPath(Common::FS::PathType::TempDataDir) / id;
|
2024-07-04 07:51:46 +00:00
|
|
|
if (!std::filesystem::exists(mount_temp_dir)) {
|
|
|
|
std::filesystem::create_directory(mount_temp_dir);
|
|
|
|
}
|
2024-06-15 14:51:51 +00:00
|
|
|
mnt->Mount(mount_temp_dir, "/temp0"); // called in app_content ==> stat/mkdir
|
2024-08-09 09:56:03 +00:00
|
|
|
mnt->Mount(mount_temp_dir, "/temp");
|
2024-06-15 14:51:51 +00:00
|
|
|
|
2024-07-26 14:07:22 +00:00
|
|
|
const auto& mount_download_dir =
|
|
|
|
Common::FS::GetUserPath(Common::FS::PathType::DownloadDir) / id;
|
|
|
|
if (!std::filesystem::exists(mount_download_dir)) {
|
|
|
|
std::filesystem::create_directory(mount_download_dir);
|
|
|
|
}
|
|
|
|
mnt->Mount(mount_download_dir, "/download0");
|
|
|
|
|
2024-07-28 13:54:09 +00:00
|
|
|
const auto& mount_captures_dir = Common::FS::GetUserPath(Common::FS::PathType::CapturesDir);
|
|
|
|
if (!std::filesystem::exists(mount_captures_dir)) {
|
|
|
|
std::filesystem::create_directory(mount_captures_dir);
|
|
|
|
}
|
|
|
|
VideoCore::SetOutputDir(mount_captures_dir.generic_string(), id);
|
|
|
|
|
2024-06-30 16:22:39 +00:00
|
|
|
// Initialize kernel and library facilities.
|
|
|
|
Libraries::Kernel::init_pthreads();
|
|
|
|
Libraries::InitHLELibs(&linker->GetHLESymbols());
|
|
|
|
|
2024-06-10 12:18:42 +00:00
|
|
|
// Load the module with the linker
|
|
|
|
linker->LoadModule(file);
|
|
|
|
|
|
|
|
// check if we have system modules to load
|
2024-06-10 15:26:37 +00:00
|
|
|
LoadSystemModules(file);
|
|
|
|
|
2024-06-10 12:18:42 +00:00
|
|
|
// Check if there is a libc.prx in sce_module folder
|
|
|
|
bool found = false;
|
|
|
|
if (Config::isLleLibc()) {
|
|
|
|
std::filesystem::path sce_module_folder = file.parent_path() / "sce_module";
|
|
|
|
if (std::filesystem::is_directory(sce_module_folder)) {
|
|
|
|
for (const auto& entry : std::filesystem::directory_iterator(sce_module_folder)) {
|
2024-07-18 15:34:08 +00:00
|
|
|
if (entry.path().filename() == "libc.prx") {
|
2024-06-10 12:18:42 +00:00
|
|
|
found = true;
|
|
|
|
}
|
2024-07-18 15:34:08 +00:00
|
|
|
LOG_INFO(Loader, "Loading {}", entry.path().string().c_str());
|
|
|
|
linker->LoadModule(entry.path());
|
2024-06-10 12:18:42 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (!found) {
|
|
|
|
Libraries::LibC::libcSymbolsRegister(&linker->GetHLESymbols());
|
|
|
|
}
|
|
|
|
|
|
|
|
// start execution
|
2024-06-10 15:26:37 +00:00
|
|
|
std::jthread mainthread =
|
|
|
|
std::jthread([this](std::stop_token stop_token) { linker->Execute(); });
|
2024-06-10 12:18:42 +00:00
|
|
|
|
2024-07-15 21:22:47 +00:00
|
|
|
while (window->isOpen()) {
|
|
|
|
window->waitEvent();
|
2024-06-10 12:18:42 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
std::exit(0);
|
|
|
|
}
|
|
|
|
|
2024-06-10 15:26:37 +00:00
|
|
|
void Emulator::LoadSystemModules(const std::filesystem::path& file) {
|
2024-08-06 16:28:17 +00:00
|
|
|
constexpr std::array<SysModules, 9> ModulesToLoad{
|
2024-06-26 11:43:01 +00:00
|
|
|
{{"libSceNgs2.sprx", nullptr},
|
2024-07-17 12:37:32 +00:00
|
|
|
{"libSceFiber.sprx", nullptr},
|
2024-07-17 12:44:20 +00:00
|
|
|
{"libSceUlt.sprx", nullptr},
|
2024-07-25 20:01:12 +00:00
|
|
|
{"libSceJson.sprx", nullptr},
|
|
|
|
{"libSceJson2.sprx", nullptr},
|
2024-06-26 11:43:01 +00:00
|
|
|
{"libSceLibcInternal.sprx", &Libraries::LibcInternal::RegisterlibSceLibcInternal},
|
|
|
|
{"libSceDiscMap.sprx", &Libraries::DiscMap::RegisterlibSceDiscMap},
|
2024-06-30 07:12:36 +00:00
|
|
|
{"libSceRtc.sprx", &Libraries::Rtc::RegisterlibSceRtc},
|
2024-08-06 16:28:17 +00:00
|
|
|
{"libSceJpegEnc.sprx", nullptr}},
|
2024-07-17 12:37:32 +00:00
|
|
|
};
|
2024-06-26 11:43:01 +00:00
|
|
|
|
|
|
|
std::vector<std::filesystem::path> found_modules;
|
2024-06-10 15:26:37 +00:00
|
|
|
const auto& sys_module_path = Common::FS::GetUserPath(Common::FS::PathType::SysModuleDir);
|
|
|
|
for (const auto& entry : std::filesystem::directory_iterator(sys_module_path)) {
|
2024-06-26 11:43:01 +00:00
|
|
|
found_modules.push_back(entry.path());
|
|
|
|
}
|
2024-07-17 12:37:32 +00:00
|
|
|
for (const auto& [module_name, init_func] : ModulesToLoad) {
|
2024-07-17 14:57:54 +00:00
|
|
|
const auto it = std::ranges::find_if(
|
|
|
|
found_modules, [&](const auto& path) { return path.filename() == module_name; });
|
2024-07-17 12:37:32 +00:00
|
|
|
if (it != found_modules.end()) {
|
|
|
|
LOG_INFO(Loader, "Loading {}", it->string());
|
|
|
|
linker->LoadModule(*it);
|
|
|
|
continue;
|
2024-06-26 11:43:01 +00:00
|
|
|
}
|
2024-07-17 12:37:32 +00:00
|
|
|
if (init_func) {
|
|
|
|
LOG_INFO(Loader, "Can't Load {} switching to HLE", module_name);
|
|
|
|
init_func(&linker->GetHLESymbols());
|
2024-06-26 11:43:01 +00:00
|
|
|
} else {
|
2024-07-17 12:37:32 +00:00
|
|
|
LOG_INFO(Loader, "No HLE available for {} module", module_name);
|
2024-06-10 15:26:37 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-06-13 21:58:57 +00:00
|
|
|
} // namespace Core
|