2013-04-18 03:09:55 +00:00
|
|
|
// Copyright 2013 Dolphin Emulator Project
|
|
|
|
// Licensed under GPLv2
|
|
|
|
// Refer to the license.txt file included.
|
2010-06-09 01:37:08 +00:00
|
|
|
|
2014-02-17 10:18:15 +00:00
|
|
|
#include <algorithm>
|
2014-02-21 00:47:53 +00:00
|
|
|
#include <cctype>
|
|
|
|
#include <cstring>
|
2010-06-09 01:37:08 +00:00
|
|
|
#include <string>
|
2014-03-04 13:39:25 +00:00
|
|
|
#include <unordered_set>
|
2014-02-17 10:18:15 +00:00
|
|
|
#include <vector>
|
2010-06-09 01:37:08 +00:00
|
|
|
|
2014-09-08 01:06:58 +00:00
|
|
|
#include "Common/CommonTypes.h"
|
2014-02-21 00:47:53 +00:00
|
|
|
#include "Common/StringUtil.h"
|
2014-06-05 23:29:54 +00:00
|
|
|
#include "Common/Logging/LogManager.h"
|
2010-06-09 01:37:08 +00:00
|
|
|
|
2014-02-17 10:18:15 +00:00
|
|
|
#include "Core/ConfigManager.h"
|
|
|
|
#include "Core/Core.h"
|
2014-02-21 00:47:53 +00:00
|
|
|
#include "Core/Boot/Boot.h"
|
2010-06-09 01:37:08 +00:00
|
|
|
|
2014-07-08 12:29:26 +00:00
|
|
|
#include "DiscIO/FileMonitor.h"
|
2014-02-21 00:47:53 +00:00
|
|
|
#include "DiscIO/Filesystem.h"
|
|
|
|
#include "DiscIO/Volume.h"
|
2014-02-17 10:18:15 +00:00
|
|
|
#include "DiscIO/VolumeCreator.h"
|
2010-06-09 01:37:08 +00:00
|
|
|
|
|
|
|
namespace FileMon
|
|
|
|
{
|
|
|
|
|
2014-07-08 13:58:25 +00:00
|
|
|
static DiscIO::IVolume *OpenISO = nullptr;
|
|
|
|
static DiscIO::IFileSystem *pFileSystem = nullptr;
|
|
|
|
static std::vector<const DiscIO::SFileInfo *> GCFiles;
|
|
|
|
static std::string ISOFile = "", CurrentFile = "";
|
|
|
|
static bool FileAccess = true;
|
2010-06-09 01:37:08 +00:00
|
|
|
|
|
|
|
// Filtered files
|
2014-03-04 13:39:25 +00:00
|
|
|
bool IsSoundFile(const std::string& filename)
|
2010-06-09 01:37:08 +00:00
|
|
|
{
|
2014-03-04 13:39:25 +00:00
|
|
|
std::string extension;
|
2014-03-09 20:14:26 +00:00
|
|
|
SplitPath(filename, nullptr, nullptr, &extension);
|
2014-03-04 13:39:25 +00:00
|
|
|
std::transform(extension.begin(), extension.end(), extension.begin(), ::tolower);
|
|
|
|
|
2014-08-25 23:56:03 +00:00
|
|
|
static std::unordered_set<std::string> extensions = {
|
2014-03-04 13:57:07 +00:00
|
|
|
".adp", // 1080 Avalanche, Crash Bandicoot, etc.
|
|
|
|
".adx", // Sonic Adventure 2 Battle, etc.
|
2014-03-04 13:39:25 +00:00
|
|
|
".afc", // Zelda WW
|
|
|
|
".ast", // Zelda TP, Mario Kart
|
2014-03-04 13:57:07 +00:00
|
|
|
".brstm", // Wii Sports, Wario Land, etc.
|
2014-03-04 13:39:25 +00:00
|
|
|
".dsp", // Metroid Prime
|
|
|
|
".hps", // SSB Melee
|
2014-03-04 13:57:07 +00:00
|
|
|
".ogg", // Tony Hawk's Underground 2
|
2014-03-09 16:46:01 +00:00
|
|
|
".sad", // Disaster
|
|
|
|
".snd", // Tales of Symphonia
|
|
|
|
".song", // Tales of Symphonia
|
|
|
|
".ssm", // Custom Robo, Kirby Air Ride, etc.
|
|
|
|
".str", // Harry Potter & the Sorcerer's Stone
|
2014-03-04 13:39:25 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
return extensions.find(extension) != extensions.end();
|
2010-06-09 01:37:08 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// Read the GC file system
|
2014-03-04 13:39:25 +00:00
|
|
|
void ReadGC(const std::string& filename)
|
2010-06-09 01:37:08 +00:00
|
|
|
{
|
|
|
|
// Should have an actual Shutdown procedure or something
|
2014-03-09 20:14:26 +00:00
|
|
|
if (OpenISO != nullptr)
|
2010-06-09 01:37:08 +00:00
|
|
|
{
|
|
|
|
delete OpenISO;
|
2014-03-09 20:14:26 +00:00
|
|
|
OpenISO = nullptr;
|
2010-06-09 01:37:08 +00:00
|
|
|
}
|
2014-03-09 20:14:26 +00:00
|
|
|
if (pFileSystem != nullptr)
|
2010-06-09 01:37:08 +00:00
|
|
|
{
|
|
|
|
delete pFileSystem;
|
2014-03-09 20:14:26 +00:00
|
|
|
pFileSystem = nullptr;
|
2010-06-09 01:37:08 +00:00
|
|
|
}
|
2013-04-17 03:14:36 +00:00
|
|
|
|
2010-06-09 01:37:08 +00:00
|
|
|
// GCFiles' pointers are no longer valid after pFileSystem is cleared
|
|
|
|
GCFiles.clear();
|
2014-03-04 13:39:25 +00:00
|
|
|
OpenISO = DiscIO::CreateVolumeFromFilename(filename);
|
|
|
|
if (!OpenISO)
|
|
|
|
return;
|
|
|
|
|
2014-12-25 10:01:18 +00:00
|
|
|
if (!DiscIO::IsVolumeWadFile(OpenISO))
|
2010-06-09 01:37:08 +00:00
|
|
|
{
|
|
|
|
pFileSystem = DiscIO::CreateFileSystem(OpenISO);
|
2014-03-04 13:39:25 +00:00
|
|
|
|
|
|
|
if (!pFileSystem)
|
|
|
|
return;
|
|
|
|
|
2010-06-09 01:37:08 +00:00
|
|
|
pFileSystem->GetFileList(GCFiles);
|
|
|
|
}
|
|
|
|
FileAccess = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check if we should play this file
|
2014-03-04 13:39:25 +00:00
|
|
|
void CheckFile(const std::string& file, u64 size)
|
2010-06-09 01:37:08 +00:00
|
|
|
{
|
|
|
|
// Don't do anything if the log is unselected
|
2011-04-01 07:43:02 +00:00
|
|
|
if (!LogManager::GetInstance()->IsEnabled(LogTypes::FILEMON))
|
|
|
|
return;
|
2010-06-09 01:37:08 +00:00
|
|
|
// Do nothing if we found the same file again
|
2014-03-04 13:39:25 +00:00
|
|
|
if (CurrentFile == file)
|
2011-04-01 07:43:02 +00:00
|
|
|
return;
|
2010-06-09 01:37:08 +00:00
|
|
|
|
2014-03-04 13:39:25 +00:00
|
|
|
if (size > 0)
|
|
|
|
size = (size / 1000);
|
2013-04-17 03:14:36 +00:00
|
|
|
|
2014-03-04 13:39:25 +00:00
|
|
|
std::string str = StringFromFormat("%s kB %s", ThousandSeparate(size, 7).c_str(), file.c_str());
|
|
|
|
if (IsSoundFile(file))
|
2010-06-09 01:37:08 +00:00
|
|
|
{
|
2014-03-04 13:39:25 +00:00
|
|
|
INFO_LOG(FILEMON, "%s", str.c_str());
|
2010-06-09 01:37:08 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2014-03-04 13:39:25 +00:00
|
|
|
WARN_LOG(FILEMON, "%s", str.c_str());
|
2010-06-09 01:37:08 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Update the current file
|
2014-03-04 13:39:25 +00:00
|
|
|
CurrentFile = file;
|
2010-06-09 01:37:08 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// Find the GC filename
|
|
|
|
void FindFilename(u64 offset)
|
|
|
|
{
|
|
|
|
// Don't do anything if a game is not running
|
2013-04-17 03:14:36 +00:00
|
|
|
if (Core::GetState() != Core::CORE_RUN)
|
|
|
|
return;
|
|
|
|
|
2010-06-09 01:37:08 +00:00
|
|
|
// Or if the log is unselected
|
2013-04-17 03:14:36 +00:00
|
|
|
if (!LogManager::GetInstance()->IsEnabled(LogTypes::FILEMON))
|
|
|
|
return;
|
|
|
|
|
|
|
|
// Or if we don't have file access
|
|
|
|
if (!FileAccess)
|
|
|
|
return;
|
2010-06-09 01:37:08 +00:00
|
|
|
|
|
|
|
if (!pFileSystem || ISOFile != SConfig::GetInstance().m_LastFilename)
|
|
|
|
{
|
|
|
|
FileAccess = false;
|
|
|
|
ReadGC(SConfig::GetInstance().m_LastFilename);
|
|
|
|
ISOFile = SConfig::GetInstance().m_LastFilename;
|
2011-02-02 18:21:20 +00:00
|
|
|
INFO_LOG(FILEMON, "Opening '%s'", ISOFile.c_str());
|
2010-06-09 01:37:08 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2014-03-15 03:38:14 +00:00
|
|
|
const std::string filename = pFileSystem->GetFileName(offset);
|
2014-03-29 10:05:44 +00:00
|
|
|
|
2014-03-24 03:29:30 +00:00
|
|
|
if (filename.empty())
|
|
|
|
return;
|
2010-06-09 01:37:08 +00:00
|
|
|
|
2014-03-15 03:38:14 +00:00
|
|
|
CheckFile(filename, pFileSystem->GetFileSize(filename));
|
2010-06-09 01:37:08 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void Close()
|
|
|
|
{
|
2014-03-09 20:14:26 +00:00
|
|
|
if (OpenISO != nullptr)
|
2010-06-09 01:37:08 +00:00
|
|
|
{
|
|
|
|
delete OpenISO;
|
2014-03-09 20:14:26 +00:00
|
|
|
OpenISO = nullptr;
|
2010-06-09 01:37:08 +00:00
|
|
|
}
|
|
|
|
|
2014-03-09 20:14:26 +00:00
|
|
|
if (pFileSystem != nullptr)
|
2010-06-09 01:37:08 +00:00
|
|
|
{
|
|
|
|
delete pFileSystem;
|
2014-03-09 20:14:26 +00:00
|
|
|
pFileSystem = nullptr;
|
2010-06-09 01:37:08 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// GCFiles' pointers are no longer valid after pFileSystem is cleared
|
|
|
|
GCFiles.clear();
|
|
|
|
|
|
|
|
ISOFile = "";
|
|
|
|
CurrentFile = "";
|
|
|
|
FileAccess = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
} // FileMon
|