2025-02-08 19:58:54 -05:00
|
|
|
#include "fastfile.h"
|
2025-02-19 19:17:31 -05:00
|
|
|
|
2025-05-03 09:57:47 -04:00
|
|
|
#include "fastfile_factory.h"
|
2025-03-01 20:38:52 -05:00
|
|
|
#include "logmanager.h"
|
2025-02-08 19:58:54 -05:00
|
|
|
|
|
|
|
|
#include <QFile>
|
|
|
|
|
#include <QDebug>
|
|
|
|
|
|
2025-02-19 19:17:31 -05:00
|
|
|
FastFile::FastFile()
|
|
|
|
|
: mStem(""),
|
|
|
|
|
mType(FILETYPE_NONE),
|
|
|
|
|
mCompany(COMPANY_NONE),
|
|
|
|
|
mSignage(SIGNAGE_NONE),
|
|
|
|
|
mMagic(""),
|
|
|
|
|
mVersion(0),
|
|
|
|
|
mZoneFile(nullptr),
|
|
|
|
|
mGame(""),
|
|
|
|
|
mPlatform("") {
|
2025-02-08 19:58:54 -05:00
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
2025-05-03 09:57:47 -04:00
|
|
|
FastFile::FastFile(const QByteArray &aData) {
|
|
|
|
|
Q_UNUSED(aData);
|
|
|
|
|
}
|
|
|
|
|
|
2025-02-19 19:17:31 -05:00
|
|
|
FastFile::FastFile(FastFile &fastFile)
|
|
|
|
|
: mStem(fastFile.GetStem()),
|
|
|
|
|
mType(fastFile.GetType()),
|
|
|
|
|
mCompany(fastFile.GetCompany()),
|
|
|
|
|
mSignage(fastFile.GetSignage()),
|
|
|
|
|
mMagic(fastFile.GetMagic()),
|
|
|
|
|
mVersion(fastFile.GetVersion()),
|
|
|
|
|
mZoneFile(fastFile.GetZoneFile()),
|
|
|
|
|
mGame(fastFile.GetGame()),
|
|
|
|
|
mPlatform(fastFile.GetPlatform()) {
|
2025-02-08 19:58:54 -05:00
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
2025-02-19 19:17:31 -05:00
|
|
|
FastFile::~FastFile() {
|
2025-02-08 19:58:54 -05:00
|
|
|
|
2025-02-19 19:17:31 -05:00
|
|
|
}
|
2025-02-08 19:58:54 -05:00
|
|
|
|
2025-04-04 20:40:37 -04:00
|
|
|
QString FastFile::GetStem() const {
|
|
|
|
|
return mStem;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
FF_FILETYPE FastFile::GetType() const {
|
|
|
|
|
return mType;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
FF_COMPANY FastFile::GetCompany() const {
|
|
|
|
|
return mCompany;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
FF_SIGNAGE FastFile::GetSignage() const {
|
|
|
|
|
return mSignage;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
QString FastFile::GetMagic() const {
|
|
|
|
|
return mMagic;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
quint32 FastFile::GetVersion() const {
|
|
|
|
|
return mVersion;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
std::shared_ptr<ZoneFile> FastFile::GetZoneFile() const {
|
|
|
|
|
return mZoneFile;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
QString FastFile::GetGame() const {
|
|
|
|
|
return mGame;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
QString FastFile::GetPlatform() const {
|
|
|
|
|
return mPlatform;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void FastFile::SetStem(const QString aStem) {
|
|
|
|
|
mStem = aStem;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void FastFile::SetType(const FF_FILETYPE aType) {
|
|
|
|
|
mType = aType;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void FastFile::SetCompany(const FF_COMPANY aCompany) {
|
|
|
|
|
mCompany = aCompany;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void FastFile::SetSignage(const FF_SIGNAGE aSignage) {
|
|
|
|
|
mSignage = aSignage;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void FastFile::SetMagic(const QString aMagic) {
|
|
|
|
|
mMagic = aMagic;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void FastFile::SetVersion(const quint32 aVersion) {
|
|
|
|
|
mVersion = aVersion;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void FastFile::SetZoneFile(const std::shared_ptr<ZoneFile> aZoneFile) {
|
|
|
|
|
mZoneFile = aZoneFile;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void FastFile::SetGame(const QString aGame) { mGame = aGame;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void FastFile::SetPlatform(const QString aPlatform) {
|
|
|
|
|
mPlatform = aPlatform;
|
|
|
|
|
}
|
|
|
|
|
|
2025-02-19 19:17:31 -05:00
|
|
|
FF_COMPANY FastFile::pParseFFCompany(QDataStream *afastFileStream, quint32 &aCompanyInt) {
|
2025-03-01 20:38:52 -05:00
|
|
|
LogManager::instance().addEntry("Parsing company into reference...");
|
2025-02-19 19:17:31 -05:00
|
|
|
// Check for null datastream ptr
|
|
|
|
|
if (!afastFileStream) { return COMPANY_NONE; }
|
|
|
|
|
// Parse company
|
|
|
|
|
QByteArray companyData(2, Qt::Uninitialized);
|
|
|
|
|
afastFileStream->readRawData(companyData.data(), 2);
|
|
|
|
|
aCompanyInt = companyData.toUInt();
|
2025-02-08 19:58:54 -05:00
|
|
|
|
2025-02-19 19:17:31 -05:00
|
|
|
if (companyData == "IW") {
|
|
|
|
|
return COMPANY_INFINITY_WARD;
|
|
|
|
|
} else if (companyData == "TA") {
|
|
|
|
|
return COMPANY_TREYARCH;
|
|
|
|
|
} else if (companyData == "Sl") {
|
|
|
|
|
return COMPANY_SLEDGEHAMMER;
|
|
|
|
|
} else if (companyData == "NX") {
|
|
|
|
|
return COMPANY_NEVERSOFT;
|
2025-02-08 19:58:54 -05:00
|
|
|
}
|
2025-04-04 20:40:37 -04:00
|
|
|
LogManager::instance().addEntry(QString("Failed to find company, found '%1'!").arg(companyData));
|
2025-02-19 19:17:31 -05:00
|
|
|
return COMPANY_NONE;
|
2025-02-08 19:58:54 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
FF_COMPANY FastFile::pParseFFCompany(QDataStream *afastFileStream) {
|
2025-03-01 20:38:52 -05:00
|
|
|
LogManager::instance().addEntry("Parsing company...");
|
2025-02-08 19:58:54 -05:00
|
|
|
// Check for null datastream ptr
|
|
|
|
|
if (!afastFileStream) { return COMPANY_NONE; }
|
|
|
|
|
// Parse company
|
|
|
|
|
QByteArray companyData(2, Qt::Uninitialized);
|
|
|
|
|
afastFileStream->readRawData(companyData.data(), 2);
|
2025-02-19 19:17:31 -05:00
|
|
|
|
2025-02-08 19:58:54 -05:00
|
|
|
if (companyData == "IW") {
|
|
|
|
|
return COMPANY_INFINITY_WARD;
|
|
|
|
|
} else if (companyData == "TA") {
|
|
|
|
|
return COMPANY_TREYARCH;
|
|
|
|
|
} else if (companyData == "Sl") {
|
|
|
|
|
} else if (companyData == "NX") {
|
|
|
|
|
return COMPANY_NEVERSOFT;
|
|
|
|
|
}
|
2025-04-04 20:40:37 -04:00
|
|
|
LogManager::instance().addEntry(QString("Failed to find company, found '%1'!").arg(companyData));
|
2025-02-08 19:58:54 -05:00
|
|
|
return COMPANY_NONE;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
FF_FILETYPE FastFile::pParseFFFileType(QDataStream *afastFileStream) {
|
|
|
|
|
// Parse filetype
|
|
|
|
|
QByteArray fileTypeData(2, Qt::Uninitialized);
|
|
|
|
|
afastFileStream->readRawData(fileTypeData.data(), 2);
|
|
|
|
|
if (fileTypeData == "ff") {
|
|
|
|
|
return FILETYPE_FAST_FILE;
|
|
|
|
|
}
|
2025-04-04 20:40:37 -04:00
|
|
|
LogManager::instance().addEntry("Failed to find file type!");
|
2025-02-08 19:58:54 -05:00
|
|
|
return FILETYPE_NONE;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
FF_SIGNAGE FastFile::pParseFFSignage(QDataStream *afastFileStream) {
|
|
|
|
|
// Parse filetype
|
|
|
|
|
QByteArray signedData(1, Qt::Uninitialized);
|
|
|
|
|
afastFileStream->readRawData(signedData.data(), 1);
|
|
|
|
|
if (signedData == "u") {
|
|
|
|
|
return SIGNAGE_UNSIGNED;
|
|
|
|
|
} else if (signedData == "0" || signedData == "x") {
|
|
|
|
|
return SIGNAGE_SIGNED;
|
|
|
|
|
}
|
2025-04-04 20:40:37 -04:00
|
|
|
LogManager::instance().addEntry("Failed to determine signage of fastfile!");
|
2025-02-08 19:58:54 -05:00
|
|
|
return SIGNAGE_NONE;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
QString FastFile::pParseFFMagic(QDataStream *afastFileStream) {
|
|
|
|
|
// Parse magic
|
|
|
|
|
QByteArray magicData(3, Qt::Uninitialized);
|
|
|
|
|
afastFileStream->readRawData(magicData.data(), 3);
|
|
|
|
|
if (magicData == "100") {
|
|
|
|
|
return magicData;
|
|
|
|
|
}
|
2025-04-04 20:40:37 -04:00
|
|
|
LogManager::instance().addEntry("Magic invalid!");
|
2025-02-08 19:58:54 -05:00
|
|
|
return "";
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
quint32 FastFile::pParseFFVersion(QDataStream *afastFileStream) {
|
|
|
|
|
// Parse version
|
|
|
|
|
quint32 version;
|
|
|
|
|
*afastFileStream >> version;
|
|
|
|
|
qDebug() << QString("Found version: '%1'").arg(version);
|
|
|
|
|
return version;
|
|
|
|
|
}
|
|
|
|
|
|
2025-02-19 19:17:31 -05:00
|
|
|
QString FastFile::pCalculateFFPlatform(quint32 aVersion) {
|
|
|
|
|
switch (aVersion) {
|
2025-02-08 19:58:54 -05:00
|
|
|
case 387: // PC World at War
|
|
|
|
|
case 473: // PC Black Ops 1
|
|
|
|
|
case 1: // PC Modern Warfare 3
|
|
|
|
|
case 147: // PC Black Ops 2
|
2025-04-04 20:40:37 -04:00
|
|
|
return "PC";
|
2025-02-08 19:58:54 -05:00
|
|
|
case 3640721408: // Xbox 360 Black Ops 1
|
|
|
|
|
case 2449473536: // Xbox 360 Black Ops 2
|
2025-04-04 20:40:37 -04:00
|
|
|
return "360";
|
|
|
|
|
case 3707830272: // Wii Black Ops 1
|
|
|
|
|
return "Wii";
|
2025-02-08 19:58:54 -05:00
|
|
|
}
|
2025-04-04 20:40:37 -04:00
|
|
|
return "NONE";
|
2025-02-08 19:58:54 -05:00
|
|
|
}
|
|
|
|
|
|
2025-02-19 19:17:31 -05:00
|
|
|
QString FastFile::pCalculateFFGame(quint32 aVersion) {
|
2025-02-08 19:58:54 -05:00
|
|
|
QString result = "NONE";
|
2025-02-19 19:17:31 -05:00
|
|
|
switch (aVersion) {
|
2025-02-08 19:58:54 -05:00
|
|
|
case 387: // PC World at War
|
|
|
|
|
result = "COD5";
|
|
|
|
|
break;
|
|
|
|
|
case 473: // PC Black Ops 1
|
|
|
|
|
break;
|
|
|
|
|
case 3640721408: // Xbox 360 Black Ops 1
|
2025-04-04 20:40:37 -04:00
|
|
|
case 3707830272: // Wii Black Ops 1
|
2025-02-08 19:58:54 -05:00
|
|
|
result = "COD7";
|
|
|
|
|
break;
|
|
|
|
|
case 1: // PC Modern Warfare 3
|
|
|
|
|
result = "COD8";
|
|
|
|
|
break;
|
|
|
|
|
case 147: // PC Black Ops 2
|
|
|
|
|
case 2449473536: // Xbox 360 Black Ops 2
|
|
|
|
|
result = "COD9";
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
qDebug() << QString("Found game: '%1'").arg(result);
|
|
|
|
|
return result;
|
|
|
|
|
}
|
2025-02-19 19:17:31 -05:00
|
|
|
|
2025-03-01 20:38:52 -05:00
|
|
|
std::shared_ptr<FastFile> FastFile::Open(const QString &aFilePath) {
|
|
|
|
|
LogManager::instance().addEntry("Processing Fastfile...");
|
|
|
|
|
|
2025-02-19 19:17:31 -05:00
|
|
|
if (aFilePath.isEmpty()) {
|
2025-03-01 20:38:52 -05:00
|
|
|
LogManager::instance().addError("Attempted to open file w/no name!");
|
2025-02-19 19:17:31 -05:00
|
|
|
return nullptr;
|
|
|
|
|
}
|
2025-03-01 20:38:52 -05:00
|
|
|
LogManager::instance().addEntry("File Path: " + aFilePath);
|
2025-02-19 19:17:31 -05:00
|
|
|
|
|
|
|
|
// Check fastfile can be read
|
|
|
|
|
QFile *file = new QFile(aFilePath);
|
|
|
|
|
if (!file->open(QIODevice::ReadOnly)) {
|
2025-03-01 20:38:52 -05:00
|
|
|
LogManager::instance().addError(QString("File failed to open: %1").arg(file->errorString()));
|
2025-02-19 19:17:31 -05:00
|
|
|
return nullptr;
|
|
|
|
|
}
|
2025-03-01 20:38:52 -05:00
|
|
|
LogManager::instance().addEntry("File opened");
|
2025-02-19 19:17:31 -05:00
|
|
|
|
|
|
|
|
const QByteArray data = file->readAll();
|
2025-03-01 20:38:52 -05:00
|
|
|
LogManager::instance().addEntry("Contents read in");
|
|
|
|
|
LogManager::instance().addEntry(QString("- Size: %1 B").arg(data.size()));
|
|
|
|
|
|
|
|
|
|
LogManager::instance().addEntry("File closed");
|
2025-02-19 19:17:31 -05:00
|
|
|
file->close();
|
|
|
|
|
|
|
|
|
|
// Create a QDataStream on the input data.
|
|
|
|
|
QDataStream fastFileStream(data);
|
|
|
|
|
fastFileStream.setByteOrder(QDataStream::LittleEndian);
|
|
|
|
|
|
|
|
|
|
quint32 companyInt;
|
|
|
|
|
FF_COMPANY company = pParseFFCompany(&fastFileStream, companyInt);
|
2025-04-04 20:40:37 -04:00
|
|
|
qDebug() << "Company: " << company;
|
2025-02-19 19:17:31 -05:00
|
|
|
FF_FILETYPE fileType;
|
|
|
|
|
FF_SIGNAGE signage;
|
|
|
|
|
QString magic;
|
|
|
|
|
quint32 version;
|
|
|
|
|
QString platform;
|
|
|
|
|
QString game;
|
|
|
|
|
|
|
|
|
|
if ((company == COMPANY_NONE) && (companyInt == 0)) {
|
|
|
|
|
company = COMPANY_INFINITY_WARD;
|
|
|
|
|
game = "COD2";
|
|
|
|
|
platform = "360";
|
|
|
|
|
} else {
|
|
|
|
|
fileType = pParseFFFileType(&fastFileStream);
|
|
|
|
|
signage = pParseFFSignage(&fastFileStream);
|
|
|
|
|
magic = pParseFFMagic(&fastFileStream);
|
|
|
|
|
version = pParseFFVersion(&fastFileStream);
|
|
|
|
|
platform = pCalculateFFPlatform(version);
|
|
|
|
|
game = pCalculateFFGame(version);
|
|
|
|
|
|
2025-03-01 20:38:52 -05:00
|
|
|
LogManager::instance().addEntry(QString("Type: %1").arg(fileType));
|
|
|
|
|
LogManager::instance().addEntry(QString("Signage: %1").arg(signage));
|
|
|
|
|
LogManager::instance().addEntry(QString("Magic: %1").arg(magic));
|
|
|
|
|
LogManager::instance().addEntry(QString("Version: %1").arg(version));
|
2025-02-19 19:17:31 -05:00
|
|
|
}
|
2025-03-01 20:38:52 -05:00
|
|
|
LogManager::instance().addEntry(QString("Company: %1").arg(company));
|
|
|
|
|
LogManager::instance().addEntry(QString("Game: %1").arg(game));
|
|
|
|
|
LogManager::instance().addEntry(QString("Platform: %1").arg(platform));
|
2025-02-19 19:17:31 -05:00
|
|
|
|
|
|
|
|
const QString fastFileStem = aFilePath.section("/", -1, -1).section('.', 0, 0);
|
2025-03-01 20:38:52 -05:00
|
|
|
LogManager::instance().addEntry(QString("Stem: %1").arg(fastFileStem));
|
2025-02-19 19:17:31 -05:00
|
|
|
|
2025-05-03 09:57:47 -04:00
|
|
|
std::shared_ptr<FastFile> fastFile = FastFileFactory::Create(data);
|
2025-02-19 19:17:31 -05:00
|
|
|
|
2025-05-03 09:57:47 -04:00
|
|
|
fastFile->SetCompany(company);
|
|
|
|
|
fastFile->SetStem(fastFileStem);
|
2025-03-01 20:38:52 -05:00
|
|
|
|
2025-05-03 09:57:47 -04:00
|
|
|
return fastFile;
|
2025-02-19 19:17:31 -05:00
|
|
|
}
|
2025-04-04 20:40:37 -04:00
|
|
|
bool FastFile::ExportFastFile(const QString aFastFilePath) {
|
|
|
|
|
QFile fastFile(aFastFilePath);
|
|
|
|
|
if (!fastFile.open(QIODevice::WriteOnly)) {
|
|
|
|
|
LogManager::instance().addEntry("Failed to write fast file! " +
|
|
|
|
|
fastFile.errorString());
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
fastFile.write(GetBinaryData());
|
|
|
|
|
fastFile.close();
|
|
|
|
|
|
|
|
|
|
return true;
|
|
|
|
|
}
|