forked from dolphin-emu/dolphin
		
	Macros should be all upper-cased. This is also kind of a wart that's been sticking out for quite a while now (we avoid prefixing underscores).
		
			
				
	
	
		
			113 lines
		
	
	
		
			3.2 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			113 lines
		
	
	
		
			3.2 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| // Copyright 2009 Dolphin Emulator Project
 | |
| // Licensed under GPLv2+
 | |
| // Refer to the license.txt file included.
 | |
| 
 | |
| #include <cstddef>
 | |
| #include <memory>
 | |
| #include <optional>
 | |
| #include <string>
 | |
| 
 | |
| #include "Common/Align.h"
 | |
| #include "Common/Assert.h"
 | |
| #include "Common/CommonTypes.h"
 | |
| #include "Common/FileUtil.h"
 | |
| #include "Common/Logging/Log.h"
 | |
| #include "DiscIO/Blob.h"
 | |
| #include "DiscIO/WiiWad.h"
 | |
| 
 | |
| namespace DiscIO
 | |
| {
 | |
| namespace
 | |
| {
 | |
| std::vector<u8> CreateWADEntry(BlobReader& reader, u32 size, u64 offset)
 | |
| {
 | |
|   if (size == 0)
 | |
|     return {};
 | |
| 
 | |
|   std::vector<u8> buffer(size);
 | |
| 
 | |
|   if (!reader.Read(offset, size, buffer.data()))
 | |
|   {
 | |
|     ERROR_LOG(DISCIO, "WiiWAD: Could not read from file");
 | |
|     PanicAlertT("WiiWAD: Could not read from file");
 | |
|   }
 | |
| 
 | |
|   return buffer;
 | |
| }
 | |
| 
 | |
| bool IsWiiWAD(BlobReader& reader)
 | |
| {
 | |
|   const std::optional<u32> header_size = reader.ReadSwapped<u32>(0x0);
 | |
|   const std::optional<u32> header_type = reader.ReadSwapped<u32>(0x4);
 | |
|   return header_size == u32(0x20) &&
 | |
|          (header_type == u32(0x49730000) || header_type == u32(0x69620000));
 | |
| }
 | |
| }  // Anonymous namespace
 | |
| 
 | |
| WiiWAD::WiiWAD(const std::string& name) : WiiWAD(CreateBlobReader(name))
 | |
| {
 | |
| }
 | |
| 
 | |
| WiiWAD::WiiWAD(std::unique_ptr<BlobReader> blob_reader) : m_reader(std::move(blob_reader))
 | |
| {
 | |
|   if (m_reader)
 | |
|     m_valid = ParseWAD();
 | |
| }
 | |
| 
 | |
| WiiWAD::~WiiWAD() = default;
 | |
| 
 | |
| bool WiiWAD::ParseWAD()
 | |
| {
 | |
|   if (!IsWiiWAD(*m_reader))
 | |
|     return false;
 | |
| 
 | |
|   std::optional<u32> certificate_chain_size = m_reader->ReadSwapped<u32>(0x08);
 | |
|   std::optional<u32> reserved = m_reader->ReadSwapped<u32>(0x0C);
 | |
|   std::optional<u32> ticket_size = m_reader->ReadSwapped<u32>(0x10);
 | |
|   std::optional<u32> tmd_size = m_reader->ReadSwapped<u32>(0x14);
 | |
|   std::optional<u32> data_app_size = m_reader->ReadSwapped<u32>(0x18);
 | |
|   std::optional<u32> footer_size = m_reader->ReadSwapped<u32>(0x1C);
 | |
|   if (!certificate_chain_size || !reserved || !ticket_size || !tmd_size || !data_app_size ||
 | |
|       !footer_size)
 | |
|   {
 | |
|     return false;
 | |
|   }
 | |
| 
 | |
|   if (MAX_LOGLEVEL >= LogTypes::LOG_LEVELS::LDEBUG)
 | |
|     DEBUG_ASSERT_MSG(BOOT, *reserved == 0x00, "WiiWAD: Reserved must be 0x00");
 | |
| 
 | |
|   u32 offset = 0x40;
 | |
|   m_certificate_chain = CreateWADEntry(*m_reader, *certificate_chain_size, offset);
 | |
|   offset += Common::AlignUp(*certificate_chain_size, 0x40);
 | |
|   m_ticket.SetBytes(CreateWADEntry(*m_reader, *ticket_size, offset));
 | |
|   offset += Common::AlignUp(*ticket_size, 0x40);
 | |
|   m_tmd.SetBytes(CreateWADEntry(*m_reader, *tmd_size, offset));
 | |
|   offset += Common::AlignUp(*tmd_size, 0x40);
 | |
|   m_data_app_offset = offset;
 | |
|   m_data_app = CreateWADEntry(*m_reader, *data_app_size, offset);
 | |
|   offset += Common::AlignUp(*data_app_size, 0x40);
 | |
|   m_footer = CreateWADEntry(*m_reader, *footer_size, offset);
 | |
|   offset += Common::AlignUp(*footer_size, 0x40);
 | |
| 
 | |
|   return true;
 | |
| }
 | |
| 
 | |
| std::vector<u8> WiiWAD::GetContent(u16 index) const
 | |
| {
 | |
|   u64 offset = m_data_app_offset;
 | |
|   for (const IOS::ES::Content& content : m_tmd.GetContents())
 | |
|   {
 | |
|     const u64 aligned_size = Common::AlignUp(content.size, 0x40);
 | |
|     if (content.index == index)
 | |
|     {
 | |
|       std::vector<u8> data(aligned_size);
 | |
|       if (!m_reader->Read(offset, aligned_size, data.data()))
 | |
|         return {};
 | |
|       return data;
 | |
|     }
 | |
|     offset += aligned_size;
 | |
|   }
 | |
|   return {};
 | |
| }
 | |
| }  // namespace DiscIO
 |