| 
									
										
										
										
											2015-05-24 06:55:12 +02:00
										 |  |  | // Copyright 2012 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.
 | 
					
						
							| 
									
										
										
										
											2012-05-04 00:09:01 +10:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-02-21 01:47:53 +01:00
										 |  |  | #include <algorithm>
 | 
					
						
							|  |  |  | #include <cstdio>
 | 
					
						
							|  |  |  | #include <cstring>
 | 
					
						
							| 
									
										
										
										
											2015-12-06 23:15:51 -05:00
										 |  |  | #include <memory>
 | 
					
						
							| 
									
										
										
										
											2014-02-21 01:47:53 +01:00
										 |  |  | #include <string>
 | 
					
						
							| 
									
										
										
										
											2016-12-21 14:01:00 +01:00
										 |  |  | #include <utility>
 | 
					
						
							| 
									
										
										
										
											2014-02-21 01:47:53 +01:00
										 |  |  | #include <vector>
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-27 11:56:22 +01:00
										 |  |  | #include "Common/Align.h"
 | 
					
						
							| 
									
										
										
										
											2016-12-21 14:01:00 +01:00
										 |  |  | #include "Common/Assert.h"
 | 
					
						
							| 
									
										
										
										
											2015-09-26 17:13:07 -04:00
										 |  |  | #include "Common/CommonFuncs.h"
 | 
					
						
							| 
									
										
										
										
											2014-09-07 20:06:58 -05:00
										 |  |  | #include "Common/CommonTypes.h"
 | 
					
						
							| 
									
										
										
										
											2014-02-17 05:18:15 -05:00
										 |  |  | #include "Common/FileUtil.h"
 | 
					
						
							| 
									
										
										
										
											2015-09-26 17:13:07 -04:00
										 |  |  | #include "Common/Logging/Log.h"
 | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02:00
										 |  |  | #include "Common/MsgHandler.h"
 | 
					
						
							| 
									
										
										
										
											2014-02-17 05:18:15 -05:00
										 |  |  | #include "DiscIO/WbfsBlob.h"
 | 
					
						
							| 
									
										
										
										
											2012-05-04 00:09:01 +10:00
										 |  |  | 
 | 
					
						
							|  |  |  | namespace DiscIO | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2014-09-01 15:48:02 -04:00
										 |  |  | static const u64 WII_SECTOR_SIZE = 0x8000; | 
					
						
							|  |  |  | static const u64 WII_SECTOR_COUNT = 143432 * 2; | 
					
						
							|  |  |  | static const u64 WII_DISC_HEADER_SIZE = 256; | 
					
						
							| 
									
										
										
										
											2012-05-04 00:09:01 +10:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-21 14:01:00 +01:00
										 |  |  | WbfsFileReader::WbfsFileReader(File::IOFile file, const std::string& path) | 
					
						
							| 
									
										
										
										
											2016-12-21 14:05:33 +01:00
										 |  |  |     : m_size(0), m_good(false) | 
					
						
							| 
									
										
										
										
											2012-05-04 00:09:01 +10:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2016-12-21 14:01:00 +01:00
										 |  |  |   if (!AddFileToList(std::move(file))) | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02:00
										 |  |  |     return; | 
					
						
							| 
									
										
										
										
											2016-12-21 14:01:00 +01:00
										 |  |  |   OpenAdditionalFiles(path); | 
					
						
							|  |  |  |   if (!ReadHeader()) | 
					
						
							|  |  |  |     return; | 
					
						
							|  |  |  |   m_good = true; | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |   // Grab disc info (assume slot 0, checked in ReadHeader())
 | 
					
						
							|  |  |  |   m_wlba_table.resize(m_blocks_per_disc); | 
					
						
							| 
									
										
										
										
											2016-12-31 22:51:16 +01:00
										 |  |  |   m_files[0].file.Seek(m_hd_sector_size + WII_DISC_HEADER_SIZE /*+ i * m_disc_info_size*/, | 
					
						
							|  |  |  |                        SEEK_SET); | 
					
						
							|  |  |  |   m_files[0].file.ReadBytes(m_wlba_table.data(), m_blocks_per_disc * sizeof(u16)); | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02:00
										 |  |  |   for (size_t i = 0; i < m_blocks_per_disc; i++) | 
					
						
							|  |  |  |     m_wlba_table[i] = Common::swap16(m_wlba_table[i]); | 
					
						
							| 
									
										
										
										
											2012-05-04 00:09:01 +10:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | WbfsFileReader::~WbfsFileReader() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-09-17 10:58:54 +02:00
										 |  |  | u64 WbfsFileReader::GetDataSize() const | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02:00
										 |  |  |   return WII_SECTOR_COUNT * WII_SECTOR_SIZE; | 
					
						
							| 
									
										
										
										
											2015-09-17 10:58:54 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-21 14:01:00 +01:00
										 |  |  | void WbfsFileReader::OpenAdditionalFiles(const std::string& path) | 
					
						
							| 
									
										
										
										
											2012-05-04 00:09:01 +10:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2016-12-21 14:01:00 +01:00
										 |  |  |   if (path.length() < 4) | 
					
						
							|  |  |  |     return; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-21 14:05:33 +01:00
										 |  |  |   _assert_(m_files.size() > 0);  // The code below gives .wbf0 for index 0, but it should be .wbfs
 | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |   while (true) | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     // Replace last character with index (e.g. wbfs = wbf1)
 | 
					
						
							| 
									
										
										
										
											2016-12-21 14:05:33 +01:00
										 |  |  |     if (m_files.size() >= 10) | 
					
						
							|  |  |  |       return; | 
					
						
							| 
									
										
										
										
											2016-12-21 14:01:00 +01:00
										 |  |  |     std::string current_path = path; | 
					
						
							| 
									
										
										
										
											2016-12-21 14:05:33 +01:00
										 |  |  |     current_path.back() = static_cast<char>('0' + m_files.size()); | 
					
						
							| 
									
										
										
										
											2016-12-21 14:01:00 +01:00
										 |  |  |     if (!AddFileToList(File::IOFile(current_path, "rb"))) | 
					
						
							|  |  |  |       return; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-21 14:01:00 +01:00
										 |  |  | bool WbfsFileReader::AddFileToList(File::IOFile file) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   if (!file.IsOpen()) | 
					
						
							|  |  |  |     return false; | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-21 14:01:00 +01:00
										 |  |  |   const u64 file_size = file.GetSize(); | 
					
						
							| 
									
										
										
										
											2016-12-31 22:51:16 +01:00
										 |  |  |   m_files.emplace_back(std::move(file), m_size, file_size); | 
					
						
							| 
									
										
										
										
											2016-12-21 14:01:00 +01:00
										 |  |  |   m_size += file_size; | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-21 14:01:00 +01:00
										 |  |  |   return true; | 
					
						
							| 
									
										
										
										
											2012-05-04 00:09:01 +10:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool WbfsFileReader::ReadHeader() | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02:00
										 |  |  |   // Read hd size info
 | 
					
						
							| 
									
										
										
										
											2016-12-31 22:51:16 +01:00
										 |  |  |   m_files[0].file.Seek(0, SEEK_SET); | 
					
						
							|  |  |  |   m_files[0].file.ReadBytes(&m_header, sizeof(WbfsHeader)); | 
					
						
							| 
									
										
										
										
											2016-12-21 11:30:12 +01:00
										 |  |  |   if (m_header.magic != WBFS_MAGIC) | 
					
						
							|  |  |  |     return false; | 
					
						
							| 
									
										
										
										
											2013-03-19 09:59:41 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-21 11:30:12 +01:00
										 |  |  |   m_header.hd_sector_count = Common::swap32(m_header.hd_sector_count); | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02:00
										 |  |  |   m_hd_sector_size = 1ull << m_header.hd_sector_shift; | 
					
						
							| 
									
										
										
										
											2013-03-19 09:59:41 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02:00
										 |  |  |   if (m_size != (m_header.hd_sector_count * m_hd_sector_size)) | 
					
						
							|  |  |  |     return false; | 
					
						
							| 
									
										
										
										
											2013-03-19 09:59:41 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02:00
										 |  |  |   // Read wbfs cluster info
 | 
					
						
							|  |  |  |   m_wbfs_sector_size = 1ull << m_header.wbfs_sector_shift; | 
					
						
							|  |  |  |   m_wbfs_sector_count = m_size / m_wbfs_sector_size; | 
					
						
							| 
									
										
										
										
											2013-03-19 09:59:41 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02:00
										 |  |  |   if (m_wbfs_sector_size < WII_SECTOR_SIZE) | 
					
						
							|  |  |  |     return false; | 
					
						
							| 
									
										
										
										
											2013-03-19 09:59:41 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02:00
										 |  |  |   m_blocks_per_disc = | 
					
						
							|  |  |  |       (WII_SECTOR_COUNT * WII_SECTOR_SIZE + m_wbfs_sector_size - 1) / m_wbfs_sector_size; | 
					
						
							|  |  |  |   m_disc_info_size = | 
					
						
							| 
									
										
										
										
											2016-11-27 11:56:22 +01:00
										 |  |  |       Common::AlignUp(WII_DISC_HEADER_SIZE + m_blocks_per_disc * sizeof(u16), m_hd_sector_size); | 
					
						
							| 
									
										
										
										
											2013-03-19 09:59:41 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02:00
										 |  |  |   return m_header.disc_table[0] != 0; | 
					
						
							| 
									
										
										
										
											2012-05-04 00:09:01 +10:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool WbfsFileReader::Read(u64 offset, u64 nbytes, u8* out_ptr) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02:00
										 |  |  |   while (nbytes) | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     u64 read_size; | 
					
						
							|  |  |  |     File::IOFile& data_file = SeekToCluster(offset, &read_size); | 
					
						
							|  |  |  |     if (read_size == 0) | 
					
						
							|  |  |  |       return false; | 
					
						
							|  |  |  |     read_size = std::min(read_size, nbytes); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (!data_file.ReadBytes(out_ptr, read_size)) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       data_file.Clear(); | 
					
						
							|  |  |  |       return false; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     out_ptr += read_size; | 
					
						
							|  |  |  |     nbytes -= read_size; | 
					
						
							|  |  |  |     offset += read_size; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return true; | 
					
						
							| 
									
										
										
										
											2012-05-04 00:09:01 +10:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | File::IOFile& WbfsFileReader::SeekToCluster(u64 offset, u64* available) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02:00
										 |  |  |   u64 base_cluster = (offset >> m_header.wbfs_sector_shift); | 
					
						
							|  |  |  |   if (base_cluster < m_blocks_per_disc) | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     u64 cluster_address = m_wbfs_sector_size * m_wlba_table[base_cluster]; | 
					
						
							|  |  |  |     u64 cluster_offset = offset & (m_wbfs_sector_size - 1); | 
					
						
							|  |  |  |     u64 final_address = cluster_address + cluster_offset; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-22 17:27:37 +01:00
										 |  |  |     for (FileEntry& file_entry : m_files) | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2016-12-31 22:51:16 +01:00
										 |  |  |       if (final_address < (file_entry.base_address + file_entry.size)) | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02:00
										 |  |  |       { | 
					
						
							| 
									
										
										
										
											2016-12-31 22:51:16 +01:00
										 |  |  |         file_entry.file.Seek(final_address - file_entry.base_address, SEEK_SET); | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02:00
										 |  |  |         if (available) | 
					
						
							|  |  |  |         { | 
					
						
							| 
									
										
										
										
											2016-12-31 22:51:16 +01:00
										 |  |  |           u64 till_end_of_file = file_entry.size - (final_address - file_entry.base_address); | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02:00
										 |  |  |           u64 till_end_of_sector = m_wbfs_sector_size - cluster_offset; | 
					
						
							|  |  |  |           *available = std::min(till_end_of_file, till_end_of_sector); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-31 22:51:16 +01:00
										 |  |  |         return file_entry.file; | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02:00
										 |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   PanicAlert("Read beyond end of disc"); | 
					
						
							|  |  |  |   if (available) | 
					
						
							|  |  |  |     *available = 0; | 
					
						
							| 
									
										
										
										
											2016-12-31 22:51:16 +01:00
										 |  |  |   m_files[0].file.Seek(0, SEEK_SET); | 
					
						
							|  |  |  |   return m_files[0].file; | 
					
						
							| 
									
										
										
										
											2012-05-04 00:09:01 +10:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-21 14:01:00 +01:00
										 |  |  | std::unique_ptr<WbfsFileReader> WbfsFileReader::Create(File::IOFile file, const std::string& path) | 
					
						
							| 
									
										
										
										
											2013-03-19 09:59:41 -04:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2016-12-21 14:01:00 +01:00
										 |  |  |   auto reader = std::unique_ptr<WbfsFileReader>(new WbfsFileReader(std::move(file), path)); | 
					
						
							| 
									
										
										
										
											2013-03-19 09:59:41 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02:00
										 |  |  |   if (!reader->IsGood()) | 
					
						
							|  |  |  |     reader.reset(); | 
					
						
							| 
									
										
										
										
											2015-12-06 23:15:51 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02:00
										 |  |  |   return reader; | 
					
						
							| 
									
										
										
										
											2012-05-04 00:09:01 +10:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | }  // namespace
 |