Mark which Wii root to use in the NAND path code.
It's used by both the GUI to do things like install WADs and check up on the system menu, in which case the global root should be used, and by /dev/es, in which case the local one should. The latter isn't *terribly* useful today, since no contents will ever be installed in temporary roots (although it's still relevant for data directories), but converting the whole thing makes sense because then it will Just Work once the entire NAND is synced. Because it would have been a bit of work to split it up (but I can if desired), this commit also contains some basic cleanup of NANDContentLoader: (1) The useless interface class INANDContentLoader is removed and the methods are changed to just return CNANDContentLoader (the only implementation); (2) CNANDContentManager is changed to use unique_ptr and cleaned up a bit.
This commit is contained in:
parent
6d4128ddcc
commit
c22d1d68ab
|
@ -56,34 +56,40 @@ void ShutdownWiiRoot()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
std::string GetTicketFileName(u64 _titleID)
|
static std::string RootUserPath(FromWhichRoot from)
|
||||||
|
{
|
||||||
|
int idx = from == FROM_CONFIGURED_ROOT ? D_WIIROOT_IDX : D_SESSION_WIIROOT_IDX;
|
||||||
|
return File::GetUserPath(idx);
|
||||||
|
}
|
||||||
|
|
||||||
|
std::string GetTicketFileName(u64 _titleID, FromWhichRoot from)
|
||||||
{
|
{
|
||||||
return StringFromFormat("%s/ticket/%08x/%08x.tik",
|
return StringFromFormat("%s/ticket/%08x/%08x.tik",
|
||||||
File::GetUserPath(D_SESSION_WIIROOT_IDX).c_str(),
|
RootUserPath(from).c_str(),
|
||||||
(u32)(_titleID >> 32), (u32)_titleID);
|
(u32)(_titleID >> 32), (u32)_titleID);
|
||||||
}
|
}
|
||||||
|
|
||||||
std::string GetTitleDataPath(u64 _titleID)
|
std::string GetTitleDataPath(u64 _titleID, FromWhichRoot from)
|
||||||
{
|
{
|
||||||
return StringFromFormat("%s/title/%08x/%08x/data/",
|
return StringFromFormat("%s/title/%08x/%08x/data/",
|
||||||
File::GetUserPath(D_SESSION_WIIROOT_IDX).c_str(),
|
RootUserPath(from).c_str(),
|
||||||
(u32)(_titleID >> 32), (u32)_titleID);
|
(u32)(_titleID >> 32), (u32)_titleID);
|
||||||
}
|
}
|
||||||
|
|
||||||
std::string GetTMDFileName(u64 _titleID)
|
std::string GetTMDFileName(u64 _titleID, FromWhichRoot from)
|
||||||
{
|
{
|
||||||
return GetTitleContentPath(_titleID) + "title.tmd";
|
return GetTitleContentPath(_titleID, from) + "title.tmd";
|
||||||
}
|
}
|
||||||
std::string GetTitleContentPath(u64 _titleID)
|
std::string GetTitleContentPath(u64 _titleID, FromWhichRoot from)
|
||||||
{
|
{
|
||||||
return StringFromFormat("%s/title/%08x/%08x/content/",
|
return StringFromFormat("%s/title/%08x/%08x/content/",
|
||||||
File::GetUserPath(D_SESSION_WIIROOT_IDX).c_str(),
|
RootUserPath(from).c_str(),
|
||||||
(u32)(_titleID >> 32), (u32)_titleID);
|
(u32)(_titleID >> 32), (u32)_titleID);
|
||||||
}
|
}
|
||||||
|
|
||||||
bool CheckTitleTMD(u64 _titleID)
|
bool CheckTitleTMD(u64 _titleID, FromWhichRoot from)
|
||||||
{
|
{
|
||||||
const std::string TitlePath = GetTMDFileName(_titleID);
|
const std::string TitlePath = GetTMDFileName(_titleID, from);
|
||||||
if (File::Exists(TitlePath))
|
if (File::Exists(TitlePath))
|
||||||
{
|
{
|
||||||
File::IOFile pTMDFile(TitlePath, "rb");
|
File::IOFile pTMDFile(TitlePath, "rb");
|
||||||
|
@ -96,9 +102,9 @@ bool CheckTitleTMD(u64 _titleID)
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool CheckTitleTIK(u64 _titleID)
|
bool CheckTitleTIK(u64 _titleID, FromWhichRoot from)
|
||||||
{
|
{
|
||||||
const std::string ticketFileName = Common::GetTicketFileName(_titleID);
|
const std::string ticketFileName = Common::GetTicketFileName(_titleID, from);
|
||||||
if (File::Exists(ticketFileName))
|
if (File::Exists(ticketFileName))
|
||||||
{
|
{
|
||||||
File::IOFile pTIKFile(ticketFileName, "rb");
|
File::IOFile pTIKFile(ticketFileName, "rb");
|
||||||
|
|
|
@ -21,11 +21,17 @@ namespace Common
|
||||||
void InitializeWiiRoot(bool use_temporary);
|
void InitializeWiiRoot(bool use_temporary);
|
||||||
void ShutdownWiiRoot();
|
void ShutdownWiiRoot();
|
||||||
|
|
||||||
std::string GetTicketFileName(u64 _titleID);
|
enum FromWhichRoot
|
||||||
std::string GetTMDFileName(u64 _titleID);
|
{
|
||||||
std::string GetTitleDataPath(u64 _titleID);
|
FROM_CONFIGURED_ROOT, // not related to currently running game - use D_WIIROOT_IDX
|
||||||
std::string GetTitleContentPath(u64 _titleID);
|
FROM_SESSION_ROOT, // request from currently running game - use D_SESSION_WIIROOT_IDX
|
||||||
bool CheckTitleTMD(u64 _titleID);
|
};
|
||||||
bool CheckTitleTIK(u64 _titleID);
|
|
||||||
|
std::string GetTicketFileName(u64 _titleID, FromWhichRoot from);
|
||||||
|
std::string GetTMDFileName(u64 _titleID, FromWhichRoot from);
|
||||||
|
std::string GetTitleDataPath(u64 _titleID, FromWhichRoot from);
|
||||||
|
std::string GetTitleContentPath(u64 _titleID, FromWhichRoot from);
|
||||||
|
bool CheckTitleTMD(u64 _titleID, FromWhichRoot from);
|
||||||
|
bool CheckTitleTIK(u64 _titleID, FromWhichRoot from);
|
||||||
void ReadReplacements(replace_v& replacements);
|
void ReadReplacements(replace_v& replacements);
|
||||||
}
|
}
|
||||||
|
|
|
@ -79,7 +79,7 @@ bool CBoot::FindMapFile(std::string* existing_map_file,
|
||||||
{
|
{
|
||||||
case SConfig::BOOT_WII_NAND:
|
case SConfig::BOOT_WII_NAND:
|
||||||
{
|
{
|
||||||
const DiscIO::INANDContentLoader& Loader =
|
const DiscIO::CNANDContentLoader& Loader =
|
||||||
DiscIO::CNANDContentManager::Access().GetNANDLoader(_StartupPara.m_strFilename);
|
DiscIO::CNANDContentManager::Access().GetNANDLoader(_StartupPara.m_strFilename);
|
||||||
if (Loader.IsValid())
|
if (Loader.IsValid())
|
||||||
{
|
{
|
||||||
|
|
|
@ -187,7 +187,7 @@ bool CBoot::SetupWiiMemory(DiscIO::IVolume::ECountry country)
|
||||||
|
|
||||||
SettingsHandler gen;
|
SettingsHandler gen;
|
||||||
std::string serno;
|
std::string serno;
|
||||||
std::string settings_Filename(Common::GetTitleDataPath(TITLEID_SYSMENU) + WII_SETTING);
|
std::string settings_Filename(Common::GetTitleDataPath(TITLEID_SYSMENU, Common::FROM_SESSION_ROOT) + WII_SETTING);
|
||||||
if (File::Exists(settings_Filename))
|
if (File::Exists(settings_Filename))
|
||||||
{
|
{
|
||||||
File::IOFile settingsFileHandle(settings_Filename, "rb");
|
File::IOFile settingsFileHandle(settings_Filename, "rb");
|
||||||
|
|
|
@ -49,7 +49,7 @@ struct StateFlags
|
||||||
|
|
||||||
bool CBoot::Boot_WiiWAD(const std::string& _pFilename)
|
bool CBoot::Boot_WiiWAD(const std::string& _pFilename)
|
||||||
{
|
{
|
||||||
std::string state_filename(Common::GetTitleDataPath(TITLEID_SYSMENU) + WII_STATE);
|
std::string state_filename(Common::GetTitleDataPath(TITLEID_SYSMENU, Common::FROM_SESSION_ROOT) + WII_STATE);
|
||||||
|
|
||||||
if (File::Exists(state_filename))
|
if (File::Exists(state_filename))
|
||||||
{
|
{
|
||||||
|
@ -75,13 +75,13 @@ bool CBoot::Boot_WiiWAD(const std::string& _pFilename)
|
||||||
state_file.WriteBytes(&state, sizeof(StateFlags));
|
state_file.WriteBytes(&state, sizeof(StateFlags));
|
||||||
}
|
}
|
||||||
|
|
||||||
const DiscIO::INANDContentLoader& ContentLoader = DiscIO::CNANDContentManager::Access().GetNANDLoader(_pFilename);
|
const DiscIO::CNANDContentLoader& ContentLoader = DiscIO::CNANDContentManager::Access().GetNANDLoader(_pFilename);
|
||||||
if (!ContentLoader.IsValid())
|
if (!ContentLoader.IsValid())
|
||||||
return false;
|
return false;
|
||||||
|
|
||||||
u64 titleID = ContentLoader.GetTitleID();
|
u64 titleID = ContentLoader.GetTitleID();
|
||||||
// create data directory
|
// create data directory
|
||||||
File::CreateFullPath(Common::GetTitleDataPath(titleID));
|
File::CreateFullPath(Common::GetTitleDataPath(titleID, Common::FROM_SESSION_ROOT));
|
||||||
|
|
||||||
if (titleID == TITLEID_SYSMENU)
|
if (titleID == TITLEID_SYSMENU)
|
||||||
HLE_IPC_CreateVirtualFATFilesystem();
|
HLE_IPC_CreateVirtualFATFilesystem();
|
||||||
|
|
|
@ -757,7 +757,7 @@ bool SConfig::AutoSetup(EBootBS2 _BootBS2)
|
||||||
else if (DiscIO::CNANDContentManager::Access().GetNANDLoader(m_strFilename).IsValid())
|
else if (DiscIO::CNANDContentManager::Access().GetNANDLoader(m_strFilename).IsValid())
|
||||||
{
|
{
|
||||||
std::unique_ptr<DiscIO::IVolume> pVolume(DiscIO::CreateVolumeFromFilename(m_strFilename));
|
std::unique_ptr<DiscIO::IVolume> pVolume(DiscIO::CreateVolumeFromFilename(m_strFilename));
|
||||||
const DiscIO::INANDContentLoader& ContentLoader = DiscIO::CNANDContentManager::Access().GetNANDLoader(m_strFilename);
|
const DiscIO::CNANDContentLoader& ContentLoader = DiscIO::CNANDContentManager::Access().GetNANDLoader(m_strFilename);
|
||||||
|
|
||||||
if (ContentLoader.GetContentByIndex(ContentLoader.GetBootIndex()) == nullptr)
|
if (ContentLoader.GetContentByIndex(ContentLoader.GetBootIndex()) == nullptr)
|
||||||
{
|
{
|
||||||
|
|
|
@ -145,7 +145,7 @@ void CEXIMemoryCard::SetupGciFolder(u16 sizeMb)
|
||||||
u32 CurrentGameId = 0;
|
u32 CurrentGameId = 0;
|
||||||
if (strUniqueID == TITLEID_SYSMENU_STRING)
|
if (strUniqueID == TITLEID_SYSMENU_STRING)
|
||||||
{
|
{
|
||||||
const DiscIO::INANDContentLoader & SysMenu_Loader = DiscIO::CNANDContentManager::Access().GetNANDLoader(TITLEID_SYSMENU, false);
|
const DiscIO::CNANDContentLoader & SysMenu_Loader = DiscIO::CNANDContentManager::Access().GetNANDLoader(TITLEID_SYSMENU, Common::FROM_SESSION_ROOT);
|
||||||
if (SysMenu_Loader.IsValid())
|
if (SysMenu_Loader.IsValid())
|
||||||
{
|
{
|
||||||
country_code = DiscIO::CountrySwitch(SysMenu_Loader.GetCountryChar());
|
country_code = DiscIO::CountrySwitch(SysMenu_Loader.GetCountryChar());
|
||||||
|
|
|
@ -569,7 +569,8 @@ bool CWiiSaveCrypted::getPaths(bool for_export)
|
||||||
{
|
{
|
||||||
if (m_title_id)
|
if (m_title_id)
|
||||||
{
|
{
|
||||||
m_wii_title_path = Common::GetTitleDataPath(m_title_id);
|
// CONFIGURED because this whole class is only used from the GUI, not directly by games.
|
||||||
|
m_wii_title_path = Common::GetTitleDataPath(m_title_id, Common::FROM_CONFIGURED_ROOT);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (for_export)
|
if (for_export)
|
||||||
|
|
|
@ -41,7 +41,7 @@ std::string HLE_IPC_BuildFilename(std::string path_wii)
|
||||||
void HLE_IPC_CreateVirtualFATFilesystem()
|
void HLE_IPC_CreateVirtualFATFilesystem()
|
||||||
{
|
{
|
||||||
const int cdbSize = 0x01400000;
|
const int cdbSize = 0x01400000;
|
||||||
const std::string cdbPath = Common::GetTitleDataPath(TITLEID_SYSMENU) + "cdb.vff";
|
const std::string cdbPath = Common::GetTitleDataPath(TITLEID_SYSMENU, Common::FROM_SESSION_ROOT) + "cdb.vff";
|
||||||
if ((int)File::GetSize(cdbPath) < cdbSize)
|
if ((int)File::GetSize(cdbPath) < cdbSize)
|
||||||
{
|
{
|
||||||
// cdb.vff is a virtual Fat filesystem created on first launch of sysmenu
|
// cdb.vff is a virtual Fat filesystem created on first launch of sysmenu
|
||||||
|
|
|
@ -185,7 +185,7 @@ IPCCommandResult CWII_IPC_HLE_Device_es::Open(u32 _CommandAddress, u32 _Mode)
|
||||||
|
|
||||||
IPCCommandResult CWII_IPC_HLE_Device_es::Close(u32 _CommandAddress, bool _bForce)
|
IPCCommandResult CWII_IPC_HLE_Device_es::Close(u32 _CommandAddress, bool _bForce)
|
||||||
{
|
{
|
||||||
// Leave deletion of the INANDContentLoader objects to CNANDContentManager, don't do it here!
|
// Leave deletion of the CNANDContentLoader objects to CNANDContentManager, don't do it here!
|
||||||
m_NANDContent.clear();
|
m_NANDContent.clear();
|
||||||
for (auto& pair : m_ContentAccessMap)
|
for (auto& pair : m_ContentAccessMap)
|
||||||
{
|
{
|
||||||
|
@ -206,7 +206,7 @@ IPCCommandResult CWII_IPC_HLE_Device_es::Close(u32 _CommandAddress, bool _bForce
|
||||||
|
|
||||||
u32 CWII_IPC_HLE_Device_es::OpenTitleContent(u32 CFD, u64 TitleID, u16 Index)
|
u32 CWII_IPC_HLE_Device_es::OpenTitleContent(u32 CFD, u64 TitleID, u16 Index)
|
||||||
{
|
{
|
||||||
const DiscIO::INANDContentLoader& Loader = AccessContentDevice(TitleID);
|
const DiscIO::CNANDContentLoader& Loader = AccessContentDevice(TitleID);
|
||||||
|
|
||||||
if (!Loader.IsValid())
|
if (!Loader.IsValid())
|
||||||
{
|
{
|
||||||
|
@ -292,7 +292,7 @@ IPCCommandResult CWII_IPC_HLE_Device_es::IOCtlV(u32 _CommandAddress)
|
||||||
|
|
||||||
u64 TitleID = Memory::Read_U64(Buffer.InBuffer[0].m_Address);
|
u64 TitleID = Memory::Read_U64(Buffer.InBuffer[0].m_Address);
|
||||||
|
|
||||||
const DiscIO::INANDContentLoader& rNANDContent = AccessContentDevice(TitleID);
|
const DiscIO::CNANDContentLoader& rNANDContent = AccessContentDevice(TitleID);
|
||||||
u16 NumberOfPrivateContent = 0;
|
u16 NumberOfPrivateContent = 0;
|
||||||
if (rNANDContent.IsValid()) // Not sure if dolphin will ever fail this check
|
if (rNANDContent.IsValid()) // Not sure if dolphin will ever fail this check
|
||||||
{
|
{
|
||||||
|
@ -322,7 +322,7 @@ IPCCommandResult CWII_IPC_HLE_Device_es::IOCtlV(u32 _CommandAddress)
|
||||||
|
|
||||||
u64 TitleID = Memory::Read_U64(Buffer.InBuffer[0].m_Address);
|
u64 TitleID = Memory::Read_U64(Buffer.InBuffer[0].m_Address);
|
||||||
|
|
||||||
const DiscIO::INANDContentLoader& rNANDCOntent = AccessContentDevice(TitleID);
|
const DiscIO::CNANDContentLoader& rNANDCOntent = AccessContentDevice(TitleID);
|
||||||
if (rNANDCOntent.IsValid()) // Not sure if dolphin will ever fail this check
|
if (rNANDCOntent.IsValid()) // Not sure if dolphin will ever fail this check
|
||||||
{
|
{
|
||||||
for (u16 i = 0; i < rNANDCOntent.GetNumEntries(); i++)
|
for (u16 i = 0; i < rNANDCOntent.GetNumEntries(); i++)
|
||||||
|
@ -579,18 +579,18 @@ IPCCommandResult CWII_IPC_HLE_Device_es::IOCtlV(u32 _CommandAddress)
|
||||||
u64 TitleID = Memory::Read_U64(Buffer.InBuffer[0].m_Address);
|
u64 TitleID = Memory::Read_U64(Buffer.InBuffer[0].m_Address);
|
||||||
|
|
||||||
u32 retVal = 0;
|
u32 retVal = 0;
|
||||||
const DiscIO::INANDContentLoader& Loader = AccessContentDevice(TitleID);
|
const DiscIO::CNANDContentLoader& Loader = AccessContentDevice(TitleID);
|
||||||
u32 ViewCount = Loader.GetTIKSize() / DiscIO::INANDContentLoader::TICKET_SIZE;
|
u32 ViewCount = Loader.GetTIKSize() / DiscIO::CNANDContentLoader::TICKET_SIZE;
|
||||||
|
|
||||||
if (!ViewCount)
|
if (!ViewCount)
|
||||||
{
|
{
|
||||||
std::string TicketFilename = Common::GetTicketFileName(TitleID);
|
std::string TicketFilename = Common::GetTicketFileName(TitleID, Common::FROM_SESSION_ROOT);
|
||||||
if (File::Exists(TicketFilename))
|
if (File::Exists(TicketFilename))
|
||||||
{
|
{
|
||||||
u32 FileSize = (u32)File::GetSize(TicketFilename);
|
u32 FileSize = (u32)File::GetSize(TicketFilename);
|
||||||
_dbg_assert_msg_(WII_IPC_ES, (FileSize % DiscIO::INANDContentLoader::TICKET_SIZE) == 0, "IOCTL_ES_GETVIEWCNT ticket file size seems to be wrong");
|
_dbg_assert_msg_(WII_IPC_ES, (FileSize % DiscIO::CNANDContentLoader::TICKET_SIZE) == 0, "IOCTL_ES_GETVIEWCNT ticket file size seems to be wrong");
|
||||||
|
|
||||||
ViewCount = FileSize / DiscIO::INANDContentLoader::TICKET_SIZE;
|
ViewCount = FileSize / DiscIO::CNANDContentLoader::TICKET_SIZE;
|
||||||
_dbg_assert_msg_(WII_IPC_ES, (ViewCount>0) && (ViewCount<=4), "IOCTL_ES_GETVIEWCNT ticket count seems to be wrong");
|
_dbg_assert_msg_(WII_IPC_ES, (ViewCount>0) && (ViewCount<=4), "IOCTL_ES_GETVIEWCNT ticket count seems to be wrong");
|
||||||
}
|
}
|
||||||
else if (TitleID >> 32 == 0x00000001)
|
else if (TitleID >> 32 == 0x00000001)
|
||||||
|
@ -626,29 +626,29 @@ IPCCommandResult CWII_IPC_HLE_Device_es::IOCtlV(u32 _CommandAddress)
|
||||||
u32 maxViews = Memory::Read_U32(Buffer.InBuffer[1].m_Address);
|
u32 maxViews = Memory::Read_U32(Buffer.InBuffer[1].m_Address);
|
||||||
u32 retVal = 0;
|
u32 retVal = 0;
|
||||||
|
|
||||||
const DiscIO::INANDContentLoader& Loader = AccessContentDevice(TitleID);
|
const DiscIO::CNANDContentLoader& Loader = AccessContentDevice(TitleID);
|
||||||
|
|
||||||
const u8 *Ticket = Loader.GetTIK();
|
const u8 *Ticket = Loader.GetTIK();
|
||||||
if (Ticket)
|
if (Ticket)
|
||||||
{
|
{
|
||||||
u32 viewCnt = Loader.GetTIKSize() / DiscIO::INANDContentLoader::TICKET_SIZE;
|
u32 viewCnt = Loader.GetTIKSize() / DiscIO::CNANDContentLoader::TICKET_SIZE;
|
||||||
for (unsigned int View = 0; View != maxViews && View < viewCnt; ++View)
|
for (unsigned int View = 0; View != maxViews && View < viewCnt; ++View)
|
||||||
{
|
{
|
||||||
Memory::Write_U32(View, Buffer.PayloadBuffer[0].m_Address + View * 0xD8);
|
Memory::Write_U32(View, Buffer.PayloadBuffer[0].m_Address + View * 0xD8);
|
||||||
Memory::CopyToEmu(Buffer.PayloadBuffer[0].m_Address + 4 + View * 0xD8,
|
Memory::CopyToEmu(Buffer.PayloadBuffer[0].m_Address + 4 + View * 0xD8,
|
||||||
Ticket + 0x1D0 + (View * DiscIO::INANDContentLoader::TICKET_SIZE), 212);
|
Ticket + 0x1D0 + (View * DiscIO::CNANDContentLoader::TICKET_SIZE), 212);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
std::string TicketFilename = Common::GetTicketFileName(TitleID);
|
std::string TicketFilename = Common::GetTicketFileName(TitleID, Common::FROM_SESSION_ROOT);
|
||||||
if (File::Exists(TicketFilename))
|
if (File::Exists(TicketFilename))
|
||||||
{
|
{
|
||||||
File::IOFile pFile(TicketFilename, "rb");
|
File::IOFile pFile(TicketFilename, "rb");
|
||||||
if (pFile)
|
if (pFile)
|
||||||
{
|
{
|
||||||
u8 FileTicket[DiscIO::INANDContentLoader::TICKET_SIZE];
|
u8 FileTicket[DiscIO::CNANDContentLoader::TICKET_SIZE];
|
||||||
for (unsigned int View = 0; View != maxViews && pFile.ReadBytes(FileTicket, DiscIO::INANDContentLoader::TICKET_SIZE); ++View)
|
for (unsigned int View = 0; View != maxViews && pFile.ReadBytes(FileTicket, DiscIO::CNANDContentLoader::TICKET_SIZE); ++View)
|
||||||
{
|
{
|
||||||
Memory::Write_U32(View, Buffer.PayloadBuffer[0].m_Address + View * 0xD8);
|
Memory::Write_U32(View, Buffer.PayloadBuffer[0].m_Address + View * 0xD8);
|
||||||
Memory::CopyToEmu(Buffer.PayloadBuffer[0].m_Address + 4 + View * 0xD8, FileTicket+0x1D0, 212);
|
Memory::CopyToEmu(Buffer.PayloadBuffer[0].m_Address + 4 + View * 0xD8, FileTicket+0x1D0, 212);
|
||||||
|
@ -688,12 +688,12 @@ IPCCommandResult CWII_IPC_HLE_Device_es::IOCtlV(u32 _CommandAddress)
|
||||||
|
|
||||||
u64 TitleID = Memory::Read_U64(Buffer.InBuffer[0].m_Address);
|
u64 TitleID = Memory::Read_U64(Buffer.InBuffer[0].m_Address);
|
||||||
|
|
||||||
const DiscIO::INANDContentLoader& Loader = AccessContentDevice(TitleID);
|
const DiscIO::CNANDContentLoader& Loader = AccessContentDevice(TitleID);
|
||||||
|
|
||||||
u32 TMDViewCnt = 0;
|
u32 TMDViewCnt = 0;
|
||||||
if (Loader.IsValid())
|
if (Loader.IsValid())
|
||||||
{
|
{
|
||||||
TMDViewCnt += DiscIO::INANDContentLoader::TMD_VIEW_SIZE;
|
TMDViewCnt += DiscIO::CNANDContentLoader::TMD_VIEW_SIZE;
|
||||||
TMDViewCnt += 2; // title version
|
TMDViewCnt += 2; // title version
|
||||||
TMDViewCnt += 2; // num entries
|
TMDViewCnt += 2; // num entries
|
||||||
TMDViewCnt += (u32)Loader.GetContentSize() * (4+2+2+8); // content id, index, type, size
|
TMDViewCnt += (u32)Loader.GetContentSize() * (4+2+2+8); // content id, index, type, size
|
||||||
|
@ -715,7 +715,7 @@ IPCCommandResult CWII_IPC_HLE_Device_es::IOCtlV(u32 _CommandAddress)
|
||||||
u64 TitleID = Memory::Read_U64(Buffer.InBuffer[0].m_Address);
|
u64 TitleID = Memory::Read_U64(Buffer.InBuffer[0].m_Address);
|
||||||
u32 MaxCount = Memory::Read_U32(Buffer.InBuffer[1].m_Address);
|
u32 MaxCount = Memory::Read_U32(Buffer.InBuffer[1].m_Address);
|
||||||
|
|
||||||
const DiscIO::INANDContentLoader& Loader = AccessContentDevice(TitleID);
|
const DiscIO::CNANDContentLoader& Loader = AccessContentDevice(TitleID);
|
||||||
|
|
||||||
INFO_LOG(WII_IPC_ES, "IOCTL_ES_GETTMDVIEWCNT: title: %08x/%08x buffer size: %i", (u32)(TitleID >> 32), (u32)TitleID, MaxCount);
|
INFO_LOG(WII_IPC_ES, "IOCTL_ES_GETTMDVIEWCNT: title: %08x/%08x buffer size: %i", (u32)(TitleID >> 32), (u32)TitleID, MaxCount);
|
||||||
|
|
||||||
|
@ -723,8 +723,8 @@ IPCCommandResult CWII_IPC_HLE_Device_es::IOCtlV(u32 _CommandAddress)
|
||||||
{
|
{
|
||||||
u32 Address = Buffer.PayloadBuffer[0].m_Address;
|
u32 Address = Buffer.PayloadBuffer[0].m_Address;
|
||||||
|
|
||||||
Memory::CopyToEmu(Address, Loader.GetTMDView(), DiscIO::INANDContentLoader::TMD_VIEW_SIZE);
|
Memory::CopyToEmu(Address, Loader.GetTMDView(), DiscIO::CNANDContentLoader::TMD_VIEW_SIZE);
|
||||||
Address += DiscIO::INANDContentLoader::TMD_VIEW_SIZE;
|
Address += DiscIO::CNANDContentLoader::TMD_VIEW_SIZE;
|
||||||
|
|
||||||
Memory::Write_U16(Loader.GetTitleVersion(), Address); Address += 2;
|
Memory::Write_U16(Loader.GetTitleVersion(), Address); Address += 2;
|
||||||
Memory::Write_U16(Loader.GetNumEntries(), Address); Address += 2;
|
Memory::Write_U16(Loader.GetNumEntries(), Address); Address += 2;
|
||||||
|
@ -757,7 +757,7 @@ IPCCommandResult CWII_IPC_HLE_Device_es::IOCtlV(u32 _CommandAddress)
|
||||||
{
|
{
|
||||||
u64 TitleID = Memory::Read_U64(Buffer.InBuffer[0].m_Address);
|
u64 TitleID = Memory::Read_U64(Buffer.InBuffer[0].m_Address);
|
||||||
INFO_LOG(WII_IPC_ES, "IOCTL_ES_DELETETICKET: title: %08x/%08x", (u32)(TitleID >> 32), (u32)TitleID);
|
INFO_LOG(WII_IPC_ES, "IOCTL_ES_DELETETICKET: title: %08x/%08x", (u32)(TitleID >> 32), (u32)TitleID);
|
||||||
if (File::Delete(Common::GetTicketFileName(TitleID)))
|
if (File::Delete(Common::GetTicketFileName(TitleID, Common::FROM_SESSION_ROOT)))
|
||||||
{
|
{
|
||||||
Memory::Write_U32(0, _CommandAddress + 0x4);
|
Memory::Write_U32(0, _CommandAddress + 0x4);
|
||||||
}
|
}
|
||||||
|
@ -772,7 +772,7 @@ IPCCommandResult CWII_IPC_HLE_Device_es::IOCtlV(u32 _CommandAddress)
|
||||||
{
|
{
|
||||||
u64 TitleID = Memory::Read_U64(Buffer.InBuffer[0].m_Address);
|
u64 TitleID = Memory::Read_U64(Buffer.InBuffer[0].m_Address);
|
||||||
INFO_LOG(WII_IPC_ES, "IOCTL_ES_DELETETITLECONTENT: title: %08x/%08x", (u32)(TitleID >> 32), (u32)TitleID);
|
INFO_LOG(WII_IPC_ES, "IOCTL_ES_DELETETITLECONTENT: title: %08x/%08x", (u32)(TitleID >> 32), (u32)TitleID);
|
||||||
if (DiscIO::CNANDContentManager::Access().RemoveTitle(TitleID))
|
if (DiscIO::CNANDContentManager::Access().RemoveTitle(TitleID, Common::FROM_SESSION_ROOT))
|
||||||
{
|
{
|
||||||
Memory::Write_U32(0, _CommandAddress + 0x4);
|
Memory::Write_U32(0, _CommandAddress + 0x4);
|
||||||
}
|
}
|
||||||
|
@ -790,14 +790,14 @@ IPCCommandResult CWII_IPC_HLE_Device_es::IOCtlV(u32 _CommandAddress)
|
||||||
// _dbg_assert_msg_(WII_IPC_ES, Buffer.NumberPayloadBuffer == 1, "IOCTL_ES_ES_GETSTOREDTMDSIZE no out buffer");
|
// _dbg_assert_msg_(WII_IPC_ES, Buffer.NumberPayloadBuffer == 1, "IOCTL_ES_ES_GETSTOREDTMDSIZE no out buffer");
|
||||||
|
|
||||||
u64 TitleID = Memory::Read_U64(Buffer.InBuffer[0].m_Address);
|
u64 TitleID = Memory::Read_U64(Buffer.InBuffer[0].m_Address);
|
||||||
const DiscIO::INANDContentLoader& Loader = AccessContentDevice(TitleID);
|
const DiscIO::CNANDContentLoader& Loader = AccessContentDevice(TitleID);
|
||||||
|
|
||||||
_dbg_assert_(WII_IPC_ES, Loader.IsValid());
|
_dbg_assert_(WII_IPC_ES, Loader.IsValid());
|
||||||
u32 TMDCnt = 0;
|
u32 TMDCnt = 0;
|
||||||
if (Loader.IsValid())
|
if (Loader.IsValid())
|
||||||
{
|
{
|
||||||
TMDCnt += DiscIO::INANDContentLoader::TMD_HEADER_SIZE;
|
TMDCnt += DiscIO::CNANDContentLoader::TMD_HEADER_SIZE;
|
||||||
TMDCnt += (u32)Loader.GetContentSize() * DiscIO::INANDContentLoader::CONTENT_HEADER_SIZE;
|
TMDCnt += (u32)Loader.GetContentSize() * DiscIO::CNANDContentLoader::CONTENT_HEADER_SIZE;
|
||||||
}
|
}
|
||||||
if (Buffer.NumberPayloadBuffer)
|
if (Buffer.NumberPayloadBuffer)
|
||||||
Memory::Write_U32(TMDCnt, Buffer.PayloadBuffer[0].m_Address);
|
Memory::Write_U32(TMDCnt, Buffer.PayloadBuffer[0].m_Address);
|
||||||
|
@ -822,7 +822,7 @@ IPCCommandResult CWII_IPC_HLE_Device_es::IOCtlV(u32 _CommandAddress)
|
||||||
// TODO: actually use this param in when writing to the outbuffer :/
|
// TODO: actually use this param in when writing to the outbuffer :/
|
||||||
MaxCount = Memory::Read_U32(Buffer.InBuffer[1].m_Address);
|
MaxCount = Memory::Read_U32(Buffer.InBuffer[1].m_Address);
|
||||||
}
|
}
|
||||||
const DiscIO::INANDContentLoader& Loader = AccessContentDevice(TitleID);
|
const DiscIO::CNANDContentLoader& Loader = AccessContentDevice(TitleID);
|
||||||
|
|
||||||
|
|
||||||
INFO_LOG(WII_IPC_ES, "IOCTL_ES_GETSTOREDTMD: title: %08x/%08x buffer size: %i", (u32)(TitleID >> 32), (u32)TitleID, MaxCount);
|
INFO_LOG(WII_IPC_ES, "IOCTL_ES_GETSTOREDTMD: title: %08x/%08x buffer size: %i", (u32)(TitleID >> 32), (u32)TitleID, MaxCount);
|
||||||
|
@ -831,14 +831,14 @@ IPCCommandResult CWII_IPC_HLE_Device_es::IOCtlV(u32 _CommandAddress)
|
||||||
{
|
{
|
||||||
u32 Address = Buffer.PayloadBuffer[0].m_Address;
|
u32 Address = Buffer.PayloadBuffer[0].m_Address;
|
||||||
|
|
||||||
Memory::CopyToEmu(Address, Loader.GetTMDHeader(), DiscIO::INANDContentLoader::TMD_HEADER_SIZE);
|
Memory::CopyToEmu(Address, Loader.GetTMDHeader(), DiscIO::CNANDContentLoader::TMD_HEADER_SIZE);
|
||||||
Address += DiscIO::INANDContentLoader::TMD_HEADER_SIZE;
|
Address += DiscIO::CNANDContentLoader::TMD_HEADER_SIZE;
|
||||||
|
|
||||||
const std::vector<DiscIO::SNANDContent>& rContent = Loader.GetContent();
|
const std::vector<DiscIO::SNANDContent>& rContent = Loader.GetContent();
|
||||||
for (size_t i=0; i<Loader.GetContentSize(); i++)
|
for (size_t i=0; i<Loader.GetContentSize(); i++)
|
||||||
{
|
{
|
||||||
Memory::CopyToEmu(Address, rContent[i].m_Header, DiscIO::INANDContentLoader::CONTENT_HEADER_SIZE);
|
Memory::CopyToEmu(Address, rContent[i].m_Header, DiscIO::CNANDContentLoader::CONTENT_HEADER_SIZE);
|
||||||
Address += DiscIO::INANDContentLoader::CONTENT_HEADER_SIZE;
|
Address += DiscIO::CNANDContentLoader::CONTENT_HEADER_SIZE;
|
||||||
}
|
}
|
||||||
|
|
||||||
_dbg_assert_(WII_IPC_ES, (Address-Buffer.PayloadBuffer[0].m_Address) == Buffer.PayloadBuffer[0].m_Size);
|
_dbg_assert_(WII_IPC_ES, (Address-Buffer.PayloadBuffer[0].m_Address) == Buffer.PayloadBuffer[0].m_Size);
|
||||||
|
@ -903,14 +903,14 @@ IPCCommandResult CWII_IPC_HLE_Device_es::IOCtlV(u32 _CommandAddress)
|
||||||
std::string tContentFile;
|
std::string tContentFile;
|
||||||
if ((u32)(TitleID>>32) != 0x00000001 || TitleID == TITLEID_SYSMENU)
|
if ((u32)(TitleID>>32) != 0x00000001 || TitleID == TITLEID_SYSMENU)
|
||||||
{
|
{
|
||||||
const DiscIO::INANDContentLoader& ContentLoader = AccessContentDevice(TitleID);
|
const DiscIO::CNANDContentLoader& ContentLoader = AccessContentDevice(TitleID);
|
||||||
if (ContentLoader.IsValid())
|
if (ContentLoader.IsValid())
|
||||||
{
|
{
|
||||||
u32 bootInd = ContentLoader.GetBootIndex();
|
u32 bootInd = ContentLoader.GetBootIndex();
|
||||||
const DiscIO::SNANDContent* pContent = ContentLoader.GetContentByIndex(bootInd);
|
const DiscIO::SNANDContent* pContent = ContentLoader.GetContentByIndex(bootInd);
|
||||||
if (pContent)
|
if (pContent)
|
||||||
{
|
{
|
||||||
tContentFile = Common::GetTitleContentPath(TitleID);
|
tContentFile = Common::GetTitleContentPath(TitleID, Common::FROM_SESSION_ROOT);
|
||||||
std::unique_ptr<CDolLoader> pDolLoader;
|
std::unique_ptr<CDolLoader> pDolLoader;
|
||||||
if (pContent->m_pData)
|
if (pContent->m_pData)
|
||||||
{
|
{
|
||||||
|
@ -1071,7 +1071,8 @@ IPCCommandResult CWII_IPC_HLE_Device_es::IOCtlV(u32 _CommandAddress)
|
||||||
return GetDefaultReply();
|
return GetDefaultReply();
|
||||||
}
|
}
|
||||||
|
|
||||||
const DiscIO::INANDContentLoader& CWII_IPC_HLE_Device_es::AccessContentDevice(u64 _TitleID)
|
// TODO: This cache is redundant with the one in CNANDContentManager.h
|
||||||
|
const DiscIO::CNANDContentLoader& CWII_IPC_HLE_Device_es::AccessContentDevice(u64 _TitleID)
|
||||||
{
|
{
|
||||||
if (m_pContentLoader->IsValid() && m_pContentLoader->GetTitleID() == _TitleID)
|
if (m_pContentLoader->IsValid() && m_pContentLoader->GetTitleID() == _TitleID)
|
||||||
return *m_pContentLoader;
|
return *m_pContentLoader;
|
||||||
|
@ -1080,7 +1081,7 @@ const DiscIO::INANDContentLoader& CWII_IPC_HLE_Device_es::AccessContentDevice(u6
|
||||||
if (itr != m_NANDContent.end())
|
if (itr != m_NANDContent.end())
|
||||||
return *itr->second;
|
return *itr->second;
|
||||||
|
|
||||||
m_NANDContent[_TitleID] = &DiscIO::CNANDContentManager::Access().GetNANDLoader(_TitleID);
|
m_NANDContent[_TitleID] = &DiscIO::CNANDContentManager::Access().GetNANDLoader(_TitleID, Common::FROM_SESSION_ROOT);
|
||||||
|
|
||||||
_dbg_assert_msg_(WII_IPC_ES, ((u32)(_TitleID >> 32) == 0x00010000) || m_NANDContent[_TitleID]->IsValid(), "NandContent not valid for TitleID %08x/%08x", (u32)(_TitleID >> 32), (u32)_TitleID);
|
_dbg_assert_msg_(WII_IPC_ES, ((u32)(_TitleID >> 32) == 0x00010000) || m_NANDContent[_TitleID]->IsValid(), "NandContent not valid for TitleID %08x/%08x", (u32)(_TitleID >> 32), (u32)_TitleID);
|
||||||
return *m_NANDContent[_TitleID];
|
return *m_NANDContent[_TitleID];
|
||||||
|
@ -1104,13 +1105,13 @@ u32 CWII_IPC_HLE_Device_es::ES_DIVerify(u8* _pTMD, u32 _sz)
|
||||||
{
|
{
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
std::string tmdPath = Common::GetTMDFileName(tmdTitleID);
|
std::string tmdPath = Common::GetTMDFileName(tmdTitleID, Common::FROM_SESSION_ROOT);
|
||||||
|
|
||||||
File::CreateFullPath(tmdPath);
|
File::CreateFullPath(tmdPath);
|
||||||
File::CreateFullPath(Common::GetTitleDataPath(tmdTitleID));
|
File::CreateFullPath(Common::GetTitleDataPath(tmdTitleID, Common::FROM_SESSION_ROOT));
|
||||||
|
|
||||||
Movie::g_titleID = tmdTitleID;
|
Movie::g_titleID = tmdTitleID;
|
||||||
std::string savePath = Common::GetTitleDataPath(tmdTitleID);
|
std::string savePath = Common::GetTitleDataPath(tmdTitleID, Common::FROM_SESSION_ROOT);
|
||||||
if (Movie::IsRecordingInput())
|
if (Movie::IsRecordingInput())
|
||||||
{
|
{
|
||||||
// TODO: Check for the actual save data
|
// TODO: Check for the actual save data
|
||||||
|
@ -1161,5 +1162,6 @@ u32 CWII_IPC_HLE_Device_es::ES_DIVerify(u8* _pTMD, u32 _sz)
|
||||||
ERROR_LOG(WII_IPC_ES, "DIVerify failed to write disc TMD to NAND.");
|
ERROR_LOG(WII_IPC_ES, "DIVerify failed to write disc TMD to NAND.");
|
||||||
}
|
}
|
||||||
DiscIO::cUIDsys::AccessInstance().AddTitle(tmdTitleID);
|
DiscIO::cUIDsys::AccessInstance().AddTitle(tmdTitleID);
|
||||||
|
DiscIO::CNANDContentManager::Access().ClearCache();
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
|
@ -14,7 +14,7 @@
|
||||||
class PointerWrap;
|
class PointerWrap;
|
||||||
namespace DiscIO
|
namespace DiscIO
|
||||||
{
|
{
|
||||||
class INANDContentLoader;
|
class CNANDContentLoader;
|
||||||
struct SNANDContent;
|
struct SNANDContent;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -133,10 +133,10 @@ private:
|
||||||
typedef std::map<u32, SContentAccess> CContentAccessMap;
|
typedef std::map<u32, SContentAccess> CContentAccessMap;
|
||||||
CContentAccessMap m_ContentAccessMap;
|
CContentAccessMap m_ContentAccessMap;
|
||||||
|
|
||||||
typedef std::map<u64, const DiscIO::INANDContentLoader*> CTitleToContentMap;
|
typedef std::map<u64, const DiscIO::CNANDContentLoader*> CTitleToContentMap;
|
||||||
CTitleToContentMap m_NANDContent;
|
CTitleToContentMap m_NANDContent;
|
||||||
|
|
||||||
const DiscIO::INANDContentLoader* m_pContentLoader;
|
const DiscIO::CNANDContentLoader* m_pContentLoader;
|
||||||
|
|
||||||
std::vector<u64> m_TitleIDs;
|
std::vector<u64> m_TitleIDs;
|
||||||
u64 m_TitleID;
|
u64 m_TitleID;
|
||||||
|
@ -144,7 +144,7 @@ private:
|
||||||
|
|
||||||
static u8 *keyTable[11];
|
static u8 *keyTable[11];
|
||||||
|
|
||||||
const DiscIO::INANDContentLoader& AccessContentDevice(u64 _TitleID);
|
const DiscIO::CNANDContentLoader& AccessContentDevice(u64 _TitleID);
|
||||||
u32 OpenTitleContent(u32 CFD, u64 TitleID, u16 Index);
|
u32 OpenTitleContent(u32 CFD, u64 TitleID, u16 Index);
|
||||||
|
|
||||||
bool IsValid(u64 _TitleID) const;
|
bool IsValid(u64 _TitleID) const;
|
||||||
|
|
|
@ -134,7 +134,7 @@ IPCCommandResult CWII_IPC_HLE_Device_net_kd_request::IOCtl(u32 _CommandAddress)
|
||||||
INFO_LOG(WII_IPC_WC24, "NET_KD_REQ: IOCTL_NWC24_REQUEST_GENERATED_USER_ID");
|
INFO_LOG(WII_IPC_WC24, "NET_KD_REQ: IOCTL_NWC24_REQUEST_GENERATED_USER_ID");
|
||||||
if (config.CreationStage() == nwc24_config_t::NWC24_IDCS_INITIAL)
|
if (config.CreationStage() == nwc24_config_t::NWC24_IDCS_INITIAL)
|
||||||
{
|
{
|
||||||
std::string settings_Filename(Common::GetTitleDataPath(TITLEID_SYSMENU) + WII_SETTING);
|
std::string settings_Filename(Common::GetTitleDataPath(TITLEID_SYSMENU, Common::FROM_SESSION_ROOT) + WII_SETTING);
|
||||||
SettingsHandler gen;
|
SettingsHandler gen;
|
||||||
std::string area, model;
|
std::string area, model;
|
||||||
bool _GotSettings = false;
|
bool _GotSettings = false;
|
||||||
|
|
|
@ -491,7 +491,7 @@ bool BeginRecordingInput(int controllers)
|
||||||
// TODO: find a way to GetTitleDataPath() from Movie::Init()
|
// TODO: find a way to GetTitleDataPath() from Movie::Init()
|
||||||
if (SConfig::GetInstance().bWii)
|
if (SConfig::GetInstance().bWii)
|
||||||
{
|
{
|
||||||
if (File::Exists(Common::GetTitleDataPath(g_titleID) + "banner.bin"))
|
if (File::Exists(Common::GetTitleDataPath(g_titleID, Common::FROM_SESSION_ROOT) + "banner.bin"))
|
||||||
Movie::g_bClearSave = false;
|
Movie::g_bClearSave = false;
|
||||||
else
|
else
|
||||||
Movie::g_bClearSave = true;
|
Movie::g_bClearSave = true;
|
||||||
|
|
|
@ -89,58 +89,6 @@ std::string CSharedContent::AddSharedContent(const u8* _pHash)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
// this classes must be created by the CNANDContentManager
|
|
||||||
class CNANDContentLoader : public INANDContentLoader
|
|
||||||
{
|
|
||||||
public:
|
|
||||||
CNANDContentLoader(const std::string& _rName);
|
|
||||||
virtual ~CNANDContentLoader();
|
|
||||||
|
|
||||||
bool IsValid() const override { return m_Valid; }
|
|
||||||
void RemoveTitle() const override;
|
|
||||||
u64 GetTitleID() const override { return m_TitleID; }
|
|
||||||
u16 GetIosVersion() const override { return m_IosVersion; }
|
|
||||||
u32 GetBootIndex() const override { return m_BootIndex; }
|
|
||||||
size_t GetContentSize() const override { return m_Content.size(); }
|
|
||||||
const SNANDContent* GetContentByIndex(int _Index) const override;
|
|
||||||
const u8* GetTMDView() const override { return m_TMDView; }
|
|
||||||
const u8* GetTMDHeader() const override { return m_TMDHeader; }
|
|
||||||
u32 GetTIKSize() const override { return m_TIKSize; }
|
|
||||||
const u8* GetTIK() const override { return m_TIK; }
|
|
||||||
|
|
||||||
const std::vector<SNANDContent>& GetContent() const override { return m_Content; }
|
|
||||||
|
|
||||||
u16 GetTitleVersion() const override {return m_TitleVersion;}
|
|
||||||
u16 GetNumEntries() const override {return m_numEntries;}
|
|
||||||
DiscIO::IVolume::ECountry GetCountry() const override;
|
|
||||||
u8 GetCountryChar() const override {return m_Country; }
|
|
||||||
|
|
||||||
private:
|
|
||||||
bool m_Valid;
|
|
||||||
bool m_isWAD;
|
|
||||||
std::string m_Path;
|
|
||||||
u64 m_TitleID;
|
|
||||||
u16 m_IosVersion;
|
|
||||||
u32 m_BootIndex;
|
|
||||||
u16 m_numEntries;
|
|
||||||
u16 m_TitleVersion;
|
|
||||||
u8 m_TMDView[TMD_VIEW_SIZE];
|
|
||||||
u8 m_TMDHeader[TMD_HEADER_SIZE];
|
|
||||||
u32 m_TIKSize;
|
|
||||||
u8* m_TIK;
|
|
||||||
u8 m_Country;
|
|
||||||
|
|
||||||
std::vector<SNANDContent> m_Content;
|
|
||||||
|
|
||||||
|
|
||||||
bool Initialize(const std::string& _rName);
|
|
||||||
|
|
||||||
void AESDecode(u8* _pKey, u8* _IV, u8* _pSrc, u32 _Size, u8* _pDest);
|
|
||||||
|
|
||||||
void GetKeyFromTicket(u8* pTicket, u8* pTicketKey);
|
|
||||||
};
|
|
||||||
|
|
||||||
|
|
||||||
CNANDContentLoader::CNANDContentLoader(const std::string& _rName)
|
CNANDContentLoader::CNANDContentLoader(const std::string& _rName)
|
||||||
: m_Valid(false)
|
: m_Valid(false)
|
||||||
, m_isWAD(false)
|
, m_isWAD(false)
|
||||||
|
@ -306,43 +254,34 @@ DiscIO::IVolume::ECountry CNANDContentLoader::GetCountry() const
|
||||||
|
|
||||||
CNANDContentManager::~CNANDContentManager()
|
CNANDContentManager::~CNANDContentManager()
|
||||||
{
|
{
|
||||||
for (auto& entry : m_Map)
|
|
||||||
{
|
|
||||||
delete entry.second;
|
|
||||||
}
|
|
||||||
m_Map.clear();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
const INANDContentLoader& CNANDContentManager::GetNANDLoader(const std::string& _rName, bool forceReload)
|
const CNANDContentLoader& CNANDContentManager::GetNANDLoader(const std::string& content_path)
|
||||||
{
|
{
|
||||||
CNANDContentMap::iterator lb = m_Map.lower_bound(_rName);
|
auto it = m_map.find(content_path);
|
||||||
|
if (it != m_map.end())
|
||||||
if (lb == m_Map.end() || (m_Map.key_comp()(_rName, lb->first)))
|
return *it->second;
|
||||||
{
|
return *m_map.emplace_hint(it, std::make_pair(content_path, std::make_unique<CNANDContentLoader>(content_path)))->second;
|
||||||
m_Map.insert(lb, CNANDContentMap::value_type(_rName, new CNANDContentLoader(_rName)));
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
if (!lb->second->IsValid() || forceReload)
|
|
||||||
{
|
|
||||||
delete lb->second;
|
|
||||||
lb->second = new CNANDContentLoader(_rName);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return *m_Map[_rName];
|
|
||||||
}
|
}
|
||||||
|
|
||||||
const INANDContentLoader& CNANDContentManager::GetNANDLoader(u64 _titleId, bool forceReload)
|
const CNANDContentLoader& CNANDContentManager::GetNANDLoader(u64 title_id, Common::FromWhichRoot from)
|
||||||
{
|
{
|
||||||
std::string _rName = Common::GetTitleContentPath(_titleId);
|
std::string path = Common::GetTitleContentPath(title_id, from);
|
||||||
return GetNANDLoader(_rName, forceReload);
|
return GetNANDLoader(path);
|
||||||
}
|
}
|
||||||
bool CNANDContentManager::RemoveTitle(u64 _titleID)
|
|
||||||
|
bool CNANDContentManager::RemoveTitle(u64 title_id, Common::FromWhichRoot from)
|
||||||
{
|
{
|
||||||
if (!GetNANDLoader(_titleID).IsValid())
|
auto& loader = GetNANDLoader(title_id, from);
|
||||||
|
if (!loader.IsValid())
|
||||||
return false;
|
return false;
|
||||||
GetNANDLoader(_titleID).RemoveTitle();
|
loader.RemoveTitle();
|
||||||
return GetNANDLoader(_titleID, true).IsValid();
|
return GetNANDLoader(title_id, from).IsValid();
|
||||||
|
}
|
||||||
|
|
||||||
|
void CNANDContentManager::ClearCache()
|
||||||
|
{
|
||||||
|
m_map.clear();
|
||||||
}
|
}
|
||||||
|
|
||||||
void CNANDContentLoader::RemoveTitle() const
|
void CNANDContentLoader::RemoveTitle() const
|
||||||
|
@ -355,11 +294,12 @@ void CNANDContentLoader::RemoveTitle() const
|
||||||
{
|
{
|
||||||
if (!(m_Content[i].m_Type & 0x8000)) // skip shared apps
|
if (!(m_Content[i].m_Type & 0x8000)) // skip shared apps
|
||||||
{
|
{
|
||||||
std::string filename = StringFromFormat("%s%08x.app", Common::GetTitleContentPath(m_TitleID).c_str(), m_Content[i].m_ContentID);
|
std::string filename = StringFromFormat("%s/%08x.app", m_Path.c_str(), m_Content[i].m_ContentID);
|
||||||
INFO_LOG(DISCIO, "Delete %s", filename.c_str());
|
INFO_LOG(DISCIO, "Delete %s", filename.c_str());
|
||||||
File::Delete(filename);
|
File::Delete(filename);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
CNANDContentManager::Access().ClearCache(); // deletes 'this'
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -434,8 +374,8 @@ void cUIDsys::GetTitleIDs(std::vector<u64>& _TitleIDs, bool _owned)
|
||||||
{
|
{
|
||||||
for (auto& Element : m_Elements)
|
for (auto& Element : m_Elements)
|
||||||
{
|
{
|
||||||
if ((_owned && Common::CheckTitleTIK(Common::swap64(Element.titleID))) ||
|
if ((_owned && Common::CheckTitleTIK(Common::swap64(Element.titleID), Common::FROM_SESSION_ROOT)) ||
|
||||||
(!_owned && Common::CheckTitleTMD(Common::swap64(Element.titleID))))
|
(!_owned && Common::CheckTitleTMD(Common::swap64(Element.titleID), Common::FROM_SESSION_ROOT)))
|
||||||
_TitleIDs.push_back(Common::swap64(Element.titleID));
|
_TitleIDs.push_back(Common::swap64(Element.titleID));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -444,7 +384,7 @@ u64 CNANDContentManager::Install_WiiWAD(const std::string& fileName)
|
||||||
{
|
{
|
||||||
if (fileName.find(".wad") == std::string::npos)
|
if (fileName.find(".wad") == std::string::npos)
|
||||||
return 0;
|
return 0;
|
||||||
const INANDContentLoader& ContentLoader = GetNANDLoader(fileName);
|
const CNANDContentLoader& ContentLoader = GetNANDLoader(fileName);
|
||||||
if (ContentLoader.IsValid() == false)
|
if (ContentLoader.IsValid() == false)
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
|
@ -452,8 +392,8 @@ u64 CNANDContentManager::Install_WiiWAD(const std::string& fileName)
|
||||||
|
|
||||||
//copy WAD's TMD header and contents to content directory
|
//copy WAD's TMD header and contents to content directory
|
||||||
|
|
||||||
std::string ContentPath(Common::GetTitleContentPath(TitleID));
|
std::string ContentPath(Common::GetTitleContentPath(TitleID, Common::FROM_CONFIGURED_ROOT));
|
||||||
std::string TMDFileName(Common::GetTMDFileName(TitleID));
|
std::string TMDFileName(Common::GetTMDFileName(TitleID, Common::FROM_CONFIGURED_ROOT));
|
||||||
File::CreateFullPath(TMDFileName);
|
File::CreateFullPath(TMDFileName);
|
||||||
|
|
||||||
File::IOFile pTMDFile(TMDFileName, "wb");
|
File::IOFile pTMDFile(TMDFileName, "wb");
|
||||||
|
@ -463,13 +403,13 @@ u64 CNANDContentManager::Install_WiiWAD(const std::string& fileName)
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
pTMDFile.WriteBytes(ContentLoader.GetTMDHeader(), INANDContentLoader::TMD_HEADER_SIZE);
|
pTMDFile.WriteBytes(ContentLoader.GetTMDHeader(), CNANDContentLoader::TMD_HEADER_SIZE);
|
||||||
|
|
||||||
for (u32 i = 0; i < ContentLoader.GetContentSize(); i++)
|
for (u32 i = 0; i < ContentLoader.GetContentSize(); i++)
|
||||||
{
|
{
|
||||||
const SNANDContent& Content = ContentLoader.GetContent()[i];
|
const SNANDContent& Content = ContentLoader.GetContent()[i];
|
||||||
|
|
||||||
pTMDFile.WriteBytes(Content.m_Header, INANDContentLoader::CONTENT_HEADER_SIZE);
|
pTMDFile.WriteBytes(Content.m_Header, CNANDContentLoader::CONTENT_HEADER_SIZE);
|
||||||
|
|
||||||
std::string APPFileName;
|
std::string APPFileName;
|
||||||
if (Content.m_Type & 0x8000) //shared
|
if (Content.m_Type & 0x8000) //shared
|
||||||
|
@ -504,12 +444,14 @@ u64 CNANDContentManager::Install_WiiWAD(const std::string& fileName)
|
||||||
|
|
||||||
cUIDsys::AccessInstance().AddTitle(TitleID);
|
cUIDsys::AccessInstance().AddTitle(TitleID);
|
||||||
|
|
||||||
|
ClearCache();
|
||||||
|
|
||||||
return TitleID;
|
return TitleID;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool Add_Ticket(u64 TitleID, const u8* p_tik, u32 tikSize)
|
bool Add_Ticket(u64 TitleID, const u8* p_tik, u32 tikSize)
|
||||||
{
|
{
|
||||||
std::string TicketFileName = Common::GetTicketFileName(TitleID);
|
std::string TicketFileName = Common::GetTicketFileName(TitleID, Common::FROM_CONFIGURED_ROOT);
|
||||||
File::CreateFullPath(TicketFileName);
|
File::CreateFullPath(TicketFileName);
|
||||||
File::IOFile pTicketFile(TicketFileName, "wb");
|
File::IOFile pTicketFile(TicketFileName, "wb");
|
||||||
if (!pTicketFile || !p_tik)
|
if (!pTicketFile || !p_tik)
|
||||||
|
|
|
@ -5,11 +5,12 @@
|
||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
#include <cstddef>
|
#include <cstddef>
|
||||||
#include <map>
|
|
||||||
#include <string>
|
#include <string>
|
||||||
|
#include <unordered_map>
|
||||||
#include <vector>
|
#include <vector>
|
||||||
|
|
||||||
#include "Common/CommonTypes.h"
|
#include "Common/CommonTypes.h"
|
||||||
|
#include "Common/NandPaths.h"
|
||||||
#include "DiscIO/Volume.h"
|
#include "DiscIO/Volume.h"
|
||||||
|
|
||||||
namespace DiscIO
|
namespace DiscIO
|
||||||
|
@ -28,31 +29,31 @@ struct SNANDContent
|
||||||
u8* m_pData;
|
u8* m_pData;
|
||||||
};
|
};
|
||||||
|
|
||||||
// pure virtual interface so just the NANDContentManager can create these files only
|
// Instances of this class must be created by CNANDContentManager
|
||||||
class INANDContentLoader
|
class CNANDContentLoader final
|
||||||
{
|
{
|
||||||
public:
|
public:
|
||||||
|
CNANDContentLoader(const std::string& _rName);
|
||||||
|
virtual ~CNANDContentLoader();
|
||||||
|
|
||||||
INANDContentLoader() {}
|
bool IsValid() const { return m_Valid; }
|
||||||
|
void RemoveTitle() const;
|
||||||
|
u64 GetTitleID() const { return m_TitleID; }
|
||||||
|
u16 GetIosVersion() const { return m_IosVersion; }
|
||||||
|
u32 GetBootIndex() const { return m_BootIndex; }
|
||||||
|
size_t GetContentSize() const { return m_Content.size(); }
|
||||||
|
const SNANDContent* GetContentByIndex(int _Index) const;
|
||||||
|
const u8* GetTMDView() const { return m_TMDView; }
|
||||||
|
const u8* GetTMDHeader() const { return m_TMDHeader; }
|
||||||
|
u32 GetTIKSize() const { return m_TIKSize; }
|
||||||
|
const u8* GetTIK() const { return m_TIK; }
|
||||||
|
|
||||||
virtual ~INANDContentLoader() {}
|
const std::vector<SNANDContent>& GetContent() const { return m_Content; }
|
||||||
|
|
||||||
virtual bool IsValid() const = 0;
|
u16 GetTitleVersion() const {return m_TitleVersion;}
|
||||||
virtual void RemoveTitle() const = 0;
|
u16 GetNumEntries() const {return m_numEntries;}
|
||||||
virtual u64 GetTitleID() const = 0;
|
DiscIO::IVolume::ECountry GetCountry() const;
|
||||||
virtual u16 GetIosVersion() const = 0;
|
u8 GetCountryChar() const {return m_Country; }
|
||||||
virtual u32 GetBootIndex() const = 0;
|
|
||||||
virtual size_t GetContentSize() const = 0;
|
|
||||||
virtual const SNANDContent* GetContentByIndex(int _Index) const = 0;
|
|
||||||
virtual const u8* GetTMDView() const = 0;
|
|
||||||
virtual const u8* GetTMDHeader() const = 0;
|
|
||||||
virtual u32 GetTIKSize() const = 0;
|
|
||||||
virtual const u8* GetTIK() const = 0;
|
|
||||||
virtual const std::vector<SNANDContent>& GetContent() const = 0;
|
|
||||||
virtual u16 GetTitleVersion() const = 0;
|
|
||||||
virtual u16 GetNumEntries() const = 0;
|
|
||||||
virtual DiscIO::IVolume::ECountry GetCountry() const = 0;
|
|
||||||
virtual u8 GetCountryChar() const = 0;
|
|
||||||
|
|
||||||
enum
|
enum
|
||||||
{
|
{
|
||||||
|
@ -61,6 +62,30 @@ public:
|
||||||
CONTENT_HEADER_SIZE = 0x24,
|
CONTENT_HEADER_SIZE = 0x24,
|
||||||
TICKET_SIZE = 0x2A4
|
TICKET_SIZE = 0x2A4
|
||||||
};
|
};
|
||||||
|
|
||||||
|
private:
|
||||||
|
bool m_Valid;
|
||||||
|
bool m_isWAD;
|
||||||
|
std::string m_Path;
|
||||||
|
u64 m_TitleID;
|
||||||
|
u16 m_IosVersion;
|
||||||
|
u32 m_BootIndex;
|
||||||
|
u16 m_numEntries;
|
||||||
|
u16 m_TitleVersion;
|
||||||
|
u8 m_TMDView[TMD_VIEW_SIZE];
|
||||||
|
u8 m_TMDHeader[TMD_HEADER_SIZE];
|
||||||
|
u32 m_TIKSize;
|
||||||
|
u8* m_TIK;
|
||||||
|
u8 m_Country;
|
||||||
|
|
||||||
|
std::vector<SNANDContent> m_Content;
|
||||||
|
|
||||||
|
|
||||||
|
bool Initialize(const std::string& _rName);
|
||||||
|
|
||||||
|
void AESDecode(u8* _pKey, u8* _IV, u8* _pSrc, u32 _Size, u8* _pDest);
|
||||||
|
|
||||||
|
void GetKeyFromTicket(u8* pTicket, u8* pTicketKey);
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
|
@ -71,9 +96,10 @@ public:
|
||||||
static CNANDContentManager& Access() { static CNANDContentManager instance; return instance; }
|
static CNANDContentManager& Access() { static CNANDContentManager instance; return instance; }
|
||||||
u64 Install_WiiWAD(const std::string& fileName);
|
u64 Install_WiiWAD(const std::string& fileName);
|
||||||
|
|
||||||
const INANDContentLoader& GetNANDLoader(const std::string& _rName, bool forceReload = false);
|
const CNANDContentLoader& GetNANDLoader(const std::string& content_path);
|
||||||
const INANDContentLoader& GetNANDLoader(u64 _titleId, bool forceReload = false);
|
const CNANDContentLoader& GetNANDLoader(u64 title_id, Common::FromWhichRoot from);
|
||||||
bool RemoveTitle(u64 _titleID);
|
bool RemoveTitle(u64 titl_id, Common::FromWhichRoot from);
|
||||||
|
void ClearCache();
|
||||||
|
|
||||||
private:
|
private:
|
||||||
CNANDContentManager() {}
|
CNANDContentManager() {}
|
||||||
|
@ -82,8 +108,7 @@ private:
|
||||||
CNANDContentManager(CNANDContentManager const&) = delete;
|
CNANDContentManager(CNANDContentManager const&) = delete;
|
||||||
void operator=(CNANDContentManager const&) = delete;
|
void operator=(CNANDContentManager const&) = delete;
|
||||||
|
|
||||||
typedef std::map<std::string, INANDContentLoader*> CNANDContentMap;
|
std::unordered_map<std::string, std::unique_ptr<CNANDContentLoader>> m_map;
|
||||||
CNANDContentMap m_Map;
|
|
||||||
};
|
};
|
||||||
|
|
||||||
class CSharedContent
|
class CSharedContent
|
||||||
|
|
|
@ -182,6 +182,7 @@ void PathConfigPane::OnNANDRootChanged(wxCommandEvent& event)
|
||||||
m_nand_root_dirpicker->SetPath(StrToWxStr(nand_path));
|
m_nand_root_dirpicker->SetPath(StrToWxStr(nand_path));
|
||||||
|
|
||||||
SConfig::GetInstance().m_SYSCONF->UpdateLocation();
|
SConfig::GetInstance().m_SYSCONF->UpdateLocation();
|
||||||
|
DiscIO::CNANDContentManager::Access().ClearCache();
|
||||||
|
|
||||||
main_frame->UpdateWiiMenuChoice();
|
main_frame->UpdateWiiMenuChoice();
|
||||||
}
|
}
|
||||||
|
|
|
@ -1471,7 +1471,7 @@ void CFrame::OnShowCheatsWindow(wxCommandEvent& WXUNUSED (event))
|
||||||
|
|
||||||
void CFrame::OnLoadWiiMenu(wxCommandEvent& WXUNUSED(event))
|
void CFrame::OnLoadWiiMenu(wxCommandEvent& WXUNUSED(event))
|
||||||
{
|
{
|
||||||
BootGame(Common::GetTitleContentPath(TITLEID_SYSMENU));
|
BootGame(Common::GetTitleContentPath(TITLEID_SYSMENU, Common::FROM_CONFIGURED_ROOT));
|
||||||
}
|
}
|
||||||
|
|
||||||
void CFrame::OnInstallWAD(wxCommandEvent& event)
|
void CFrame::OnInstallWAD(wxCommandEvent& event)
|
||||||
|
@ -1527,7 +1527,7 @@ void CFrame::UpdateWiiMenuChoice(wxMenuItem *WiiMenuItem)
|
||||||
WiiMenuItem = GetMenuBar()->FindItem(IDM_LOAD_WII_MENU);
|
WiiMenuItem = GetMenuBar()->FindItem(IDM_LOAD_WII_MENU);
|
||||||
}
|
}
|
||||||
|
|
||||||
const DiscIO::INANDContentLoader & SysMenu_Loader = DiscIO::CNANDContentManager::Access().GetNANDLoader(TITLEID_SYSMENU, true);
|
const DiscIO::CNANDContentLoader & SysMenu_Loader = DiscIO::CNANDContentManager::Access().GetNANDLoader(TITLEID_SYSMENU, Common::FROM_CONFIGURED_ROOT);
|
||||||
if (SysMenu_Loader.IsValid())
|
if (SysMenu_Loader.IsValid())
|
||||||
{
|
{
|
||||||
int sysmenuVersion = SysMenu_Loader.GetTitleVersion();
|
int sysmenuVersion = SysMenu_Loader.GetTitleVersion();
|
||||||
|
@ -1758,7 +1758,7 @@ void CFrame::UpdateGUI()
|
||||||
GetMenuBar()->FindItem(IDM_SAVE_STATE)->Enable(Initialized);
|
GetMenuBar()->FindItem(IDM_SAVE_STATE)->Enable(Initialized);
|
||||||
// Misc
|
// Misc
|
||||||
GetMenuBar()->FindItem(IDM_CHANGE_DISC)->Enable(Initialized);
|
GetMenuBar()->FindItem(IDM_CHANGE_DISC)->Enable(Initialized);
|
||||||
if (DiscIO::CNANDContentManager::Access().GetNANDLoader(TITLEID_SYSMENU).IsValid())
|
if (DiscIO::CNANDContentManager::Access().GetNANDLoader(TITLEID_SYSMENU, Common::FROM_CONFIGURED_ROOT).IsValid())
|
||||||
GetMenuBar()->FindItem(IDM_LOAD_WII_MENU)->Enable(!Initialized);
|
GetMenuBar()->FindItem(IDM_LOAD_WII_MENU)->Enable(!Initialized);
|
||||||
|
|
||||||
// Tools
|
// Tools
|
||||||
|
|
Loading…
Reference in New Issue