mirror of
https://github.com/dolphin-emu/dolphin.git
synced 2025-01-25 15:31:17 +01:00
cec601f1fb
The header of a Wii disc can be read from two places: The unencrypted area at the beginning of the disc, or the beginning of the game partition. The two copies are usually identical (except for 0x60 and 0x61), but there are exceptions. For most of Dolphin's history, we have been reading from the header inside the game partition when getting metadata. This was however not the case starting with 4.0-4901 and ending with 5.0-3762. This commit once again makes Dolphin read metadata from the unencrypted header, because of the following reasons that I recently was informed about: - The "pink fish" disc has the game ID 410E01 in the unencrypted header but the placeholder game ID RELSAB in the partition header. - The revisions of some games differ between the two headers, with the unencrypted one making more sense. (See https://bugs.dolphin-emu.org/issues/11387) For better or worse, this also means that sloppily hacked games where only the game ID in the unencrypted header has been changed now will use that modified game ID. And unlike with the partition header, there is no signing or hashing that can tell us whether the unencrypted header has been modified by someone other than Nintendo.
138 lines
5.3 KiB
C++
138 lines
5.3 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;
|
|
|
|
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 {}; }
|
|
virtual std::optional<u64> GetTitleID(const Partition& partition = PARTITION_NONE) 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; }
|
|
// 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 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 CheckIntegrity(const Partition& partition) const { return false; }
|
|
// May be inaccurate for WADs
|
|
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;
|
|
// 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;
|
|
};
|
|
|
|
std::unique_ptr<Volume> CreateVolumeFromFilename(const std::string& filename);
|
|
|
|
} // namespace
|