2015-05-24 06:55:12 +02:00
|
|
|
// Copyright 2008 Dolphin Emulator Project
|
2015-05-18 01:08:10 +02:00
|
|
|
// Licensed under GPLv2+
|
2013-04-17 23:09:55 -04:00
|
|
|
// Refer to the license.txt file included.
|
2008-12-08 05:30:24 +00:00
|
|
|
|
2014-02-21 01:47:53 +01:00
|
|
|
#include <cstddef>
|
|
|
|
#include <cstring>
|
2015-08-31 19:27:18 -04:00
|
|
|
#include <memory>
|
2014-02-21 01:47:53 +01:00
|
|
|
#include <string>
|
2015-08-31 19:27:18 -04:00
|
|
|
#include <utility>
|
2008-12-08 05:30:24 +00:00
|
|
|
#include <vector>
|
|
|
|
|
2015-09-24 00:39:34 +02:00
|
|
|
#include <mbedtls/aes.h>
|
2008-12-08 05:30:24 +00:00
|
|
|
|
2014-09-07 20:06:58 -05:00
|
|
|
#include "Common/CommonTypes.h"
|
2014-02-17 05:18:15 -05:00
|
|
|
#include "Common/StringUtil.h"
|
2015-09-26 17:13:07 -04:00
|
|
|
#include "Common/Logging/Log.h"
|
2014-02-21 01:47:53 +01:00
|
|
|
#include "DiscIO/Blob.h"
|
2014-02-17 05:18:15 -05:00
|
|
|
#include "DiscIO/Volume.h"
|
2014-07-08 14:29:26 +02:00
|
|
|
#include "DiscIO/VolumeCreator.h"
|
2014-02-17 05:18:15 -05:00
|
|
|
#include "DiscIO/VolumeDirectory.h"
|
|
|
|
#include "DiscIO/VolumeGC.h"
|
|
|
|
#include "DiscIO/VolumeWad.h"
|
|
|
|
#include "DiscIO/VolumeWiiCrypted.h"
|
2008-12-08 05:30:24 +00:00
|
|
|
|
|
|
|
|
|
|
|
namespace DiscIO
|
|
|
|
{
|
|
|
|
enum EDiscType
|
|
|
|
{
|
|
|
|
DISC_TYPE_UNK,
|
|
|
|
DISC_TYPE_WII,
|
|
|
|
DISC_TYPE_WII_CONTAINER,
|
2009-06-07 02:54:07 +00:00
|
|
|
DISC_TYPE_GC,
|
|
|
|
DISC_TYPE_WAD
|
2008-12-08 05:30:24 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
class CBlobBigEndianReader
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
CBlobBigEndianReader(IBlobReader& _rReader) : m_rReader(_rReader) {}
|
|
|
|
|
|
|
|
u32 Read32(u64 _Offset)
|
|
|
|
{
|
|
|
|
u32 Temp;
|
|
|
|
m_rReader.Read(_Offset, 4, (u8*)&Temp);
|
2013-12-06 20:40:12 -06:00
|
|
|
return Common::swap32(Temp);
|
|
|
|
}
|
|
|
|
u16 Read16(u64 _Offset)
|
|
|
|
{
|
|
|
|
u16 Temp;
|
|
|
|
m_rReader.Read(_Offset, 2, (u8*)&Temp);
|
|
|
|
return Common::swap16(Temp);
|
|
|
|
}
|
|
|
|
u8 Read8(u64 _Offset)
|
|
|
|
{
|
|
|
|
u8 Temp;
|
|
|
|
m_rReader.Read(_Offset, 1, &Temp);
|
|
|
|
return Temp;
|
2008-12-08 05:30:24 +00:00
|
|
|
}
|
|
|
|
private:
|
|
|
|
IBlobReader& m_rReader;
|
|
|
|
};
|
|
|
|
|
2014-10-12 15:05:11 -04:00
|
|
|
static const unsigned char s_master_key[16] = {
|
|
|
|
0xeb,0xe4,0x2a,0x22,0x5e,0x85,0x93,0xe4,
|
|
|
|
0x48,0xd9,0xc5,0x45,0x73,0x81,0xaa,0xf7
|
|
|
|
};
|
|
|
|
|
|
|
|
static const unsigned char s_master_key_korean[16] = {
|
|
|
|
0x63,0xb8,0x2b,0xb4,0xf4,0x61,0x4e,0x2e,
|
|
|
|
0x13,0xf2,0xfe,0xfb,0xba,0x4c,0x9b,0x7e
|
|
|
|
};
|
2008-12-08 05:30:24 +00:00
|
|
|
|
2015-12-06 23:15:51 -05:00
|
|
|
static std::unique_ptr<IVolume> CreateVolumeFromCryptedWiiImage(std::unique_ptr<IBlobReader> reader, u32 partition_group, u32 volume_type, u32 volume_number);
|
2008-12-08 05:30:24 +00:00
|
|
|
EDiscType GetDiscType(IBlobReader& _rReader);
|
|
|
|
|
2015-12-06 23:15:51 -05:00
|
|
|
std::unique_ptr<IVolume> CreateVolumeFromFilename(const std::string& filename, u32 partition_group, u32 volume_number)
|
2008-12-08 05:30:24 +00:00
|
|
|
{
|
2015-12-06 23:15:51 -05:00
|
|
|
std::unique_ptr<IBlobReader> reader(CreateBlobReader(filename));
|
2015-08-31 19:27:18 -04:00
|
|
|
if (reader == nullptr)
|
2014-03-09 21:14:26 +01:00
|
|
|
return nullptr;
|
2008-12-08 05:30:24 +00:00
|
|
|
|
2015-08-31 19:27:18 -04:00
|
|
|
switch (GetDiscType(*reader))
|
2008-12-08 05:30:24 +00:00
|
|
|
{
|
2009-02-16 06:17:21 +00:00
|
|
|
case DISC_TYPE_WII:
|
|
|
|
case DISC_TYPE_GC:
|
2015-12-06 23:15:51 -05:00
|
|
|
return std::make_unique<CVolumeGC>(std::move(reader));
|
2009-02-16 06:17:21 +00:00
|
|
|
|
2009-06-07 02:54:07 +00:00
|
|
|
case DISC_TYPE_WAD:
|
2015-12-06 23:15:51 -05:00
|
|
|
return std::make_unique<CVolumeWAD>(std::move(reader));
|
2009-06-07 02:54:07 +00:00
|
|
|
|
2009-02-16 06:17:21 +00:00
|
|
|
case DISC_TYPE_WII_CONTAINER:
|
2015-12-06 23:15:51 -05:00
|
|
|
return CreateVolumeFromCryptedWiiImage(std::move(reader), partition_group, 0, volume_number);
|
2009-02-16 06:17:21 +00:00
|
|
|
|
|
|
|
case DISC_TYPE_UNK:
|
|
|
|
default:
|
2015-12-06 23:15:51 -05:00
|
|
|
std::string name, extension;
|
|
|
|
SplitPath(filename, nullptr, &name, &extension);
|
|
|
|
name += extension;
|
2010-01-11 05:07:56 +00:00
|
|
|
NOTICE_LOG(DISCIO, "%s does not have the Magic word for a gcm, wiidisc or wad file\n"
|
2015-12-06 23:15:51 -05:00
|
|
|
"Set Log Verbosity to Warning and attempt to load the game again to view the values", name.c_str());
|
2008-12-08 05:30:24 +00:00
|
|
|
}
|
|
|
|
|
2014-03-09 21:14:26 +01:00
|
|
|
return nullptr;
|
2008-12-08 05:30:24 +00:00
|
|
|
}
|
|
|
|
|
2015-12-06 23:15:51 -05:00
|
|
|
std::unique_ptr<IVolume> CreateVolumeFromDirectory(const std::string& directory, bool is_wii, const std::string& apploader, const std::string& dol)
|
2008-12-08 05:30:24 +00:00
|
|
|
{
|
2015-12-06 23:15:51 -05:00
|
|
|
if (CVolumeDirectory::IsValidDirectory(directory))
|
|
|
|
return std::make_unique<CVolumeDirectory>(directory, is_wii, apploader, dol);
|
2013-10-29 01:23:17 -04:00
|
|
|
|
2014-03-09 21:14:26 +01:00
|
|
|
return nullptr;
|
2008-12-08 05:30:24 +00:00
|
|
|
}
|
|
|
|
|
2015-08-31 19:30:32 -04:00
|
|
|
void VolumeKeyForPartition(IBlobReader& _rReader, u64 offset, u8* VolumeKey)
|
2014-05-29 03:38:39 -07:00
|
|
|
{
|
|
|
|
CBlobBigEndianReader Reader(_rReader);
|
|
|
|
|
|
|
|
u8 SubKey[16];
|
|
|
|
_rReader.Read(offset + 0x1bf, 16, SubKey);
|
|
|
|
|
|
|
|
u8 IV[16];
|
|
|
|
memset(IV, 0, 16);
|
|
|
|
_rReader.Read(offset + 0x44c, 8, IV);
|
|
|
|
|
|
|
|
bool usingKoreanKey = false;
|
|
|
|
// Issue: 6813
|
|
|
|
// Magic value is at partition's offset + 0x1f1 (1byte)
|
|
|
|
// If encrypted with the Korean key, the magic value would be 1
|
|
|
|
// Otherwise it is zero
|
|
|
|
if (Reader.Read8(0x3) == 'K' && Reader.Read8(offset + 0x1f1) == 1)
|
|
|
|
usingKoreanKey = true;
|
|
|
|
|
2015-09-24 00:39:34 +02:00
|
|
|
mbedtls_aes_context AES_ctx;
|
|
|
|
mbedtls_aes_setkey_dec(&AES_ctx, (usingKoreanKey ? s_master_key_korean : s_master_key), 128);
|
2014-05-29 03:38:39 -07:00
|
|
|
|
2015-09-24 00:39:34 +02:00
|
|
|
mbedtls_aes_crypt_cbc(&AES_ctx, MBEDTLS_AES_DECRYPT, 16, IV, SubKey, VolumeKey);
|
2014-05-29 03:38:39 -07:00
|
|
|
}
|
|
|
|
|
2015-12-06 23:15:51 -05:00
|
|
|
static std::unique_ptr<IVolume> CreateVolumeFromCryptedWiiImage(std::unique_ptr<IBlobReader> reader, u32 partition_group, u32 volume_type, u32 volume_number)
|
2008-12-08 05:30:24 +00:00
|
|
|
{
|
2015-08-31 19:27:18 -04:00
|
|
|
CBlobBigEndianReader big_endian_reader(*reader);
|
2008-12-08 05:30:24 +00:00
|
|
|
|
2015-12-06 23:15:51 -05:00
|
|
|
u32 numPartitions = big_endian_reader.Read32(0x40000 + (partition_group * 8));
|
|
|
|
u64 PartitionsOffset = (u64)big_endian_reader.Read32(0x40000 + (partition_group * 8) + 4) << 2;
|
2009-05-22 00:36:44 +00:00
|
|
|
|
2009-03-10 17:19:30 +00:00
|
|
|
// Check if we're looking for a valid partition
|
2015-12-06 23:15:51 -05:00
|
|
|
if ((int)volume_number != -1 && volume_number > numPartitions)
|
2014-03-09 21:14:26 +01:00
|
|
|
return nullptr;
|
2009-05-22 00:36:44 +00:00
|
|
|
|
2008-12-08 05:30:24 +00:00
|
|
|
struct SPartition
|
|
|
|
{
|
2015-12-06 23:15:51 -05:00
|
|
|
u64 offset;
|
|
|
|
u32 type;
|
2008-12-08 05:30:24 +00:00
|
|
|
};
|
2014-08-29 20:07:55 -04:00
|
|
|
|
2009-05-22 00:36:44 +00:00
|
|
|
struct SPartitionGroup
|
|
|
|
{
|
2015-12-06 23:15:51 -05:00
|
|
|
u32 num_partitions;
|
|
|
|
u64 partitions_offset;
|
|
|
|
std::vector<SPartition> partitions;
|
2009-05-22 00:36:44 +00:00
|
|
|
};
|
2015-12-06 23:15:51 -05:00
|
|
|
SPartitionGroup partition_groups[4];
|
2013-10-29 01:23:17 -04:00
|
|
|
|
2015-04-02 15:50:14 -04:00
|
|
|
// Read all partitions
|
2015-12-06 23:15:51 -05:00
|
|
|
for (SPartitionGroup& group : partition_groups)
|
2008-12-08 05:30:24 +00:00
|
|
|
{
|
2009-05-22 00:36:44 +00:00
|
|
|
for (u32 i = 0; i < numPartitions; i++)
|
|
|
|
{
|
2015-12-06 23:15:51 -05:00
|
|
|
SPartition partition;
|
|
|
|
partition.offset = ((u64)big_endian_reader.Read32(PartitionsOffset + (i * 8) + 0)) << 2;
|
|
|
|
partition.type = big_endian_reader.Read32(PartitionsOffset + (i * 8) + 4);
|
|
|
|
group.partitions.push_back(partition);
|
2009-05-22 00:36:44 +00:00
|
|
|
}
|
2008-12-08 05:30:24 +00:00
|
|
|
}
|
|
|
|
|
2015-04-02 15:50:14 -04:00
|
|
|
// Return the partition type specified or number
|
2009-03-10 17:19:30 +00:00
|
|
|
// types: 0 = game, 1 = firmware update, 2 = channel installer
|
2015-04-02 15:50:14 -04:00
|
|
|
// some partitions on SSBB use the ASCII title id of the demo VC game they hold...
|
2015-12-06 23:15:51 -05:00
|
|
|
for (size_t i = 0; i < partition_groups[partition_group].partitions.size(); i++)
|
2008-12-08 05:30:24 +00:00
|
|
|
{
|
2015-12-06 23:15:51 -05:00
|
|
|
const SPartition& partition = partition_groups[partition_group].partitions.at(i);
|
2008-12-08 05:30:24 +00:00
|
|
|
|
2015-12-06 23:15:51 -05:00
|
|
|
if ((partition.type == volume_type && (int)volume_number == -1) || i == volume_number)
|
2008-12-08 05:30:24 +00:00
|
|
|
{
|
2015-12-06 23:15:51 -05:00
|
|
|
u8 volume_key[16];
|
|
|
|
VolumeKeyForPartition(*reader, partition.offset, volume_key);
|
|
|
|
return std::make_unique<CVolumeWiiCrypted>(std::move(reader), partition.offset, volume_key);
|
2008-12-08 05:30:24 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-03-09 21:14:26 +01:00
|
|
|
return nullptr;
|
2008-12-08 05:30:24 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
EDiscType GetDiscType(IBlobReader& _rReader)
|
|
|
|
{
|
|
|
|
CBlobBigEndianReader Reader(_rReader);
|
2010-06-01 20:47:07 +00:00
|
|
|
u32 WiiMagic = Reader.Read32(0x18);
|
|
|
|
u32 WiiContainerMagic = Reader.Read32(0x60);
|
|
|
|
u32 WADMagic = Reader.Read32(0x02);
|
|
|
|
u32 GCMagic = Reader.Read32(0x1C);
|
2008-12-08 05:30:24 +00:00
|
|
|
|
2015-04-02 15:50:14 -04:00
|
|
|
// Check for Wii
|
2010-06-01 20:47:07 +00:00
|
|
|
if (WiiMagic == 0x5D1C9EA3 && WiiContainerMagic != 0)
|
|
|
|
return DISC_TYPE_WII;
|
|
|
|
if (WiiMagic == 0x5D1C9EA3 && WiiContainerMagic == 0)
|
|
|
|
return DISC_TYPE_WII_CONTAINER;
|
2008-12-08 05:30:24 +00:00
|
|
|
|
2015-04-02 15:50:14 -04:00
|
|
|
// Check for WAD
|
2010-06-01 20:47:07 +00:00
|
|
|
// 0x206962 for boot2 wads
|
|
|
|
if (WADMagic == 0x00204973 || WADMagic == 0x00206962)
|
|
|
|
return DISC_TYPE_WAD;
|
2009-06-07 02:54:07 +00:00
|
|
|
|
2015-04-02 15:50:14 -04:00
|
|
|
// Check for GC
|
2010-06-01 20:47:07 +00:00
|
|
|
if (GCMagic == 0xC2339F3D)
|
|
|
|
return DISC_TYPE_GC;
|
|
|
|
|
|
|
|
WARN_LOG(DISCIO, "No known magic words found");
|
|
|
|
WARN_LOG(DISCIO, "Wii offset: 0x18 value: 0x%08x", WiiMagic);
|
|
|
|
WARN_LOG(DISCIO, "WiiC offset: 0x60 value: 0x%08x", WiiContainerMagic);
|
|
|
|
WARN_LOG(DISCIO, "WAD offset: 0x02 value: 0x%08x", WADMagic);
|
|
|
|
WARN_LOG(DISCIO, "GC offset: 0x1C value: 0x%08x", GCMagic);
|
2008-12-08 05:30:24 +00:00
|
|
|
|
|
|
|
return DISC_TYPE_UNK;
|
|
|
|
}
|
|
|
|
|
|
|
|
} // namespace
|