diff options
Diffstat (limited to 'src/core')
| -rw-r--r-- | src/core/arm/dynarmic/arm_dynarmic_32.cpp | 30 | ||||
| -rw-r--r-- | src/core/arm/dynarmic/arm_dynarmic_64.cpp | 34 | ||||
| -rw-r--r-- | src/core/file_sys/bis_factory.cpp | 23 | ||||
| -rw-r--r-- | src/core/file_sys/registered_cache.cpp | 98 | ||||
| -rw-r--r-- | src/core/file_sys/registered_cache.h | 4 | ||||
| -rw-r--r-- | src/core/file_sys/sdmc_factory.cpp | 4 | ||||
| -rw-r--r-- | src/core/file_sys/vfs_real.cpp | 27 | ||||
| -rw-r--r-- | src/core/hle/kernel/handle_table.cpp | 8 | ||||
| -rw-r--r-- | src/core/hle/kernel/handle_table.h | 7 | ||||
| -rw-r--r-- | src/core/hle/kernel/kernel.cpp | 5 | ||||
| -rw-r--r-- | src/core/hle/kernel/memory/memory_layout.h | 2 | ||||
| -rw-r--r-- | src/core/hle/kernel/process.cpp | 2 | ||||
| -rw-r--r-- | src/core/hle/kernel/process.h | 12 | ||||
| -rw-r--r-- | src/core/hle/kernel/thread.cpp | 22 | ||||
| -rw-r--r-- | src/core/hle/kernel/thread.h | 5 | ||||
| -rw-r--r-- | src/core/hle/service/am/am.cpp | 14 | ||||
| -rw-r--r-- | src/core/settings.h | 47 | 
17 files changed, 235 insertions, 109 deletions
| diff --git a/src/core/arm/dynarmic/arm_dynarmic_32.cpp b/src/core/arm/dynarmic/arm_dynarmic_32.cpp index 0d4ab95b7..443ca72eb 100644 --- a/src/core/arm/dynarmic/arm_dynarmic_32.cpp +++ b/src/core/arm/dynarmic/arm_dynarmic_32.cpp @@ -142,10 +142,32 @@ std::shared_ptr<Dynarmic::A32::Jit> ARM_Dynarmic_32::MakeJit(Common::PageTable&      // Timing      config.wall_clock_cntpct = uses_wall_clock; -    // Optimizations -    if (Settings::values.disable_cpu_opt) { -        config.enable_optimizations = false; -        config.enable_fast_dispatch = false; +    // Safe optimizations +    if (Settings::values.cpu_accuracy != Settings::CPUAccuracy::Accurate) { +        if (!Settings::values.cpuopt_page_tables) { +            config.page_table = nullptr; +        } +        if (!Settings::values.cpuopt_block_linking) { +            config.optimizations &= ~Dynarmic::OptimizationFlag::BlockLinking; +        } +        if (!Settings::values.cpuopt_return_stack_buffer) { +            config.optimizations &= ~Dynarmic::OptimizationFlag::ReturnStackBuffer; +        } +        if (!Settings::values.cpuopt_fast_dispatcher) { +            config.optimizations &= ~Dynarmic::OptimizationFlag::FastDispatch; +        } +        if (!Settings::values.cpuopt_context_elimination) { +            config.optimizations &= ~Dynarmic::OptimizationFlag::GetSetElimination; +        } +        if (!Settings::values.cpuopt_const_prop) { +            config.optimizations &= ~Dynarmic::OptimizationFlag::ConstProp; +        } +        if (!Settings::values.cpuopt_misc_ir) { +            config.optimizations &= ~Dynarmic::OptimizationFlag::MiscIROpt; +        } +        if (!Settings::values.cpuopt_reduce_misalign_checks) { +            config.only_detect_misalignment_via_page_table_on_page_boundary = false; +        }      }      return std::make_unique<Dynarmic::A32::Jit>(config); diff --git a/src/core/arm/dynarmic/arm_dynarmic_64.cpp b/src/core/arm/dynarmic/arm_dynarmic_64.cpp index 790981034..a63a04a25 100644 --- a/src/core/arm/dynarmic/arm_dynarmic_64.cpp +++ b/src/core/arm/dynarmic/arm_dynarmic_64.cpp @@ -191,15 +191,37 @@ std::shared_ptr<Dynarmic::A64::Jit> ARM_Dynarmic_64::MakeJit(Common::PageTable&      // Unpredictable instructions      config.define_unpredictable_behaviour = true; -    // Optimizations -    if (Settings::values.disable_cpu_opt) { -        config.enable_optimizations = false; -        config.enable_fast_dispatch = false; -    } -      // Timing      config.wall_clock_cntpct = uses_wall_clock; +    // Safe optimizations +    if (Settings::values.cpu_accuracy != Settings::CPUAccuracy::Accurate) { +        if (!Settings::values.cpuopt_page_tables) { +            config.page_table = nullptr; +        } +        if (!Settings::values.cpuopt_block_linking) { +            config.optimizations &= ~Dynarmic::OptimizationFlag::BlockLinking; +        } +        if (!Settings::values.cpuopt_return_stack_buffer) { +            config.optimizations &= ~Dynarmic::OptimizationFlag::ReturnStackBuffer; +        } +        if (!Settings::values.cpuopt_fast_dispatcher) { +            config.optimizations &= ~Dynarmic::OptimizationFlag::FastDispatch; +        } +        if (!Settings::values.cpuopt_context_elimination) { +            config.optimizations &= ~Dynarmic::OptimizationFlag::GetSetElimination; +        } +        if (!Settings::values.cpuopt_const_prop) { +            config.optimizations &= ~Dynarmic::OptimizationFlag::ConstProp; +        } +        if (!Settings::values.cpuopt_misc_ir) { +            config.optimizations &= ~Dynarmic::OptimizationFlag::MiscIROpt; +        } +        if (!Settings::values.cpuopt_reduce_misalign_checks) { +            config.only_detect_misalignment_via_page_table_on_page_boundary = false; +        } +    } +      return std::make_shared<Dynarmic::A64::Jit>(config);  } diff --git a/src/core/file_sys/bis_factory.cpp b/src/core/file_sys/bis_factory.cpp index 8935a62c3..285277ef8 100644 --- a/src/core/file_sys/bis_factory.cpp +++ b/src/core/file_sys/bis_factory.cpp @@ -12,6 +12,10 @@  namespace FileSys { +constexpr u64 NAND_USER_SIZE = 0x680000000;  // 26624 MiB +constexpr u64 NAND_SYSTEM_SIZE = 0xA0000000; // 2560 MiB +constexpr u64 NAND_TOTAL_SIZE = 0x747C00000; // 29820 MiB +  BISFactory::BISFactory(VirtualDir nand_root_, VirtualDir load_root_, VirtualDir dump_root_)      : nand_root(std::move(nand_root_)), load_root(std::move(load_root_)),        dump_root(std::move(dump_root_)), @@ -110,30 +114,29 @@ VirtualDir BISFactory::GetImageDirectory() const {  u64 BISFactory::GetSystemNANDFreeSpace() const {      const auto sys_dir = GetOrCreateDirectoryRelative(nand_root, "/system"); -    if (sys_dir == nullptr) -        return 0; +    if (sys_dir == nullptr) { +        return GetSystemNANDTotalSpace(); +    }      return GetSystemNANDTotalSpace() - sys_dir->GetSize();  }  u64 BISFactory::GetSystemNANDTotalSpace() const { -    return static_cast<u64>(Settings::values.nand_system_size); +    return NAND_SYSTEM_SIZE;  }  u64 BISFactory::GetUserNANDFreeSpace() const { -    const auto usr_dir = GetOrCreateDirectoryRelative(nand_root, "/user"); -    if (usr_dir == nullptr) -        return 0; - -    return GetUserNANDTotalSpace() - usr_dir->GetSize(); +    // For some reason games such as BioShock 1 checks whether this is exactly 0x680000000 bytes. +    // Set the free space to be 1 MiB less than the total as a workaround to this issue. +    return GetUserNANDTotalSpace() - 0x100000;  }  u64 BISFactory::GetUserNANDTotalSpace() const { -    return static_cast<u64>(Settings::values.nand_user_size); +    return NAND_USER_SIZE;  }  u64 BISFactory::GetFullNANDTotalSpace() const { -    return static_cast<u64>(Settings::values.nand_total_size); +    return NAND_TOTAL_SIZE;  }  VirtualDir BISFactory::GetBCATDirectory(u64 title_id) const { diff --git a/src/core/file_sys/registered_cache.cpp b/src/core/file_sys/registered_cache.cpp index 27c1b0233..37351c561 100644 --- a/src/core/file_sys/registered_cache.cpp +++ b/src/core/file_sys/registered_cache.cpp @@ -547,6 +547,56 @@ InstallResult RegisteredCache::InstallEntry(const XCI& xci, bool overwrite_if_ex      return InstallEntry(*xci.GetSecurePartitionNSP(), overwrite_if_exists, copy);  } +bool RegisteredCache::RemoveExistingEntry(u64 title_id) { +    const auto delete_nca = [this](const NcaID& id) { +        const auto path = GetRelativePathFromNcaID(id, false, true, false); + +        if (dir->GetFileRelative(path) == nullptr) { +            return false; +        } + +        Core::Crypto::SHA256Hash hash{}; +        mbedtls_sha256_ret(id.data(), id.size(), hash.data(), 0); +        const auto dirname = fmt::format("000000{:02X}", hash[0]); + +        const auto dir2 = GetOrCreateDirectoryRelative(dir, dirname); + +        const auto res = dir2->DeleteFile(fmt::format("{}.nca", Common::HexToString(id, false))); + +        return res; +    }; + +    // If an entry exists in the registered cache, remove it +    if (HasEntry(title_id, ContentRecordType::Meta)) { +        LOG_INFO(Loader, +                 "Previously installed entry (v{}) for title_id={:016X} detected! " +                 "Attempting to remove...", +                 GetEntryVersion(title_id).value_or(0), title_id); +        // Get all the ncas associated with the current CNMT and delete them +        const auto meta_old_id = +            GetNcaIDFromMetadata(title_id, ContentRecordType::Meta).value_or(NcaID{}); +        const auto program_id = +            GetNcaIDFromMetadata(title_id, ContentRecordType::Program).value_or(NcaID{}); +        const auto data_id = +            GetNcaIDFromMetadata(title_id, ContentRecordType::Data).value_or(NcaID{}); +        const auto control_id = +            GetNcaIDFromMetadata(title_id, ContentRecordType::Control).value_or(NcaID{}); +        const auto html_id = +            GetNcaIDFromMetadata(title_id, ContentRecordType::HtmlDocument).value_or(NcaID{}); +        const auto legal_id = +            GetNcaIDFromMetadata(title_id, ContentRecordType::LegalInformation).value_or(NcaID{}); + +        delete_nca(meta_old_id); +        delete_nca(program_id); +        delete_nca(data_id); +        delete_nca(control_id); +        delete_nca(html_id); +        delete_nca(legal_id); +        return true; +    } +    return false; +} +  InstallResult RegisteredCache::InstallEntry(const NSP& nsp, bool overwrite_if_exists,                                              const VfsCopyFunction& copy) {      const auto ncas = nsp.GetNCAsCollapsed(); @@ -560,31 +610,57 @@ InstallResult RegisteredCache::InstallEntry(const NSP& nsp, bool overwrite_if_ex          return InstallResult::ErrorMetaFailed;      } -    // Install Metadata File      const auto meta_id_raw = (*meta_iter)->GetName().substr(0, 32);      const auto meta_id = Common::HexStringToArray<16>(meta_id_raw); -    const auto res = RawInstallNCA(**meta_iter, copy, overwrite_if_exists, meta_id); -    if (res != InstallResult::Success) -        return res; +    if ((*meta_iter)->GetSubdirectories().empty()) { +        LOG_ERROR(Loader, +                  "The file you are attempting to install does not contain a section0 within the " +                  "metadata NCA and is therefore malformed. Verify that the file is valid."); +        return InstallResult::ErrorMetaFailed; +    } -    // Install all the other NCAs      const auto section0 = (*meta_iter)->GetSubdirectories()[0]; + +    if (section0->GetFiles().empty()) { +        LOG_ERROR(Loader, +                  "The file you are attempting to install does not contain a CNMT within the " +                  "metadata NCA and is therefore malformed. Verify that the file is valid."); +        return InstallResult::ErrorMetaFailed; +    } +      const auto cnmt_file = section0->GetFiles()[0];      const CNMT cnmt(cnmt_file); + +    const auto title_id = cnmt.GetTitleID(); +    const auto result = RemoveExistingEntry(title_id); + +    // Install Metadata File +    const auto res = RawInstallNCA(**meta_iter, copy, overwrite_if_exists, meta_id); +    if (res != InstallResult::Success) { +        return res; +    } + +    // Install all the other NCAs      for (const auto& record : cnmt.GetContentRecords()) {          // Ignore DeltaFragments, they are not useful to us -        if (record.type == ContentRecordType::DeltaFragment) +        if (record.type == ContentRecordType::DeltaFragment) {              continue; +        }          const auto nca = GetNCAFromNSPForID(nsp, record.nca_id); -        if (nca == nullptr) +        if (nca == nullptr) {              return InstallResult::ErrorCopyFailed; +        }          const auto res2 = RawInstallNCA(*nca, copy, overwrite_if_exists, record.nca_id); -        if (res2 != InstallResult::Success) +        if (res2 != InstallResult::Success) {              return res2; +        }      }      Refresh(); +    if (result) { +        return InstallResult::OverwriteExisting; +    }      return InstallResult::Success;  } @@ -610,8 +686,9 @@ InstallResult RegisteredCache::InstallEntry(const NCA& nca, TitleType type,      mbedtls_sha256_ret(data.data(), data.size(), c_rec.hash.data(), 0);      memcpy(&c_rec.nca_id, &c_rec.hash, 16);      const CNMT new_cnmt(header, opt_header, {c_rec}, {}); -    if (!RawInstallYuzuMeta(new_cnmt)) +    if (!RawInstallYuzuMeta(new_cnmt)) {          return InstallResult::ErrorMetaFailed; +    }      return RawInstallNCA(nca, copy, overwrite_if_exists, c_rec.nca_id);  } @@ -649,8 +726,9 @@ InstallResult RegisteredCache::RawInstallNCA(const NCA& nca, const VfsCopyFuncti      }      auto out = dir->CreateFileRelative(path); -    if (out == nullptr) +    if (out == nullptr) {          return InstallResult::ErrorCopyFailed; +    }      return copy(in, out, VFS_RC_LARGE_COPY_BLOCK) ? InstallResult::Success                                                    : InstallResult::ErrorCopyFailed;  } diff --git a/src/core/file_sys/registered_cache.h b/src/core/file_sys/registered_cache.h index f339cd17b..29cf0d40c 100644 --- a/src/core/file_sys/registered_cache.h +++ b/src/core/file_sys/registered_cache.h @@ -34,6 +34,7 @@ using VfsCopyFunction = std::function<bool(const VirtualFile&, const VirtualFile  enum class InstallResult {      Success, +    OverwriteExisting,      ErrorAlreadyExists,      ErrorCopyFailed,      ErrorMetaFailed, @@ -154,6 +155,9 @@ public:          std::optional<TitleType> title_type = {}, std::optional<ContentRecordType> record_type = {},          std::optional<u64> title_id = {}) const override; +    // Removes an existing entry based on title id +    bool RemoveExistingEntry(u64 title_id); +      // Raw copies all the ncas from the xci/nsp to the csache. Does some quick checks to make sure      // there is a meta NCA and all of them are accessible.      InstallResult InstallEntry(const XCI& xci, bool overwrite_if_exists = false, diff --git a/src/core/file_sys/sdmc_factory.cpp b/src/core/file_sys/sdmc_factory.cpp index 5113a1ca6..6f732e4d8 100644 --- a/src/core/file_sys/sdmc_factory.cpp +++ b/src/core/file_sys/sdmc_factory.cpp @@ -10,6 +10,8 @@  namespace FileSys { +constexpr u64 SDMC_TOTAL_SIZE = 0x10000000000; // 1 TiB +  SDMCFactory::SDMCFactory(VirtualDir dir_)      : dir(std::move(dir_)), contents(std::make_unique<RegisteredCache>(                                  GetOrCreateDirectoryRelative(dir, "/Nintendo/Contents/registered"), @@ -46,7 +48,7 @@ u64 SDMCFactory::GetSDMCFreeSpace() const {  }  u64 SDMCFactory::GetSDMCTotalSpace() const { -    return static_cast<u64>(Settings::values.sdmc_size); +    return SDMC_TOTAL_SIZE;  }  } // namespace FileSys diff --git a/src/core/file_sys/vfs_real.cpp b/src/core/file_sys/vfs_real.cpp index e21300a7c..96ce5957c 100644 --- a/src/core/file_sys/vfs_real.cpp +++ b/src/core/file_sys/vfs_real.cpp @@ -112,19 +112,26 @@ VirtualFile RealVfsFilesystem::MoveFile(std::string_view old_path_, std::string_      const auto new_path =          FileUtil::SanitizePath(new_path_, FileUtil::DirectorySeparator::PlatformDefault); -    if (!FileUtil::Exists(old_path) || FileUtil::Exists(new_path) || -        FileUtil::IsDirectory(old_path) || !FileUtil::Rename(old_path, new_path)) -        return nullptr; -      if (cache.find(old_path) != cache.end()) { -        auto cached = cache[old_path]; -        if (!cached.expired()) { -            auto file = cached.lock(); -            file->Open(new_path, "r+b"); -            cache.erase(old_path); -            cache[new_path] = file; +        auto file = cache[old_path].lock(); + +        if (!cache[old_path].expired()) { +            file->Close(); +        } + +        if (!FileUtil::Exists(old_path) || FileUtil::Exists(new_path) || +            FileUtil::IsDirectory(old_path) || !FileUtil::Rename(old_path, new_path)) { +            return nullptr;          } + +        cache.erase(old_path); +        file->Open(new_path, "r+b"); +        cache[new_path] = file; +    } else { +        UNREACHABLE(); +        return nullptr;      } +      return OpenFile(new_path, Mode::ReadWrite);  } diff --git a/src/core/hle/kernel/handle_table.cpp b/src/core/hle/kernel/handle_table.cpp index 35448b576..fb30b6f8b 100644 --- a/src/core/hle/kernel/handle_table.cpp +++ b/src/core/hle/kernel/handle_table.cpp @@ -8,7 +8,9 @@  #include "core/core.h"  #include "core/hle/kernel/errors.h"  #include "core/hle/kernel/handle_table.h" +#include "core/hle/kernel/kernel.h"  #include "core/hle/kernel/process.h" +#include "core/hle/kernel/scheduler.h"  #include "core/hle/kernel/thread.h"  namespace Kernel { @@ -22,7 +24,7 @@ constexpr u16 GetGeneration(Handle handle) {  }  } // Anonymous namespace -HandleTable::HandleTable() { +HandleTable::HandleTable(KernelCore& kernel) : kernel{kernel} {      Clear();  } @@ -103,9 +105,9 @@ bool HandleTable::IsValid(Handle handle) const {  std::shared_ptr<Object> HandleTable::GetGeneric(Handle handle) const {      if (handle == CurrentThread) { -        return SharedFrom(GetCurrentThread()); +        return SharedFrom(kernel.CurrentScheduler().GetCurrentThread());      } else if (handle == CurrentProcess) { -        return SharedFrom(Core::System::GetInstance().CurrentProcess()); +        return SharedFrom(kernel.CurrentProcess());      }      if (!IsValid(handle)) { diff --git a/src/core/hle/kernel/handle_table.h b/src/core/hle/kernel/handle_table.h index 8029660ed..c9dab8cdd 100644 --- a/src/core/hle/kernel/handle_table.h +++ b/src/core/hle/kernel/handle_table.h @@ -14,6 +14,8 @@  namespace Kernel { +class KernelCore; +  enum KernelHandle : Handle {      InvalidHandle = 0,      CurrentThread = 0xFFFF8000, @@ -48,7 +50,7 @@ public:      /// This is the maximum limit of handles allowed per process in Horizon      static constexpr std::size_t MAX_COUNT = 1024; -    HandleTable(); +    explicit HandleTable(KernelCore& kernel);      ~HandleTable();      /** @@ -134,6 +136,9 @@ private:      /// Head of the free slots linked list.      u16 next_free_slot = 0; + +    /// Underlying kernel instance that this handle table operates under. +    KernelCore& kernel;  };  } // namespace Kernel diff --git a/src/core/hle/kernel/kernel.cpp b/src/core/hle/kernel/kernel.cpp index 1f2af7a1b..6e2014e08 100644 --- a/src/core/hle/kernel/kernel.cpp +++ b/src/core/hle/kernel/kernel.cpp @@ -50,7 +50,8 @@ namespace Kernel {  struct KernelCore::Impl {      explicit Impl(Core::System& system, KernelCore& kernel) -        : global_scheduler{kernel}, synchronization{system}, time_manager{system}, system{system} {} +        : global_scheduler{kernel}, synchronization{system}, time_manager{system}, +          global_handle_table{kernel}, system{system} {}      void SetMulticore(bool is_multicore) {          this->is_multicore = is_multicore; @@ -307,7 +308,7 @@ struct KernelCore::Impl {      // This is the kernel's handle table or supervisor handle table which      // stores all the objects in place. -    Kernel::HandleTable global_handle_table; +    HandleTable global_handle_table;      /// Map of named ports managed by the kernel, which can be retrieved using      /// the ConnectToPort SVC. diff --git a/src/core/hle/kernel/memory/memory_layout.h b/src/core/hle/kernel/memory/memory_layout.h index 830c6f0d7..9b3d6267a 100644 --- a/src/core/hle/kernel/memory/memory_layout.h +++ b/src/core/hle/kernel/memory/memory_layout.h @@ -66,8 +66,6 @@ private:      const MemoryRegion application;      const MemoryRegion applet;      const MemoryRegion system; - -    const PAddr start_address{};  };  } // namespace Kernel::Memory diff --git a/src/core/hle/kernel/process.cpp b/src/core/hle/kernel/process.cpp index c6fcb56ad..ff9d9248b 100644 --- a/src/core/hle/kernel/process.cpp +++ b/src/core/hle/kernel/process.cpp @@ -408,7 +408,7 @@ void Process::LoadModule(CodeSet code_set, VAddr base_addr) {  Process::Process(Core::System& system)      : SynchronizationObject{system.Kernel()}, page_table{std::make_unique<Memory::PageTable>(                                                    system)}, -      address_arbiter{system}, mutex{system}, system{system} {} +      handle_table{system.Kernel()}, address_arbiter{system}, mutex{system}, system{system} {}  Process::~Process() = default; diff --git a/src/core/hle/kernel/process.h b/src/core/hle/kernel/process.h index 9dabe3568..f45cb5674 100644 --- a/src/core/hle/kernel/process.h +++ b/src/core/hle/kernel/process.h @@ -382,12 +382,6 @@ private:      /// List of threads waiting for a condition variable      std::unordered_map<VAddr, std::list<std::shared_ptr<Thread>>> cond_var_threads; -    /// System context -    Core::System& system; - -    /// Name of this process -    std::string name; -      /// Address of the top of the main thread's stack      VAddr main_thread_stack_top{}; @@ -399,6 +393,12 @@ private:      /// Process total image size      std::size_t image_size{}; + +    /// Name of this process +    std::string name; + +    /// System context +    Core::System& system;  };  } // namespace Kernel diff --git a/src/core/hle/kernel/thread.cpp b/src/core/hle/kernel/thread.cpp index 2b1092697..67148fa6d 100644 --- a/src/core/hle/kernel/thread.cpp +++ b/src/core/hle/kernel/thread.cpp @@ -13,16 +13,8 @@  #include "common/logging/log.h"  #include "common/thread_queue_list.h"  #include "core/arm/arm_interface.h" -#ifdef ARCHITECTURE_x86_64 -#include "core/arm/dynarmic/arm_dynarmic_32.h" -#include "core/arm/dynarmic/arm_dynarmic_64.h" -#endif -#include "core/arm/cpu_interrupt_handler.h" -#include "core/arm/exclusive_monitor.h"  #include "core/arm/unicorn/arm_unicorn.h"  #include "core/core.h" -#include "core/core_timing.h" -#include "core/core_timing_util.h"  #include "core/cpu_manager.h"  #include "core/hardware_properties.h"  #include "core/hle/kernel/errors.h" @@ -36,6 +28,11 @@  #include "core/hle/result.h"  #include "core/memory.h" +#ifdef ARCHITECTURE_x86_64 +#include "core/arm/dynarmic/arm_dynarmic_32.h" +#include "core/arm/dynarmic/arm_dynarmic_64.h" +#endif +  namespace Kernel {  bool Thread::ShouldWait(const Thread* thread) const { @@ -540,13 +537,4 @@ ResultCode Thread::SetCoreAndAffinityMask(s32 new_core, u64 new_affinity_mask) {      return RESULT_SUCCESS;  } -//////////////////////////////////////////////////////////////////////////////////////////////////// - -/** - * Gets the current thread - */ -Thread* GetCurrentThread() { -    return Core::System::GetInstance().CurrentScheduler().GetCurrentThread(); -} -  } // namespace Kernel diff --git a/src/core/hle/kernel/thread.h b/src/core/hle/kernel/thread.h index c0342c462..9808767e5 100644 --- a/src/core/hle/kernel/thread.h +++ b/src/core/hle/kernel/thread.h @@ -680,9 +680,4 @@ private:      std::string name;  }; -/** - * Gets the current thread - */ -Thread* GetCurrentThread(); -  } // namespace Kernel diff --git a/src/core/hle/service/am/am.cpp b/src/core/hle/service/am/am.cpp index 256449aa7..4e7a0bec9 100644 --- a/src/core/hle/service/am/am.cpp +++ b/src/core/hle/service/am/am.cpp @@ -1407,7 +1407,19 @@ void IApplicationFunctions::GetDesiredLanguage(Kernel::HLERequestContext& ctx) {      u32 supported_languages = 0;      FileSys::PatchManager pm{system.CurrentProcess()->GetTitleID()}; -    const auto res = pm.GetControlMetadata(); +    const auto res = [this] { +        const auto title_id = system.CurrentProcess()->GetTitleID(); + +        FileSys::PatchManager pm{title_id}; +        auto res = pm.GetControlMetadata(); +        if (res.first != nullptr) { +            return res; +        } + +        FileSys::PatchManager pm_update{FileSys::GetUpdateTitleID(title_id)}; +        return pm_update.GetControlMetadata(); +    }(); +      if (res.first != nullptr) {          supported_languages = res.first->GetSupportedLanguages();      } diff --git a/src/core/settings.h b/src/core/settings.h index 850ca4072..3eb336f75 100644 --- a/src/core/settings.h +++ b/src/core/settings.h @@ -346,31 +346,6 @@ struct TouchscreenInput {      u32 rotation_angle;  }; -enum class NANDTotalSize : u64 { -    S29_1GB = 0x747C00000ULL, -}; - -enum class NANDUserSize : u64 { -    S26GB = 0x680000000ULL, -}; - -enum class NANDSystemSize : u64 { -    S2_5GB = 0xA0000000, -}; - -enum class SDMCSize : u64 { -    S1GB = 0x40000000, -    S2GB = 0x80000000, -    S4GB = 0x100000000ULL, -    S8GB = 0x200000000ULL, -    S16GB = 0x400000000ULL, -    S32GB = 0x800000000ULL, -    S64GB = 0x1000000000ULL, -    S128GB = 0x2000000000ULL, -    S256GB = 0x4000000000ULL, -    S1TB = 0x10000000000ULL, -}; -  enum class RendererBackend {      OpenGL = 0,      Vulkan = 1, @@ -382,6 +357,11 @@ enum class GPUAccuracy : u32 {      Extreme = 2,  }; +enum class CPUAccuracy { +    Accurate = 0, +    DebugMode = 1, +}; +  extern bool configuring_global;  template <typename Type> @@ -427,6 +407,18 @@ struct Values {      // Core      Setting<bool> use_multi_core; +    // Cpu +    CPUAccuracy cpu_accuracy; + +    bool cpuopt_page_tables; +    bool cpuopt_block_linking; +    bool cpuopt_return_stack_buffer; +    bool cpuopt_fast_dispatcher; +    bool cpuopt_context_elimination; +    bool cpuopt_const_prop; +    bool cpuopt_misc_ir; +    bool cpuopt_reduce_misalign_checks; +      // Renderer      Setting<RendererBackend> renderer_backend;      bool renderer_debug; @@ -491,10 +483,6 @@ struct Values {      bool gamecard_inserted;      bool gamecard_current_game;      std::string gamecard_path; -    NANDTotalSize nand_total_size; -    NANDSystemSize nand_system_size; -    NANDUserSize nand_user_size; -    SDMCSize sdmc_size;      // Debugging      bool record_frame_times; @@ -505,7 +493,6 @@ struct Values {      bool dump_nso;      bool reporting_services;      bool quest_flag; -    bool disable_cpu_opt;      bool disable_macro_jit;      // Misceallaneous | 
