2020-02-15 23:36:20 +01:00
|
|
|
|
#include "stdafx.h"
|
|
|
|
|
|
#include "Emu/VFS.h"
|
2014-02-15 01:06:12 +01:00
|
|
|
|
#include "TRP.h"
|
2017-04-13 19:29:47 +02:00
|
|
|
|
#include "Crypto/sha1.h"
|
2018-10-10 06:59:33 +02:00
|
|
|
|
#include "Utilities/StrUtil.h"
|
2014-02-15 01:06:12 +01:00
|
|
|
|
|
2020-02-02 12:09:20 +01:00
|
|
|
|
LOG_CHANNEL(trp_log, "Trophy");
|
2020-02-01 05:36:53 +01:00
|
|
|
|
|
2016-02-01 22:47:09 +01:00
|
|
|
|
TRPLoader::TRPLoader(const fs::file& f)
|
|
|
|
|
|
: trp_f(f)
|
2014-02-15 01:06:12 +01:00
|
|
|
|
{
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2016-02-01 22:47:09 +01:00
|
|
|
|
bool TRPLoader::Install(const std::string& dest, bool show)
|
2014-03-17 20:34:19 +01:00
|
|
|
|
{
|
2016-02-01 22:47:09 +01:00
|
|
|
|
if (!trp_f)
|
2015-07-26 10:14:56 +02:00
|
|
|
|
{
|
2014-03-17 20:34:19 +01:00
|
|
|
|
return false;
|
2015-07-26 10:14:56 +02:00
|
|
|
|
}
|
2014-02-16 02:51:04 +01:00
|
|
|
|
|
2016-02-01 22:47:09 +01:00
|
|
|
|
const std::string& local_path = vfs::get(dest);
|
2014-02-16 02:51:04 +01:00
|
|
|
|
|
2019-10-15 13:19:34 +02:00
|
|
|
|
if (!fs::is_dir(local_path) && !fs::create_dir(local_path))
|
2015-07-15 15:21:41 +02:00
|
|
|
|
{
|
2016-02-01 22:47:09 +01:00
|
|
|
|
return false;
|
2015-07-15 15:21:41 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
2017-09-02 13:43:44 +02:00
|
|
|
|
std::vector<char> buffer(65536);
|
2016-02-01 22:47:09 +01:00
|
|
|
|
|
2014-02-15 01:06:12 +01:00
|
|
|
|
for (const TRPEntry& entry : m_entries)
|
|
|
|
|
|
{
|
2016-02-01 22:47:09 +01:00
|
|
|
|
trp_f.seek(entry.offset);
|
|
|
|
|
|
buffer.resize(entry.size);
|
|
|
|
|
|
if (!trp_f.read(buffer)) continue; // ???
|
|
|
|
|
|
fs::file(local_path + '/' + entry.name, fs::rewrite).write(buffer);
|
2014-02-15 01:06:12 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
return true;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool TRPLoader::LoadHeader(bool show)
|
|
|
|
|
|
{
|
2016-02-01 22:47:09 +01:00
|
|
|
|
if (!trp_f)
|
2015-07-26 10:14:56 +02:00
|
|
|
|
{
|
2014-03-17 20:34:19 +01:00
|
|
|
|
return false;
|
2015-07-26 10:14:56 +02:00
|
|
|
|
}
|
2014-03-17 20:34:19 +01:00
|
|
|
|
|
2016-02-01 22:47:09 +01:00
|
|
|
|
trp_f.seek(0);
|
2015-07-26 10:14:56 +02:00
|
|
|
|
|
2016-02-01 22:47:09 +01:00
|
|
|
|
if (!trp_f.read(m_header))
|
2015-07-26 10:14:56 +02:00
|
|
|
|
{
|
2014-02-15 01:06:12 +01:00
|
|
|
|
return false;
|
2015-07-26 10:14:56 +02:00
|
|
|
|
}
|
2014-02-15 01:06:12 +01:00
|
|
|
|
|
2014-02-16 02:51:04 +01:00
|
|
|
|
if (m_header.trp_magic != 0xDCA24D00)
|
2015-07-26 10:14:56 +02:00
|
|
|
|
{
|
2014-02-15 01:06:12 +01:00
|
|
|
|
return false;
|
2015-07-26 10:14:56 +02:00
|
|
|
|
}
|
2014-02-15 01:06:12 +01:00
|
|
|
|
|
|
|
|
|
|
if (show)
|
2015-07-26 10:14:56 +02:00
|
|
|
|
{
|
2020-02-01 05:36:53 +01:00
|
|
|
|
trp_log.notice("TRP version: 0x%x", m_header.trp_version);
|
2015-07-26 10:14:56 +02:00
|
|
|
|
}
|
2014-02-15 01:06:12 +01:00
|
|
|
|
|
2017-04-13 19:29:47 +02:00
|
|
|
|
if (m_header.trp_version >= 2)
|
|
|
|
|
|
{
|
|
|
|
|
|
unsigned char hash[20];
|
|
|
|
|
|
std::vector<unsigned char> file_contents(m_header.trp_file_size);
|
|
|
|
|
|
|
|
|
|
|
|
trp_f.seek(0);
|
|
|
|
|
|
if (!trp_f.read(file_contents))
|
|
|
|
|
|
{
|
2020-02-01 05:36:53 +01:00
|
|
|
|
trp_log.notice("Failed verifying checksum");
|
2017-04-13 19:29:47 +02:00
|
|
|
|
}
|
|
|
|
|
|
else
|
|
|
|
|
|
{
|
|
|
|
|
|
memset(&(reinterpret_cast<TRPHeader*>(file_contents.data()))->sha1, 0, 20);
|
|
|
|
|
|
sha1(reinterpret_cast<const unsigned char*>(file_contents.data()), m_header.trp_file_size, hash);
|
|
|
|
|
|
|
|
|
|
|
|
if (memcmp(hash, m_header.sha1, 20) != 0)
|
|
|
|
|
|
{
|
2020-02-01 05:36:53 +01:00
|
|
|
|
trp_log.error("Invalid checksum of TROPHY.TRP file");
|
2017-04-13 19:29:47 +02:00
|
|
|
|
return false;
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
trp_f.seek(sizeof(m_header));
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2014-02-15 01:06:12 +01:00
|
|
|
|
m_entries.clear();
|
|
|
|
|
|
m_entries.resize(m_header.trp_files_count);
|
|
|
|
|
|
|
2015-07-26 10:14:56 +02:00
|
|
|
|
for (u32 i = 0; i < m_header.trp_files_count; i++)
|
2014-02-15 01:06:12 +01:00
|
|
|
|
{
|
2016-02-01 22:47:09 +01:00
|
|
|
|
if (!trp_f.read(m_entries[i]))
|
2015-07-26 10:14:56 +02:00
|
|
|
|
{
|
2014-02-15 01:06:12 +01:00
|
|
|
|
return false;
|
2015-07-26 10:14:56 +02:00
|
|
|
|
}
|
2014-02-15 01:06:12 +01:00
|
|
|
|
|
|
|
|
|
|
if (show)
|
2015-07-26 10:14:56 +02:00
|
|
|
|
{
|
2020-02-01 05:36:53 +01:00
|
|
|
|
trp_log.notice("TRP entry #%d: %s", m_entries[i].name);
|
2015-07-26 10:14:56 +02:00
|
|
|
|
}
|
2014-02-15 01:06:12 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
return true;
|
|
|
|
|
|
}
|
2014-03-17 20:34:19 +01:00
|
|
|
|
|
2017-11-20 14:08:35 +01:00
|
|
|
|
u64 TRPLoader::GetRequiredSpace() const
|
|
|
|
|
|
{
|
|
|
|
|
|
const u64 file_size = m_header.trp_file_size;
|
|
|
|
|
|
const u64 file_element_size = u64{1} * m_header.trp_files_count * m_header.trp_element_size;
|
|
|
|
|
|
|
|
|
|
|
|
return file_size - sizeof(m_header) - file_element_size;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2014-04-28 05:41:51 +02:00
|
|
|
|
bool TRPLoader::ContainsEntry(const char *filename)
|
2014-03-17 20:34:19 +01:00
|
|
|
|
{
|
2015-07-26 10:14:56 +02:00
|
|
|
|
for (const TRPEntry& entry : m_entries)
|
|
|
|
|
|
{
|
2014-03-17 20:34:19 +01:00
|
|
|
|
if (!strcmp(entry.name, filename))
|
2015-07-26 10:14:56 +02:00
|
|
|
|
{
|
2014-03-17 20:34:19 +01:00
|
|
|
|
return true;
|
2015-07-26 10:14:56 +02:00
|
|
|
|
}
|
2014-03-17 20:34:19 +01:00
|
|
|
|
}
|
|
|
|
|
|
return false;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2014-04-28 05:41:51 +02:00
|
|
|
|
void TRPLoader::RemoveEntry(const char *filename)
|
2014-03-17 20:34:19 +01:00
|
|
|
|
{
|
|
|
|
|
|
std::vector<TRPEntry>::iterator i = m_entries.begin();
|
2015-07-26 10:14:56 +02:00
|
|
|
|
while (i != m_entries.end())
|
|
|
|
|
|
{
|
2014-03-17 20:34:19 +01:00
|
|
|
|
if (!strcmp(i->name, filename))
|
2015-07-26 10:14:56 +02:00
|
|
|
|
{
|
2014-03-17 20:34:19 +01:00
|
|
|
|
i = m_entries.erase(i);
|
2015-07-26 10:14:56 +02:00
|
|
|
|
}
|
2014-03-17 20:34:19 +01:00
|
|
|
|
else
|
2015-07-26 10:14:56 +02:00
|
|
|
|
{
|
2014-03-17 20:34:19 +01:00
|
|
|
|
i++;
|
2015-07-26 10:14:56 +02:00
|
|
|
|
}
|
2014-03-17 20:34:19 +01:00
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2014-04-28 05:41:51 +02:00
|
|
|
|
void TRPLoader::RenameEntry(const char *oldname, const char *newname)
|
2014-03-17 20:34:19 +01:00
|
|
|
|
{
|
2018-10-10 06:59:33 +02:00
|
|
|
|
for (TRPEntry& entry : m_entries)
|
2015-07-26 10:14:56 +02:00
|
|
|
|
{
|
2014-03-17 20:34:19 +01:00
|
|
|
|
if (!strcmp(entry.name, oldname))
|
2015-07-26 10:14:56 +02:00
|
|
|
|
{
|
2018-10-10 06:59:33 +02:00
|
|
|
|
strcpy_trunc(entry.name, newname);
|
2015-07-26 10:14:56 +02:00
|
|
|
|
}
|
2014-03-17 20:34:19 +01:00
|
|
|
|
}
|
|
|
|
|
|
}
|