diff options
Diffstat (limited to 'src/core')
| -rw-r--r-- | src/core/core.cpp | 2 | ||||
| -rw-r--r-- | src/core/crypto/key_manager.cpp | 39 | ||||
| -rw-r--r-- | src/core/crypto/key_manager.h | 2 | ||||
| -rw-r--r-- | src/core/crypto/partition_data_manager.cpp | 80 | ||||
| -rw-r--r-- | src/core/crypto/partition_data_manager.h | 18 | ||||
| -rw-r--r-- | src/core/file_sys/control_metadata.cpp | 13 | ||||
| -rw-r--r-- | src/core/file_sys/control_metadata.h | 1 | ||||
| -rw-r--r-- | src/core/hle/kernel/process.h | 21 | ||||
| -rw-r--r-- | src/core/hle/kernel/svc.cpp | 31 | ||||
| -rw-r--r-- | src/core/hle/kernel/svc_wrap.h | 8 | ||||
| -rw-r--r-- | src/core/hle/service/aoc/aoc_u.cpp | 18 | ||||
| -rw-r--r-- | src/core/hle/service/filesystem/filesystem.cpp | 16 | ||||
| -rw-r--r-- | src/core/hle/service/filesystem/filesystem.h | 4 | ||||
| -rw-r--r-- | src/core/hle/service/service.cpp | 4 | ||||
| -rw-r--r-- | src/core/hle/service/service.h | 3 | ||||
| -rw-r--r-- | src/core/loader/deconstructed_rom_directory.cpp | 22 | ||||
| -rw-r--r-- | src/core/loader/loader.cpp | 3 | ||||
| -rw-r--r-- | src/core/loader/loader.h | 1 | ||||
| -rw-r--r-- | src/core/loader/nro.cpp | 10 | ||||
| -rw-r--r-- | src/core/loader/nro.h | 2 | ||||
| -rw-r--r-- | src/core/loader/nso.cpp | 23 | ||||
| -rw-r--r-- | src/core/loader/nso.h | 6 | 
22 files changed, 209 insertions, 118 deletions
| diff --git a/src/core/core.cpp b/src/core/core.cpp index e2fb9e038..32baa40dc 100644 --- a/src/core/core.cpp +++ b/src/core/core.cpp @@ -148,7 +148,7 @@ struct System::Impl {          telemetry_session = std::make_unique<Core::TelemetrySession>();          service_manager = std::make_shared<Service::SM::ServiceManager>(); -        Service::Init(service_manager, virtual_filesystem); +        Service::Init(service_manager, *virtual_filesystem);          GDBStub::Init();          renderer = VideoCore::CreateRenderer(emu_window); diff --git a/src/core/crypto/key_manager.cpp b/src/core/crypto/key_manager.cpp index a59a7e1f5..fd0786068 100644 --- a/src/core/crypto/key_manager.cpp +++ b/src/core/crypto/key_manager.cpp @@ -98,7 +98,7 @@ std::array<u8, 144> DecryptKeyblob(const std::array<u8, 176>& encrypted_keyblob,      return keyblob;  } -void KeyManager::DeriveGeneralPurposeKeys(u8 crypto_revision) { +void KeyManager::DeriveGeneralPurposeKeys(std::size_t crypto_revision) {      const auto kek_generation_source =          GetKey(S128KeyType::Source, static_cast<u64>(SourceKeyType::AESKekGeneration));      const auto key_generation_source = @@ -147,31 +147,38 @@ boost::optional<Key128> DeriveSDSeed() {                                     "rb+");      if (!save_43.IsOpen())          return boost::none; +      const FileUtil::IOFile sd_private(          FileUtil::GetUserPath(FileUtil::UserPath::SDMCDir) + "/Nintendo/Contents/private", "rb+");      if (!sd_private.IsOpen())          return boost::none; -    sd_private.Seek(0, SEEK_SET);      std::array<u8, 0x10> private_seed{}; -    if (sd_private.ReadBytes(private_seed.data(), private_seed.size()) != 0x10) +    if (sd_private.ReadBytes(private_seed.data(), private_seed.size()) != private_seed.size()) {          return boost::none; +    }      std::array<u8, 0x10> buffer{};      std::size_t offset = 0;      for (; offset + 0x10 < save_43.GetSize(); ++offset) { -        save_43.Seek(offset, SEEK_SET); +        if (!save_43.Seek(offset, SEEK_SET)) { +            return boost::none; +        } +          save_43.ReadBytes(buffer.data(), buffer.size()); -        if (buffer == private_seed) +        if (buffer == private_seed) {              break; +        }      } -    if (offset + 0x10 >= save_43.GetSize()) +    if (!save_43.Seek(offset + 0x10, SEEK_SET)) {          return boost::none; +    }      Key128 seed{}; -    save_43.Seek(offset + 0x10, SEEK_SET); -    save_43.ReadBytes(seed.data(), seed.size()); +    if (save_43.ReadBytes(seed.data(), seed.size()) != seed.size()) { +        return boost::none; +    }      return seed;  } @@ -234,7 +241,9 @@ std::vector<TicketRaw> GetTicketblob(const FileUtil::IOFile& ticket_save) {          return {};      std::vector<u8> buffer(ticket_save.GetSize()); -    ticket_save.ReadBytes(buffer.data(), buffer.size()); +    if (ticket_save.ReadBytes(buffer.data(), buffer.size()) != buffer.size()) { +        return {}; +    }      std::vector<TicketRaw> out;      u32 magic{}; @@ -261,6 +270,9 @@ static std::array<u8, size> operator^(const std::array<u8, size>& lhs,  template <size_t target_size, size_t in_size>  static std::array<u8, target_size> MGF1(const std::array<u8, in_size>& seed) { +    // Avoids truncation overflow within the loop below. +    static_assert(target_size <= 0xFF); +      std::array<u8, in_size + 4> seed_exp{};      std::memcpy(seed_exp.data(), seed.data(), in_size); @@ -268,7 +280,7 @@ static std::array<u8, target_size> MGF1(const std::array<u8, in_size>& seed) {      size_t i = 0;      while (out.size() < target_size) {          out.resize(out.size() + 0x20); -        seed_exp[in_size + 3] = i; +        seed_exp[in_size + 3] = static_cast<u8>(i);          mbedtls_sha256(seed_exp.data(), seed_exp.size(), out.data() + out.size() - 0x20, 0);          ++i;      } @@ -299,10 +311,11 @@ boost::optional<std::pair<Key128, Key128>> ParseTicket(const TicketRaw& ticket,      std::memcpy(&cert_authority, ticket.data() + 0x140, sizeof(cert_authority));      if (cert_authority == 0)          return boost::none; -    if (cert_authority != Common::MakeMagic('R', 'o', 'o', 't')) +    if (cert_authority != Common::MakeMagic('R', 'o', 'o', 't')) {          LOG_INFO(Crypto,                   "Attempting to parse ticket with non-standard certificate authority {:08X}.",                   cert_authority); +    }      Key128 rights_id;      std::memcpy(rights_id.data(), ticket.data() + 0x2A0, sizeof(Key128)); @@ -871,9 +884,9 @@ void KeyManager::DeriveETicket(PartitionDataManager& data) {                                       "/system/save/80000000000000e2",                                   "rb+"); +    const auto blob2 = GetTicketblob(save2);      auto res = GetTicketblob(save1); -    const auto res2 = GetTicketblob(save2); -    std::copy(res2.begin(), res2.end(), std::back_inserter(res)); +    res.insert(res.end(), blob2.begin(), blob2.end());      for (const auto& raw : res) {          const auto pair = ParseTicket(raw, rsa_key); diff --git a/src/core/crypto/key_manager.h b/src/core/crypto/key_manager.h index a41abbdfc..cccb3c0ae 100644 --- a/src/core/crypto/key_manager.h +++ b/src/core/crypto/key_manager.h @@ -175,7 +175,7 @@ private:      void WriteKeyToFile(KeyCategory category, std::string_view keyname,                          const std::array<u8, Size>& key); -    void DeriveGeneralPurposeKeys(u8 crypto_revision); +    void DeriveGeneralPurposeKeys(std::size_t crypto_revision);      void SetKeyWrapped(S128KeyType id, Key128 key, u64 field1 = 0, u64 field2 = 0);      void SetKeyWrapped(S256KeyType id, Key256 key, u64 field1 = 0, u64 field2 = 0); diff --git a/src/core/crypto/partition_data_manager.cpp b/src/core/crypto/partition_data_manager.cpp index d1c04e98d..25cee1f3a 100644 --- a/src/core/crypto/partition_data_manager.cpp +++ b/src/core/crypto/partition_data_manager.cpp @@ -11,7 +11,6 @@  #include <array>  #include <cctype>  #include <cstring> -#include <boost/optional/optional.hpp>  #include <mbedtls/sha256.h>  #include "common/assert.h"  #include "common/common_funcs.h" @@ -19,7 +18,7 @@  #include "common/hex_util.h"  #include "common/logging/log.h"  #include "common/string_util.h" -#include "core/crypto/ctr_encryption_layer.h" +#include "common/swap.h"  #include "core/crypto/key_manager.h"  #include "core/crypto/partition_data_manager.h"  #include "core/crypto/xts_encryption_layer.h" @@ -302,10 +301,10 @@ FileSys::VirtualFile FindFileInDirWithNames(const FileSys::VirtualDir& dir,      return nullptr;  } -PartitionDataManager::PartitionDataManager(FileSys::VirtualDir sysdata_dir) +PartitionDataManager::PartitionDataManager(const FileSys::VirtualDir& sysdata_dir)      : boot0(FindFileInDirWithNames(sysdata_dir, "BOOT0")), -      fuses(FindFileInDirWithNames(sysdata_dir, "fuse")), -      kfuses(FindFileInDirWithNames(sysdata_dir, "kfuse")), +      fuses(FindFileInDirWithNames(sysdata_dir, "fuses")), +      kfuses(FindFileInDirWithNames(sysdata_dir, "kfuses")),        package2({            FindFileInDirWithNames(sysdata_dir, "BCPKG2-1-Normal-Main"),            FindFileInDirWithNames(sysdata_dir, "BCPKG2-2-Normal-Sub"), @@ -314,13 +313,14 @@ PartitionDataManager::PartitionDataManager(FileSys::VirtualDir sysdata_dir)            FindFileInDirWithNames(sysdata_dir, "BCPKG2-5-Repair-Main"),            FindFileInDirWithNames(sysdata_dir, "BCPKG2-6-Repair-Sub"),        }), +      prodinfo(FindFileInDirWithNames(sysdata_dir, "PRODINFO")),        secure_monitor(FindFileInDirWithNames(sysdata_dir, "secmon")),        package1_decrypted(FindFileInDirWithNames(sysdata_dir, "pkg1_decr")),        secure_monitor_bytes(secure_monitor == nullptr ? std::vector<u8>{}                                                       : secure_monitor->ReadAllBytes()),        package1_decrypted_bytes(package1_decrypted == nullptr ? std::vector<u8>{} -                                                             : package1_decrypted->ReadAllBytes()), -      prodinfo(FindFileInDirWithNames(sysdata_dir, "PRODINFO")) {} +                                                             : package1_decrypted->ReadAllBytes()) { +}  PartitionDataManager::~PartitionDataManager() = default; @@ -332,18 +332,19 @@ FileSys::VirtualFile PartitionDataManager::GetBoot0Raw() const {      return boot0;  } -std::array<u8, 176> PartitionDataManager::GetEncryptedKeyblob(u8 index) const { -    if (HasBoot0() && index < 32) +PartitionDataManager::EncryptedKeyBlob PartitionDataManager::GetEncryptedKeyblob( +    std::size_t index) const { +    if (HasBoot0() && index < NUM_ENCRYPTED_KEYBLOBS)          return GetEncryptedKeyblobs()[index];      return {};  } -std::array<std::array<u8, 176>, 32> PartitionDataManager::GetEncryptedKeyblobs() const { +PartitionDataManager::EncryptedKeyBlobs PartitionDataManager::GetEncryptedKeyblobs() const {      if (!HasBoot0())          return {}; -    std::array<std::array<u8, 176>, 32> out{}; -    for (size_t i = 0; i < 0x20; ++i) +    EncryptedKeyBlobs out{}; +    for (size_t i = 0; i < out.size(); ++i)          boot0->Read(out[i].data(), out[i].size(), 0x180000 + i * 0x200);      return out;  } @@ -389,7 +390,7 @@ std::array<u8, 16> PartitionDataManager::GetKeyblobMACKeySource() const {      return FindKeyFromHex(package1_decrypted_bytes, source_hashes[0]);  } -std::array<u8, 16> PartitionDataManager::GetKeyblobKeySource(u8 revision) const { +std::array<u8, 16> PartitionDataManager::GetKeyblobKeySource(std::size_t revision) const {      if (keyblob_source_hashes[revision] == SHA256Hash{}) {          LOG_WARNING(Crypto,                      "No keyblob source hash for crypto revision {:02X}! Cannot derive keys...", @@ -446,7 +447,7 @@ bool AttemptDecrypt(const std::array<u8, 16>& key, Package2Header& header) {      return false;  } -void PartitionDataManager::DecryptPackage2(std::array<std::array<u8, 16>, 0x20> package2_keys, +void PartitionDataManager::DecryptPackage2(const std::array<Key128, 0x20>& package2_keys,                                             Package2Type type) {      FileSys::VirtualFile file = std::make_shared<FileSys::OffsetVfsFile>(          package2[static_cast<size_t>(type)], @@ -456,43 +457,38 @@ void PartitionDataManager::DecryptPackage2(std::array<std::array<u8, 16>, 0x20>      if (file->ReadObject(&header) != sizeof(Package2Header))          return; -    u8 revision = 0xFF; +    std::size_t revision = 0xFF;      if (header.magic != Common::MakeMagic('P', 'K', '2', '1')) { -        for (size_t i = 0; i < package2_keys.size(); ++i) { -            if (AttemptDecrypt(package2_keys[i], header)) +        for (std::size_t i = 0; i < package2_keys.size(); ++i) { +            if (AttemptDecrypt(package2_keys[i], header)) {                  revision = i; +            }          }      }      if (header.magic != Common::MakeMagic('P', 'K', '2', '1'))          return; -    const std::vector<u8> s1_iv(header.section_ctr[1].begin(), header.section_ctr[1].end()); -      const auto a = std::make_shared<FileSys::OffsetVfsFile>(          file, header.section_size[1], header.section_size[0] + sizeof(Package2Header));      auto c = a->ReadAllBytes();      AESCipher<Key128> cipher(package2_keys[revision], Mode::CTR); -    cipher.SetIV(s1_iv); +    cipher.SetIV({header.section_ctr[1].begin(), header.section_ctr[1].end()});      cipher.Transcode(c.data(), c.size(), c.data(), Op::Decrypt); -    // package2_decrypted[static_cast<size_t>(type)] = s1; -      INIHeader ini;      std::memcpy(&ini, c.data(), sizeof(INIHeader));      if (ini.magic != Common::MakeMagic('I', 'N', 'I', '1'))          return; -    std::map<u64, KIPHeader> kips{};      u64 offset = sizeof(INIHeader);      for (size_t i = 0; i < ini.process_count; ++i) {          KIPHeader kip;          std::memcpy(&kip, c.data() + offset, sizeof(KIPHeader));          if (kip.magic != Common::MakeMagic('K', 'I', 'P', '1'))              return; -        kips.emplace(offset, kip);          const auto name =              Common::StringFromFixedZeroTerminatedBuffer(kip.name.data(), kip.name.size()); @@ -503,33 +499,29 @@ void PartitionDataManager::DecryptPackage2(std::array<std::array<u8, 16>, 0x20>              continue;          } -        std::vector<u8> text(kip.sections[0].size_compressed); -        std::vector<u8> rodata(kip.sections[1].size_compressed); -        std::vector<u8> data(kip.sections[2].size_compressed); +        const u64 initial_offset = sizeof(KIPHeader) + offset; +        const auto text_begin = c.cbegin() + initial_offset; +        const auto text_end = text_begin + kip.sections[0].size_compressed; +        const std::vector<u8> text = DecompressBLZ({text_begin, text_end}); -        u64 offset_sec = sizeof(KIPHeader) + offset; -        std::memcpy(text.data(), c.data() + offset_sec, text.size()); -        offset_sec += text.size(); -        std::memcpy(rodata.data(), c.data() + offset_sec, rodata.size()); -        offset_sec += rodata.size(); -        std::memcpy(data.data(), c.data() + offset_sec, data.size()); +        const auto rodata_end = text_end + kip.sections[1].size_compressed; +        const std::vector<u8> rodata = DecompressBLZ({text_end, rodata_end}); -        offset += sizeof(KIPHeader) + kip.sections[0].size_compressed + -                  kip.sections[1].size_compressed + kip.sections[2].size_compressed; +        const auto data_end = rodata_end + kip.sections[2].size_compressed; +        const std::vector<u8> data = DecompressBLZ({rodata_end, data_end}); -        text = DecompressBLZ(text); -        rodata = DecompressBLZ(rodata); -        data = DecompressBLZ(data); +        std::vector<u8> out; +        out.reserve(text.size() + rodata.size() + data.size()); +        out.insert(out.end(), text.begin(), text.end()); +        out.insert(out.end(), rodata.begin(), rodata.end()); +        out.insert(out.end(), data.begin(), data.end()); -        std::vector<u8> out(text.size() + rodata.size() + data.size()); -        std::memcpy(out.data(), text.data(), text.size()); -        std::memcpy(out.data() + text.size(), rodata.data(), rodata.size()); -        std::memcpy(out.data() + text.size() + rodata.size(), data.data(), data.size()); +        offset += sizeof(KIPHeader) + out.size();          if (name == "FS") -            package2_fs[static_cast<size_t>(type)] = out; +            package2_fs[static_cast<size_t>(type)] = std::move(out);          else if (name == "spl") -            package2_spl[static_cast<size_t>(type)] = out; +            package2_spl[static_cast<size_t>(type)] = std::move(out);      }  } diff --git a/src/core/crypto/partition_data_manager.h b/src/core/crypto/partition_data_manager.h index 45c7fecfa..0ad007c72 100644 --- a/src/core/crypto/partition_data_manager.h +++ b/src/core/crypto/partition_data_manager.h @@ -5,9 +5,7 @@  #pragma once  #include <vector> -#include "common/common_funcs.h"  #include "common/common_types.h" -#include "common/swap.h"  #include "core/file_sys/vfs_types.h"  namespace Core::Crypto { @@ -24,15 +22,20 @@ enum class Package2Type {  class PartitionDataManager {  public:      static const u8 MAX_KEYBLOB_SOURCE_HASH; +    static constexpr std::size_t NUM_ENCRYPTED_KEYBLOBS = 32; +    static constexpr std::size_t ENCRYPTED_KEYBLOB_SIZE = 0xB0; -    explicit PartitionDataManager(FileSys::VirtualDir sysdata_dir); +    using EncryptedKeyBlob = std::array<u8, ENCRYPTED_KEYBLOB_SIZE>; +    using EncryptedKeyBlobs = std::array<EncryptedKeyBlob, NUM_ENCRYPTED_KEYBLOBS>; + +    explicit PartitionDataManager(const FileSys::VirtualDir& sysdata_dir);      ~PartitionDataManager();      // BOOT0      bool HasBoot0() const;      FileSys::VirtualFile GetBoot0Raw() const; -    std::array<u8, 0xB0> GetEncryptedKeyblob(u8 index) const; -    std::array<std::array<u8, 0xB0>, 0x20> GetEncryptedKeyblobs() const; +    EncryptedKeyBlob GetEncryptedKeyblob(std::size_t index) const; +    EncryptedKeyBlobs GetEncryptedKeyblobs() const;      std::vector<u8> GetSecureMonitor() const;      std::array<u8, 0x10> GetPackage2KeySource() const;      std::array<u8, 0x10> GetAESKekGenerationSource() const; @@ -43,7 +46,7 @@ public:      std::vector<u8> GetPackage1Decrypted() const;      std::array<u8, 0x10> GetMasterKeySource() const;      std::array<u8, 0x10> GetKeyblobMACKeySource() const; -    std::array<u8, 0x10> GetKeyblobKeySource(u8 revision) const; +    std::array<u8, 0x10> GetKeyblobKeySource(std::size_t revision) const;      // Fuses      bool HasFuses() const; @@ -57,7 +60,8 @@ public:      // Package2      bool HasPackage2(Package2Type type = Package2Type::NormalMain) const;      FileSys::VirtualFile GetPackage2Raw(Package2Type type = Package2Type::NormalMain) const; -    void DecryptPackage2(std::array<std::array<u8, 16>, 0x20> package2, Package2Type type); +    void DecryptPackage2(const std::array<std::array<u8, 16>, 0x20>& package2_keys, +                         Package2Type type);      const std::vector<u8>& GetPackage2FSDecompressed(          Package2Type type = Package2Type::NormalMain) const;      std::array<u8, 0x10> GetKeyAreaKeyApplicationSource( diff --git a/src/core/file_sys/control_metadata.cpp b/src/core/file_sys/control_metadata.cpp index 5b1177a03..a012c2be9 100644 --- a/src/core/file_sys/control_metadata.cpp +++ b/src/core/file_sys/control_metadata.cpp @@ -17,11 +17,13 @@ const std::array<const char*, 15> LANGUAGE_NAMES = {  };  std::string LanguageEntry::GetApplicationName() const { -    return Common::StringFromFixedZeroTerminatedBuffer(application_name.data(), 0x200); +    return Common::StringFromFixedZeroTerminatedBuffer(application_name.data(), +                                                       application_name.size());  }  std::string LanguageEntry::GetDeveloperName() const { -    return Common::StringFromFixedZeroTerminatedBuffer(developer_name.data(), 0x100); +    return Common::StringFromFixedZeroTerminatedBuffer(developer_name.data(), +                                                       developer_name.size());  }  NACP::NACP(VirtualFile file) : raw(std::make_unique<RawNACP>()) { @@ -56,7 +58,12 @@ u64 NACP::GetTitleId() const {      return raw->title_id;  } +u64 NACP::GetDLCBaseTitleId() const { +    return raw->dlc_base_title_id; +} +  std::string NACP::GetVersionString() const { -    return Common::StringFromFixedZeroTerminatedBuffer(raw->version_string.data(), 0x10); +    return Common::StringFromFixedZeroTerminatedBuffer(raw->version_string.data(), +                                                       raw->version_string.size());  }  } // namespace FileSys diff --git a/src/core/file_sys/control_metadata.h b/src/core/file_sys/control_metadata.h index 43d6f0719..141f7e056 100644 --- a/src/core/file_sys/control_metadata.h +++ b/src/core/file_sys/control_metadata.h @@ -79,6 +79,7 @@ public:      std::string GetApplicationName(Language language = Language::Default) const;      std::string GetDeveloperName(Language language = Language::Default) const;      u64 GetTitleId() const; +    u64 GetDLCBaseTitleId() const;      std::string GetVersionString() const;  private: diff --git a/src/core/hle/kernel/process.h b/src/core/hle/kernel/process.h index 73ec01e11..f2816943a 100644 --- a/src/core/hle/kernel/process.h +++ b/src/core/hle/kernel/process.h @@ -24,6 +24,7 @@ class ProgramMetadata;  namespace Kernel {  class KernelCore; +class ResourceLimit;  struct AddressMapping {      // Address and size must be page-aligned @@ -57,9 +58,23 @@ union ProcessFlags {      BitField<12, 1, u16> loaded_high; ///< Application loaded high (not at 0x00100000).  }; -enum class ProcessStatus { Created, Running, Exited }; - -class ResourceLimit; +/** + * Indicates the status of a Process instance. + * + * @note These match the values as used by kernel, + *       so new entries should only be added if RE + *       shows that a new value has been introduced. + */ +enum class ProcessStatus { +    Created, +    CreatedWithDebuggerAttached, +    Running, +    WaitingForDebuggerToAttach, +    DebuggerAttached, +    Exiting, +    Exited, +    DebugBreak, +};  struct CodeSet final {      struct Segment { diff --git a/src/core/hle/kernel/svc.cpp b/src/core/hle/kernel/svc.cpp index e406df829..7a053da1e 100644 --- a/src/core/hle/kernel/svc.cpp +++ b/src/core/hle/kernel/svc.cpp @@ -389,6 +389,12 @@ static void Break(u32 reason, u64 info1, u64 info2) {              "Emulated program broke execution! reason=0x{:016X}, info1=0x{:016X}, info2=0x{:016X}",              reason, info1, info2);          ASSERT(false); + +        Core::CurrentProcess()->PrepareForTermination(); + +        // Kill the current thread +        GetCurrentThread()->Stop(); +        Core::System::GetInstance().PrepareReschedule();      }  } @@ -1092,6 +1098,29 @@ static ResultCode ClearEvent(Handle handle) {      return RESULT_SUCCESS;  } +static ResultCode GetProcessInfo(u64* out, Handle process_handle, u32 type) { +    LOG_DEBUG(Kernel_SVC, "called, handle=0x{:08X}, type=0x{:X}", process_handle, type); + +    // This function currently only allows retrieving a process' status. +    enum class InfoType { +        Status, +    }; + +    const auto& kernel = Core::System::GetInstance().Kernel(); +    const auto process = kernel.HandleTable().Get<Process>(process_handle); +    if (!process) { +        return ERR_INVALID_HANDLE; +    } + +    const auto info_type = static_cast<InfoType>(type); +    if (info_type != InfoType::Status) { +        return ERR_INVALID_ENUM_VALUE; +    } + +    *out = static_cast<u64>(process->GetStatus()); +    return RESULT_SUCCESS; +} +  namespace {  struct FunctionDef {      using Func = void(); @@ -1227,7 +1256,7 @@ static const FunctionDef SVC_Table[] = {      {0x79, nullptr, "CreateProcess"},      {0x7A, nullptr, "StartProcess"},      {0x7B, nullptr, "TerminateProcess"}, -    {0x7C, nullptr, "GetProcessInfo"}, +    {0x7C, SvcWrap<GetProcessInfo>, "GetProcessInfo"},      {0x7D, nullptr, "CreateResourceLimit"},      {0x7E, nullptr, "SetResourceLimitLimitValue"},      {0x7F, nullptr, "CallSecureMonitor"}, diff --git a/src/core/hle/kernel/svc_wrap.h b/src/core/hle/kernel/svc_wrap.h index cbb80c3c4..b09753c80 100644 --- a/src/core/hle/kernel/svc_wrap.h +++ b/src/core/hle/kernel/svc_wrap.h @@ -77,6 +77,14 @@ void SvcWrap() {      FuncReturn(retval);  } +template <ResultCode func(u64*, u32, u32)> +void SvcWrap() { +    u64 param_1 = 0; +    u32 retval = func(¶m_1, static_cast<u32>(Param(1)), static_cast<u32>(Param(2))).raw; +    Core::CurrentArmInterface().SetReg(1, param_1); +    FuncReturn(retval); +} +  template <ResultCode func(u32, u64)>  void SvcWrap() {      FuncReturn(func(static_cast<u32>(Param(0)), Param(1)).raw); diff --git a/src/core/hle/service/aoc/aoc_u.cpp b/src/core/hle/service/aoc/aoc_u.cpp index 0ecfb5af1..518161bf7 100644 --- a/src/core/hle/service/aoc/aoc_u.cpp +++ b/src/core/hle/service/aoc/aoc_u.cpp @@ -7,8 +7,10 @@  #include <vector>  #include "common/logging/log.h"  #include "core/file_sys/content_archive.h" +#include "core/file_sys/control_metadata.h"  #include "core/file_sys/nca_metadata.h"  #include "core/file_sys/partition_filesystem.h" +#include "core/file_sys/patch_manager.h"  #include "core/file_sys/registered_cache.h"  #include "core/hle/ipc_helpers.h"  #include "core/hle/kernel/process.h" @@ -19,7 +21,7 @@  namespace Service::AOC {  constexpr u64 DLC_BASE_TITLE_ID_MASK = 0xFFFFFFFFFFFFE000; -constexpr u64 DLC_BASE_TO_AOC_ID_MASK = 0x1000; +constexpr u64 DLC_BASE_TO_AOC_ID = 0x1000;  static bool CheckAOCTitleIDMatchesBase(u64 base, u64 aoc) {      return (aoc & DLC_BASE_TITLE_ID_MASK) == base; @@ -97,14 +99,24 @@ void AOC_U::ListAddOnContent(Kernel::HLERequestContext& ctx) {      ctx.WriteBuffer(out); -    IPC::ResponseBuilder rb{ctx, 2}; +    IPC::ResponseBuilder rb{ctx, 3};      rb.Push(RESULT_SUCCESS); +    rb.Push(count);  }  void AOC_U::GetAddOnContentBaseId(Kernel::HLERequestContext& ctx) {      IPC::ResponseBuilder rb{ctx, 4};      rb.Push(RESULT_SUCCESS); -    rb.Push(Core::System::GetInstance().CurrentProcess()->GetTitleID() | DLC_BASE_TO_AOC_ID_MASK); +    const auto title_id = Core::System::GetInstance().CurrentProcess()->GetTitleID(); +    FileSys::PatchManager pm{title_id}; + +    const auto res = pm.GetControlMetadata(); +    if (res.first == nullptr) { +        rb.Push(title_id + DLC_BASE_TO_AOC_ID); +        return; +    } + +    rb.Push(res.first->GetDLCBaseTitleId());  }  void AOC_U::PrepareAddOnContent(Kernel::HLERequestContext& ctx) { diff --git a/src/core/hle/service/filesystem/filesystem.cpp b/src/core/hle/service/filesystem/filesystem.cpp index 439e62d27..e06712603 100644 --- a/src/core/hle/service/filesystem/filesystem.cpp +++ b/src/core/hle/service/filesystem/filesystem.cpp @@ -361,19 +361,19 @@ FileSys::VirtualDir GetModificationLoadRoot(u64 title_id) {      return bis_factory->GetModificationLoadRoot(title_id);  } -void CreateFactories(const FileSys::VirtualFilesystem& vfs, bool overwrite) { +void CreateFactories(FileSys::VfsFilesystem& vfs, bool overwrite) {      if (overwrite) {          bis_factory = nullptr;          save_data_factory = nullptr;          sdmc_factory = nullptr;      } -    auto nand_directory = vfs->OpenDirectory(FileUtil::GetUserPath(FileUtil::UserPath::NANDDir), -                                             FileSys::Mode::ReadWrite); -    auto sd_directory = vfs->OpenDirectory(FileUtil::GetUserPath(FileUtil::UserPath::SDMCDir), -                                           FileSys::Mode::ReadWrite); -    auto load_directory = vfs->OpenDirectory(FileUtil::GetUserPath(FileUtil::UserPath::LoadDir), -                                             FileSys::Mode::ReadWrite); +    auto nand_directory = vfs.OpenDirectory(FileUtil::GetUserPath(FileUtil::UserPath::NANDDir), +                                            FileSys::Mode::ReadWrite); +    auto sd_directory = vfs.OpenDirectory(FileUtil::GetUserPath(FileUtil::UserPath::SDMCDir), +                                          FileSys::Mode::ReadWrite); +    auto load_directory = vfs.OpenDirectory(FileUtil::GetUserPath(FileUtil::UserPath::LoadDir), +                                            FileSys::Mode::ReadWrite);      if (bis_factory == nullptr)          bis_factory = std::make_unique<FileSys::BISFactory>(nand_directory, load_directory); @@ -383,7 +383,7 @@ void CreateFactories(const FileSys::VirtualFilesystem& vfs, bool overwrite) {          sdmc_factory = std::make_unique<FileSys::SDMCFactory>(std::move(sd_directory));  } -void InstallInterfaces(SM::ServiceManager& service_manager, const FileSys::VirtualFilesystem& vfs) { +void InstallInterfaces(SM::ServiceManager& service_manager, FileSys::VfsFilesystem& vfs) {      romfs_factory = nullptr;      CreateFactories(vfs, false);      std::make_shared<FSP_LDR>()->InstallAsService(service_manager); diff --git a/src/core/hle/service/filesystem/filesystem.h b/src/core/hle/service/filesystem/filesystem.h index 53b01bb01..2df1faeb0 100644 --- a/src/core/hle/service/filesystem/filesystem.h +++ b/src/core/hle/service/filesystem/filesystem.h @@ -57,9 +57,9 @@ FileSys::VirtualDir GetModificationLoadRoot(u64 title_id);  // Creates the SaveData, SDMC, and BIS Factories. Should be called once and before any function  // above is called. -void CreateFactories(const FileSys::VirtualFilesystem& vfs, bool overwrite = true); +void CreateFactories(FileSys::VfsFilesystem& vfs, bool overwrite = true); -void InstallInterfaces(SM::ServiceManager& service_manager, const FileSys::VirtualFilesystem& vfs); +void InstallInterfaces(SM::ServiceManager& service_manager, FileSys::VfsFilesystem& vfs);  // A class that wraps a VfsDirectory with methods that return ResultVal and ResultCode instead of  // pointers and booleans. This makes using a VfsDirectory with switch services much easier and diff --git a/src/core/hle/service/service.cpp b/src/core/hle/service/service.cpp index 62f049660..a225cb4cb 100644 --- a/src/core/hle/service/service.cpp +++ b/src/core/hle/service/service.cpp @@ -197,7 +197,7 @@ ResultCode ServiceFrameworkBase::HandleSyncRequest(Kernel::HLERequestContext& co  // Module interface  /// Initialize ServiceManager -void Init(std::shared_ptr<SM::ServiceManager>& sm, const FileSys::VirtualFilesystem& rfs) { +void Init(std::shared_ptr<SM::ServiceManager>& sm, FileSys::VfsFilesystem& vfs) {      // NVFlinger needs to be accessed by several services like Vi and AppletOE so we instantiate it      // here and pass it into the respective InstallInterfaces functions.      auto nv_flinger = std::make_shared<NVFlinger::NVFlinger>(); @@ -220,7 +220,7 @@ void Init(std::shared_ptr<SM::ServiceManager>& sm, const FileSys::VirtualFilesys      EUPLD::InstallInterfaces(*sm);      Fatal::InstallInterfaces(*sm);      FGM::InstallInterfaces(*sm); -    FileSystem::InstallInterfaces(*sm, rfs); +    FileSystem::InstallInterfaces(*sm, vfs);      Friend::InstallInterfaces(*sm);      GRC::InstallInterfaces(*sm);      HID::InstallInterfaces(*sm); diff --git a/src/core/hle/service/service.h b/src/core/hle/service/service.h index 2fc57a82e..98483ecf1 100644 --- a/src/core/hle/service/service.h +++ b/src/core/hle/service/service.h @@ -180,8 +180,7 @@ private:  };  /// Initialize ServiceManager -void Init(std::shared_ptr<SM::ServiceManager>& sm, -          const std::shared_ptr<FileSys::VfsFilesystem>& vfs); +void Init(std::shared_ptr<SM::ServiceManager>& sm, FileSys::VfsFilesystem& vfs);  /// Shutdown ServiceManager  void Shutdown(); diff --git a/src/core/loader/deconstructed_rom_directory.cpp b/src/core/loader/deconstructed_rom_directory.cpp index 951fd8257..8518dddcb 100644 --- a/src/core/loader/deconstructed_rom_directory.cpp +++ b/src/core/loader/deconstructed_rom_directory.cpp @@ -139,14 +139,22 @@ ResultStatus AppLoader_DeconstructedRomDirectory::Load(Kernel::Process& process)      for (const auto& module : {"rtld", "main", "subsdk0", "subsdk1", "subsdk2", "subsdk3",                                 "subsdk4", "subsdk5", "subsdk6", "subsdk7", "sdk"}) {          const FileSys::VirtualFile module_file = dir->GetFile(module); -        if (module_file != nullptr) { -            const VAddr load_addr = next_load_addr; -            next_load_addr = AppLoader_NSO::LoadModule(module_file, load_addr, -                                                       std::strcmp(module, "rtld") == 0, pm); -            LOG_DEBUG(Loader, "loaded module {} @ 0x{:X}", module, load_addr); -            // Register module with GDBStub -            GDBStub::RegisterModule(module, load_addr, next_load_addr - 1, false); +        if (module_file == nullptr) { +            continue;          } + +        const VAddr load_addr = next_load_addr; +        const bool should_pass_arguments = std::strcmp(module, "rtld") == 0; +        const auto tentative_next_load_addr = +            AppLoader_NSO::LoadModule(*module_file, load_addr, should_pass_arguments, pm); +        if (!tentative_next_load_addr) { +            return ResultStatus::ErrorLoadingNSO; +        } + +        next_load_addr = *tentative_next_load_addr; +        LOG_DEBUG(Loader, "loaded module {} @ 0x{:X}", module, load_addr); +        // Register module with GDBStub +        GDBStub::RegisterModule(module, load_addr, next_load_addr - 1, false);      }      process.Run(base_address, metadata.GetMainThreadPriority(), metadata.GetMainThreadStackSize()); diff --git a/src/core/loader/loader.cpp b/src/core/loader/loader.cpp index 91659ec17..9cd0b0ccd 100644 --- a/src/core/loader/loader.cpp +++ b/src/core/loader/loader.cpp @@ -93,7 +93,7 @@ std::string GetFileTypeString(FileType type) {      return "unknown";  } -constexpr std::array<const char*, 59> RESULT_MESSAGES{ +constexpr std::array<const char*, 60> RESULT_MESSAGES{      "The operation completed successfully.",      "The loader requested to load is already loaded.",      "The operation is not implemented.", @@ -128,6 +128,7 @@ constexpr std::array<const char*, 59> RESULT_MESSAGES{      "The RomFS could not be found.",      "The ELF file has incorrect size as determined by the header.",      "There was a general error loading the NRO into emulated memory.", +    "There was a general error loading the NSO into emulated memory.",      "There is no icon available.",      "There is no control data available.",      "The NAX file has a bad header.", diff --git a/src/core/loader/loader.h b/src/core/loader/loader.h index 0e0333db5..e562b3a04 100644 --- a/src/core/loader/loader.h +++ b/src/core/loader/loader.h @@ -90,6 +90,7 @@ enum class ResultStatus : u16 {      ErrorNoRomFS,      ErrorIncorrectELFFileSize,      ErrorLoadingNRO, +    ErrorLoadingNSO,      ErrorNoIcon,      ErrorNoControl,      ErrorBadNAXHeader, diff --git a/src/core/loader/nro.cpp b/src/core/loader/nro.cpp index 576fe692a..243b499f2 100644 --- a/src/core/loader/nro.cpp +++ b/src/core/loader/nro.cpp @@ -127,10 +127,10 @@ static constexpr u32 PageAlignSize(u32 size) {      return (size + Memory::PAGE_MASK) & ~Memory::PAGE_MASK;  } -bool AppLoader_NRO::LoadNro(FileSys::VirtualFile file, VAddr load_base) { +bool AppLoader_NRO::LoadNro(const FileSys::VfsFile& file, VAddr load_base) {      // Read NSO header      NroHeader nro_header{}; -    if (sizeof(NroHeader) != file->ReadObject(&nro_header)) { +    if (sizeof(NroHeader) != file.ReadObject(&nro_header)) {          return {};      }      if (nro_header.magic != Common::MakeMagic('N', 'R', 'O', '0')) { @@ -138,7 +138,7 @@ bool AppLoader_NRO::LoadNro(FileSys::VirtualFile file, VAddr load_base) {      }      // Build program image -    std::vector<u8> program_image = file->ReadBytes(PageAlignSize(nro_header.file_size)); +    std::vector<u8> program_image = file.ReadBytes(PageAlignSize(nro_header.file_size));      if (program_image.size() != PageAlignSize(nro_header.file_size)) {          return {};      } @@ -182,7 +182,7 @@ bool AppLoader_NRO::LoadNro(FileSys::VirtualFile file, VAddr load_base) {      Core::CurrentProcess()->LoadModule(std::move(codeset), load_base);      // Register module with GDBStub -    GDBStub::RegisterModule(file->GetName(), load_base, load_base); +    GDBStub::RegisterModule(file.GetName(), load_base, load_base);      return true;  } @@ -195,7 +195,7 @@ ResultStatus AppLoader_NRO::Load(Kernel::Process& process) {      // Load NRO      const VAddr base_address = process.VMManager().GetCodeRegionBaseAddress(); -    if (!LoadNro(file, base_address)) { +    if (!LoadNro(*file, base_address)) {          return ResultStatus::ErrorLoadingNRO;      } diff --git a/src/core/loader/nro.h b/src/core/loader/nro.h index 04b46119a..50ee5a78a 100644 --- a/src/core/loader/nro.h +++ b/src/core/loader/nro.h @@ -41,7 +41,7 @@ public:      bool IsRomFSUpdatable() const override;  private: -    bool LoadNro(FileSys::VirtualFile file, VAddr load_base); +    bool LoadNro(const FileSys::VfsFile& file, VAddr load_base);      std::vector<u8> icon_data;      std::unique_ptr<FileSys::NACP> nacp; diff --git a/src/core/loader/nso.cpp b/src/core/loader/nso.cpp index ba57db9bf..68efca5c0 100644 --- a/src/core/loader/nso.cpp +++ b/src/core/loader/nso.cpp @@ -93,17 +93,14 @@ static constexpr u32 PageAlignSize(u32 size) {      return (size + Memory::PAGE_MASK) & ~Memory::PAGE_MASK;  } -VAddr AppLoader_NSO::LoadModule(FileSys::VirtualFile file, VAddr load_base, -                                bool should_pass_arguments, -                                boost::optional<FileSys::PatchManager> pm) { -    if (file == nullptr) -        return {}; - -    if (file->GetSize() < sizeof(NsoHeader)) +std::optional<VAddr> AppLoader_NSO::LoadModule(const FileSys::VfsFile& file, VAddr load_base, +                                               bool should_pass_arguments, +                                               std::optional<FileSys::PatchManager> pm) { +    if (file.GetSize() < sizeof(NsoHeader))          return {};      NsoHeader nso_header{}; -    if (sizeof(NsoHeader) != file->ReadObject(&nso_header)) +    if (sizeof(NsoHeader) != file.ReadObject(&nso_header))          return {};      if (nso_header.magic != Common::MakeMagic('N', 'S', 'O', '0')) @@ -114,7 +111,7 @@ VAddr AppLoader_NSO::LoadModule(FileSys::VirtualFile file, VAddr load_base,      std::vector<u8> program_image;      for (std::size_t i = 0; i < nso_header.segments.size(); ++i) {          std::vector<u8> data = -            file->ReadBytes(nso_header.segments_compressed_size[i], nso_header.segments[i].offset); +            file.ReadBytes(nso_header.segments_compressed_size[i], nso_header.segments[i].offset);          if (nso_header.IsSegmentCompressed(i)) {              data = DecompressSegment(data, nso_header.segments[i]);          } @@ -157,7 +154,7 @@ VAddr AppLoader_NSO::LoadModule(FileSys::VirtualFile file, VAddr load_base,      program_image.resize(image_size);      // Apply patches if necessary -    if (pm != boost::none && pm->HasNSOPatch(nso_header.build_id)) { +    if (pm && pm->HasNSOPatch(nso_header.build_id)) {          std::vector<u8> pi_header(program_image.size() + 0x100);          std::memcpy(pi_header.data(), &nso_header, sizeof(NsoHeader));          std::memcpy(pi_header.data() + 0x100, program_image.data(), program_image.size()); @@ -172,7 +169,7 @@ VAddr AppLoader_NSO::LoadModule(FileSys::VirtualFile file, VAddr load_base,      Core::CurrentProcess()->LoadModule(std::move(codeset), load_base);      // Register module with GDBStub -    GDBStub::RegisterModule(file->GetName(), load_base, load_base); +    GDBStub::RegisterModule(file.GetName(), load_base, load_base);      return load_base + image_size;  } @@ -184,7 +181,9 @@ ResultStatus AppLoader_NSO::Load(Kernel::Process& process) {      // Load module      const VAddr base_address = process.VMManager().GetCodeRegionBaseAddress(); -    LoadModule(file, base_address, true); +    if (!LoadModule(*file, base_address, true)) { +        return ResultStatus::ErrorLoadingNSO; +    }      LOG_DEBUG(Loader, "loaded module {} @ 0x{:X}", file->GetName(), base_address);      process.Run(base_address, Kernel::THREADPRIO_DEFAULT, Memory::DEFAULT_STACK_SIZE); diff --git a/src/core/loader/nso.h b/src/core/loader/nso.h index 70ab3b718..433306139 100644 --- a/src/core/loader/nso.h +++ b/src/core/loader/nso.h @@ -4,6 +4,7 @@  #pragma once +#include <optional>  #include "common/common_types.h"  #include "core/file_sys/patch_manager.h"  #include "core/loader/linker.h" @@ -36,8 +37,9 @@ public:          return IdentifyType(file);      } -    static VAddr LoadModule(FileSys::VirtualFile file, VAddr load_base, bool should_pass_arguments, -                            boost::optional<FileSys::PatchManager> pm = boost::none); +    static std::optional<VAddr> LoadModule(const FileSys::VfsFile& file, VAddr load_base, +                                           bool should_pass_arguments, +                                           std::optional<FileSys::PatchManager> pm = {});      ResultStatus Load(Kernel::Process& process) override;  }; | 
