| 
									
										
										
										
											2015-05-24 06:55:12 +02:00
										 |  |  | // Copyright 2008 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.
 | 
					
						
							| 
									
										
										
										
											2008-12-08 05:30:24 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-02-21 01:47:53 +01:00
										 |  |  | #include <algorithm>
 | 
					
						
							|  |  |  | #include <cstddef>
 | 
					
						
							|  |  |  | #include <cstring>
 | 
					
						
							|  |  |  | #include <map>
 | 
					
						
							| 
									
										
										
										
											2014-08-25 23:40:05 -04:00
										 |  |  | #include <memory>
 | 
					
						
							| 
									
										
										
										
											2014-02-21 01:47:53 +01:00
										 |  |  | #include <string>
 | 
					
						
							|  |  |  | #include <vector>
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-27 11:56:22 +01:00
										 |  |  | #include "Common/Align.h"
 | 
					
						
							| 
									
										
										
										
											2015-09-26 16:39:47 -04:00
										 |  |  | #include "Common/Assert.h"
 | 
					
						
							| 
									
										
										
										
											2014-02-17 05:18:15 -05:00
										 |  |  | #include "Common/CommonPaths.h"
 | 
					
						
							| 
									
										
										
										
											2014-09-07 20:06:58 -05:00
										 |  |  | #include "Common/CommonTypes.h"
 | 
					
						
							| 
									
										
										
										
											2014-02-21 01:47:53 +01:00
										 |  |  | #include "Common/FileUtil.h"
 | 
					
						
							| 
									
										
										
										
											2015-09-26 16:39:47 -04:00
										 |  |  | #include "Common/Logging/Log.h"
 | 
					
						
							| 
									
										
										
										
											2015-09-27 14:01:12 +02:00
										 |  |  | #include "DiscIO/Blob.h"
 | 
					
						
							| 
									
										
										
										
											2016-07-06 20:33:05 +02:00
										 |  |  | #include "DiscIO/Enums.h"
 | 
					
						
							| 
									
										
										
										
											2014-12-27 20:13:31 +01:00
										 |  |  | #include "DiscIO/FileMonitor.h"
 | 
					
						
							| 
									
										
										
										
											2014-02-21 01:47:53 +01:00
										 |  |  | #include "DiscIO/Volume.h"
 | 
					
						
							| 
									
										
										
										
											2014-02-17 05:18:15 -05:00
										 |  |  | #include "DiscIO/VolumeDirectory.h"
 | 
					
						
							| 
									
										
										
										
											2008-12-08 05:30:24 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | namespace DiscIO | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2016-12-26 11:45:22 +01:00
										 |  |  | static u32 ComputeNameSize(const File::FSTEntry& parent_entry); | 
					
						
							| 
									
										
										
										
											2016-12-25 22:57:14 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-11-17 10:09:54 +01:00
										 |  |  | const size_t CVolumeDirectory::MAX_NAME_LENGTH; | 
					
						
							|  |  |  | const size_t CVolumeDirectory::MAX_ID_LENGTH; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-26 11:45:22 +01:00
										 |  |  | CVolumeDirectory::CVolumeDirectory(const std::string& directory, bool is_wii, | 
					
						
							|  |  |  |                                    const std::string& apploader, const std::string& dol) | 
					
						
							|  |  |  |     : m_data_start_address(-1), m_disk_header(DISKHEADERINFO_ADDRESS), | 
					
						
							|  |  |  |       m_disk_header_info(std::make_unique<SDiskHeaderInfo>()), m_fst_address(0), m_dol_address(0) | 
					
						
							| 
									
										
										
										
											2008-12-08 05:30:24 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2016-12-26 11:45:22 +01:00
										 |  |  |   m_root_directory = ExtractDirectoryName(directory); | 
					
						
							| 
									
										
										
										
											2008-12-08 05:30:24 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02:00
										 |  |  |   // create the default disk header
 | 
					
						
							| 
									
										
										
										
											2016-10-29 14:42:43 +02:00
										 |  |  |   SetGameID("AGBJ01"); | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02:00
										 |  |  |   SetName("Default name"); | 
					
						
							| 
									
										
										
										
											2008-12-08 05:30:24 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-26 11:45:22 +01:00
										 |  |  |   if (is_wii) | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02:00
										 |  |  |     SetDiskTypeWii(); | 
					
						
							|  |  |  |   else | 
					
						
							|  |  |  |     SetDiskTypeGC(); | 
					
						
							| 
									
										
										
										
											2008-12-08 05:30:24 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02:00
										 |  |  |   // Don't load the DOL if we don't have an apploader
 | 
					
						
							| 
									
										
										
										
											2016-12-26 11:45:22 +01:00
										 |  |  |   if (SetApploader(apploader)) | 
					
						
							|  |  |  |     SetDOL(dol); | 
					
						
							| 
									
										
										
										
											2009-12-10 09:16:10 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02:00
										 |  |  |   BuildFST(); | 
					
						
							| 
									
										
										
										
											2008-12-08 05:30:24 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | CVolumeDirectory::~CVolumeDirectory() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-26 11:45:22 +01:00
										 |  |  | bool CVolumeDirectory::IsValidDirectory(const std::string& directory) | 
					
						
							| 
									
										
										
										
											2008-12-08 05:30:24 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2016-12-26 11:45:22 +01:00
										 |  |  |   return File::IsDirectory(ExtractDirectoryName(directory)); | 
					
						
							| 
									
										
										
										
											2008-12-08 05:30:24 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-26 11:45:22 +01:00
										 |  |  | bool CVolumeDirectory::Read(u64 offset, u64 length, u8* buffer, bool decrypt) const | 
					
						
							| 
									
										
										
										
											2009-02-22 13:59:06 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2016-12-26 11:45:22 +01:00
										 |  |  |   if (!decrypt && (offset + length >= 0x400) && m_is_wii) | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02:00
										 |  |  |   { | 
					
						
							|  |  |  |     // Fully supporting this would require re-encrypting every file that's read.
 | 
					
						
							| 
									
										
										
										
											2016-06-25 18:07:10 +02:00
										 |  |  |     // Only supporting the areas that IOS allows software to read could be more feasible.
 | 
					
						
							|  |  |  |     // Currently, only the header (up to 0x400) is supported, though we're cheating a bit
 | 
					
						
							|  |  |  |     // with it by reading the header inside the current partition instead. Supporting the
 | 
					
						
							|  |  |  |     // header is enough for booting games, but not for running things like the Disc Channel.
 | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02:00
										 |  |  |     return false; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (decrypt && !m_is_wii) | 
					
						
							|  |  |  |     PanicAlertT("Tried to decrypt data from a non-Wii volume"); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   // header
 | 
					
						
							| 
									
										
										
										
											2016-12-26 11:45:22 +01:00
										 |  |  |   if (offset < DISKHEADERINFO_ADDRESS) | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02:00
										 |  |  |   { | 
					
						
							| 
									
										
										
										
											2016-12-26 11:45:22 +01:00
										 |  |  |     WriteToBuffer(DISKHEADER_ADDRESS, DISKHEADERINFO_ADDRESS, m_disk_header.data(), &offset, | 
					
						
							|  |  |  |                   &length, &buffer); | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02:00
										 |  |  |   } | 
					
						
							|  |  |  |   // header info
 | 
					
						
							| 
									
										
										
										
											2016-12-26 11:45:22 +01:00
										 |  |  |   if (offset >= DISKHEADERINFO_ADDRESS && offset < APPLOADER_ADDRESS) | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02:00
										 |  |  |   { | 
					
						
							| 
									
										
										
										
											2016-12-26 11:45:22 +01:00
										 |  |  |     WriteToBuffer(DISKHEADERINFO_ADDRESS, sizeof(m_disk_header_info), (u8*)m_disk_header_info.get(), | 
					
						
							|  |  |  |                   &offset, &length, &buffer); | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02:00
										 |  |  |   } | 
					
						
							|  |  |  |   // apploader
 | 
					
						
							| 
									
										
										
										
											2016-12-26 11:45:22 +01:00
										 |  |  |   if (offset >= APPLOADER_ADDRESS && offset < APPLOADER_ADDRESS + m_apploader.size()) | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02:00
										 |  |  |   { | 
					
						
							| 
									
										
										
										
											2016-12-26 11:45:22 +01:00
										 |  |  |     WriteToBuffer(APPLOADER_ADDRESS, m_apploader.size(), m_apploader.data(), &offset, &length, | 
					
						
							|  |  |  |                   &buffer); | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02:00
										 |  |  |   } | 
					
						
							|  |  |  |   // dol
 | 
					
						
							| 
									
										
										
										
											2016-12-26 11:45:22 +01:00
										 |  |  |   if (offset >= m_dol_address && offset < m_dol_address + m_dol.size()) | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02:00
										 |  |  |   { | 
					
						
							| 
									
										
										
										
											2016-12-26 11:45:22 +01:00
										 |  |  |     WriteToBuffer(m_dol_address, m_dol.size(), m_dol.data(), &offset, &length, &buffer); | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02:00
										 |  |  |   } | 
					
						
							|  |  |  |   // fst
 | 
					
						
							| 
									
										
										
										
											2016-12-26 11:45:22 +01:00
										 |  |  |   if (offset >= m_fst_address && offset < m_data_start_address) | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02:00
										 |  |  |   { | 
					
						
							| 
									
										
										
										
											2016-12-26 11:45:22 +01:00
										 |  |  |     WriteToBuffer(m_fst_address, m_fst_data.size(), m_fst_data.data(), &offset, &length, &buffer); | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-26 11:45:22 +01:00
										 |  |  |   if (m_virtual_disk.empty()) | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02:00
										 |  |  |     return true; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   // Determine which file the offset refers to
 | 
					
						
							| 
									
										
										
										
											2016-12-26 11:45:22 +01:00
										 |  |  |   std::map<u64, std::string>::const_iterator fileIter = m_virtual_disk.lower_bound(offset); | 
					
						
							|  |  |  |   if (fileIter->first > offset && fileIter != m_virtual_disk.begin()) | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02:00
										 |  |  |     --fileIter; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   // zero fill to start of file data
 | 
					
						
							| 
									
										
										
										
											2016-12-26 11:45:22 +01:00
										 |  |  |   PadToAddress(fileIter->first, &offset, &length, &buffer); | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-26 11:45:22 +01:00
										 |  |  |   while (fileIter != m_virtual_disk.end() && length > 0) | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02:00
										 |  |  |   { | 
					
						
							| 
									
										
										
										
											2016-12-26 11:45:22 +01:00
										 |  |  |     _dbg_assert_(DVDINTERFACE, fileIter->first <= offset); | 
					
						
							|  |  |  |     u64 fileOffset = offset - fileIter->first; | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02:00
										 |  |  |     const std::string fileName = fileIter->second; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     File::IOFile file(fileName, "rb"); | 
					
						
							|  |  |  |     if (!file) | 
					
						
							|  |  |  |       return false; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     u64 fileSize = file.GetSize(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     FileMon::CheckFile(fileName, fileSize); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (fileOffset < fileSize) | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2016-12-26 11:45:22 +01:00
										 |  |  |       u64 fileBytes = std::min(fileSize - fileOffset, length); | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |       if (!file.Seek(fileOffset, SEEK_SET)) | 
					
						
							|  |  |  |         return false; | 
					
						
							| 
									
										
										
										
											2016-12-26 11:45:22 +01:00
										 |  |  |       if (!file.ReadBytes(buffer, fileBytes)) | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02:00
										 |  |  |         return false; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-26 11:45:22 +01:00
										 |  |  |       length -= fileBytes; | 
					
						
							|  |  |  |       buffer += fileBytes; | 
					
						
							|  |  |  |       offset += fileBytes; | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     ++fileIter; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-26 11:45:22 +01:00
										 |  |  |     if (fileIter != m_virtual_disk.end()) | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2016-12-26 11:45:22 +01:00
										 |  |  |       _dbg_assert_(DVDINTERFACE, fileIter->first >= offset); | 
					
						
							|  |  |  |       PadToAddress(fileIter->first, &offset, &length, &buffer); | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02:00
										 |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return true; | 
					
						
							| 
									
										
										
										
											2008-12-08 05:30:24 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-29 14:42:43 +02:00
										 |  |  | std::string CVolumeDirectory::GetGameID() const | 
					
						
							| 
									
										
										
										
											2008-12-08 05:30:24 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2016-12-26 11:45:22 +01:00
										 |  |  |   return std::string(m_disk_header.begin(), m_disk_header.begin() + MAX_ID_LENGTH); | 
					
						
							| 
									
										
										
										
											2008-12-08 05:30:24 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-29 14:42:43 +02:00
										 |  |  | void CVolumeDirectory::SetGameID(const std::string& id) | 
					
						
							| 
									
										
										
										
											2008-12-08 05:30:24 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2016-12-26 11:45:22 +01:00
										 |  |  |   memcpy(m_disk_header.data(), id.c_str(), std::min(id.length(), MAX_ID_LENGTH)); | 
					
						
							| 
									
										
										
										
											2008-12-08 05:30:24 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-23 18:41:21 +01:00
										 |  |  | Region CVolumeDirectory::GetRegion() const | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   if (m_is_wii) | 
					
						
							|  |  |  |     return RegionSwitchWii(m_disk_header[3]); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return RegionSwitchGC(m_disk_header[3]); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-07-06 20:33:05 +02:00
										 |  |  | Country CVolumeDirectory::GetCountry() const | 
					
						
							| 
									
										
										
										
											2008-12-08 05:30:24 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2016-12-26 11:45:22 +01:00
										 |  |  |   return CountrySwitch(m_disk_header[3]); | 
					
						
							| 
									
										
										
										
											2008-12-08 05:30:24 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | std::string CVolumeDirectory::GetMakerID() const | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02:00
										 |  |  |   // Not implemented
 | 
					
						
							|  |  |  |   return "00"; | 
					
						
							| 
									
										
										
										
											2008-12-08 05:30:24 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-05-11 11:19:30 +02:00
										 |  |  | std::string CVolumeDirectory::GetInternalName() const | 
					
						
							| 
									
										
										
										
											2015-04-10 22:10:49 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02:00
										 |  |  |   char name[0x60]; | 
					
						
							|  |  |  |   if (Read(0x20, 0x60, (u8*)name, false)) | 
					
						
							|  |  |  |     return DecodeString(name); | 
					
						
							|  |  |  |   else | 
					
						
							|  |  |  |     return ""; | 
					
						
							| 
									
										
										
										
											2015-04-10 22:10:49 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-07-06 20:33:05 +02:00
										 |  |  | std::map<Language, std::string> CVolumeDirectory::GetLongNames() const | 
					
						
							| 
									
										
										
										
											2008-12-08 05:30:24 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02:00
										 |  |  |   std::string name = GetInternalName(); | 
					
						
							|  |  |  |   if (name.empty()) | 
					
						
							|  |  |  |     return {{}}; | 
					
						
							| 
									
										
										
										
											2016-07-06 20:33:05 +02:00
										 |  |  |   return {{Language::LANGUAGE_UNKNOWN, name}}; | 
					
						
							| 
									
										
										
										
											2008-12-08 05:30:24 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-12-03 17:29:59 +01:00
										 |  |  | std::vector<u32> CVolumeDirectory::GetBanner(int* width, int* height) const | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02:00
										 |  |  |   // Not implemented
 | 
					
						
							|  |  |  |   *width = 0; | 
					
						
							|  |  |  |   *height = 0; | 
					
						
							|  |  |  |   return std::vector<u32>(); | 
					
						
							| 
									
										
										
										
											2015-12-03 17:29:59 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-10-28 13:52:36 -04:00
										 |  |  | void CVolumeDirectory::SetName(const std::string& name) | 
					
						
							| 
									
										
										
										
											2008-12-08 05:30:24 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02:00
										 |  |  |   size_t length = std::min(name.length(), MAX_NAME_LENGTH); | 
					
						
							| 
									
										
										
										
											2016-12-26 11:45:22 +01:00
										 |  |  |   memcpy(&m_disk_header[0x20], name.c_str(), length); | 
					
						
							|  |  |  |   m_disk_header[length + 0x20] = 0; | 
					
						
							| 
									
										
										
										
											2008-12-08 05:30:24 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-08-10 16:35:23 +02:00
										 |  |  | u64 CVolumeDirectory::GetFSTSize() const | 
					
						
							| 
									
										
										
										
											2008-12-08 05:30:24 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02:00
										 |  |  |   // Not implemented
 | 
					
						
							|  |  |  |   return 0; | 
					
						
							| 
									
										
										
										
											2008-12-08 05:30:24 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | std::string CVolumeDirectory::GetApploaderDate() const | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02:00
										 |  |  |   // Not implemented
 | 
					
						
							|  |  |  |   return "VOID"; | 
					
						
							| 
									
										
										
										
											2008-12-08 05:30:24 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-07-06 20:33:05 +02:00
										 |  |  | Platform CVolumeDirectory::GetVolumeType() const | 
					
						
							| 
									
										
										
										
											2015-01-17 13:21:02 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2016-07-06 20:33:05 +02:00
										 |  |  |   return m_is_wii ? Platform::WII_DISC : Platform::GAMECUBE_DISC; | 
					
						
							| 
									
										
										
										
											2015-01-17 13:21:02 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-09-27 14:01:12 +02:00
										 |  |  | BlobType CVolumeDirectory::GetBlobType() const | 
					
						
							| 
									
										
										
										
											2015-09-26 15:24:29 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02:00
										 |  |  |   // VolumeDirectory isn't actually a blob, but it sort of acts
 | 
					
						
							|  |  |  |   // like one, so it makes sense that it has its own blob type.
 | 
					
						
							|  |  |  |   // It should be made into a proper blob in the future.
 | 
					
						
							|  |  |  |   return BlobType::DIRECTORY; | 
					
						
							| 
									
										
										
										
											2015-09-26 15:24:29 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-08 05:30:24 +00:00
										 |  |  | u64 CVolumeDirectory::GetSize() const | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02:00
										 |  |  |   // Not implemented
 | 
					
						
							|  |  |  |   return 0; | 
					
						
							| 
									
										
										
										
											2008-12-08 05:30:24 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-04-09 12:58:56 -05:00
										 |  |  | u64 CVolumeDirectory::GetRawSize() const | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02:00
										 |  |  |   // Not implemented
 | 
					
						
							|  |  |  |   return 0; | 
					
						
							| 
									
										
										
										
											2013-04-09 12:58:56 -05:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2008-12-08 05:30:24 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-26 11:45:22 +01:00
										 |  |  | std::string CVolumeDirectory::ExtractDirectoryName(const std::string& directory) | 
					
						
							| 
									
										
										
										
											2008-12-08 05:30:24 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2016-12-26 11:45:22 +01:00
										 |  |  |   std::string result = directory; | 
					
						
							| 
									
										
										
										
											2008-12-08 05:30:24 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-26 11:45:22 +01:00
										 |  |  |   size_t last_separator = result.find_last_of(DIR_SEP_CHR); | 
					
						
							| 
									
										
										
										
											2008-12-08 05:30:24 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-26 11:45:22 +01:00
										 |  |  |   if (last_separator != result.size() - 1) | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02:00
										 |  |  |   { | 
					
						
							|  |  |  |     // TODO: This assumes that file names will always have a dot in them
 | 
					
						
							|  |  |  |     //       and directory names never will; both assumptions are often
 | 
					
						
							|  |  |  |     //       right but in general wrong.
 | 
					
						
							| 
									
										
										
										
											2016-12-26 11:45:22 +01:00
										 |  |  |     size_t extension_start = result.find_last_of('.'); | 
					
						
							|  |  |  |     if (extension_start != std::string::npos && extension_start > last_separator) | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2016-12-26 11:45:22 +01:00
										 |  |  |       result.resize(last_separator); | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02:00
										 |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   else | 
					
						
							|  |  |  |   { | 
					
						
							| 
									
										
										
										
											2016-12-26 11:45:22 +01:00
										 |  |  |     result.resize(last_separator); | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2008-12-08 05:30:24 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-26 11:45:22 +01:00
										 |  |  |   return result; | 
					
						
							| 
									
										
										
										
											2008-12-08 05:30:24 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void CVolumeDirectory::SetDiskTypeWii() | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2016-12-26 11:45:22 +01:00
										 |  |  |   Write32(0x5d1c9ea3, 0x18, &m_disk_header); | 
					
						
							|  |  |  |   memset(&m_disk_header[0x1c], 0, 4); | 
					
						
							| 
									
										
										
										
											2008-12-08 05:30:24 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02:00
										 |  |  |   m_is_wii = true; | 
					
						
							| 
									
										
										
										
											2016-12-26 11:45:22 +01:00
										 |  |  |   m_address_shift = 2; | 
					
						
							| 
									
										
										
										
											2008-12-08 05:30:24 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void CVolumeDirectory::SetDiskTypeGC() | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2016-12-26 11:45:22 +01:00
										 |  |  |   memset(&m_disk_header[0x18], 0, 4); | 
					
						
							|  |  |  |   Write32(0xc2339f3d, 0x1c, &m_disk_header); | 
					
						
							| 
									
										
										
										
											2008-12-08 05:30:24 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02:00
										 |  |  |   m_is_wii = false; | 
					
						
							| 
									
										
										
										
											2016-12-26 11:45:22 +01:00
										 |  |  |   m_address_shift = 0; | 
					
						
							| 
									
										
										
										
											2008-12-08 05:30:24 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-26 11:45:22 +01:00
										 |  |  | bool CVolumeDirectory::SetApploader(const std::string& apploader) | 
					
						
							| 
									
										
										
										
											2009-12-10 09:16:10 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2016-12-26 11:45:22 +01:00
										 |  |  |   if (!apploader.empty()) | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02:00
										 |  |  |   { | 
					
						
							|  |  |  |     std::string data; | 
					
						
							| 
									
										
										
										
											2016-12-26 11:45:22 +01:00
										 |  |  |     if (!File::ReadFileToString(apploader, data)) | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02:00
										 |  |  |     { | 
					
						
							|  |  |  |       PanicAlertT("Apploader unable to load from file"); | 
					
						
							|  |  |  |       return false; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2016-12-26 11:45:22 +01:00
										 |  |  |     size_t apploader_size = 0x20 + Common::swap32(*(u32*)&data.data()[0x14]) + | 
					
						
							|  |  |  |                             Common::swap32(*(u32*)&data.data()[0x18]); | 
					
						
							|  |  |  |     if (apploader_size != data.size()) | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02:00
										 |  |  |     { | 
					
						
							|  |  |  |       PanicAlertT("Apploader is the wrong size...is it really an apploader?"); | 
					
						
							|  |  |  |       return false; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2016-12-26 11:45:22 +01:00
										 |  |  |     m_apploader.resize(apploader_size); | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02:00
										 |  |  |     std::copy(data.begin(), data.end(), m_apploader.begin()); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // 32byte aligned (plus 0x20 padding)
 | 
					
						
							| 
									
										
										
										
											2016-11-27 11:56:22 +01:00
										 |  |  |     m_dol_address = Common::AlignUp(APPLOADER_ADDRESS + m_apploader.size() + 0x20, 0x20ull); | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02:00
										 |  |  |     return true; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   else | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     m_apploader.resize(0x20); | 
					
						
							|  |  |  |     // Make sure BS2 HLE doesn't try to run the apploader
 | 
					
						
							|  |  |  |     *(u32*)&m_apploader[0x10] = (u32)-1; | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2009-12-10 09:16:10 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-26 11:45:22 +01:00
										 |  |  | void CVolumeDirectory::SetDOL(const std::string& dol) | 
					
						
							| 
									
										
										
										
											2009-12-10 09:16:10 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2016-12-26 11:45:22 +01:00
										 |  |  |   if (!dol.empty()) | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02:00
										 |  |  |   { | 
					
						
							|  |  |  |     std::string data; | 
					
						
							| 
									
										
										
										
											2016-12-26 11:45:22 +01:00
										 |  |  |     File::ReadFileToString(dol, data); | 
					
						
							|  |  |  |     m_dol.resize(data.size()); | 
					
						
							|  |  |  |     std::copy(data.begin(), data.end(), m_dol.begin()); | 
					
						
							| 
									
										
										
										
											2009-12-10 09:16:10 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-26 11:45:22 +01:00
										 |  |  |     Write32((u32)(m_dol_address >> m_address_shift), 0x0420, &m_disk_header); | 
					
						
							| 
									
										
										
										
											2009-12-10 09:16:10 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02:00
										 |  |  |     // 32byte aligned (plus 0x20 padding)
 | 
					
						
							| 
									
										
										
										
											2016-12-26 11:45:22 +01:00
										 |  |  |     m_fst_address = Common::AlignUp(m_dol_address + m_dol.size() + 0x20, 0x20ull); | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2009-12-10 09:16:10 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-08 05:30:24 +00:00
										 |  |  | void CVolumeDirectory::BuildFST() | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2016-12-26 11:45:22 +01:00
										 |  |  |   m_fst_data.clear(); | 
					
						
							| 
									
										
										
										
											2008-12-08 05:30:24 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-26 11:45:22 +01:00
										 |  |  |   File::FSTEntry rootEntry = File::ScanDirectoryTree(m_root_directory, true); | 
					
						
							| 
									
										
										
										
											2016-12-25 22:57:14 +01:00
										 |  |  |   u32 name_table_size = ComputeNameSize(rootEntry); | 
					
						
							| 
									
										
										
										
											2008-12-08 05:30:24 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-26 11:45:22 +01:00
										 |  |  |   m_fst_name_offset = rootEntry.size * ENTRY_SIZE;  // offset of name table in FST
 | 
					
						
							|  |  |  |   m_fst_data.resize(m_fst_name_offset + name_table_size); | 
					
						
							| 
									
										
										
										
											2008-12-08 05:30:24 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02:00
										 |  |  |   // if FST hasn't been assigned (ie no apploader/dol setup), set to default
 | 
					
						
							|  |  |  |   if (m_fst_address == 0) | 
					
						
							|  |  |  |     m_fst_address = APPLOADER_ADDRESS + 0x2000; | 
					
						
							| 
									
										
										
										
											2012-04-23 00:38:58 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02:00
										 |  |  |   // 4 byte aligned start of data on disk
 | 
					
						
							| 
									
										
										
										
											2016-12-26 11:45:22 +01:00
										 |  |  |   m_data_start_address = Common::AlignUp(m_fst_address + m_fst_data.size(), 0x8000ull); | 
					
						
							|  |  |  |   u64 current_data_address = m_data_start_address; | 
					
						
							| 
									
										
										
										
											2008-12-08 05:30:24 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-26 11:45:22 +01:00
										 |  |  |   u32 fst_offset = 0;   // Offset within FST data
 | 
					
						
							|  |  |  |   u32 name_offset = 0;  // Offset within name table
 | 
					
						
							|  |  |  |   u32 root_offset = 0;  // Offset of root of FST
 | 
					
						
							| 
									
										
										
										
											2008-12-08 05:30:24 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02:00
										 |  |  |   // write root entry
 | 
					
						
							| 
									
										
										
										
											2016-12-26 11:45:22 +01:00
										 |  |  |   WriteEntryData(&fst_offset, DIRECTORY_ENTRY, 0, 0, rootEntry.size); | 
					
						
							| 
									
										
										
										
											2008-12-08 05:30:24 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-26 11:45:22 +01:00
										 |  |  |   WriteDirectory(rootEntry, &fst_offset, &name_offset, ¤t_data_address, root_offset); | 
					
						
							| 
									
										
										
										
											2008-12-08 05:30:24 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02:00
										 |  |  |   // overflow check
 | 
					
						
							| 
									
										
										
										
											2016-12-26 11:45:22 +01:00
										 |  |  |   _dbg_assert_(DVDINTERFACE, name_offset == name_table_size); | 
					
						
							| 
									
										
										
										
											2008-12-08 05:30:24 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02:00
										 |  |  |   // write FST size and location
 | 
					
						
							| 
									
										
										
										
											2016-12-26 11:45:22 +01:00
										 |  |  |   Write32((u32)(m_fst_address >> m_address_shift), 0x0424, &m_disk_header); | 
					
						
							|  |  |  |   Write32((u32)(m_fst_data.size() >> m_address_shift), 0x0428, &m_disk_header); | 
					
						
							|  |  |  |   Write32((u32)(m_fst_data.size() >> m_address_shift), 0x042c, &m_disk_header); | 
					
						
							| 
									
										
										
										
											2008-12-08 05:30:24 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-26 11:45:22 +01:00
										 |  |  | void CVolumeDirectory::WriteToBuffer(u64 source_start_address, u64 source_length, const u8* source, | 
					
						
							|  |  |  |                                      u64* address, u64* length, u8** buffer) const | 
					
						
							| 
									
										
										
										
											2008-12-08 05:30:24 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2016-12-26 11:45:22 +01:00
										 |  |  |   if (*length == 0) | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02:00
										 |  |  |     return; | 
					
						
							| 
									
										
										
										
											2009-01-11 22:25:57 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-26 11:45:22 +01:00
										 |  |  |   _dbg_assert_(DVDINTERFACE, *address >= source_start_address); | 
					
						
							| 
									
										
										
										
											2008-12-08 05:30:24 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-26 11:45:22 +01:00
										 |  |  |   u64 source_offset = *address - source_start_address; | 
					
						
							| 
									
										
										
										
											2008-12-08 05:30:24 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-26 11:45:22 +01:00
										 |  |  |   if (source_offset < source_length) | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02:00
										 |  |  |   { | 
					
						
							| 
									
										
										
										
											2016-12-26 11:45:22 +01:00
										 |  |  |     size_t bytes_to_read = std::min(source_length - source_offset, *length); | 
					
						
							| 
									
										
										
										
											2008-12-08 05:30:24 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-26 11:45:22 +01:00
										 |  |  |     memcpy(*buffer, source + source_offset, bytes_to_read); | 
					
						
							| 
									
										
										
										
											2008-12-08 05:30:24 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-26 11:45:22 +01:00
										 |  |  |     *length -= bytes_to_read; | 
					
						
							|  |  |  |     *buffer += bytes_to_read; | 
					
						
							|  |  |  |     *address += bytes_to_read; | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2008-12-08 05:30:24 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-26 11:45:22 +01:00
										 |  |  | void CVolumeDirectory::PadToAddress(u64 start_address, u64* address, u64* length, u8** buffer) const | 
					
						
							| 
									
										
										
										
											2008-12-08 05:30:24 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2016-12-26 11:45:22 +01:00
										 |  |  |   if (start_address > *address && *length > 0) | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02:00
										 |  |  |   { | 
					
						
							| 
									
										
										
										
											2016-12-26 11:45:22 +01:00
										 |  |  |     u64 padBytes = std::min(start_address - *address, *length); | 
					
						
							|  |  |  |     memset(*buffer, 0, (size_t)padBytes); | 
					
						
							|  |  |  |     *length -= padBytes; | 
					
						
							|  |  |  |     *buffer += padBytes; | 
					
						
							|  |  |  |     *address += padBytes; | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2008-12-08 05:30:24 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-08-25 23:40:05 -04:00
										 |  |  | void CVolumeDirectory::Write32(u32 data, u32 offset, std::vector<u8>* const buffer) | 
					
						
							| 
									
										
										
										
											2008-12-08 05:30:24 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02:00
										 |  |  |   (*buffer)[offset++] = (data >> 24); | 
					
						
							|  |  |  |   (*buffer)[offset++] = (data >> 16) & 0xff; | 
					
						
							|  |  |  |   (*buffer)[offset++] = (data >> 8) & 0xff; | 
					
						
							|  |  |  |   (*buffer)[offset] = (data)&0xff; | 
					
						
							| 
									
										
										
										
											2008-12-08 05:30:24 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-26 11:45:22 +01:00
										 |  |  | void CVolumeDirectory::WriteEntryData(u32* entry_offset, u8 type, u32 name_offset, u64 data_offset, | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02:00
										 |  |  |                                       u64 length) | 
					
						
							| 
									
										
										
										
											2008-12-08 05:30:24 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2016-12-26 11:45:22 +01:00
										 |  |  |   m_fst_data[(*entry_offset)++] = type; | 
					
						
							| 
									
										
										
										
											2008-12-08 05:30:24 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-26 11:45:22 +01:00
										 |  |  |   m_fst_data[(*entry_offset)++] = (name_offset >> 16) & 0xff; | 
					
						
							|  |  |  |   m_fst_data[(*entry_offset)++] = (name_offset >> 8) & 0xff; | 
					
						
							|  |  |  |   m_fst_data[(*entry_offset)++] = (name_offset)&0xff; | 
					
						
							| 
									
										
										
										
											2008-12-08 05:30:24 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-26 11:45:22 +01:00
										 |  |  |   Write32((u32)(data_offset >> m_address_shift), *entry_offset, &m_fst_data); | 
					
						
							|  |  |  |   *entry_offset += 4; | 
					
						
							| 
									
										
										
										
											2008-12-08 05:30:24 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-26 11:45:22 +01:00
										 |  |  |   Write32((u32)length, *entry_offset, &m_fst_data); | 
					
						
							|  |  |  |   *entry_offset += 4; | 
					
						
							| 
									
										
										
										
											2008-12-08 05:30:24 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-26 11:45:22 +01:00
										 |  |  | void CVolumeDirectory::WriteEntryName(u32* name_offset, const std::string& name) | 
					
						
							| 
									
										
										
										
											2008-12-08 05:30:24 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2016-12-26 11:45:22 +01:00
										 |  |  |   strncpy((char*)&m_fst_data[*name_offset + m_fst_name_offset], name.c_str(), name.length() + 1); | 
					
						
							| 
									
										
										
										
											2013-10-29 01:23:17 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-26 11:45:22 +01:00
										 |  |  |   *name_offset += (u32)(name.length() + 1); | 
					
						
							| 
									
										
										
										
											2008-12-08 05:30:24 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-26 11:45:22 +01:00
										 |  |  | void CVolumeDirectory::WriteDirectory(const File::FSTEntry& parent_entry, u32* fst_offset, | 
					
						
							|  |  |  |                                       u32* name_offset, u64* data_offset, u32 parent_entry_index) | 
					
						
							| 
									
										
										
										
											2013-10-29 01:23:17 -04:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2016-12-25 22:35:38 +01:00
										 |  |  |   std::vector<File::FSTEntry> sorted_entries = parent_entry.children; | 
					
						
							| 
									
										
										
										
											2008-12-08 05:30:24 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-25 22:35:38 +01:00
										 |  |  |   std::sort(sorted_entries.begin(), sorted_entries.end(), | 
					
						
							|  |  |  |             [](const File::FSTEntry& one, const File::FSTEntry& two) { | 
					
						
							|  |  |  |               return one.virtualName < two.virtualName; | 
					
						
							|  |  |  |             }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   for (const File::FSTEntry& entry : sorted_entries) | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     if (entry.isDirectory) | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2016-12-26 11:45:22 +01:00
										 |  |  |       u32 entry_index = *fst_offset / ENTRY_SIZE; | 
					
						
							|  |  |  |       WriteEntryData(fst_offset, DIRECTORY_ENTRY, *name_offset, parent_entry_index, | 
					
						
							|  |  |  |                      entry_index + entry.size + 1); | 
					
						
							|  |  |  |       WriteEntryName(name_offset, entry.virtualName); | 
					
						
							| 
									
										
										
										
											2016-12-25 22:35:38 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-26 11:45:22 +01:00
										 |  |  |       WriteDirectory(entry, fst_offset, name_offset, data_offset, entry_index); | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2016-12-25 22:35:38 +01:00
										 |  |  |     else | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       // put entry in FST
 | 
					
						
							| 
									
										
										
										
											2016-12-26 11:45:22 +01:00
										 |  |  |       WriteEntryData(fst_offset, FILE_ENTRY, *name_offset, *data_offset, entry.size); | 
					
						
							|  |  |  |       WriteEntryName(name_offset, entry.virtualName); | 
					
						
							| 
									
										
										
										
											2008-12-08 05:30:24 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-25 22:35:38 +01:00
										 |  |  |       // write entry to virtual disk
 | 
					
						
							| 
									
										
										
										
											2016-12-26 11:45:22 +01:00
										 |  |  |       _dbg_assert_(DVDINTERFACE, m_virtual_disk.find(*data_offset) == m_virtual_disk.end()); | 
					
						
							|  |  |  |       m_virtual_disk.emplace(*data_offset, entry.physicalName); | 
					
						
							| 
									
										
										
										
											2008-12-08 05:30:24 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-25 22:35:38 +01:00
										 |  |  |       // 4 byte aligned
 | 
					
						
							| 
									
										
										
										
											2016-12-26 11:45:22 +01:00
										 |  |  |       *data_offset = Common::AlignUp(*data_offset + std::max<u64>(entry.size, 1ull), 0x8000ull); | 
					
						
							| 
									
										
										
										
											2016-12-25 22:35:38 +01:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2008-12-08 05:30:24 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-26 11:45:22 +01:00
										 |  |  | static u32 ComputeNameSize(const File::FSTEntry& parent_entry) | 
					
						
							| 
									
										
										
										
											2008-12-08 05:30:24 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2016-12-26 11:45:22 +01:00
										 |  |  |   u32 name_size = 0; | 
					
						
							|  |  |  |   for (const File::FSTEntry& entry : parent_entry.children) | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02:00
										 |  |  |   { | 
					
						
							|  |  |  |     if (entry.isDirectory) | 
					
						
							| 
									
										
										
										
											2016-12-26 11:45:22 +01:00
										 |  |  |       name_size += ComputeNameSize(entry); | 
					
						
							| 
									
										
										
										
											2016-12-25 23:01:42 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-26 11:45:22 +01:00
										 |  |  |     name_size += (u32)entry.virtualName.length() + 1; | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2016-12-26 11:45:22 +01:00
										 |  |  |   return name_size; | 
					
						
							| 
									
										
										
										
											2008-12-08 05:30:24 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02:00
										 |  |  | }  // namespace
 |