| 
									
										
										
										
											2009-07-28 21:32:10 +00:00
										 |  |  | // Copyright (C) 2003 Dolphin Project.
 | 
					
						
							| 
									
										
										
										
											2009-07-06 02:10:26 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | // 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 "NANDContentLoader.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include <algorithm>
 | 
					
						
							|  |  |  | #include <cctype> 
 | 
					
						
							|  |  |  | #include "MathUtil.h"
 | 
					
						
							|  |  |  | #include "FileUtil.h"
 | 
					
						
							|  |  |  | #include "Log.h"
 | 
					
						
							|  |  |  | #include "WiiWad.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | namespace DiscIO | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class CBlobBigEndianReader | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | public: | 
					
						
							|  |  |  | 	CBlobBigEndianReader(DiscIO::IBlobReader& _rReader) : m_rReader(_rReader) {} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	u32 Read32(u64 _Offset) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 		u32 Temp; | 
					
						
							|  |  |  | 		m_rReader.Read(_Offset, 4, (u8*)&Temp); | 
					
						
							|  |  |  | 		return(Common::swap32(Temp)); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | private: | 
					
						
							|  |  |  | 	DiscIO::IBlobReader& m_rReader; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | WiiWAD::WiiWAD(const std::string& _rName) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     DiscIO::IBlobReader* pReader = DiscIO::CreateBlobReader(_rName.c_str()); | 
					
						
							| 
									
										
										
										
											2011-07-18 01:47:55 +00:00
										 |  |  |     if (pReader == NULL || File::IsDirectory(_rName)) | 
					
						
							| 
									
										
										
										
											2009-07-06 02:10:26 +00:00
										 |  |  | 	{ | 
					
						
							|  |  |  | 		m_Valid = false; | 
					
						
							| 
									
										
										
										
											2013-03-01 19:21:34 +13:00
										 |  |  |         if(pReader) delete pReader; | 
					
						
							|  |  |  |         return; | 
					
						
							| 
									
										
										
										
											2009-07-06 02:10:26 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	m_Valid = ParseWAD(*pReader); | 
					
						
							|  |  |  | 	delete pReader; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | WiiWAD::~WiiWAD() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	if (m_Valid) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 		delete m_pCertificateChain; | 
					
						
							|  |  |  | 		delete m_pTicket; | 
					
						
							|  |  |  | 		delete m_pTMD; | 
					
						
							|  |  |  | 		delete m_pDataApp; | 
					
						
							|  |  |  | 		delete m_pFooter; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | u8* WiiWAD::CreateWADEntry(DiscIO::IBlobReader& _rReader, u32 _Size, u64 _Offset) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (_Size > 0) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         u8* pTmpBuffer = new u8[_Size]; | 
					
						
							|  |  |  |         _dbg_assert_msg_(BOOT, pTmpBuffer!=0, "WiiWAD: Cant allocate memory for WAD entry"); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (!_rReader.Read(_Offset, _Size, pTmpBuffer)) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  | 			ERROR_LOG(DISCIO, "WiiWAD: Could not read from file"); | 
					
						
							| 
									
										
										
										
											2011-01-13 02:05:58 +00:00
										 |  |  |             PanicAlertT("WiiWAD: Could not read from file"); | 
					
						
							| 
									
										
										
										
											2009-07-06 02:10:26 +00:00
										 |  |  |         } | 
					
						
							|  |  |  |         return pTmpBuffer; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 	return NULL; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool WiiWAD::ParseWAD(DiscIO::IBlobReader& _rReader) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     CBlobBigEndianReader ReaderBig(_rReader); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // get header size	
 | 
					
						
							|  |  |  | 	u32 HeaderSize = ReaderBig.Read32(0); | 
					
						
							|  |  |  |     if (HeaderSize != 0x20)  | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         _dbg_assert_msg_(BOOT, (HeaderSize==0x20), "WiiWAD: Header size != 0x20"); | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  |     }     | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // get header 
 | 
					
						
							|  |  |  |     u8 Header[0x20]; | 
					
						
							|  |  |  |     _rReader.Read(0, HeaderSize, Header); | 
					
						
							|  |  |  | 	u32 HeaderType = ReaderBig.Read32(0x4); | 
					
						
							|  |  |  |     if ((0x49730000 != HeaderType) && (0x69620000 != HeaderType)) | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     m_CertificateChainSize    = ReaderBig.Read32(0x8); | 
					
						
							| 
									
										
										
										
											2010-05-29 21:28:27 +00:00
										 |  |  |     u32 Reserved              = ReaderBig.Read32(0xC); | 
					
						
							| 
									
										
										
										
											2009-07-06 02:10:26 +00:00
										 |  |  |     m_TicketSize              = ReaderBig.Read32(0x10); | 
					
						
							|  |  |  |     m_TMDSize                 = ReaderBig.Read32(0x14); | 
					
						
							|  |  |  |     m_DataAppSize             = ReaderBig.Read32(0x18); | 
					
						
							|  |  |  |     m_FooterSize              = ReaderBig.Read32(0x1C); | 
					
						
							| 
									
										
										
										
											2010-05-29 21:28:27 +00:00
										 |  |  | #if MAX_LOGLEVEL >= DEBUG_LEVEL
 | 
					
						
							| 
									
										
										
										
											2009-07-06 02:10:26 +00:00
										 |  |  |     _dbg_assert_msg_(BOOT, Reserved==0x00, "WiiWAD: Reserved must be 0x00"); | 
					
						
							| 
									
										
										
										
											2010-05-29 21:28:27 +00:00
										 |  |  | #else
 | 
					
						
							|  |  |  |     (void)Reserved; | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2009-07-06 02:10:26 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     u32 Offset = 0x40; | 
					
						
							|  |  |  |     m_pCertificateChain   = CreateWADEntry(_rReader, m_CertificateChainSize, Offset);  Offset += ROUND_UP(m_CertificateChainSize, 0x40); | 
					
						
							|  |  |  |     m_pTicket             = CreateWADEntry(_rReader, m_TicketSize, Offset);            Offset += ROUND_UP(m_TicketSize, 0x40); | 
					
						
							|  |  |  |     m_pTMD                = CreateWADEntry(_rReader, m_TMDSize, Offset);               Offset += ROUND_UP(m_TMDSize, 0x40); | 
					
						
							|  |  |  |     m_pDataApp            = CreateWADEntry(_rReader, m_DataAppSize, Offset);           Offset += ROUND_UP(m_DataAppSize, 0x40); | 
					
						
							|  |  |  |     m_pFooter             = CreateWADEntry(_rReader, m_FooterSize, Offset);            Offset += ROUND_UP(m_FooterSize, 0x40); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool WiiWAD::IsWiiWAD(const std::string& _rName) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     DiscIO::IBlobReader* pReader = DiscIO::CreateBlobReader(_rName.c_str()); | 
					
						
							|  |  |  |     if (pReader == NULL) | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     CBlobBigEndianReader Reader(*pReader); | 
					
						
							|  |  |  |     bool Result = false; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // check for wii wad
 | 
					
						
							|  |  |  |     if (Reader.Read32(0x00) == 0x20) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         u32 WADTYpe = Reader.Read32(0x04); | 
					
						
							|  |  |  |         switch(WADTYpe) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |         case 0x49730000: | 
					
						
							|  |  |  |         case 0x69620000: | 
					
						
							|  |  |  |             Result = true; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     delete pReader; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return Result; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | } // namespace end
 | 
					
						
							|  |  |  | 
 |