mirror of
https://github.com/dolphin-emu/dolphin.git
synced 2025-01-11 00:29:11 +01:00
d210fbac2c
Currently scrubbing will display a warning that it removes the garbage data PERMENENTLY from the original file (mainly for speed of scrubbing). It could be removed in the future if people agree with me, since it is indeed just garbage. Any tips for making scrubbing faster would be welcomed :) git-svn-id: https://dolphin-emu.googlecode.com/svn/trunk@3267 8ced0084-cf51-0410-be5f-012b33b47a6e
479 lines
11 KiB
C++
479 lines
11 KiB
C++
// Copyright (C) 2003-2009 Dolphin Project.
|
|
|
|
// This program is free software: you can redistribute it and/or modify
|
|
// it under the terms of the GNU General Public License as published by
|
|
// the Free Software Foundation, version 2.0.
|
|
|
|
// This program is distributed in the hope that it will be useful,
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
// GNU General Public License 2.0 for more details.
|
|
|
|
// A copy of the GPL 2.0 should have been included with the program.
|
|
// If not, see http://www.gnu.org/licenses/
|
|
|
|
// Official SVN repository and contact information can be found at
|
|
// http://code.google.com/p/dolphin-emu/
|
|
#include "stdafx.h"
|
|
|
|
#include "VolumeDirectory.h"
|
|
#include "FileBlob.h"
|
|
|
|
namespace DiscIO
|
|
{
|
|
|
|
static const u8 ENTRY_SIZE = 0x0c;
|
|
static const u8 FILE_ENTRY = 0;
|
|
static const u8 DIRECTORY_ENTRY = 1;
|
|
static const u64 FST_ADDRESS = 0x440;
|
|
static const u32 MAX_NAME_LENGTH = 0x3df;
|
|
|
|
CVolumeDirectory::CVolumeDirectory(const std::string& _rDirectory, bool _bIsWii)
|
|
: m_totalNameSize(0)
|
|
, m_dataStartAddress(-1)
|
|
, m_fstSize(0)
|
|
, m_FSTData(NULL)
|
|
{
|
|
m_rootDirectory = ExtractDirectoryName(_rDirectory);
|
|
|
|
// create the default disk header
|
|
m_diskHeader = new u8[FST_ADDRESS];
|
|
memset(m_diskHeader, 0, (size_t)FST_ADDRESS);
|
|
SetUniqueID("RZDE01");
|
|
SetName("Default name");
|
|
|
|
if(_bIsWii)
|
|
{
|
|
SetDiskTypeWii();
|
|
}
|
|
else
|
|
{
|
|
SetDiskTypeGC();
|
|
}
|
|
|
|
BuildFST();
|
|
}
|
|
|
|
CVolumeDirectory::~CVolumeDirectory()
|
|
{
|
|
delete m_FSTData;
|
|
m_FSTData = NULL;
|
|
|
|
delete m_diskHeader;
|
|
m_diskHeader = NULL;
|
|
}
|
|
|
|
bool CVolumeDirectory::IsValidDirectory(const std::string& _rDirectory)
|
|
{
|
|
std::string directoryName = ExtractDirectoryName(_rDirectory);
|
|
return File::IsDirectory(directoryName.c_str());
|
|
}
|
|
|
|
bool CVolumeDirectory::RAWRead( u64 _Offset, u64 _Length, u8* _pBuffer ) const
|
|
{
|
|
return false;
|
|
}
|
|
|
|
bool CVolumeDirectory::Read(u64 _Offset, u64 _Length, u8* _pBuffer) const
|
|
{
|
|
if(_Offset < FST_ADDRESS)
|
|
{
|
|
WriteToBuffer(0, FST_ADDRESS, m_diskHeader, _Offset, _Length, _pBuffer);
|
|
}
|
|
|
|
if(_Offset < m_dataStartAddress)
|
|
{
|
|
WriteToBuffer(FST_ADDRESS, m_fstSize, m_FSTData, _Offset, _Length, _pBuffer);
|
|
}
|
|
|
|
if(m_virtualDisk.size() == 0)
|
|
return true;
|
|
|
|
// Determine which file the offset refers to
|
|
std::map<u64, std::string>::const_iterator fileIter = m_virtualDisk.lower_bound(_Offset);
|
|
if(fileIter->first > _Offset && fileIter != m_virtualDisk.begin())
|
|
--fileIter;
|
|
|
|
// zero fill to start of file data
|
|
PadToAddress(fileIter->first, _Offset, _Length, _pBuffer);
|
|
|
|
while(fileIter != m_virtualDisk.end() && _Length > 0)
|
|
{
|
|
_dbg_assert_(DVDINTERFACE, fileIter->first <= _Offset);
|
|
u64 fileOffset = _Offset - fileIter->first;
|
|
|
|
PlainFileReader* reader = PlainFileReader::Create(fileIter->second.c_str());
|
|
if(reader == NULL)
|
|
return false;
|
|
|
|
u64 fileSize = reader->GetDataSize();
|
|
|
|
if(fileOffset < fileSize)
|
|
{
|
|
u64 fileBytes = fileSize - fileOffset;
|
|
if(_Length < fileBytes)
|
|
fileBytes = _Length;
|
|
|
|
if(!reader->Read(fileOffset, fileBytes, _pBuffer))
|
|
return false;
|
|
|
|
_Length -= fileBytes;
|
|
_pBuffer += fileBytes;
|
|
_Offset += fileBytes;
|
|
}
|
|
|
|
++fileIter;
|
|
|
|
if(fileIter != m_virtualDisk.end())
|
|
{
|
|
_dbg_assert_(DVDINTERFACE, fileIter->first >= _Offset);
|
|
PadToAddress(fileIter->first, _Offset, _Length, _pBuffer);
|
|
}
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
std::string CVolumeDirectory::GetUniqueID() const
|
|
{
|
|
_dbg_assert_(DVDINTERFACE, m_diskHeader);
|
|
|
|
char buffer[7];
|
|
memcpy(buffer, m_diskHeader, 6);
|
|
buffer[6] = 0;
|
|
|
|
std::string id = buffer;
|
|
return id;
|
|
}
|
|
|
|
void CVolumeDirectory::SetUniqueID(std::string _ID)
|
|
{
|
|
_dbg_assert_(DVDINTERFACE, m_diskHeader);
|
|
|
|
u32 length = (u32)_ID.length();
|
|
if(length > 6)
|
|
length = 6;
|
|
|
|
memcpy(m_diskHeader, _ID.c_str(), length);
|
|
}
|
|
|
|
IVolume::ECountry CVolumeDirectory::GetCountry() const
|
|
{
|
|
_dbg_assert_(DVDINTERFACE, m_diskHeader);
|
|
|
|
u8 CountryCode = m_diskHeader[3];
|
|
|
|
ECountry country = COUNTRY_UNKNOWN;
|
|
|
|
switch (CountryCode)
|
|
{
|
|
case 'S':
|
|
country = COUNTRY_EUROPE;
|
|
break; // PAL <- that is shitty :) zelda demo disc
|
|
|
|
case 'P':
|
|
country = COUNTRY_EUROPE;
|
|
break; // PAL
|
|
|
|
case 'D':
|
|
country = COUNTRY_EUROPE;
|
|
break; // PAL
|
|
|
|
case 'F':
|
|
country = COUNTRY_FRANCE;
|
|
break; // PAL
|
|
|
|
case 'I':
|
|
country = COUNTRY_ITALY;
|
|
break; // PAL
|
|
|
|
case 'X':
|
|
country = COUNTRY_EUROPE;
|
|
break; // XIII <- uses X but is PAL rip
|
|
|
|
case 'E':
|
|
country = COUNTRY_USA;
|
|
break; // USA
|
|
|
|
case 'J':
|
|
country = COUNTRY_JAP;
|
|
break; // JAP
|
|
|
|
case 'O':
|
|
country = COUNTRY_UNKNOWN;
|
|
break; // SDK
|
|
|
|
default:
|
|
country = COUNTRY_UNKNOWN;
|
|
break;
|
|
}
|
|
|
|
return(country);
|
|
}
|
|
|
|
std::string CVolumeDirectory::GetMakerID() const
|
|
{
|
|
return "VOID";
|
|
}
|
|
|
|
std::string CVolumeDirectory::GetName() const
|
|
{
|
|
_dbg_assert_(DVDINTERFACE, m_diskHeader);
|
|
std::string name = (char*)(m_diskHeader + 0x20);
|
|
return name;
|
|
}
|
|
|
|
void CVolumeDirectory::SetName(std::string _Name)
|
|
{
|
|
_dbg_assert_(DVDINTERFACE, m_diskHeader);
|
|
|
|
u32 length = (u32)_Name.length();
|
|
if(length > MAX_NAME_LENGTH)
|
|
length = MAX_NAME_LENGTH;
|
|
|
|
memcpy(m_diskHeader + 0x20, _Name.c_str(), length);
|
|
m_diskHeader[length + 0x20] = 0;
|
|
}
|
|
|
|
u32 CVolumeDirectory::GetFSTSize() const
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
std::string CVolumeDirectory::GetApploaderDate() const
|
|
{
|
|
return "VOID";
|
|
}
|
|
|
|
u64 CVolumeDirectory::GetSize() const
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
static const char DIR_SEPARATOR =
|
|
#ifdef _WIN32
|
|
'\\';
|
|
#else
|
|
'/';
|
|
#endif
|
|
|
|
std::string CVolumeDirectory::ExtractDirectoryName(const std::string& _rDirectory)
|
|
{
|
|
std::string directoryName = _rDirectory;
|
|
|
|
size_t lastSep = directoryName.find_last_of(DIR_SEPARATOR);
|
|
|
|
if(lastSep != directoryName.size() - 1)
|
|
{
|
|
// 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.
|
|
size_t extensionStart = directoryName.find_last_of('.');
|
|
if(extensionStart != std::string::npos && extensionStart > lastSep)
|
|
{
|
|
directoryName.resize(lastSep);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
directoryName.resize(lastSep);
|
|
}
|
|
|
|
return directoryName;
|
|
}
|
|
|
|
void CVolumeDirectory::SetDiskTypeWii()
|
|
{
|
|
_dbg_assert_(DVDINTERFACE, m_diskHeader);
|
|
|
|
m_diskHeader[0x18] = 0x5d;
|
|
m_diskHeader[0x19] = 0x1c;
|
|
m_diskHeader[0x1a] = 0x9e;
|
|
m_diskHeader[0x1b] = 0xa3;
|
|
memset(m_diskHeader + 0x1c, 0, 4);
|
|
|
|
m_addressShift = 2;
|
|
}
|
|
|
|
void CVolumeDirectory::SetDiskTypeGC()
|
|
{
|
|
_dbg_assert_(DVDINTERFACE, m_diskHeader);
|
|
|
|
memset(m_diskHeader + 0x18, 0, 4);
|
|
m_diskHeader[0x1c] = 0xc2;
|
|
m_diskHeader[0x1d] = 0x33;
|
|
m_diskHeader[0x1e] = 0x9f;
|
|
m_diskHeader[0x1f] = 0x3d;
|
|
|
|
m_addressShift = 0;
|
|
}
|
|
|
|
void CVolumeDirectory::BuildFST()
|
|
{
|
|
if(m_FSTData)
|
|
{
|
|
delete m_FSTData;
|
|
}
|
|
|
|
File::FSTEntry rootEntry;
|
|
|
|
// read data from physical disk to rootEntry
|
|
u32 totalEntries = AddDirectoryEntries(m_rootDirectory, rootEntry) + 1;
|
|
|
|
m_fstNameOffset = totalEntries * ENTRY_SIZE; // offset in FST nameTable
|
|
m_fstSize = m_fstNameOffset + m_totalNameSize;
|
|
m_FSTData = new u8[(u32)m_fstSize];
|
|
|
|
// 4 byte aligned start of data on disk
|
|
m_dataStartAddress = (FST_ADDRESS + m_fstSize + 3) & ~3;
|
|
u64 curDataAddress = m_dataStartAddress;
|
|
|
|
u32 fstOffset = 0; // offset within FST data
|
|
u32 nameOffset = 0; // offset within name table
|
|
u32 rootOffset = 0; // offset of root of FST
|
|
|
|
// write root entry
|
|
WriteEntryData(fstOffset, DIRECTORY_ENTRY, 0, 0, totalEntries);
|
|
|
|
for(std::vector<File::FSTEntry>::iterator iter = rootEntry.children.begin(); iter != rootEntry.children.end(); ++iter)
|
|
{
|
|
WriteEntry(*iter, fstOffset, nameOffset, curDataAddress, rootOffset);
|
|
}
|
|
|
|
// overflow check
|
|
_dbg_assert_(DVDINTERFACE, nameOffset == m_totalNameSize);
|
|
|
|
// write FST size and location
|
|
_dbg_assert_(DVDINTERFACE, m_diskHeader);
|
|
Write32((u32)(FST_ADDRESS >> m_addressShift), 0x0424, m_diskHeader);
|
|
Write32((u32)m_fstSize, 0x0428, m_diskHeader);
|
|
Write32((u32)m_fstSize, 0x042c, m_diskHeader);
|
|
}
|
|
|
|
void CVolumeDirectory::WriteToBuffer(u64 _SrcStartAddress, u64 _SrcLength, u8* _Src,
|
|
u64& _Address, u64& _Length, u8*& _pBuffer) const
|
|
{
|
|
if(_Length == 0)
|
|
return;
|
|
|
|
_dbg_assert_(DVDINTERFACE, _Address >= _SrcStartAddress);
|
|
|
|
u64 srcOffset = _Address - _SrcStartAddress;
|
|
|
|
if(srcOffset < _SrcLength)
|
|
{
|
|
u64 srcBytes = _SrcLength - srcOffset;
|
|
if(_Length < srcBytes)
|
|
srcBytes = _Length;
|
|
|
|
memcpy(_pBuffer, _Src + srcOffset, (size_t)srcBytes);
|
|
|
|
_Length -= srcBytes;
|
|
_pBuffer += srcBytes;
|
|
_Address += srcBytes;
|
|
}
|
|
}
|
|
|
|
void CVolumeDirectory::PadToAddress(u64 _StartAddress, u64& _Address, u64& _Length, u8*& _pBuffer) const
|
|
{
|
|
if(_StartAddress <= _Address)
|
|
return;
|
|
|
|
u64 padBytes = _StartAddress - _Address;
|
|
if(padBytes > _Length)
|
|
padBytes = _Length;
|
|
|
|
if(_Length > 0)
|
|
{
|
|
memset(_pBuffer, 0, (size_t)padBytes);
|
|
_Length -= padBytes;
|
|
_pBuffer += padBytes;
|
|
_Address += padBytes;
|
|
}
|
|
}
|
|
|
|
void CVolumeDirectory::Write32(u32 data, u32 offset, u8* buffer)
|
|
{
|
|
buffer[offset++] = (data >> 24);
|
|
buffer[offset++] = (data >> 16) & 0xff;
|
|
buffer[offset++] = (data >> 8) & 0xff;
|
|
buffer[offset] = (data) & 0xff;
|
|
}
|
|
|
|
void CVolumeDirectory::WriteEntryData(u32& entryOffset, u8 type, u32 nameOffset, u64 dataOffset, u32 length)
|
|
{
|
|
m_FSTData[entryOffset++] = type;
|
|
|
|
m_FSTData[entryOffset++] = (nameOffset >> 16) & 0xff;
|
|
m_FSTData[entryOffset++] = (nameOffset >> 8) & 0xff;
|
|
m_FSTData[entryOffset++] = (nameOffset) & 0xff;
|
|
|
|
Write32((u32)(dataOffset >> m_addressShift), entryOffset, m_FSTData);
|
|
entryOffset += 4;
|
|
|
|
Write32((u32)length, entryOffset, m_FSTData);
|
|
entryOffset += 4;
|
|
}
|
|
|
|
void CVolumeDirectory::WriteEntryName(u32& nameOffset, const std::string& name)
|
|
{
|
|
strncpy((char*)(m_FSTData + nameOffset + m_fstNameOffset), name.c_str(), name.length() + 1);
|
|
|
|
nameOffset += (u32)(name.length() + 1);
|
|
}
|
|
|
|
void CVolumeDirectory::WriteEntry(const File::FSTEntry& entry, u32& fstOffset, u32& nameOffset, u64& dataOffset, u32 parentEntryNum)
|
|
{
|
|
if(entry.isDirectory)
|
|
{
|
|
u32 myOffset = fstOffset;
|
|
u32 myEntryNum = myOffset / ENTRY_SIZE;
|
|
WriteEntryData(fstOffset, DIRECTORY_ENTRY, nameOffset, parentEntryNum, (u32)(myEntryNum + entry.size + 1));
|
|
WriteEntryName(nameOffset, entry.virtualName);
|
|
|
|
for(std::vector<File::FSTEntry>::const_iterator iter = entry.children.begin(); iter != entry.children.end(); ++iter)
|
|
{
|
|
WriteEntry(*iter, fstOffset, nameOffset, dataOffset, myEntryNum);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// put entry in FST
|
|
WriteEntryData(fstOffset, FILE_ENTRY, nameOffset, dataOffset, (u32)entry.size);
|
|
WriteEntryName(nameOffset, entry.virtualName);
|
|
|
|
// write entry to virtual disk
|
|
_dbg_assert_(DVDINTERFACE, m_virtualDisk.find(dataOffset) == m_virtualDisk.end());
|
|
m_virtualDisk.insert(make_pair(dataOffset, entry.physicalName));
|
|
|
|
// 4 byte aligned
|
|
dataOffset = (dataOffset + entry.size + 3) & ~3ULL;
|
|
}
|
|
}
|
|
|
|
static u32 ComputeNameSize(const File::FSTEntry& parentEntry)
|
|
{
|
|
u32 nameSize = 0;
|
|
const std::vector<File::FSTEntry>& children = parentEntry.children;
|
|
for (std::vector<File::FSTEntry>::const_iterator it = children.begin();
|
|
it != children.end(); ++it)
|
|
{
|
|
const File::FSTEntry& entry = *it;
|
|
if (entry.isDirectory)
|
|
{
|
|
nameSize += ComputeNameSize(entry);
|
|
}
|
|
nameSize += (u32)entry.virtualName.length() + 1;
|
|
}
|
|
return nameSize;
|
|
}
|
|
|
|
u32 CVolumeDirectory::AddDirectoryEntries(const std::string& _Directory, File::FSTEntry& parentEntry)
|
|
{
|
|
u32 foundEntries = ScanDirectoryTree(_Directory.c_str(), parentEntry);
|
|
m_totalNameSize += ComputeNameSize(parentEntry);
|
|
return foundEntries;
|
|
}
|
|
|
|
} // namespace
|