| 
									
										
										
										
											2013-04-17 23:09:55 -04:00
										 |  |  | // Copyright 2013 Dolphin Emulator Project
 | 
					
						
							|  |  |  | // Licensed under GPLv2
 | 
					
						
							|  |  |  | // 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>
 | 
					
						
							|  |  |  | #include <string>
 | 
					
						
							| 
									
										
										
										
											2008-12-08 05:30:24 +00:00
										 |  |  | #include <vector>
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-10-27 18:27:07 +00:00
										 |  |  | #include <polarssl/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"
 | 
					
						
							| 
									
										
										
										
											2008-12-08 05:30:24 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											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
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-05-29 03:38:39 -07:00
										 |  |  | static IVolume* CreateVolumeFromCryptedWiiImage(IBlobReader& _rReader, u32 _PartitionGroup, u32 _VolumeType, u32 _VolumeNum); | 
					
						
							| 
									
										
										
										
											2008-12-08 05:30:24 +00:00
										 |  |  | EDiscType GetDiscType(IBlobReader& _rReader); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-05-22 00:36:44 +00:00
										 |  |  | IVolume* CreateVolumeFromFilename(const std::string& _rFilename, u32 _PartitionGroup, u32 _VolumeNum) | 
					
						
							| 
									
										
										
										
											2008-12-08 05:30:24 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2014-03-12 15:33:41 -04:00
										 |  |  | 	IBlobReader* pReader = CreateBlobReader(_rFilename); | 
					
						
							| 
									
										
										
										
											2014-03-09 21:14:26 +01:00
										 |  |  | 	if (pReader == nullptr) | 
					
						
							|  |  |  | 		return nullptr; | 
					
						
							| 
									
										
										
										
											2008-12-08 05:30:24 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	switch (GetDiscType(*pReader)) | 
					
						
							|  |  |  | 	{ | 
					
						
							| 
									
										
										
										
											2009-02-16 06:17:21 +00:00
										 |  |  | 		case DISC_TYPE_WII: | 
					
						
							|  |  |  | 		case DISC_TYPE_GC: | 
					
						
							|  |  |  | 			return new CVolumeGC(pReader); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-06-07 02:54:07 +00:00
										 |  |  | 		case DISC_TYPE_WAD: | 
					
						
							|  |  |  | 			return new CVolumeWAD(pReader); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-16 06:17:21 +00:00
										 |  |  | 		case DISC_TYPE_WII_CONTAINER: | 
					
						
							|  |  |  | 		{ | 
					
						
							| 
									
										
										
										
											2014-05-29 03:38:39 -07:00
										 |  |  | 			IVolume* pVolume = CreateVolumeFromCryptedWiiImage(*pReader, _PartitionGroup, 0, _VolumeNum); | 
					
						
							| 
									
										
										
										
											2009-02-16 06:17:21 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-03-09 21:14:26 +01:00
										 |  |  | 			if (pVolume == nullptr) | 
					
						
							| 
									
										
										
										
											2009-02-16 06:17:21 +00:00
										 |  |  | 			{ | 
					
						
							|  |  |  | 				delete pReader; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-24 19:57:29 +00:00
										 |  |  | 			return pVolume; | 
					
						
							| 
									
										
										
										
											2009-02-16 06:17:21 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		case DISC_TYPE_UNK: | 
					
						
							|  |  |  | 		default: | 
					
						
							| 
									
										
										
										
											2010-01-11 05:07:56 +00:00
										 |  |  | 			std::string Filename, ext; | 
					
						
							| 
									
										
										
										
											2014-03-09 21:14:26 +01:00
										 |  |  | 			SplitPath(_rFilename, nullptr, &Filename, &ext); | 
					
						
							| 
									
										
										
										
											2010-01-11 05:07:56 +00:00
										 |  |  | 			Filename += ext; | 
					
						
							|  |  |  | 			NOTICE_LOG(DISCIO, "%s does not have the Magic word for a gcm, wiidisc or wad file\n" | 
					
						
							|  |  |  | 						"Set Log Verbosity to Warning and attempt to load the game again to view the values", Filename.c_str()); | 
					
						
							| 
									
										
										
										
											2009-02-16 06:17:21 +00:00
										 |  |  | 			delete pReader; | 
					
						
							| 
									
										
										
										
											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
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-10 09:16:10 +00:00
										 |  |  | IVolume* CreateVolumeFromDirectory(const std::string& _rDirectory, bool _bIsWii, const std::string& _rApploader, const std::string& _rDOL) | 
					
						
							| 
									
										
										
										
											2008-12-08 05:30:24 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	if (CVolumeDirectory::IsValidDirectory(_rDirectory)) | 
					
						
							| 
									
										
										
										
											2009-12-10 09:16:10 +00:00
										 |  |  | 		return new CVolumeDirectory(_rDirectory, _bIsWii, _rApploader, _rDOL); | 
					
						
							| 
									
										
										
										
											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
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-06-07 02:54:07 +00:00
										 |  |  | bool IsVolumeWadFile(const IVolume *_rVolume) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	u32 MagicWord = 0; | 
					
						
							| 
									
										
										
										
											2014-12-28 10:35:48 +01:00
										 |  |  | 	_rVolume->Read(0x02, 4, (u8*)&MagicWord, false); | 
					
						
							| 
									
										
										
										
											2009-06-07 02:54:07 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-06-08 02:25:21 +00:00
										 |  |  | 	return (Common::swap32(MagicWord) == 0x00204973 || Common::swap32(MagicWord) == 0x00206962); | 
					
						
							| 
									
										
										
										
											2009-06-07 02:54:07 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-05-29 03:38:39 -07:00
										 |  |  | void VolumeKeyForParition(IBlobReader& _rReader, u64 offset, u8* VolumeKey) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	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; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	aes_context AES_ctx; | 
					
						
							|  |  |  | 	aes_setkey_dec(&AES_ctx, (usingKoreanKey ? s_master_key_korean : s_master_key), 128); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	aes_crypt_cbc(&AES_ctx, AES_DECRYPT, 16, IV, SubKey, VolumeKey); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static IVolume* CreateVolumeFromCryptedWiiImage(IBlobReader& _rReader, u32 _PartitionGroup, u32 _VolumeType, u32 _VolumeNum) | 
					
						
							| 
									
										
										
										
											2008-12-08 05:30:24 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	CBlobBigEndianReader Reader(_rReader); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-05-22 00:36:44 +00:00
										 |  |  | 	u32 numPartitions = Reader.Read32(0x40000 + (_PartitionGroup * 8)); | 
					
						
							|  |  |  | 	u64 PartitionsOffset = (u64)Reader.Read32(0x40000 + (_PartitionGroup * 8) + 4) << 2; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-03-10 17:19:30 +00:00
										 |  |  | 	// Check if we're looking for a valid partition
 | 
					
						
							| 
									
										
										
										
											2010-02-24 03:38:36 +00:00
										 |  |  | 	if ((int)_VolumeNum != -1 && _VolumeNum > 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 | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 		u64 Offset; | 
					
						
							|  |  |  | 		u32 Type; | 
					
						
							|  |  |  | 	}; | 
					
						
							| 
									
										
										
										
											2014-08-29 20:07:55 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-05-22 00:36:44 +00:00
										 |  |  | 	struct SPartitionGroup | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 		u32 numPartitions; | 
					
						
							|  |  |  | 		u64 PartitionsOffset; | 
					
						
							|  |  |  | 		std::vector<SPartition> PartitionsVec; | 
					
						
							|  |  |  | 	}; | 
					
						
							|  |  |  | 	SPartitionGroup PartitionGroup[4]; | 
					
						
							| 
									
										
										
										
											2013-10-29 01:23:17 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-08 05:30:24 +00:00
										 |  |  | 	// read all partitions
 | 
					
						
							| 
									
										
										
										
											2014-03-03 06:25:15 +01:00
										 |  |  | 	for (SPartitionGroup& group : PartitionGroup) | 
					
						
							| 
									
										
										
										
											2008-12-08 05:30:24 +00:00
										 |  |  | 	{ | 
					
						
							| 
									
										
										
										
											2009-05-22 00:36:44 +00:00
										 |  |  | 		for (u32 i = 0; i < numPartitions; i++) | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			SPartition Partition; | 
					
						
							|  |  |  | 			Partition.Offset = ((u64)Reader.Read32(PartitionsOffset + (i * 8) + 0)) << 2; | 
					
						
							|  |  |  | 			Partition.Type   = Reader.Read32(PartitionsOffset + (i * 8) + 4); | 
					
						
							| 
									
										
										
										
											2014-03-03 06:25:15 +01:00
										 |  |  | 			group.PartitionsVec.push_back(Partition); | 
					
						
							| 
									
										
										
										
											2009-05-22 00:36:44 +00:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2008-12-08 05:30:24 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-03-10 17:19:30 +00:00
										 |  |  | 	// return the partition type specified or number
 | 
					
						
							|  |  |  | 	// types: 0 = game, 1 = firmware update, 2 = channel installer
 | 
					
						
							|  |  |  | 	//  some partitions on ssbb use the ascii title id of the demo VC game they hold...
 | 
					
						
							| 
									
										
										
										
											2009-05-22 00:36:44 +00:00
										 |  |  | 	for (size_t i = 0; i < PartitionGroup[_PartitionGroup].PartitionsVec.size(); i++) | 
					
						
							| 
									
										
										
										
											2008-12-08 05:30:24 +00:00
										 |  |  | 	{ | 
					
						
							| 
									
										
										
										
											2009-05-22 00:36:44 +00:00
										 |  |  | 		const SPartition& rPartition = PartitionGroup[_PartitionGroup].PartitionsVec.at(i); | 
					
						
							| 
									
										
										
										
											2008-12-08 05:30:24 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-05-29 03:38:39 -07:00
										 |  |  | 		if ((rPartition.Type == _VolumeType && (int)_VolumeNum == -1) || i == _VolumeNum) | 
					
						
							| 
									
										
										
										
											2008-12-08 05:30:24 +00:00
										 |  |  | 		{ | 
					
						
							|  |  |  | 			u8 VolumeKey[16]; | 
					
						
							| 
									
										
										
										
											2014-05-29 03:38:39 -07:00
										 |  |  | 			VolumeKeyForParition(_rReader, rPartition.Offset, VolumeKey); | 
					
						
							|  |  |  | 			return new CVolumeWiiCrypted(&_rReader, rPartition.Offset, VolumeKey); | 
					
						
							| 
									
										
										
										
											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
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-06-01 20:47:07 +00:00
										 |  |  | 	// check for Wii
 | 
					
						
							|  |  |  | 	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
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-06-07 02:54:07 +00: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
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-08 05:30:24 +00: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
 |