mirror of
https://github.com/RPCS3/rpcs3.git
synced 2025-04-28 13:28:01 +03:00

* Optimizations 1) Some headers simplified for better compilation time 2) Some templates simplified for smaller executable size 3) Eliminate std::future to fix compilation for mingw64 4) PKG installation can be cancelled now 5) cellGame fixes 6) XAudio2 fix for mingw64 7) PPUInterpreter bug fixed (Clang) * any_pod<> implemented Aliases: any16, any32, any64 rsx::make_command fixed
236 lines
5.4 KiB
C++
236 lines
5.4 KiB
C++
#include "stdafx.h"
|
|
#include "utils.h"
|
|
#include "aes.h"
|
|
#include "sha1.h"
|
|
#include "key_vault.h"
|
|
#include "unpkg.h"
|
|
|
|
bool pkg_install(const fs::file& pkg_f, const std::string& dir, atomic_t<double>& sync)
|
|
{
|
|
const std::size_t BUF_SIZE = 8192 * 1024; // 8 MB
|
|
|
|
// Save current file offset (probably zero)
|
|
const u64 start_offset = pkg_f.pos();
|
|
|
|
// Get basic PKG information
|
|
PKGHeader header;
|
|
|
|
if (!pkg_f.read(header))
|
|
{
|
|
LOG_ERROR(LOADER, "PKG file is too short!");
|
|
return false;
|
|
}
|
|
|
|
if (header.pkg_magic != "\x7FPKG"_u32)
|
|
{
|
|
LOG_ERROR(LOADER, "Not a PKG file!");
|
|
return false;
|
|
}
|
|
|
|
switch (const u16 type = header.pkg_type)
|
|
{
|
|
case PKG_RELEASE_TYPE_DEBUG: break;
|
|
case PKG_RELEASE_TYPE_RELEASE: break;
|
|
default:
|
|
{
|
|
LOG_ERROR(LOADER, "Unknown PKG type (0x%x)", type);
|
|
return false;
|
|
}
|
|
}
|
|
|
|
switch (const u16 platform = header.pkg_platform)
|
|
{
|
|
case PKG_PLATFORM_TYPE_PS3: break;
|
|
case PKG_PLATFORM_TYPE_PSP: break;
|
|
default:
|
|
{
|
|
LOG_ERROR(LOADER, "Unknown PKG platform (0x%x)", platform);
|
|
return false;
|
|
}
|
|
}
|
|
|
|
if (header.header_size != PKG_HEADER_SIZE && header.header_size != PKG_HEADER_SIZE2)
|
|
{
|
|
LOG_ERROR(LOADER, "Wrong PKG header size (0x%x)", header.header_size);
|
|
return false;
|
|
}
|
|
|
|
if (header.pkg_size > pkg_f.size())
|
|
{
|
|
LOG_ERROR(LOADER, "PKG file size mismatch (pkg_size=0x%llx)", header.pkg_size);
|
|
return false;
|
|
}
|
|
|
|
if (header.data_size + header.data_offset > header.pkg_size)
|
|
{
|
|
LOG_ERROR(LOADER, "PKG data size mismatch (data_size=0x%llx, data_offset=0x%llx, file_size=0x%llx)", header.data_size, header.data_offset, header.pkg_size);
|
|
return false;
|
|
}
|
|
|
|
// Allocate buffer with BUF_SIZE size or more if required
|
|
const std::unique_ptr<u128[]> buf(new u128[std::max<u64>(BUF_SIZE, sizeof(PKGEntry) * header.file_count) / sizeof(u128)]);
|
|
|
|
// Define decryption subfunction (`psp` arg selects the key for specific block)
|
|
auto decrypt = [&](u64 offset, u64 size, bool psp) -> u64
|
|
{
|
|
pkg_f.seek(start_offset + header.data_offset + offset);
|
|
|
|
// Read the data and set available size
|
|
const u64 read = pkg_f.read(buf.get(), size);
|
|
|
|
// Get block count
|
|
const u64 blocks = (read + 15) / 16;
|
|
|
|
if (header.pkg_type == PKG_RELEASE_TYPE_DEBUG)
|
|
{
|
|
// Debug key
|
|
be_t<u64> input[8] =
|
|
{
|
|
header.qa_digest[0],
|
|
header.qa_digest[0],
|
|
header.qa_digest[1],
|
|
header.qa_digest[1],
|
|
};
|
|
|
|
for (u64 i = 0; i < blocks; i++)
|
|
{
|
|
// Initialize "debug key" for current position
|
|
input[7] = offset / 16 + i;
|
|
|
|
u128 key;
|
|
|
|
sha1(reinterpret_cast<const u8*>(input), sizeof(input), reinterpret_cast<u8*>(&key));
|
|
|
|
buf[i] ^= key;
|
|
}
|
|
}
|
|
|
|
if (header.pkg_type == PKG_RELEASE_TYPE_RELEASE)
|
|
{
|
|
aes_context ctx;
|
|
|
|
// Set decryption key
|
|
aes_setkey_enc(&ctx, psp ? PKG_AES_KEY2 : PKG_AES_KEY, 128);
|
|
|
|
// Initialize "release key" for start position
|
|
be_t<u128> input = header.klicensee.value() + offset / 16;
|
|
|
|
// Increment "release key" for every block
|
|
for (u64 i = 0; i < blocks; i++, input++)
|
|
{
|
|
u128 key;
|
|
|
|
aes_crypt_ecb(&ctx, AES_ENCRYPT, reinterpret_cast<const u8*>(&input), reinterpret_cast<u8*>(&key));
|
|
|
|
buf[i] ^= key;
|
|
}
|
|
}
|
|
|
|
// Return the amount of data written in buf
|
|
return read;
|
|
};
|
|
|
|
decrypt(0, header.file_count * sizeof(PKGEntry), header.pkg_platform == PKG_PLATFORM_TYPE_PSP);
|
|
|
|
std::vector<PKGEntry> entries(header.file_count);
|
|
|
|
std::memcpy(entries.data(), buf.get(), entries.size() * sizeof(PKGEntry));
|
|
|
|
for (const auto& entry : entries)
|
|
{
|
|
const bool is_psp = (entry.type & PKG_FILE_ENTRY_PSP) != 0;
|
|
|
|
if (entry.name_size > 256)
|
|
{
|
|
LOG_ERROR(LOADER, "PKG name size is too big (0x%x)", entry.name_size);
|
|
continue;
|
|
}
|
|
|
|
decrypt(entry.name_offset, entry.name_size, is_psp);
|
|
|
|
const std::string name(reinterpret_cast<char*>(buf.get()), entry.name_size);
|
|
|
|
switch (entry.type & 0xff)
|
|
{
|
|
case PKG_FILE_ENTRY_NPDRM:
|
|
case PKG_FILE_ENTRY_NPDRMEDAT:
|
|
case PKG_FILE_ENTRY_SDAT:
|
|
case PKG_FILE_ENTRY_REGULAR:
|
|
case PKG_FILE_ENTRY_UNK1:
|
|
{
|
|
const std::string path = dir + name;
|
|
|
|
const bool did_overwrite = fs::is_file(path);
|
|
|
|
if (fs::file out{ path, fs::rewrite })
|
|
{
|
|
for (u64 pos = 0; pos < entry.file_size; pos += BUF_SIZE)
|
|
{
|
|
const u64 block_size = std::min<u64>(BUF_SIZE, entry.file_size - pos);
|
|
|
|
if (decrypt(entry.file_offset + pos, block_size, is_psp) != block_size)
|
|
{
|
|
LOG_ERROR(LOADER, "Failed to extract file %s", path);
|
|
break;
|
|
}
|
|
|
|
if (out.write(buf.get(), block_size) != block_size)
|
|
{
|
|
LOG_ERROR(LOADER, "Failed to write file %s", path);
|
|
break;
|
|
}
|
|
|
|
if (sync.fetch_add((block_size + 0.0) / header.data_size) < 0.)
|
|
{
|
|
LOG_ERROR(LOADER, "Package installation cancelled: %s", dir);
|
|
return false;
|
|
}
|
|
}
|
|
|
|
if (did_overwrite)
|
|
{
|
|
LOG_WARNING(LOADER, "Overwritten file %s", name);
|
|
}
|
|
else
|
|
{
|
|
LOG_NOTICE(LOADER, "Created file %s", name);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
LOG_ERROR(LOADER, "Failed to create file %s", path);
|
|
}
|
|
|
|
break;
|
|
}
|
|
|
|
case PKG_FILE_ENTRY_FOLDER:
|
|
{
|
|
const std::string path = dir + name;
|
|
|
|
if (fs::create_dir(path))
|
|
{
|
|
LOG_NOTICE(LOADER, "Created directory %s", name);
|
|
}
|
|
else if (fs::is_dir(path))
|
|
{
|
|
LOG_WARNING(LOADER, "Reused existing directory %s", name);
|
|
}
|
|
else
|
|
{
|
|
LOG_ERROR(LOADER, "Failed to create directory %s", path);
|
|
}
|
|
|
|
break;
|
|
}
|
|
|
|
default:
|
|
{
|
|
LOG_ERROR(LOADER, "Unknown PKG entry type (0x%x) %s", entry.type, name);
|
|
}
|
|
}
|
|
}
|
|
|
|
LOG_SUCCESS(LOADER, "Package successfully installed to %s", dir);
|
|
return true;
|
|
}
|