// Copyright (C) 2003-2008 Dolphin Project. // This program is free software: you can redistribute it and/or modify // it under the terms of the GNU General Public License as published by // the Free Software Foundation, version 2.0. // This program is distributed in the hope that it will be useful, // but WITHOUT ANY WARRANTY; without even the implied warranty of // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the // GNU General Public License 2.0 for more details. // A copy of the GPL 2.0 should have been included with the program. // If not, see http://www.gnu.org/licenses/ // Official SVN repository and contact information can be found at // http://code.google.com/p/dolphin-emu/ #include #include "AES/aes.h" #include "VolumeCreator.h" #include "Volume.h" #include "VolumeDirectory.h" #include "VolumeGC.h" #include "VolumeWiiCrypted.h" #include "Hash.h" namespace DiscIO { enum EDiscType { DISC_TYPE_UNK, DISC_TYPE_WII, DISC_TYPE_WII_CONTAINER, DISC_TYPE_GC }; #ifndef _WIN32 struct SPartition { u64 Offset; u32 Type; }; //gcc 4.3 cries if it's local #endif class CBlobBigEndianReader { public: CBlobBigEndianReader(IBlobReader& _rReader) : m_rReader(_rReader) {} u32 Read32(u64 _Offset) { u32 Temp; m_rReader.Read(_Offset, 4, (u8*)&Temp); return(Common::swap32(Temp)); } private: IBlobReader& m_rReader; }; unsigned char g_MasterKey[16]; unsigned char g_MasterKeyK[16]; bool g_MasterKeyInit[] = {false, false}; IVolume* CreateVolumeFromCryptedWiiImage(IBlobReader& _rReader, u32 _VolumeType, bool Korean); EDiscType GetDiscType(IBlobReader& _rReader); IVolume* CreateVolumeFromFilename(const std::string& _rFilename) { IBlobReader* pReader = CreateBlobReader(_rFilename.c_str()); if (pReader == NULL) return NULL; switch (GetDiscType(*pReader)) { case DISC_TYPE_WII: case DISC_TYPE_GC: return new CVolumeGC(pReader); case DISC_TYPE_WII_CONTAINER: { u8 region; pReader->Read(0x3,1,®ion); IVolume* pVolume = CreateVolumeFromCryptedWiiImage(*pReader, 0, region == 'K'); if (pVolume == NULL) { delete pReader; } return(pVolume); } break; case DISC_TYPE_UNK: default: delete pReader; return NULL; } // unreachable code return NULL; } IVolume* CreateVolumeFromDirectory(const std::string& _rDirectory, bool _bIsWii) { if (CVolumeDirectory::IsValidDirectory(_rDirectory)) return new CVolumeDirectory(_rDirectory, _bIsWii); return NULL; } bool IsVolumeWiiDisc(const IVolume *_rVolume) { u32 MagicWord = 0; _rVolume->Read(0x18, 4, (u8*)&MagicWord); return (Common::swap32(MagicWord) == 0x5D1C9EA3); } IVolume* CreateVolumeFromCryptedWiiImage(IBlobReader& _rReader, u32 _VolumeType, bool Korean) { if (!g_MasterKeyInit[Korean ? 1 : 0]) { const char * MasterKeyFile = Korean ? WII_MASTERKEY1_FILE : WII_MASTERKEY_FILE; const char * MasterKeyFileHex = Korean ? WII_MASTERKEY1_FILE_HEX : WII_MASTERKEY_FILE_HEX; FILE* pT = fopen(MasterKeyFile, "rb"); if (pT == NULL) { if(PanicYesNo("Can't open '%s'.\n If you know the key, now it's the time to paste it into " "'%s' before pressing [YES].", MasterKeyFile, MasterKeyFileHex)) { pT = fopen(MasterKeyFileHex, "r"); if(pT==NULL){ PanicAlert("could not open %s", MasterKeyFileHex); return NULL; } static char hexkey[32]; if(fread(hexkey,1,32,pT)<32) { PanicAlert("%s is not the right size", MasterKeyFileHex); fclose(pT); return NULL; } fclose(pT); static char binkey[16]; char *t=hexkey; for(int i=0;i<16;i++) { char h[3]={*(t++),*(t++),0}; binkey[i] = (char) strtol(h,NULL,16); } pT = fopen(MasterKeyFile, "wb"); if(pT==NULL){ PanicAlert("could not open/make '%s' for writing!", MasterKeyFile); return NULL; } fwrite(binkey,16,1,pT); fclose(pT); pT = fopen(MasterKeyFileHex, "rb"); if(pT==NULL){ PanicAlert("could not open '%s' for reading!\n did the file get deleted or locked after converting?", MasterKeyFileHex); return NULL; } } else return NULL; } fread(Korean ? g_MasterKeyK : g_MasterKey, 16, 1, pT); fclose(pT); const u32 keyhash = 0x4bc30936; const u32 keyhashK = 0x485c08e9; u32 hash = HashAdler32(Korean ? g_MasterKeyK : g_MasterKey, 16); if (hash != (Korean ? keyhashK : keyhash)) PanicAlert("Your Wii %sdisc decryption key is bad.", Korean ? "Korean " : "",keyhash, hash); else g_MasterKeyInit[Korean ? 1 : 0] = true; } CBlobBigEndianReader Reader(_rReader); u32 numPartitions = Reader.Read32(0x40000); u64 PartitionsOffset = (u64)Reader.Read32(0x40004) << 2; #ifdef _WIN32 struct SPartition { u64 Offset; u32 Type; }; #endif std::vector PartitionsVec; // read all partitions 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); PartitionsVec.push_back(Partition); } // find the partition with the game... type == 1 is prolly the firmware update partition for (size_t i = 0; i < PartitionsVec.size(); i++) { const SPartition& rPartition = PartitionsVec[i]; if (rPartition.Type == _VolumeType) { u8 SubKey[16]; _rReader.Read(rPartition.Offset + 0x1bf, 16, SubKey); u8 IV[16]; memset(IV, 0, 16); _rReader.Read(rPartition.Offset + 0x44c, 8, IV); AES_KEY AES_KEY; AES_set_decrypt_key((Korean ? g_MasterKeyK : g_MasterKey), 128, &AES_KEY); u8 VolumeKey[16]; AES_cbc_encrypt(SubKey, VolumeKey, 16, &AES_KEY, IV, AES_DECRYPT); return new CVolumeWiiCrypted(&_rReader, rPartition.Offset + 0x20000, VolumeKey); } } return NULL; } EDiscType GetDiscType(IBlobReader& _rReader) { CBlobBigEndianReader Reader(_rReader); // check for wii { u32 MagicWord = Reader.Read32(0x18); if (MagicWord == 0x5D1C9EA3) { if (Reader.Read32(0x60) != 0) return(DISC_TYPE_WII); else return(DISC_TYPE_WII_CONTAINER); } } // check for GC { u32 MagicWord = Reader.Read32(0x1C); if (MagicWord == 0xC2339F3D) return(DISC_TYPE_GC); } return DISC_TYPE_UNK; } } // namespace