mirror of
https://github.com/RPCS3/rpcs3.git
synced 2025-03-30 07:20:13 +00:00
2000 lines
52 KiB
C++
2000 lines
52 KiB
C++
#include "stdafx.h"
|
||
#include "VFS.h"
|
||
#include "Utilities/bin_patch.h"
|
||
#include "Emu/Memory/vm.h"
|
||
#include "Emu/System.h"
|
||
#include "Emu/system_progress.hpp"
|
||
#include "Emu/system_utils.hpp"
|
||
#include "Emu/perf_meter.hpp"
|
||
#include "Emu/vfs_config.h"
|
||
|
||
#include "Emu/Cell/ErrorCodes.h"
|
||
#include "Emu/Cell/PPUThread.h"
|
||
#include "Emu/Cell/PPUDisAsm.h"
|
||
#include "Emu/Cell/PPUAnalyser.h"
|
||
#include "Emu/Cell/SPUThread.h"
|
||
#include "Emu/RSX/RSXThread.h"
|
||
#include "Emu/Cell/lv2/sys_process.h"
|
||
#include "Emu/Cell/lv2/sys_sync.h"
|
||
#include "Emu/Cell/lv2/sys_prx.h"
|
||
#include "Emu/Cell/lv2/sys_overlay.h"
|
||
|
||
#include "Emu/title.h"
|
||
#include "Emu/IdManager.h"
|
||
#include "Emu/RSX/Capture/rsx_replay.h"
|
||
|
||
#include "Loader/PSF.h"
|
||
#include "Loader/ELF.h"
|
||
|
||
#include "Utilities/StrUtil.h"
|
||
|
||
#include "../Crypto/unself.h"
|
||
#include "util/yaml.hpp"
|
||
#include "util/logs.hpp"
|
||
#include "util/serialization.hpp"
|
||
|
||
#include <fstream>
|
||
#include <memory>
|
||
#include <regex>
|
||
|
||
#include "Utilities/JIT.h"
|
||
|
||
#include "display_sleep_control.h"
|
||
|
||
#if defined(HAVE_VULKAN)
|
||
#include "Emu/RSX/VK/VulkanAPI.h"
|
||
#endif
|
||
|
||
LOG_CHANNEL(sys_log, "SYS");
|
||
|
||
// Preallocate 32 MiB
|
||
stx::manual_typemap<void, 0x20'00000, 128> g_fixed_typemap;
|
||
|
||
bool g_use_rtm = false;
|
||
u64 g_rtm_tx_limit1 = 0;
|
||
u64 g_rtm_tx_limit2 = 0;
|
||
|
||
std::string g_cfg_defaults;
|
||
|
||
atomic_t<u64> g_watchdog_hold_ctr{0};
|
||
|
||
extern bool ppu_load_exec(const ppu_exec_object&);
|
||
extern void spu_load_exec(const spu_exec_object&);
|
||
extern void ppu_precompile(std::vector<std::string>& dir_queue, std::vector<lv2_prx*>* loaded_prx);
|
||
extern bool ppu_initialize(const ppu_module&, bool = false);
|
||
extern void ppu_finalize(const ppu_module&);
|
||
extern void ppu_unload_prx(const lv2_prx&);
|
||
extern std::shared_ptr<lv2_prx> ppu_load_prx(const ppu_prx_object&, const std::string&, s64 = 0);
|
||
extern std::pair<std::shared_ptr<lv2_overlay>, CellError> ppu_load_overlay(const ppu_exec_object&, const std::string& path, s64 = 0);
|
||
|
||
fs::file g_tty;
|
||
atomic_t<s64> g_tty_size{0};
|
||
std::array<std::deque<std::string>, 16> g_tty_input;
|
||
std::mutex g_tty_mutex;
|
||
|
||
// Report error and call std::abort(), defined in main.cpp
|
||
[[noreturn]] void report_fatal_error(std::string_view);
|
||
|
||
namespace atomic_wait
|
||
{
|
||
extern void parse_hashtable(bool(*cb)(u64 id, u32 refs, u64 ptr, u32 stats));
|
||
}
|
||
|
||
template<>
|
||
void fmt_class_string<game_boot_result>::format(std::string& out, u64 arg)
|
||
{
|
||
format_enum(out, arg, [](game_boot_result value)
|
||
{
|
||
switch (value)
|
||
{
|
||
case game_boot_result::no_errors: return "No errors";
|
||
case game_boot_result::generic_error: return "Generic error";
|
||
case game_boot_result::nothing_to_boot: return "Nothing to boot";
|
||
case game_boot_result::wrong_disc_location: return "Wrong disc location";
|
||
case game_boot_result::invalid_file_or_folder: return "Invalid file or folder";
|
||
case game_boot_result::install_failed: return "Game install failed";
|
||
case game_boot_result::decryption_error: return "Failed to decrypt content";
|
||
case game_boot_result::file_creation_error: return "Could not create important files";
|
||
case game_boot_result::firmware_missing: return "Firmware is missing";
|
||
case game_boot_result::unsupported_disc_type: return "This disc type is not supported yet";
|
||
}
|
||
return unknown;
|
||
});
|
||
}
|
||
|
||
template<>
|
||
void fmt_class_string<cfg_mode>::format(std::string& out, u64 arg)
|
||
{
|
||
format_enum(out, arg, [](cfg_mode value)
|
||
{
|
||
switch (value)
|
||
{
|
||
case cfg_mode::custom: return "custom config";
|
||
case cfg_mode::custom_selection: return "custom config selection";
|
||
case cfg_mode::global: return "global config";
|
||
case cfg_mode::config_override: return "config override";
|
||
case cfg_mode::continuous: return "continuous config";
|
||
case cfg_mode::default_config: return "default config";
|
||
}
|
||
return unknown;
|
||
});
|
||
}
|
||
|
||
void Emulator::Init(bool add_only)
|
||
{
|
||
jit_runtime::initialize();
|
||
|
||
if (!g_tty)
|
||
{
|
||
const auto tty_path = fs::get_cache_dir() + "TTY.log";
|
||
g_tty.open(tty_path, fs::rewrite + fs::append);
|
||
|
||
if (!g_tty)
|
||
{
|
||
sys_log.fatal("Failed to create TTY log: %s (%s)", tty_path, fs::g_tls_error);
|
||
}
|
||
}
|
||
|
||
g_fxo->reset();
|
||
|
||
// Reset defaults, cache them
|
||
g_cfg_vfs.from_default();
|
||
g_cfg.from_default();
|
||
g_cfg.name.clear();
|
||
|
||
// Not all renderers are known at compile time, so set a provided default if possible
|
||
if (m_default_renderer == video_renderer::vulkan && !m_default_graphics_adapter.empty())
|
||
{
|
||
g_cfg.video.renderer.set(m_default_renderer);
|
||
g_cfg.video.vk.adapter.from_string(m_default_graphics_adapter);
|
||
}
|
||
|
||
g_cfg_defaults = g_cfg.to_string();
|
||
|
||
// Load VFS config
|
||
g_cfg_vfs.load();
|
||
sys_log.notice("Using VFS config:\n%s", g_cfg_vfs.to_string());
|
||
|
||
// Mount all devices
|
||
const std::string emu_dir = rpcs3::utils::get_emu_dir();
|
||
const std::string elf_dir = fs::get_parent_dir(m_path);
|
||
|
||
vfs::mount("/dev_hdd0", g_cfg_vfs.get(g_cfg_vfs.dev_hdd0, emu_dir));
|
||
vfs::mount("/dev_flash", g_cfg_vfs.get_dev_flash());
|
||
vfs::mount("/dev_flash2", g_cfg_vfs.get_dev_flash2());
|
||
vfs::mount("/dev_flash3", g_cfg_vfs.get_dev_flash3());
|
||
vfs::mount("/dev_usb", g_cfg_vfs.get(g_cfg_vfs.dev_usb000, emu_dir));
|
||
vfs::mount("/dev_usb000", g_cfg_vfs.get(g_cfg_vfs.dev_usb000, emu_dir));
|
||
vfs::mount("/app_home", g_cfg_vfs.app_home.to_string().empty() ? elf_dir + '/' : g_cfg_vfs.get(g_cfg_vfs.app_home, emu_dir));
|
||
|
||
if (!hdd1.empty())
|
||
{
|
||
vfs::mount("/dev_hdd1", hdd1);
|
||
sys_log.notice("Hdd1: %s", vfs::get("/dev_hdd1"));
|
||
}
|
||
|
||
// Load config file
|
||
if (m_config_mode == cfg_mode::config_override)
|
||
{
|
||
if (const fs::file cfg_file{m_config_path, fs::read + fs::create})
|
||
{
|
||
sys_log.notice("Applying config override: %s", m_config_path);
|
||
|
||
if (!g_cfg.from_string(cfg_file.to_string()))
|
||
{
|
||
sys_log.fatal("Failed to apply config: %s. Proceeding with regular configuration.", m_config_path);
|
||
m_config_path.clear();
|
||
m_config_mode = cfg_mode::custom;
|
||
}
|
||
else
|
||
{
|
||
sys_log.success("Applied config override: %s", m_config_path);
|
||
g_cfg.name = m_config_path;
|
||
}
|
||
}
|
||
else
|
||
{
|
||
sys_log.fatal("Failed to access config: %s (%s). Proceeding with regular configuration.", m_config_path, fs::g_tls_error);
|
||
m_config_path.clear();
|
||
m_config_mode = cfg_mode::custom;
|
||
}
|
||
}
|
||
|
||
// Reload global configuration
|
||
if (m_config_mode != cfg_mode::config_override && m_config_mode != cfg_mode::default_config)
|
||
{
|
||
const auto cfg_path = fs::get_config_dir() + "/config.yml";
|
||
|
||
if (const fs::file cfg_file{cfg_path, fs::read + fs::create})
|
||
{
|
||
sys_log.notice("Applying global config: %s", cfg_path);
|
||
|
||
if (!g_cfg.from_string(cfg_file.to_string()))
|
||
{
|
||
sys_log.fatal("Failed to apply global config: %s", cfg_path);
|
||
}
|
||
|
||
g_cfg.name = cfg_path;
|
||
}
|
||
else
|
||
{
|
||
sys_log.fatal("Failed to access global config: %s (%s)", cfg_path, fs::g_tls_error);
|
||
}
|
||
}
|
||
|
||
// Create directories (can be disabled if necessary)
|
||
const std::string dev_hdd0 = rpcs3::utils::get_hdd0_dir();
|
||
const std::string dev_hdd1 = g_cfg_vfs.get(g_cfg_vfs.dev_hdd1, emu_dir);
|
||
const std::string dev_usb = g_cfg_vfs.get(g_cfg_vfs.dev_usb000, emu_dir);
|
||
const std::string dev_flsh = g_cfg_vfs.get_dev_flash();
|
||
const std::string dev_flsh2 = g_cfg_vfs.get_dev_flash2();
|
||
const std::string dev_flsh3 = g_cfg_vfs.get_dev_flash3();
|
||
|
||
auto make_path_verbose = [](const std::string& path)
|
||
{
|
||
if (!fs::create_path(path))
|
||
{
|
||
sys_log.fatal("Failed to create path: %s (%s)", path, fs::g_tls_error);
|
||
}
|
||
};
|
||
|
||
const std::string save_path = dev_hdd0 + "home/" + m_usr + "/savedata/";
|
||
const std::string user_path = dev_hdd0 + "home/" + m_usr + "/localusername";
|
||
|
||
if (g_cfg.vfs.init_dirs)
|
||
{
|
||
make_path_verbose(dev_hdd0);
|
||
make_path_verbose(dev_hdd1);
|
||
make_path_verbose(dev_flsh);
|
||
make_path_verbose(dev_flsh2);
|
||
make_path_verbose(dev_flsh3);
|
||
make_path_verbose(dev_usb);
|
||
make_path_verbose(dev_hdd0 + "game/");
|
||
make_path_verbose(dev_hdd0 + "game/TEST12345/");
|
||
make_path_verbose(dev_hdd0 + "game/TEST12345/USRDIR/");
|
||
make_path_verbose(dev_hdd0 + reinterpret_cast<const char*>(u8"game/$locks/"));
|
||
make_path_verbose(dev_hdd0 + "home/");
|
||
make_path_verbose(dev_hdd0 + "home/" + m_usr + "/");
|
||
make_path_verbose(dev_hdd0 + "home/" + m_usr + "/exdata/");
|
||
make_path_verbose(save_path);
|
||
make_path_verbose(dev_hdd0 + "home/" + m_usr + "/trophy/");
|
||
|
||
if (!fs::write_file(user_path, fs::create + fs::excl + fs::write, "User"s))
|
||
{
|
||
if (fs::g_tls_error != fs::error::exist)
|
||
{
|
||
sys_log.fatal("Failed to create file: %s (%s)", user_path, fs::g_tls_error);
|
||
}
|
||
}
|
||
|
||
make_path_verbose(dev_hdd0 + "disc/");
|
||
make_path_verbose(dev_hdd0 + "savedata/");
|
||
make_path_verbose(dev_hdd0 + "savedata/vmc/");
|
||
make_path_verbose(dev_hdd0 + "photo/");
|
||
make_path_verbose(dev_hdd1 + "caches/");
|
||
}
|
||
|
||
make_path_verbose(fs::get_cache_dir() + "shaderlog/");
|
||
make_path_verbose(fs::get_cache_dir() + "spu_progs/");
|
||
make_path_verbose(fs::get_config_dir() + "captures/");
|
||
make_path_verbose(fs::get_config_dir() + "sounds/");
|
||
make_path_verbose(patch_engine::get_patches_path());
|
||
|
||
if (add_only)
|
||
{
|
||
// We don't need to initialize the rest if we only add games
|
||
return;
|
||
}
|
||
|
||
// Log user
|
||
if (m_usr.empty())
|
||
{
|
||
sys_log.fatal("No user configured");
|
||
}
|
||
else
|
||
{
|
||
std::string username;
|
||
|
||
if (const fs::file file = fs::file(user_path))
|
||
{
|
||
if (const std::string localusername = file.to_string(); !localusername.empty())
|
||
{
|
||
username = localusername;
|
||
}
|
||
else
|
||
{
|
||
sys_log.warning("Empty username in file: '%s'. Consider setting a username for user '%s' in the user manager.", user_path, m_usr);
|
||
}
|
||
}
|
||
else
|
||
{
|
||
sys_log.error("Could not read file: '%s'", user_path);
|
||
}
|
||
|
||
if (username.empty())
|
||
{
|
||
sys_log.notice("Logged in as user '%s'", m_usr);
|
||
}
|
||
else
|
||
{
|
||
sys_log.notice("Logged in as user '%s' with the username '%s'", m_usr, username);
|
||
}
|
||
}
|
||
|
||
// Fixup savedata
|
||
for (const auto& entry : fs::dir(save_path))
|
||
{
|
||
if (entry.is_directory && entry.name.starts_with(".backup_"))
|
||
{
|
||
const std::string desired = entry.name.substr(8);
|
||
const std::string pending = save_path + ".working_" + desired;
|
||
|
||
if (fs::is_dir(pending))
|
||
{
|
||
// Finalize interrupted saving
|
||
if (!fs::rename(pending, save_path + desired, false))
|
||
{
|
||
sys_log.fatal("Failed to fix save data: %s (%s)", pending, fs::g_tls_error);
|
||
continue;
|
||
}
|
||
|
||
sys_log.success("Fixed save data: %s", desired);
|
||
}
|
||
|
||
// Remove pending backup data
|
||
if (!fs::remove_all(save_path + entry.name))
|
||
{
|
||
sys_log.fatal("Failed to remove save data backup: %s%s (%s)", save_path, entry.name, fs::g_tls_error);
|
||
}
|
||
else
|
||
{
|
||
sys_log.success("Removed save data backup: %s%s", save_path, entry.name);
|
||
}
|
||
}
|
||
}
|
||
|
||
// Limit cache size
|
||
if (g_cfg.vfs.limit_cache_size)
|
||
rpcs3::cache::limit_cache_size();
|
||
}
|
||
|
||
void Emulator::SetUsr(const std::string& user)
|
||
{
|
||
sys_log.notice("Setting user ID '%s'", user);
|
||
|
||
const u32 id = rpcs3::utils::check_user(user);
|
||
|
||
if (id == 0)
|
||
{
|
||
fmt::throw_exception("Failed to set user ID '%s'", user);
|
||
}
|
||
|
||
m_usrid = id;
|
||
m_usr = user;
|
||
}
|
||
|
||
std::string Emulator::GetBackgroundPicturePath() const
|
||
{
|
||
// Try to find a custom icon first
|
||
std::string path = fs::get_config_dir() + "/Icons/game_icons/" + GetTitleID() + "/PIC1.PNG";
|
||
|
||
if (fs::is_file(path))
|
||
{
|
||
return path;
|
||
}
|
||
|
||
path = m_sfo_dir + "/PIC1.PNG";
|
||
|
||
if (!fs::is_file(path))
|
||
{
|
||
const std::string disc_dir = vfs::get("/dev_bdvd/PS3_GAME");
|
||
|
||
if (disc_dir.empty())
|
||
{
|
||
// Fallback to ICON0.PNG
|
||
path = m_sfo_dir + "/ICON0.PNG";
|
||
}
|
||
else
|
||
{
|
||
// Fallback to PIC1.PNG in disc dir
|
||
path = disc_dir + "/PIC1.PNG";
|
||
|
||
if (!fs::is_file(path))
|
||
{
|
||
// Fallback to ICON0.PNG in update dir
|
||
path = m_sfo_dir + "/ICON0.PNG";
|
||
|
||
if (!fs::is_file(path))
|
||
{
|
||
// Fallback to ICON0.PNG in disc dir
|
||
path = disc_dir + "/ICON0.PNG";
|
||
}
|
||
}
|
||
}
|
||
}
|
||
|
||
return path;
|
||
}
|
||
|
||
bool Emulator::BootRsxCapture(const std::string& path)
|
||
{
|
||
fs::file in_file(path);
|
||
|
||
if (!in_file)
|
||
{
|
||
return false;
|
||
}
|
||
|
||
std::unique_ptr<rsx::frame_capture_data> frame = std::make_unique<rsx::frame_capture_data>();
|
||
utils::serial load_manager;
|
||
load_manager.set_reading_state(in_file.to_vector<u8>());
|
||
|
||
load_manager(*frame);
|
||
in_file.close();
|
||
|
||
if (frame->magic != rsx::c_fc_magic)
|
||
{
|
||
sys_log.error("Invalid rsx capture file!");
|
||
return false;
|
||
}
|
||
|
||
if (frame->version != rsx::c_fc_version)
|
||
{
|
||
sys_log.error("Rsx capture file version not supported! Expected %d, found %d", +rsx::c_fc_version, frame->version);
|
||
return false;
|
||
}
|
||
|
||
if (frame->LE_format != u32{std::endian::little == std::endian::native})
|
||
{
|
||
static constexpr std::string_view machines[2]{"Big-Endian", "Little-Endian"};
|
||
|
||
sys_log.error("Rsx capture byte endianness not supported! Expected %s format, found %s format"
|
||
, machines[frame->LE_format ^ 1], machines[frame->LE_format]);
|
||
|
||
return false;
|
||
}
|
||
|
||
Init();
|
||
g_cfg.video.disable_on_disk_shader_cache.set(true);
|
||
|
||
vm::init();
|
||
g_fxo->init(false);
|
||
|
||
// Initialize progress dialog
|
||
g_fxo->init<named_thread<progress_dialog_server>>();
|
||
|
||
// PS3 'executable'
|
||
m_state = system_state::ready;
|
||
GetCallbacks().on_ready();
|
||
|
||
GetCallbacks().init_gs_render();
|
||
GetCallbacks().init_pad_handler("");
|
||
|
||
GetCallbacks().on_run(false);
|
||
m_state = system_state::running;
|
||
|
||
auto replay_thr = g_fxo->init<named_thread<rsx::rsx_replay_thread>>("RSX Replay", std::move(frame));
|
||
replay_thr->state -= cpu_flag::stop;
|
||
replay_thr->state.notify_one(cpu_flag::stop);
|
||
|
||
return true;
|
||
}
|
||
|
||
game_boot_result Emulator::BootGame(const std::string& path, const std::string& title_id, bool direct, bool add_only, cfg_mode config_mode, const std::string& config_path)
|
||
{
|
||
if (!fs::exists(path))
|
||
{
|
||
return game_boot_result::invalid_file_or_folder;
|
||
}
|
||
|
||
m_path_old = m_path;
|
||
|
||
m_config_mode = config_mode;
|
||
m_config_path = config_path;
|
||
|
||
if (direct || fs::is_file(path))
|
||
{
|
||
m_path = path;
|
||
return Load(title_id, add_only);
|
||
}
|
||
|
||
game_boot_result result = game_boot_result::nothing_to_boot;
|
||
|
||
static const char* boot_list[] =
|
||
{
|
||
"/EBOOT.BIN",
|
||
"/USRDIR/EBOOT.BIN",
|
||
"/USRDIR/ISO.BIN.EDAT",
|
||
"/PS3_GAME/USRDIR/EBOOT.BIN",
|
||
};
|
||
|
||
for (std::string elf : boot_list)
|
||
{
|
||
elf = path + elf;
|
||
|
||
if (fs::is_file(elf))
|
||
{
|
||
m_path = elf;
|
||
result = Load(title_id, add_only);
|
||
break;
|
||
}
|
||
}
|
||
|
||
if (add_only)
|
||
{
|
||
for (auto&& entry : fs::dir{ path })
|
||
{
|
||
if (entry.name == "." || entry.name == "..")
|
||
{
|
||
continue;
|
||
}
|
||
|
||
if (entry.is_directory && std::regex_match(entry.name, std::regex("^PS3_GM[[:digit:]]{2}$")))
|
||
{
|
||
const std::string elf = path + "/" + entry.name + "/USRDIR/EBOOT.BIN";
|
||
|
||
if (fs::is_file(elf))
|
||
{
|
||
m_path = elf;
|
||
if (const auto err = Load(title_id, add_only); err != game_boot_result::no_errors)
|
||
{
|
||
result = err;
|
||
}
|
||
}
|
||
}
|
||
}
|
||
}
|
||
return result;
|
||
}
|
||
|
||
void Emulator::SetForceBoot(bool force_boot)
|
||
{
|
||
m_force_boot = force_boot;
|
||
}
|
||
|
||
game_boot_result Emulator::Load(const std::string& title_id, bool add_only, bool is_disc_patch)
|
||
{
|
||
const std::string resolved_path = fs::resolve_path(m_path);
|
||
|
||
if (!IsStopped())
|
||
{
|
||
Stop();
|
||
}
|
||
|
||
if (!title_id.empty())
|
||
{
|
||
m_title_id = title_id;
|
||
}
|
||
|
||
sys_log.notice("Selected config: mode=%s, path=\"%s\"", m_config_mode, m_config_path);
|
||
sys_log.notice("Path: %s", m_path);
|
||
|
||
{
|
||
Init(add_only);
|
||
|
||
// Load game list (maps ABCD12345 IDs to /dev_bdvd/ locations)
|
||
YAML::Node games;
|
||
|
||
if (fs::file f{fs::get_config_dir() + "/games.yml", fs::read + fs::create})
|
||
{
|
||
auto [result, error] = yaml_load(f.to_string());
|
||
|
||
if (!error.empty())
|
||
{
|
||
sys_log.error("Failed to load games.yml: %s", error);
|
||
}
|
||
|
||
games = result;
|
||
}
|
||
|
||
if (!games.IsMap())
|
||
{
|
||
games.reset();
|
||
}
|
||
|
||
const std::string elf_dir = fs::get_parent_dir(m_path);
|
||
|
||
// Load PARAM.SFO (TODO)
|
||
psf::registry _psf;
|
||
if (fs::file sfov{elf_dir + "/sce_sys/param.sfo"})
|
||
{
|
||
m_sfo_dir = elf_dir;
|
||
_psf = psf::load_object(sfov);
|
||
}
|
||
else
|
||
{
|
||
if (fs::is_dir(m_path))
|
||
{
|
||
// Special case (directory scan)
|
||
m_sfo_dir = rpcs3::utils::get_sfo_dir_from_game_path(m_path, m_title_id);
|
||
}
|
||
else if (!disc.empty())
|
||
{
|
||
// Check previously used category before it's overwritten
|
||
if (m_cat == "DG")
|
||
{
|
||
m_sfo_dir = disc + "/" + m_game_dir;
|
||
}
|
||
else if (m_cat == "GD")
|
||
{
|
||
m_sfo_dir = rpcs3::utils::get_hdd0_dir() + "game/" + m_title_id;
|
||
}
|
||
else
|
||
{
|
||
m_sfo_dir = rpcs3::utils::get_sfo_dir_from_game_path(disc, m_title_id);
|
||
}
|
||
}
|
||
else
|
||
{
|
||
m_sfo_dir = rpcs3::utils::get_sfo_dir_from_game_path(elf_dir + "/../", m_title_id);
|
||
}
|
||
|
||
_psf = psf::load_object(fs::file(m_sfo_dir + "/PARAM.SFO"));
|
||
}
|
||
|
||
m_title = std::string(psf::get_string(_psf, "TITLE", std::string_view(m_path).substr(m_path.find_last_of(fs::delim) + 1)));
|
||
m_title_id = std::string(psf::get_string(_psf, "TITLE_ID"));
|
||
m_cat = std::string(psf::get_string(_psf, "CATEGORY"));
|
||
|
||
const auto version_app = psf::get_string(_psf, "APP_VER", "Unknown");
|
||
const auto version_disc = psf::get_string(_psf, "VERSION", "Unknown");
|
||
m_app_version = version_app == "Unknown" ? version_disc : version_app;
|
||
|
||
if (!_psf.empty() && m_cat.empty())
|
||
{
|
||
sys_log.fatal("Corrupted PARAM.SFO found! Try reinstalling the game.");
|
||
return game_boot_result::invalid_file_or_folder;
|
||
}
|
||
|
||
sys_log.notice("Title: %s", GetTitle());
|
||
sys_log.notice("Serial: %s", GetTitleID());
|
||
sys_log.notice("Category: %s", GetCat());
|
||
sys_log.notice("Version: APP_VER=%s VERSION=%s", version_app, version_disc);
|
||
|
||
if (!add_only)
|
||
{
|
||
if (m_config_mode == cfg_mode::custom_selection || (m_config_mode == cfg_mode::continuous && !m_config_path.empty()))
|
||
{
|
||
if (fs::file cfg_file{ m_config_path })
|
||
{
|
||
sys_log.notice("Applying %s: %s", m_config_mode, m_config_path);
|
||
|
||
if (g_cfg.from_string(cfg_file.to_string()))
|
||
{
|
||
g_cfg.name = m_config_path;
|
||
}
|
||
else
|
||
{
|
||
sys_log.fatal("Failed to apply %s: %s", m_config_mode, m_config_path);
|
||
}
|
||
}
|
||
else
|
||
{
|
||
sys_log.fatal("Failed to access %s: %s", m_config_mode, m_config_path);
|
||
}
|
||
}
|
||
else if (m_config_mode == cfg_mode::custom)
|
||
{
|
||
const std::string config_path = rpcs3::utils::get_custom_config_path(m_title_id);
|
||
|
||
// Load custom config-1
|
||
if (fs::file cfg_file{ config_path })
|
||
{
|
||
sys_log.notice("Applying custom config: %s", config_path);
|
||
|
||
if (g_cfg.from_string(cfg_file.to_string()))
|
||
{
|
||
g_cfg.name = config_path;
|
||
m_config_path = config_path;
|
||
}
|
||
else
|
||
{
|
||
sys_log.fatal("Failed to apply custom config: %s", config_path);
|
||
}
|
||
}
|
||
|
||
// Load custom config-2
|
||
if (fs::file cfg_file{ m_path + ".yml" })
|
||
{
|
||
sys_log.notice("Applying custom config: %s.yml", m_path);
|
||
|
||
if (g_cfg.from_string(cfg_file.to_string()))
|
||
{
|
||
g_cfg.name = m_path + ".yml";
|
||
m_config_path = g_cfg.name;
|
||
}
|
||
else
|
||
{
|
||
sys_log.fatal("Failed to apply custom config: %s.yml", m_path);
|
||
}
|
||
}
|
||
}
|
||
}
|
||
|
||
initalize_timebased_time();
|
||
|
||
// Set RTM usage
|
||
g_use_rtm = utils::has_rtm() && (((utils::has_mpx() && !utils::has_tsx_force_abort()) && g_cfg.core.enable_TSX == tsx_usage::enabled) || g_cfg.core.enable_TSX == tsx_usage::forced);
|
||
|
||
if (!add_only)
|
||
{
|
||
// Log some extra info in case of boot
|
||
#if defined(HAVE_VULKAN)
|
||
if (g_cfg.video.renderer == video_renderer::vulkan)
|
||
{
|
||
sys_log.notice("Vulkan SDK Revision: %d", VK_HEADER_VERSION);
|
||
}
|
||
#endif
|
||
sys_log.notice("Used configuration:\n%s\n", g_cfg.to_string());
|
||
|
||
if (g_use_rtm && (!utils::has_mpx() || utils::has_tsx_force_abort()))
|
||
{
|
||
sys_log.warning("TSX forced by User");
|
||
}
|
||
|
||
// Initialize patch engine
|
||
g_fxo->need<patch_engine>();
|
||
|
||
// Load patches from different locations
|
||
g_fxo->get<patch_engine>().append_global_patches();
|
||
g_fxo->get<patch_engine>().append_title_patches(m_title_id);
|
||
}
|
||
|
||
if (g_use_rtm)
|
||
{
|
||
// Update supplementary settings
|
||
const f64 _1ns = utils::get_tsc_freq() / 1000'000'000.;
|
||
g_rtm_tx_limit1 = static_cast<u64>(g_cfg.core.tx_limit1_ns * _1ns);
|
||
g_rtm_tx_limit2 = static_cast<u64>(g_cfg.core.tx_limit2_ns * _1ns);
|
||
}
|
||
|
||
// Set bdvd_dir
|
||
std::string bdvd_dir;
|
||
|
||
if (!add_only)
|
||
{
|
||
bdvd_dir = g_cfg_vfs.dev_bdvd;
|
||
|
||
if (!bdvd_dir.empty() && bdvd_dir.back() != fs::delim[0] && bdvd_dir.back() != fs::delim[1])
|
||
{
|
||
bdvd_dir.push_back('/');
|
||
}
|
||
|
||
if (!bdvd_dir.empty() && !fs::is_file(bdvd_dir + "PS3_DISC.SFB"))
|
||
{
|
||
// Unuse if invalid
|
||
sys_log.error("Failed to use custom BDVD directory: '%s'", bdvd_dir);
|
||
bdvd_dir.clear();
|
||
}
|
||
}
|
||
|
||
// Special boot mode (directory scan)
|
||
if (!add_only && fs::is_dir(m_path))
|
||
{
|
||
m_state = system_state::ready;
|
||
GetCallbacks().on_ready();
|
||
vm::init();
|
||
g_fxo->init(false);
|
||
Run(false);
|
||
m_force_boot = false;
|
||
|
||
// Force LLVM recompiler
|
||
g_cfg.core.ppu_decoder.from_default();
|
||
|
||
// Force LLE lib loading mode
|
||
g_cfg.core.libraries_control.set_set([]()
|
||
{
|
||
std::set<std::string> set;
|
||
|
||
extern const std::map<std::string_view, int> g_prx_list;
|
||
|
||
for (const auto& lib : g_prx_list)
|
||
{
|
||
set.emplace(std::string(lib.first) + ":lle");
|
||
}
|
||
|
||
return set;
|
||
}());
|
||
|
||
// Fake arg (workaround)
|
||
argv.resize(1);
|
||
argv[0] = "/dev_bdvd/PS3_GAME/USRDIR/EBOOT.BIN";
|
||
m_dir = "/dev_bdvd/PS3_GAME";
|
||
|
||
g_fxo->init<named_thread>("SPRX Loader"sv, [this]
|
||
{
|
||
std::string path;
|
||
std::vector<std::string> dir_queue;
|
||
dir_queue.emplace_back(m_path + '/');
|
||
|
||
if (m_title_id.empty())
|
||
{
|
||
// Check if we are trying to scan vsh/module
|
||
const std::string vsh_path = g_cfg_vfs.get_dev_flash() + "vsh/module";
|
||
|
||
if (IsPathInsideDir(m_path, vsh_path))
|
||
{
|
||
// Memorize path to vsh.self
|
||
path = vsh_path + "/vsh.self";
|
||
}
|
||
}
|
||
else
|
||
{
|
||
// Find game update to use EBOOT.BIN from it, also add its directory to scan
|
||
if (m_cat == "DG")
|
||
{
|
||
const std::string hdd0_path = vfs::get("/dev_hdd0/game/") + m_title_id;
|
||
|
||
if (fs::is_file(hdd0_path + "/USRDIR/EBOOT.BIN"))
|
||
{
|
||
m_path = hdd0_path;
|
||
dir_queue.emplace_back(m_path + '/');
|
||
}
|
||
}
|
||
|
||
// Memorize path to EBOOT.BIN
|
||
path = m_path + "/USRDIR/EBOOT.BIN";
|
||
|
||
// Try to add all related directories
|
||
const std::set<std::string> dirs = GetGameDirs();
|
||
dir_queue.insert(std::end(dir_queue), std::begin(dirs), std::end(dirs));
|
||
}
|
||
|
||
if (fs::is_file(path))
|
||
{
|
||
// Compile binary first
|
||
ppu_log.notice("Trying to load binary: %s", path);
|
||
|
||
fs::file src{path};
|
||
|
||
src = decrypt_self(std::move(src));
|
||
|
||
const ppu_exec_object obj = src;
|
||
|
||
if (obj == elf_error::ok)
|
||
{
|
||
auto& _main = g_fxo->get<ppu_module>();
|
||
|
||
ppu_load_exec(obj);
|
||
|
||
_main.path = path;
|
||
|
||
ConfigurePPUCache();
|
||
|
||
ppu_initialize(_main);
|
||
}
|
||
else
|
||
{
|
||
sys_log.error("Failed to load binary '%s' (%s)", path, obj.get_error());
|
||
}
|
||
}
|
||
else
|
||
{
|
||
// Workaround for analyser glitches
|
||
ensure(vm::falloc(0x10000, 0xf0000, vm::main));
|
||
}
|
||
|
||
if (IsStopped())
|
||
{
|
||
m_path = m_path_old; // Reset m_path to fix boot from gui
|
||
GetCallbacks().on_stop(); // Call on_stop to refresh gui
|
||
return;
|
||
}
|
||
|
||
ppu_precompile(dir_queue, nullptr);
|
||
|
||
// Exit "process"
|
||
CallAfter([]
|
||
{
|
||
Emu.SetForceBoot(true);
|
||
Emu.Stop();
|
||
});
|
||
|
||
m_path = m_path_old; // Reset m_path to fix boot from gui
|
||
GetCallbacks().on_stop(); // Call on_stop to refresh gui
|
||
});
|
||
|
||
return game_boot_result::no_errors;
|
||
}
|
||
|
||
// Detect boot location
|
||
const std::string hdd0_game = vfs::get("/dev_hdd0/game/");
|
||
const std::string hdd0_disc = vfs::get("/dev_hdd0/disc/");
|
||
const bool from_hdd0_game = IsPathInsideDir(m_path, hdd0_game);
|
||
const bool from_dev_flash = IsPathInsideDir(m_path, g_cfg_vfs.get_dev_flash());
|
||
|
||
#ifdef _WIN32
|
||
// m_path might be passed from command line with differences in uppercase/lowercase on windows.
|
||
if ((!from_hdd0_game && IsPathInsideDir(fmt::to_lower(m_path), fmt::to_lower(hdd0_game))) ||
|
||
(!from_dev_flash && IsPathInsideDir(fmt::to_lower(m_path), fmt::to_lower(g_cfg_vfs.get_dev_flash()))))
|
||
{
|
||
// Let's just abort to prevent errors down the line.
|
||
sys_log.error("The boot path seems to contain incorrectly cased characters. Please adjust the path and try again.");
|
||
return game_boot_result::invalid_file_or_folder;
|
||
}
|
||
#endif
|
||
|
||
// Mount /dev_bdvd/ if necessary
|
||
if (bdvd_dir.empty() && disc.empty())
|
||
{
|
||
// Find disc directory by searching a valid PS3_DISC.SFB closest to root directory
|
||
std::string sfb_dir, main_dir;
|
||
std::string_view main_dir_name;
|
||
|
||
for (std::string search_dir = elf_dir;;)
|
||
{
|
||
std::string parent_dir = fs::get_parent_dir(search_dir);
|
||
|
||
if (parent_dir.size() == search_dir.size())
|
||
{
|
||
// Keep looking until root directory is reached
|
||
break;
|
||
}
|
||
|
||
if (fs::file sfb_file{parent_dir + "/PS3_DISC.SFB"}; sfb_file && sfb_file.size() >= 4 && sfb_file.read<u32>() == ".SFB"_u32)
|
||
{
|
||
main_dir_name = std::string_view{search_dir}.substr(search_dir.find_last_of(fs::delim) + 1);
|
||
|
||
if (main_dir_name == "PS3_GAME" || std::regex_match(main_dir_name.begin(), main_dir_name.end(), std::regex("^PS3_GM[[:digit:]]{2}$")))
|
||
{
|
||
// Remember valid disc directory
|
||
main_dir = search_dir;
|
||
sfb_dir = parent_dir;
|
||
main_dir_name = std::string_view{main_dir}.substr(main_dir.find_last_of(fs::delim) + 1);
|
||
}
|
||
}
|
||
|
||
search_dir = std::move(parent_dir);
|
||
}
|
||
|
||
if (!sfb_dir.empty())
|
||
{
|
||
{
|
||
if (from_hdd0_game)
|
||
{
|
||
// Booting disc game from wrong location
|
||
sys_log.error("Disc game %s found at invalid location /dev_hdd0/game/", m_title_id);
|
||
|
||
const std::string dst_dir = hdd0_disc + sfb_dir.substr(hdd0_game.size());
|
||
|
||
// Move and retry from correct location
|
||
if (fs::create_path(fs::get_parent_dir(dst_dir)) && fs::rename(sfb_dir, dst_dir, false))
|
||
{
|
||
sys_log.success("Disc game %s moved to special location /dev_hdd0/disc/", m_title_id);
|
||
return m_path = hdd0_disc + m_path.substr(hdd0_game.size()), Load(m_title_id, add_only);
|
||
}
|
||
|
||
sys_log.error("Failed to move disc game %s to /dev_hdd0/disc/ (%s)", m_title_id, fs::g_tls_error);
|
||
return game_boot_result::wrong_disc_location;
|
||
}
|
||
|
||
bdvd_dir = sfb_dir + "/";
|
||
|
||
// Set game dir
|
||
m_game_dir = std::string{main_dir_name};
|
||
}
|
||
}
|
||
}
|
||
|
||
if (bdvd_dir.empty() && !is_disc_patch)
|
||
{
|
||
// Reset original disc game dir if this is neither disc nor disc patch
|
||
m_game_dir = "PS3_GAME";
|
||
}
|
||
|
||
// Booting patch data
|
||
if ((is_disc_patch || m_cat == "GD") && bdvd_dir.empty() && disc.empty())
|
||
{
|
||
// Load /dev_bdvd/ from game list if available
|
||
if (auto node = games[m_title_id])
|
||
{
|
||
bdvd_dir = node.Scalar();
|
||
}
|
||
else
|
||
{
|
||
sys_log.fatal("Disc directory not found. Try to run the game from the actual game disc directory.");
|
||
return game_boot_result::invalid_file_or_folder;
|
||
}
|
||
}
|
||
|
||
// Check /dev_bdvd/
|
||
if (disc.empty() && !bdvd_dir.empty() && fs::is_dir(bdvd_dir))
|
||
{
|
||
fs::file sfb_file;
|
||
|
||
vfs::mount("/dev_bdvd", bdvd_dir);
|
||
sys_log.notice("Disc: %s", vfs::get("/dev_bdvd"));
|
||
|
||
vfs::mount("/dev_bdvd/PS3_GAME", bdvd_dir + m_game_dir + "/");
|
||
sys_log.notice("Game: %s", vfs::get("/dev_bdvd/PS3_GAME"));
|
||
|
||
const auto sfb_path = vfs::get("/dev_bdvd/PS3_DISC.SFB");
|
||
|
||
if (!sfb_file.open(sfb_path) || sfb_file.size() < 4 || sfb_file.read<u32>() != ".SFB"_u32)
|
||
{
|
||
sys_log.error("Invalid disc directory for the disc game %s. (%s)", m_title_id, sfb_path);
|
||
return game_boot_result::invalid_file_or_folder;
|
||
}
|
||
|
||
const auto game_psf = psf::load_object(fs::file{vfs::get("/dev_bdvd/PS3_GAME/PARAM.SFO")});
|
||
const auto bdvd_title_id = psf::get_string(game_psf, "TITLE_ID");
|
||
|
||
if (m_title_id.empty())
|
||
{
|
||
// We are most likely booting a binary inside a disc directory
|
||
sys_log.error("Booting binary without TITLE_ID inside disc dir of '%s'", bdvd_title_id);
|
||
}
|
||
else
|
||
{
|
||
if (bdvd_title_id != m_title_id)
|
||
{
|
||
// Not really an error just an odd situation
|
||
sys_log.error("Unexpected PARAM.SFO found in disc directory '%s' (found '%s')", m_title_id, bdvd_title_id);
|
||
}
|
||
|
||
// Store /dev_bdvd/ location
|
||
games[m_title_id] = bdvd_dir;
|
||
YAML::Emitter out;
|
||
out << games;
|
||
|
||
fs::pending_file temp(fs::get_config_dir() + "/games.yml");
|
||
|
||
// Do not update games.yml when TITLE_ID is empty
|
||
if (!temp.file || temp.file.write(out.c_str(), out.size()), !temp.commit())
|
||
{
|
||
sys_log.error("Failed to save BDVD location of title '%s' (%s)", m_title_id, fs::g_tls_error);
|
||
}
|
||
}
|
||
}
|
||
else if (m_cat == "1P" && from_hdd0_game)
|
||
{
|
||
// PS1 Classic located in dev_hdd0/game
|
||
sys_log.notice("PS1 Game: %s, %s", m_title_id, m_title);
|
||
|
||
const std::string game_path = "/dev_hdd0/game/" + m_path.substr(hdd0_game.size(), 9);
|
||
|
||
argv.resize(9);
|
||
argv[0] = "/dev_flash/ps1emu/ps1_newemu.self";
|
||
argv[1] = m_title_id + "_mc1.VM1"; // virtual mc 1 /dev_hdd0/savedata/vmc/%argv[1]%
|
||
argv[2] = m_title_id + "_mc2.VM1"; // virtual mc 2 /dev_hdd0/savedata/vmc/%argv[2]%
|
||
argv[3] = "0082"; // region target
|
||
argv[4] = "1600"; // ??? arg4 600 / 1200 / 1600, resolution scale? (purely a guess, the numbers seem to match closely to resolutions tho)
|
||
argv[5] = game_path; // ps1 game folder path (not the game serial)
|
||
argv[6] = "1"; // ??? arg6 1 ?
|
||
argv[7] = "2"; // ??? arg7 2 -- full screen on/off 2/1 ?
|
||
argv[8] = "1"; // ??? arg8 2 -- smoothing on/off = 1/0 ?
|
||
|
||
//TODO, this seems like it would normally be done by sysutil etc
|
||
//Basically make 2 128KB memory cards 0 filled and let the games handle formatting.
|
||
|
||
fs::file card_1_file(vfs::get("/dev_hdd0/savedata/vmc/" + argv[1]), fs::write + fs::create);
|
||
card_1_file.trunc(128 * 1024);
|
||
fs::file card_2_file(vfs::get("/dev_hdd0/savedata/vmc/" + argv[2]), fs::write + fs::create);
|
||
card_2_file.trunc(128 * 1024);
|
||
}
|
||
else if (m_cat == "PE" && from_hdd0_game)
|
||
{
|
||
// PSP Remaster located in dev_hdd0/game
|
||
sys_log.notice("PSP Remaster Game: %s, %s", m_title_id, m_title);
|
||
|
||
const std::string game_path = "/dev_hdd0/game/" + m_path.substr(hdd0_game.size(), 9);
|
||
|
||
argv.resize(2);
|
||
argv[0] = "/dev_flash/pspemu/psp_emulator.self";
|
||
argv[1] = game_path;
|
||
}
|
||
else if (m_cat != "DG" && m_cat != "GD")
|
||
{
|
||
// Don't need /dev_bdvd
|
||
}
|
||
else if (m_cat == "DG" && from_hdd0_game)
|
||
{
|
||
// Disc game located in dev_hdd0/game
|
||
vfs::mount("/dev_bdvd/PS3_GAME", hdd0_game + m_path.substr(hdd0_game.size(), 10));
|
||
sys_log.notice("Game: %s", vfs::get("/dev_bdvd/PS3_GAME"));
|
||
}
|
||
else if (disc.empty())
|
||
{
|
||
sys_log.error("Failed to mount disc directory for the disc game %s", m_title_id);
|
||
return game_boot_result::invalid_file_or_folder;
|
||
}
|
||
else
|
||
{
|
||
// Disc game
|
||
bdvd_dir = disc;
|
||
vfs::mount("/dev_bdvd", bdvd_dir);
|
||
sys_log.notice("Disk: %s", vfs::get("/dev_bdvd"));
|
||
}
|
||
|
||
if (add_only)
|
||
{
|
||
sys_log.notice("Finished to add data to games.yml by boot for: %s", m_path);
|
||
m_path = m_path_old; // Reset m_path to fix boot from gui
|
||
return game_boot_result::no_errors;
|
||
}
|
||
|
||
// Initialize progress dialog
|
||
g_fxo->init<named_thread<progress_dialog_server>>();
|
||
|
||
// Set title to actual disc title if necessary
|
||
const std::string disc_sfo_dir = vfs::get("/dev_bdvd/PS3_GAME/PARAM.SFO");
|
||
|
||
const auto disc_psf_obj = psf::load_object(fs::file{ disc_sfo_dir });
|
||
|
||
// Install PKGDIR, INSDIR, PS3_EXTRA
|
||
if (!bdvd_dir.empty())
|
||
{
|
||
std::string ins_dir = vfs::get("/dev_bdvd/PS3_GAME/INSDIR/");
|
||
std::string pkg_dir = vfs::get("/dev_bdvd/PS3_GAME/PKGDIR/");
|
||
std::string extra_dir = vfs::get("/dev_bdvd/PS3_GAME/PS3_EXTRA/");
|
||
fs::file lock_file;
|
||
|
||
for (const auto path_ptr : {&ins_dir, &pkg_dir, &extra_dir})
|
||
{
|
||
if (!fs::is_dir(*path_ptr))
|
||
{
|
||
path_ptr->clear();
|
||
}
|
||
}
|
||
|
||
const std::string lock_file_path = fmt::format("%s%s%s_v%s", hdd0_game, u8"$locks/", m_title_id, psf::get_string(disc_psf_obj, "APP_VER"));
|
||
|
||
if (!ins_dir.empty() || !pkg_dir.empty() || !extra_dir.empty())
|
||
{
|
||
// For backwards compatibility
|
||
if (!lock_file.open(hdd0_game + ".locks/" + m_title_id))
|
||
{
|
||
// Check if already installed
|
||
lock_file.open(lock_file_path);
|
||
}
|
||
}
|
||
|
||
if (!lock_file && !ins_dir.empty())
|
||
{
|
||
sys_log.notice("Found INSDIR: %s", ins_dir);
|
||
|
||
for (auto&& entry : fs::dir{ins_dir})
|
||
{
|
||
const std::string pkg = ins_dir + entry.name;
|
||
if (!entry.is_directory && entry.name.ends_with(".PKG") && !rpcs3::utils::install_pkg(pkg))
|
||
{
|
||
sys_log.error("Failed to install %s", pkg);
|
||
return game_boot_result::install_failed;
|
||
}
|
||
}
|
||
}
|
||
|
||
if (!lock_file && !pkg_dir.empty())
|
||
{
|
||
sys_log.notice("Found PKGDIR: %s", pkg_dir);
|
||
|
||
for (auto&& entry : fs::dir{pkg_dir})
|
||
{
|
||
if (entry.is_directory && entry.name.starts_with("PKG"))
|
||
{
|
||
const std::string pkg_file = pkg_dir + entry.name + "/INSTALL.PKG";
|
||
|
||
if (fs::is_file(pkg_file) && !rpcs3::utils::install_pkg(pkg_file))
|
||
{
|
||
sys_log.error("Failed to install %s", pkg_file);
|
||
return game_boot_result::install_failed;
|
||
}
|
||
}
|
||
}
|
||
}
|
||
|
||
if (!lock_file && !extra_dir.empty())
|
||
{
|
||
sys_log.notice("Found PS3_EXTRA: %s", extra_dir);
|
||
|
||
for (auto&& entry : fs::dir{extra_dir})
|
||
{
|
||
if (entry.is_directory && entry.name[0] == 'D')
|
||
{
|
||
const std::string pkg_file = extra_dir + entry.name + "/DATA000.PKG";
|
||
|
||
if (fs::is_file(pkg_file) && !rpcs3::utils::install_pkg(pkg_file))
|
||
{
|
||
sys_log.error("Failed to install %s", pkg_file);
|
||
return game_boot_result::install_failed;
|
||
}
|
||
}
|
||
}
|
||
}
|
||
|
||
if (!lock_file)
|
||
{
|
||
// Create lock file to prevent double installation
|
||
// Do it after installation to prevent false positives when RPCS3 closed in the middle of the operation
|
||
lock_file.open(lock_file_path, fs::read + fs::create + fs::excl);
|
||
}
|
||
}
|
||
|
||
// Check game updates
|
||
const std::string hdd0_boot = hdd0_game + m_title_id + "/USRDIR/EBOOT.BIN";
|
||
|
||
if (disc.empty() && !bdvd_dir.empty() && fs::resolve_path(m_path) != fs::resolve_path(hdd0_boot) && fs::is_file(hdd0_boot))
|
||
{
|
||
// Booting game update
|
||
sys_log.success("Updates found at /dev_hdd0/game/%s/", m_title_id);
|
||
return m_path = hdd0_boot, Load(m_title_id, false, true);
|
||
}
|
||
|
||
if (!disc_psf_obj.empty())
|
||
{
|
||
const auto bdvd_title = psf::get_string(disc_psf_obj, "TITLE");
|
||
|
||
if (!bdvd_title.empty() && bdvd_title != m_title)
|
||
{
|
||
sys_log.notice("Title was set from %s to %s", m_title, bdvd_title);
|
||
m_title = bdvd_title;
|
||
}
|
||
}
|
||
|
||
for (auto& c : m_title)
|
||
{
|
||
// Replace newlines with spaces
|
||
if (c == '\n')
|
||
c = ' ';
|
||
}
|
||
|
||
// Mount /host_root/ if necessary (special value)
|
||
if (g_cfg.vfs.host_root)
|
||
{
|
||
vfs::mount("/host_root", "/");
|
||
}
|
||
|
||
// Open SELF or ELF
|
||
std::string elf_path = m_path;
|
||
|
||
if (m_cat == "1P" || m_cat == "PE")
|
||
{
|
||
// Use emulator path
|
||
elf_path = vfs::get(argv[0]);
|
||
}
|
||
|
||
fs::file elf_file(elf_path);
|
||
|
||
if (!elf_file)
|
||
{
|
||
sys_log.error("Failed to open executable: %s", elf_path);
|
||
return game_boot_result::invalid_file_or_folder;
|
||
}
|
||
|
||
// Check SELF header
|
||
if (elf_file.size() >= 4 && elf_file.read<u32>() == "SCE\0"_u32)
|
||
{
|
||
const std::string decrypted_path = "boot.elf";
|
||
|
||
fs::stat_t encrypted_stat = elf_file.stat();
|
||
fs::stat_t decrypted_stat;
|
||
|
||
// Check modification time and try to load decrypted ELF
|
||
if (false && fs::stat(decrypted_path, decrypted_stat) && decrypted_stat.mtime == encrypted_stat.mtime)
|
||
{
|
||
elf_file.open(decrypted_path);
|
||
}
|
||
// Decrypt SELF
|
||
else if ((elf_file = decrypt_self(std::move(elf_file), klic.empty() ? nullptr : reinterpret_cast<u8*>(&klic[0]), &g_ps3_process_info.self_info)))
|
||
{
|
||
if (true)
|
||
{
|
||
}
|
||
else if (fs::file elf_out{decrypted_path, fs::rewrite})
|
||
{
|
||
elf_out.write(elf_file.to_vector<u8>());
|
||
elf_out.close();
|
||
fs::utime(decrypted_path, encrypted_stat.atime, encrypted_stat.mtime);
|
||
}
|
||
else
|
||
{
|
||
sys_log.error("Failed to create boot.elf");
|
||
}
|
||
}
|
||
}
|
||
else
|
||
{
|
||
g_ps3_process_info.self_info.valid = false;
|
||
}
|
||
|
||
if (!elf_file)
|
||
{
|
||
sys_log.error("Failed to decrypt SELF: %s", elf_path);
|
||
return game_boot_result::decryption_error;
|
||
}
|
||
|
||
m_state = system_state::ready;
|
||
vm::init();
|
||
|
||
ppu_exec_object ppu_exec;
|
||
ppu_prx_object ppu_prx;
|
||
spu_exec_object spu_exec;
|
||
|
||
if (ppu_exec.open(elf_file) == elf_error::ok)
|
||
{
|
||
// PS3 executable
|
||
GetCallbacks().on_ready();
|
||
|
||
if (argv.empty())
|
||
{
|
||
argv.resize(1);
|
||
}
|
||
|
||
if (argv[0].empty())
|
||
{
|
||
auto unescape = [](std::string_view path)
|
||
{
|
||
// Unescape from host FS
|
||
std::vector<std::string> escaped = fmt::split(path, {std::string_view{&fs::delim[0], 1}, std::string_view{&fs::delim[1], 1}});
|
||
std::vector<std::string> result;
|
||
for (auto& sv : escaped)
|
||
result.emplace_back(vfs::unescape(sv));
|
||
|
||
return fmt::merge(result, "/");
|
||
};
|
||
|
||
const std::string resolved_hdd0 = fs::resolve_path(hdd0_game) + '/';
|
||
|
||
if (from_hdd0_game && m_cat == "DG")
|
||
{
|
||
argv[0] = "/dev_bdvd/PS3_GAME/" + unescape(resolved_path.substr(resolved_hdd0.size() + 10));
|
||
m_dir = "/dev_hdd0/game/" + resolved_path.substr(resolved_hdd0.size(), 10);
|
||
sys_log.notice("Disc path: %s", m_dir);
|
||
}
|
||
else if (from_hdd0_game)
|
||
{
|
||
argv[0] = "/dev_hdd0/game/" + unescape(resolved_path.substr(resolved_hdd0.size()));
|
||
m_dir = "/dev_hdd0/game/" + resolved_path.substr(resolved_hdd0.size(), 10);
|
||
sys_log.notice("Boot path: %s", m_dir);
|
||
}
|
||
else if (!bdvd_dir.empty() && fs::is_dir(bdvd_dir))
|
||
{
|
||
// Disc games are on /dev_bdvd/
|
||
const usz pos = resolved_path.rfind(m_game_dir);
|
||
argv[0] = "/dev_bdvd/PS3_GAME/" + unescape(resolved_path.substr(pos + m_game_dir.size() + 1));
|
||
m_dir = "/dev_bdvd/PS3_GAME/";
|
||
}
|
||
else if (from_dev_flash)
|
||
{
|
||
// Firmware executables
|
||
argv[0] = "/dev_flash" + resolved_path.substr(fs::resolve_path(g_cfg_vfs.get_dev_flash()).size());
|
||
m_dir = fs::get_parent_dir(argv[0]) + '/';
|
||
}
|
||
else if (g_cfg.vfs.host_root)
|
||
{
|
||
// For homebrew
|
||
argv[0] = "/host_root/" + resolved_path;
|
||
m_dir = "/host_root/" + elf_dir + '/';
|
||
}
|
||
else
|
||
{
|
||
// Use /app_home if /host_root is disabled
|
||
argv[0] = "/app_home/" + resolved_path.substr(resolved_path.find_last_of(fs::delim) + 1);
|
||
m_dir = "/app_home/";
|
||
}
|
||
|
||
sys_log.notice("Elf path: %s", argv[0]);
|
||
}
|
||
|
||
if (!argv[0].starts_with("/dev_hdd0/game"sv) && m_cat == "HG"sv)
|
||
{
|
||
sys_log.error("Booting HG category outside of HDD0!");
|
||
}
|
||
|
||
g_fxo->init<ppu_module>();
|
||
g_fxo->init<id_manager::id_map<lv2_obj>>();
|
||
g_fxo->init<id_manager::id_map<named_thread<ppu_thread>>>();
|
||
|
||
if (ppu_load_exec(ppu_exec))
|
||
{
|
||
ConfigurePPUCache();
|
||
|
||
g_fxo->init(false);
|
||
GetCallbacks().init_gs_render();
|
||
GetCallbacks().init_pad_handler(m_title_id);
|
||
GetCallbacks().init_kb_handler();
|
||
GetCallbacks().init_mouse_handler();
|
||
}
|
||
// Overlay (OVL) executable (only load it)
|
||
else if (vm::map(0x3000'0000, 0x1000'0000, 0x200); !ppu_load_overlay(ppu_exec, m_path).first)
|
||
{
|
||
ppu_exec.set_error(elf_error::header_type);
|
||
}
|
||
else
|
||
{
|
||
// Preserve emulation state for OVL excutable
|
||
Pause(true);
|
||
}
|
||
|
||
if (ppu_exec != elf_error::ok)
|
||
{
|
||
SetForceBoot(true);
|
||
Stop();
|
||
|
||
sys_log.error("Invalid or unsupported PPU executable format: %s", elf_path);
|
||
|
||
return game_boot_result::invalid_file_or_folder;
|
||
}
|
||
}
|
||
else if (ppu_prx.open(elf_file) == elf_error::ok)
|
||
{
|
||
// PPU PRX (experimental)
|
||
GetCallbacks().on_ready();
|
||
g_fxo->init(false);
|
||
ppu_load_prx(ppu_prx, m_path);
|
||
Pause(true);
|
||
}
|
||
else if (spu_exec.open(elf_file) == elf_error::ok)
|
||
{
|
||
// SPU executable (experimental)
|
||
GetCallbacks().on_ready();
|
||
g_fxo->init(false);
|
||
spu_load_exec(spu_exec);
|
||
Pause(true);
|
||
}
|
||
else
|
||
{
|
||
sys_log.error("Invalid or unsupported file format: %s", elf_path);
|
||
|
||
sys_log.warning("** ppu_exec -> %s", ppu_exec.get_error());
|
||
sys_log.warning("** ppu_prx -> %s", ppu_prx.get_error());
|
||
sys_log.warning("** spu_exec -> %s", spu_exec.get_error());
|
||
|
||
SetForceBoot(true);
|
||
Stop();
|
||
return game_boot_result::invalid_file_or_folder;
|
||
}
|
||
|
||
if (ppu_exec == elf_error::ok && !fs::is_file(g_cfg_vfs.get_dev_flash() + "sys/external/liblv2.sprx"))
|
||
{
|
||
const auto libs = g_cfg.core.libraries_control.get_set();
|
||
|
||
extern const std::map<std::string_view, int> g_prx_list;
|
||
|
||
// Check if there are any firmware SPRX which may be LLEd during emulation
|
||
// Don't prompt GUI confirmation if there aren't any
|
||
if (std::any_of(g_prx_list.begin(), g_prx_list.end(), [&libs](auto& lib)
|
||
{
|
||
return libs.count(std::string(lib.first) + ":lle") || (!lib.second && !libs.count(std::string(lib.first) + ":hle"));
|
||
}))
|
||
{
|
||
SetForceBoot(true);
|
||
Stop();
|
||
|
||
CallAfter([this]()
|
||
{
|
||
GetCallbacks().on_missing_fw();
|
||
});
|
||
|
||
return game_boot_result::firmware_missing;
|
||
}
|
||
}
|
||
|
||
const bool autostart = (std::exchange(m_force_boot, false) || g_cfg.misc.autostart);
|
||
|
||
if (IsReady())
|
||
{
|
||
if (autostart)
|
||
{
|
||
Run(true);
|
||
}
|
||
}
|
||
|
||
return game_boot_result::no_errors;
|
||
}
|
||
}
|
||
|
||
void Emulator::Run(bool start_playtime)
|
||
{
|
||
ensure(IsReady());
|
||
|
||
GetCallbacks().on_run(start_playtime);
|
||
|
||
m_pause_start_time = 0;
|
||
m_pause_amend_time = 0;
|
||
rpcs3::utils::configure_logs();
|
||
|
||
m_state = system_state::running;
|
||
|
||
// Run main thread
|
||
idm::check<named_thread<ppu_thread>>(ppu_thread::id_base, [](named_thread<ppu_thread>& cpu)
|
||
{
|
||
ensure(cpu.state.test_and_reset(cpu_flag::stop));
|
||
cpu.state.notify_one(cpu_flag::stop);
|
||
});
|
||
|
||
if (auto thr = g_fxo->try_get<named_thread<rsx::rsx_replay_thread>>())
|
||
{
|
||
thr->state -= cpu_flag::stop;
|
||
thr->state.notify_one(cpu_flag::stop);
|
||
}
|
||
|
||
if (g_cfg.misc.prevent_display_sleep)
|
||
{
|
||
disable_display_sleep();
|
||
}
|
||
}
|
||
|
||
bool Emulator::Pause(bool freeze_emulation)
|
||
{
|
||
const u64 start = get_system_time();
|
||
|
||
const system_state pause_state = freeze_emulation ? system_state::frozen : system_state::paused;
|
||
|
||
// Try to pause
|
||
if (!m_state.compare_and_swap_test(system_state::running, pause_state))
|
||
{
|
||
if (!freeze_emulation)
|
||
{
|
||
return false;
|
||
}
|
||
|
||
if (!m_state.compare_and_swap_test(system_state::ready, pause_state))
|
||
{
|
||
if (!m_state.compare_and_swap_test(system_state::paused, pause_state))
|
||
{
|
||
return false;
|
||
}
|
||
}
|
||
}
|
||
|
||
// Signal profilers to print results (if enabled)
|
||
cpu_thread::flush_profilers();
|
||
|
||
GetCallbacks().on_pause();
|
||
|
||
static atomic_t<u32> pause_mark = 0;
|
||
|
||
if (freeze_emulation)
|
||
{
|
||
sys_log.warning("Emulation has been frozen! You can either use debugger tools to inspect current emulation state or terminate it.");
|
||
}
|
||
else
|
||
{
|
||
sys_log.success("Emulation is being paused... (mark=%u)", pause_mark++);
|
||
}
|
||
|
||
// Update pause start time
|
||
if (m_pause_start_time.exchange(start))
|
||
{
|
||
sys_log.error("Emulator::Pause() error: concurrent access");
|
||
}
|
||
|
||
auto on_select = [](u32, cpu_thread& cpu)
|
||
{
|
||
cpu.state += cpu_flag::dbg_global_pause;
|
||
};
|
||
|
||
idm::select<named_thread<ppu_thread>>(on_select);
|
||
idm::select<named_thread<spu_thread>>(on_select);
|
||
|
||
if (auto rsx = g_fxo->try_get<rsx::thread>())
|
||
{
|
||
rsx->state += cpu_flag::dbg_global_pause;
|
||
}
|
||
|
||
// Always Enable display sleep, not only if it was prevented.
|
||
enable_display_sleep();
|
||
|
||
return true;
|
||
}
|
||
|
||
void Emulator::Resume()
|
||
{
|
||
// Get pause start time
|
||
const u64 time = m_pause_start_time.exchange(0);
|
||
|
||
// Try to increment summary pause time
|
||
if (time)
|
||
{
|
||
m_pause_amend_time += get_system_time() - time;
|
||
}
|
||
|
||
// Print and reset debug data collected
|
||
if (m_state == system_state::paused && g_cfg.core.ppu_debug)
|
||
{
|
||
PPUDisAsm dis_asm(cpu_disasm_mode::dump, vm::g_sudo_addr);
|
||
|
||
std::string dump;
|
||
|
||
for (u32 i = 0x10000; i < 0xE0000000;)
|
||
{
|
||
if (vm::check_addr(i, vm::page_executable))
|
||
{
|
||
if (auto& data = *reinterpret_cast<be_t<u32>*>(vm::g_stat_addr + i))
|
||
{
|
||
dis_asm.disasm(i);
|
||
fmt::append(dump, "\n\t'%08X' %s", data, dis_asm.last_opcode);
|
||
data = 0;
|
||
}
|
||
|
||
i += sizeof(u32);
|
||
}
|
||
else
|
||
{
|
||
i += 4096;
|
||
}
|
||
}
|
||
|
||
ppu_log.notice("[RESUME] Dumping instruction stats:%s", dump);
|
||
}
|
||
|
||
perf_stat_base::report();
|
||
|
||
// Try to resume
|
||
if (!m_state.compare_and_swap_test(system_state::paused, system_state::running))
|
||
{
|
||
return;
|
||
}
|
||
|
||
if (!time)
|
||
{
|
||
sys_log.error("Emulator::Resume() error: concurrent access");
|
||
}
|
||
|
||
auto on_select = [](u32, cpu_thread& cpu)
|
||
{
|
||
cpu.state -= cpu_flag::dbg_global_pause;
|
||
cpu.state.notify_one(cpu_flag::dbg_global_pause);
|
||
};
|
||
|
||
idm::select<named_thread<ppu_thread>>(on_select);
|
||
idm::select<named_thread<spu_thread>>(on_select);
|
||
|
||
if (auto rsx = g_fxo->try_get<rsx::thread>())
|
||
{
|
||
// TODO: notify?
|
||
rsx->state -= cpu_flag::dbg_global_pause;
|
||
}
|
||
|
||
GetCallbacks().on_resume();
|
||
|
||
sys_log.success("Emulation has been resumed!");
|
||
|
||
if (g_cfg.misc.prevent_display_sleep)
|
||
{
|
||
disable_display_sleep();
|
||
}
|
||
}
|
||
|
||
void Emulator::Stop(bool restart)
|
||
{
|
||
if (m_state.exchange(system_state::stopped) == system_state::stopped)
|
||
{
|
||
if (restart)
|
||
{
|
||
// Reload with prior configs.
|
||
if (const auto error = Load(m_title_id); error != game_boot_result::no_errors)
|
||
sys_log.error("Restart failed: %s", error);
|
||
return;
|
||
}
|
||
|
||
m_force_boot = false;
|
||
return;
|
||
}
|
||
|
||
sys_log.notice("Stopping emulator...");
|
||
|
||
if (!restart)
|
||
{
|
||
// Show visual feedback to the user in case that stopping takes a while.
|
||
// This needs to be done before actually stopping, because otherwise the necessary threads will be terminated before we can show an image.
|
||
if (auto progress_dialog = g_fxo->try_get<named_thread<progress_dialog_server>>(); progress_dialog && +g_progr)
|
||
{
|
||
// We are currently showing a progress dialog. Notify it that we are going to stop emulation.
|
||
g_system_progress_stopping = true;
|
||
std::this_thread::sleep_for(20ms); // Enough for one frame to be rendered
|
||
}
|
||
}
|
||
|
||
named_thread stop_watchdog("Stop Watchdog", [&]()
|
||
{
|
||
for (uint i = 0; thread_ctrl::state() != thread_state::aborting;)
|
||
{
|
||
// We don't need accurate timekeeping, using clocks may interfere with debugging
|
||
if (i >= 1000)
|
||
{
|
||
// Total amount of waiting: about 5s
|
||
report_fatal_error("Stopping emulator took too long."
|
||
"\nSome thread has probably deadlocked. Aborting.");
|
||
}
|
||
|
||
thread_ctrl::wait_for(5'000);
|
||
|
||
if (!g_watchdog_hold_ctr)
|
||
{
|
||
// Don't count if there are still uninterruptable threads like PPU LLVM workers
|
||
i++;
|
||
}
|
||
}
|
||
});
|
||
|
||
// Signal threads
|
||
if (auto rsx = g_fxo->try_get<rsx::thread>())
|
||
{
|
||
*static_cast<cpu_thread*>(rsx) = thread_state::aborting;
|
||
}
|
||
|
||
for (const auto& [type, data] : *g_fxo)
|
||
{
|
||
if (type.stop)
|
||
{
|
||
type.stop(data, thread_state::aborting);
|
||
}
|
||
}
|
||
|
||
sys_log.notice("All emulation threads have been signaled.");
|
||
|
||
// Wait fot newly created cpu_thread to see that emulation has been stopped
|
||
id_manager::g_mutex.lock_unlock();
|
||
|
||
GetCallbacks().on_stop();
|
||
|
||
// Join threads
|
||
for (const auto& [type, data] : *g_fxo)
|
||
{
|
||
if (type.stop)
|
||
{
|
||
type.stop(data, thread_state::finished);
|
||
}
|
||
}
|
||
|
||
cpu_thread::cleanup();
|
||
|
||
g_fxo->reset();
|
||
|
||
sys_log.notice("All threads have been stopped.");
|
||
|
||
lv2_obj::cleanup();
|
||
|
||
sys_log.notice("Objects cleared...");
|
||
|
||
vm::close();
|
||
|
||
jit_runtime::finalize();
|
||
|
||
perf_stat_base::report();
|
||
|
||
static u64 aw_refs = 0;
|
||
static u64 aw_colm = 0;
|
||
static u64 aw_colc = 0;
|
||
static u64 aw_used = 0;
|
||
|
||
aw_refs = 0;
|
||
aw_colm = 0;
|
||
aw_colc = 0;
|
||
aw_used = 0;
|
||
|
||
atomic_wait::parse_hashtable([](u64 /*id*/, u32 refs, u64 ptr, u32 maxc) -> bool
|
||
{
|
||
aw_refs += refs != 0;
|
||
aw_used += ptr != 0;
|
||
|
||
aw_colm = std::max<u64>(aw_colm, maxc);
|
||
aw_colc += maxc != 0;
|
||
|
||
return false;
|
||
});
|
||
|
||
sys_log.notice("Atomic wait hashtable stats: [in_use=%u, used=%u, max_collision_weight=%u, total_collisions=%u]", aw_refs, aw_used, aw_colm, aw_colc);
|
||
|
||
stop_watchdog = thread_state::aborting;
|
||
|
||
m_stop_ctr++;
|
||
m_stop_ctr.notify_all();
|
||
|
||
if (restart)
|
||
{
|
||
// Reload with prior configs.
|
||
if (const auto error = Load(m_title_id); error != game_boot_result::no_errors)
|
||
sys_log.error("Restart failed: %s", error);
|
||
return;
|
||
}
|
||
|
||
// Boot arg cleanup (preserved in the case restarting)
|
||
argv.clear();
|
||
envp.clear();
|
||
data.clear();
|
||
disc.clear();
|
||
klic.clear();
|
||
hdd1.clear();
|
||
m_config_path.clear();
|
||
m_config_mode = cfg_mode::custom;
|
||
|
||
// Always Enable display sleep, not only if it was prevented.
|
||
enable_display_sleep();
|
||
|
||
if (!m_force_boot)
|
||
{
|
||
if (Quit(g_cfg.misc.autoexit.get()))
|
||
{
|
||
return;
|
||
}
|
||
}
|
||
|
||
m_force_boot = false;
|
||
}
|
||
|
||
bool Emulator::Quit(bool force_quit)
|
||
{
|
||
m_force_boot = false;
|
||
|
||
// The callback is only used if we actually quit RPCS3
|
||
const auto on_exit = []()
|
||
{
|
||
Emu.CleanUp();
|
||
};
|
||
|
||
if (GetCallbacks().try_to_quit)
|
||
{
|
||
return GetCallbacks().try_to_quit(force_quit, on_exit);
|
||
}
|
||
|
||
on_exit();
|
||
return true;
|
||
}
|
||
|
||
void Emulator::CleanUp()
|
||
{
|
||
// Deinitialize object manager to prevent any hanging objects at program exit
|
||
g_fxo->clear();
|
||
}
|
||
|
||
std::string Emulator::GetFormattedTitle(double fps) const
|
||
{
|
||
rpcs3::title_format_data title_data;
|
||
title_data.format = g_cfg.misc.title_format.to_string();
|
||
title_data.title = GetTitle();
|
||
title_data.title_id = GetTitleID();
|
||
title_data.renderer = g_cfg.video.renderer.to_string();
|
||
title_data.vulkan_adapter = g_cfg.video.vk.adapter.to_string();
|
||
title_data.fps = fps;
|
||
|
||
return rpcs3::get_formatted_title(title_data);
|
||
}
|
||
|
||
s32 error_code::error_report(s32 result, const char* fmt, const fmt_type_info* sup, const u64* args)
|
||
{
|
||
static thread_local std::unordered_map<std::string, usz> g_tls_error_stats;
|
||
static thread_local std::string g_tls_error_str;
|
||
|
||
if (!sup && !args)
|
||
{
|
||
if (!fmt)
|
||
{
|
||
// Report and clean error state
|
||
for (auto&& pair : g_tls_error_stats)
|
||
{
|
||
if (pair.second > 3)
|
||
{
|
||
sys_log.error("Stat: %s [x%u]", pair.first, pair.second);
|
||
}
|
||
}
|
||
|
||
g_tls_error_stats.clear();
|
||
return 0;
|
||
}
|
||
}
|
||
|
||
logs::channel* channel = &sys_log;
|
||
const char* func = "Unknown function";
|
||
|
||
if (auto ppu = get_current_cpu_thread<ppu_thread>())
|
||
{
|
||
if (ppu->current_function)
|
||
{
|
||
func = ppu->current_function;
|
||
}
|
||
}
|
||
|
||
// Format log message (use preallocated buffer)
|
||
g_tls_error_str.clear();
|
||
fmt::append(g_tls_error_str, "'%s' failed with 0x%08x", func, result);
|
||
fmt::raw_append(g_tls_error_str, fmt, sup, args);
|
||
|
||
// Update stats and check log threshold
|
||
const auto stat = ++g_tls_error_stats[g_tls_error_str];
|
||
|
||
if (stat <= 3)
|
||
{
|
||
channel->error("%s [%u]", g_tls_error_str, stat);
|
||
}
|
||
|
||
return result;
|
||
}
|
||
|
||
void Emulator::ConfigurePPUCache() const
|
||
{
|
||
auto& _main = g_fxo->get<ppu_module>();
|
||
|
||
_main.cache = rpcs3::utils::get_cache_dir();
|
||
|
||
if (!m_title_id.empty() && m_cat != "1P")
|
||
{
|
||
_main.cache += GetTitleID();
|
||
_main.cache += '/';
|
||
}
|
||
|
||
fmt::append(_main.cache, "ppu-%s-%s/", fmt::base57(_main.sha1), _main.path.substr(_main.path.find_last_of('/') + 1));
|
||
|
||
if (!fs::create_path(_main.cache))
|
||
{
|
||
sys_log.error("Failed to create cache directory: %s (%s)", _main.cache, fs::g_tls_error);
|
||
}
|
||
else
|
||
{
|
||
sys_log.notice("Cache: %s", _main.cache);
|
||
}
|
||
}
|
||
|
||
std::set<std::string> Emulator::GetGameDirs() const
|
||
{
|
||
std::set<std::string> dirs;
|
||
|
||
// Add boot directory.
|
||
// For installed titles and disc titles with updates this is usually /dev_hdd0/game/<title_id>/
|
||
// For disc titles without updates this is /dev_bdvd/PS3_GAME/
|
||
if (const std::string dir = vfs::get(GetDir()); !dir.empty())
|
||
{
|
||
dirs.insert(dir + '/');
|
||
}
|
||
|
||
// Add more paths for disc titles.
|
||
if (const std::string dev_bdvd = vfs::get("/dev_bdvd/PS3_GAME");
|
||
!dev_bdvd.empty() && !GetTitleID().empty())
|
||
{
|
||
// Add the dev_bdvd dir if available. This is necessary for disc titles with installed updates.
|
||
dirs.insert(dev_bdvd + '/');
|
||
|
||
// Naive search for all matching game data dirs.
|
||
const std::string game_dir = vfs::get("/dev_hdd0/game/");
|
||
for (auto&& entry : fs::dir(game_dir))
|
||
{
|
||
if (entry.is_directory && entry.name.starts_with(GetTitleID()))
|
||
{
|
||
const std::string sfo_dir = game_dir + entry.name + '/';
|
||
const fs::file sfo_file(sfo_dir + "PARAM.SFO");
|
||
if (!sfo_file)
|
||
{
|
||
continue;
|
||
}
|
||
|
||
const psf::registry psf = psf::load_object(sfo_file);
|
||
const std::string title_id = std::string(psf::get_string(psf, "TITLE_ID", ""));
|
||
|
||
if (title_id == GetTitleID())
|
||
{
|
||
dirs.insert(sfo_dir);
|
||
}
|
||
}
|
||
}
|
||
}
|
||
|
||
return dirs;
|
||
}
|
||
|
||
bool Emulator::IsPathInsideDir(std::string_view path, std::string_view dir) const
|
||
{
|
||
const std::string dir_path = fs::resolve_path(dir);
|
||
|
||
return !dir_path.empty() && (fs::resolve_path(path) + '/').starts_with(dir_path + '/');
|
||
};
|
||
|
||
const std::string& Emulator::GetFakeCat() const
|
||
{
|
||
if (m_cat == "DG")
|
||
{
|
||
const std::string mount_point = vfs::get("/dev_bdvd");
|
||
|
||
if (mount_point.empty() || !IsPathInsideDir(m_path, mount_point))
|
||
{
|
||
static const std::string s_hg = "HG";
|
||
return s_hg;
|
||
}
|
||
}
|
||
|
||
return m_cat;
|
||
};
|
||
|
||
Emulator Emu;
|