mirror of
https://github.com/dolphin-emu/dolphin.git
synced 2025-01-09 23:59:27 +01:00
4b73d18eaa
The old implementation of this was not able to distinguish between a title that had the common key index set to 1 because it actually was Korean and a title that had the common key index set to 1 due to fakesigning. This new implementation solves the problem by decrypting a content with each possible common key and checking which result matches the provided SHA-1 hash. The problem that the old implementation causes has only been reported to affect a certain pirated WAD of Chronos Twins DX (WC6EUP), but it's possible that the problem would start affecting more WADs if we add support for the vWii common key (which uses index 2). Adding support for the vWii common key would also prevent us from using the simpler solution of always forcing the index to 0 if the title is not Korean.
161 lines
6.4 KiB
C++
161 lines
6.4 KiB
C++
// Copyright 2008 Dolphin Emulator Project
|
|
// Licensed under GPLv2+
|
|
// Refer to the license.txt file included.
|
|
|
|
#pragma once
|
|
|
|
#include <cstring>
|
|
#include <limits>
|
|
#include <map>
|
|
#include <memory>
|
|
#include <optional>
|
|
#include <string>
|
|
#include <vector>
|
|
|
|
#include "Common/CommonTypes.h"
|
|
#include "Common/StringUtil.h"
|
|
#include "Common/Swap.h"
|
|
#include "Core/IOS/ES/Formats.h"
|
|
#include "DiscIO/Enums.h"
|
|
|
|
namespace DiscIO
|
|
{
|
|
enum class BlobType;
|
|
class FileSystem;
|
|
class VolumeWAD;
|
|
|
|
struct Partition final
|
|
{
|
|
Partition() : offset(std::numeric_limits<u64>::max()) {}
|
|
explicit Partition(u64 offset_) : offset(offset_) {}
|
|
bool operator==(const Partition& other) const { return offset == other.offset; }
|
|
bool operator!=(const Partition& other) const { return !(*this == other); }
|
|
bool operator<(const Partition& other) const { return offset < other.offset; }
|
|
bool operator>(const Partition& other) const { return other < *this; }
|
|
bool operator<=(const Partition& other) const { return !(*this < other); }
|
|
bool operator>=(const Partition& other) const { return !(*this > other); }
|
|
u64 offset;
|
|
};
|
|
|
|
const Partition PARTITION_NONE(std::numeric_limits<u64>::max() - 1);
|
|
|
|
class Volume
|
|
{
|
|
public:
|
|
Volume() {}
|
|
virtual ~Volume() {}
|
|
virtual bool Read(u64 offset, u64 length, u8* buffer, const Partition& partition) const = 0;
|
|
template <typename T>
|
|
std::optional<T> ReadSwapped(u64 offset, const Partition& partition) const
|
|
{
|
|
T temp;
|
|
if (!Read(offset, sizeof(T), reinterpret_cast<u8*>(&temp), partition))
|
|
return {};
|
|
return Common::FromBigEndian(temp);
|
|
}
|
|
std::optional<u64> ReadSwappedAndShifted(u64 offset, const Partition& partition) const
|
|
{
|
|
const std::optional<u32> temp = ReadSwapped<u32>(offset, partition);
|
|
return temp ? static_cast<u64>(*temp) << GetOffsetShift() : std::optional<u64>();
|
|
}
|
|
|
|
virtual bool IsEncryptedAndHashed() const { return false; }
|
|
virtual std::vector<Partition> GetPartitions() const { return {}; }
|
|
virtual Partition GetGamePartition() const { return PARTITION_NONE; }
|
|
virtual std::optional<u32> GetPartitionType(const Partition& partition) const { return {}; }
|
|
std::optional<u64> GetTitleID() const { return GetTitleID(GetGamePartition()); }
|
|
virtual std::optional<u64> GetTitleID(const Partition& partition) const { return {}; }
|
|
virtual const IOS::ES::TicketReader& GetTicket(const Partition& partition) const
|
|
{
|
|
return INVALID_TICKET;
|
|
}
|
|
virtual const IOS::ES::TMDReader& GetTMD(const Partition& partition) const { return INVALID_TMD; }
|
|
virtual const std::vector<u8>& GetCertificateChain(const Partition& partition) const
|
|
{
|
|
return INVALID_CERT_CHAIN;
|
|
}
|
|
virtual std::vector<u8> GetContent(u16 index) const { return {}; }
|
|
virtual std::vector<u64> GetContentOffsets() const { return {}; }
|
|
virtual bool CheckContentIntegrity(const IOS::ES::Content& content, u64 content_offset,
|
|
const IOS::ES::TicketReader& ticket) const
|
|
{
|
|
return false;
|
|
}
|
|
virtual IOS::ES::TicketReader GetTicketWithFixedCommonKey() const { return {}; }
|
|
// Returns a non-owning pointer. Returns nullptr if the file system couldn't be read.
|
|
virtual const FileSystem* GetFileSystem(const Partition& partition) const = 0;
|
|
virtual u64 PartitionOffsetToRawOffset(u64 offset, const Partition& partition) const
|
|
{
|
|
return offset;
|
|
}
|
|
virtual std::string GetGameID(const Partition& partition = PARTITION_NONE) const = 0;
|
|
virtual std::string GetGameTDBID(const Partition& partition = PARTITION_NONE) const = 0;
|
|
virtual std::string GetMakerID(const Partition& partition = PARTITION_NONE) const = 0;
|
|
virtual std::optional<u16> GetRevision(const Partition& partition = PARTITION_NONE) const = 0;
|
|
virtual std::string GetInternalName(const Partition& partition = PARTITION_NONE) const = 0;
|
|
virtual std::map<Language, std::string> GetShortNames() const { return {}; }
|
|
virtual std::map<Language, std::string> GetLongNames() const { return {}; }
|
|
virtual std::map<Language, std::string> GetShortMakers() const { return {}; }
|
|
virtual std::map<Language, std::string> GetLongMakers() const { return {}; }
|
|
virtual std::map<Language, std::string> GetDescriptions() const { return {}; }
|
|
virtual std::vector<u32> GetBanner(u32* width, u32* height) const = 0;
|
|
std::string GetApploaderDate() const { return GetApploaderDate(GetGamePartition()); }
|
|
virtual std::string GetApploaderDate(const Partition& partition) const = 0;
|
|
// 0 is the first disc, 1 is the second disc
|
|
virtual std::optional<u8> GetDiscNumber(const Partition& partition = PARTITION_NONE) const
|
|
{
|
|
return 0;
|
|
}
|
|
virtual Platform GetVolumeType() const = 0;
|
|
virtual bool SupportsIntegrityCheck() const { return false; }
|
|
virtual bool CheckH3TableIntegrity(const Partition& partition) const { return false; }
|
|
virtual bool CheckBlockIntegrity(u64 block_index, const Partition& partition) const
|
|
{
|
|
return false;
|
|
}
|
|
virtual Region GetRegion() const = 0;
|
|
virtual Country GetCountry(const Partition& partition = PARTITION_NONE) const = 0;
|
|
virtual BlobType GetBlobType() const = 0;
|
|
// Size of virtual disc (may be inaccurate depending on the blob type)
|
|
virtual u64 GetSize() const = 0;
|
|
virtual bool IsSizeAccurate() const = 0;
|
|
// Size on disc (compressed size)
|
|
virtual u64 GetRawSize() const = 0;
|
|
|
|
protected:
|
|
template <u32 N>
|
|
std::string DecodeString(const char (&data)[N]) const
|
|
{
|
|
// strnlen to trim NULLs
|
|
std::string string(data, strnlen(data, sizeof(data)));
|
|
|
|
if (GetRegion() == Region::NTSC_J)
|
|
return SHIFTJISToUTF8(string);
|
|
else
|
|
return CP1252ToUTF8(string);
|
|
}
|
|
|
|
virtual u32 GetOffsetShift() const { return 0; }
|
|
static std::map<Language, std::string> ReadWiiNames(const std::vector<char16_t>& data);
|
|
|
|
static const size_t NUMBER_OF_LANGUAGES = 10;
|
|
static const size_t NAME_CHARS_LENGTH = 42;
|
|
static const size_t NAME_BYTES_LENGTH = NAME_CHARS_LENGTH * sizeof(char16_t);
|
|
static const size_t NAMES_TOTAL_CHARS = NAME_CHARS_LENGTH * NUMBER_OF_LANGUAGES;
|
|
static const size_t NAMES_TOTAL_BYTES = NAME_BYTES_LENGTH * NUMBER_OF_LANGUAGES;
|
|
|
|
static const IOS::ES::TicketReader INVALID_TICKET;
|
|
static const IOS::ES::TMDReader INVALID_TMD;
|
|
static const std::vector<u8> INVALID_CERT_CHAIN;
|
|
};
|
|
|
|
class VolumeDisc : public Volume
|
|
{
|
|
};
|
|
|
|
std::unique_ptr<VolumeDisc> CreateDisc(const std::string& path);
|
|
std::unique_ptr<VolumeWAD> CreateWAD(const std::string& path);
|
|
std::unique_ptr<Volume> CreateVolume(const std::string& path);
|
|
|
|
} // namespace DiscIO
|