XPlor/libs/fastfile/fastfile.cpp

275 lines
8.8 KiB
C++
Raw Normal View History

2025-02-08 19:58:54 -05:00
#include "fastfile.h"
2025-02-19 19:17:31 -05:00
#include "fastfile_cod2.h"
#include "fastfile_cod5.h"
#include "fastfile_cod7.h"
#include "fastfile_cod9.h"
2025-02-08 19:58:54 -05:00
#include "compressor.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-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-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") {
2025-03-01 20:38:52 -05:00
LogManager::instance().addEntry("Company found: 'INFINITY_WARD'");
2025-02-19 19:17:31 -05:00
return COMPANY_INFINITY_WARD;
} else if (companyData == "TA") {
2025-03-01 20:38:52 -05:00
LogManager::instance().addEntry("Company found: 'TREYARCH'");
2025-02-19 19:17:31 -05:00
return COMPANY_TREYARCH;
} else if (companyData == "Sl") {
2025-03-01 20:38:52 -05:00
LogManager::instance().addEntry("Company found: 'SLEDGEHAMMER'");
2025-02-19 19:17:31 -05:00
return COMPANY_SLEDGEHAMMER;
} else if (companyData == "NX") {
2025-03-01 20:38:52 -05:00
LogManager::instance().addEntry("Company found: 'NEVERSOFT'");
2025-02-19 19:17:31 -05:00
return COMPANY_NEVERSOFT;
} else {
2025-03-01 20:38:52 -05:00
LogManager::instance().addEntry(QString("Failed to find company, found '%1'!").arg(companyData));
2025-02-08 19:58:54 -05:00
}
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") {
2025-03-01 20:38:52 -05:00
LogManager::instance().addEntry("Company found: 'INFINITY_WARD'");
2025-02-08 19:58:54 -05:00
return COMPANY_INFINITY_WARD;
} else if (companyData == "TA") {
2025-03-01 20:38:52 -05:00
LogManager::instance().addEntry("Company found: 'TREYARCH'");
2025-02-08 19:58:54 -05:00
return COMPANY_TREYARCH;
} else if (companyData == "Sl") {
2025-03-01 20:38:52 -05:00
LogManager::instance().addEntry("Company found: 'SLEDGEHAMMER'");
2025-02-08 19:58:54 -05:00
return COMPANY_SLEDGEHAMMER;
} else if (companyData == "NX") {
2025-03-01 20:38:52 -05:00
LogManager::instance().addEntry("Company found: 'NEVERSOFT'");
2025-02-08 19:58:54 -05:00
return COMPANY_NEVERSOFT;
} else {
2025-03-01 20:38:52 -05: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") {
qDebug() << "File type found: 'FAST_FILE'";
return FILETYPE_FAST_FILE;
} else {
qDebug() << "Failed to find file type!";
}
return FILETYPE_NONE;
}
FF_SIGNAGE FastFile::pParseFFSignage(QDataStream *afastFileStream) {
// Parse filetype
QByteArray signedData(1, Qt::Uninitialized);
afastFileStream->readRawData(signedData.data(), 1);
if (signedData == "u") {
qDebug() << "Found valid signage: Unsigned";
return SIGNAGE_UNSIGNED;
} else if (signedData == "0" || signedData == "x") {
qDebug() << "Found valid signage: Signed";
return SIGNAGE_SIGNED;
} else {
qDebug() << "Failed to determine signage of fastfile!";
}
return SIGNAGE_NONE;
}
QString FastFile::pParseFFMagic(QDataStream *afastFileStream) {
// Parse magic
QByteArray magicData(3, Qt::Uninitialized);
afastFileStream->readRawData(magicData.data(), 3);
if (magicData == "100") {
qDebug() << QString("Found valid magic: '%1'").arg(magicData);
return magicData;
} else {
qDebug() << "Magic invalid!";
}
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) {
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
case 473: // PC Black Ops 1
case 1: // PC Modern Warfare 3
case 147: // PC Black Ops 2
result = "PC";
break;
case 3640721408: // Xbox 360 Black Ops 1
case 2449473536: // Xbox 360 Black Ops 2
result = "360";
break;
}
qDebug() << QString("Found platform: '%1'").arg(result);
return result;
}
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
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);
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
FastFile *fastFile;
2025-03-01 20:38:52 -05:00
bool validff = true;
2025-02-19 19:17:31 -05:00
if (game == "COD2") {
fastFile = new FastFile_COD2();
} else if (game == "COD5") {
fastFile = new FastFile_COD5();
} else if (game == "COD7") {
fastFile = new FastFile_COD7();
} else if (game == "COD9") {
fastFile = new FastFile_COD9();
2025-03-01 20:38:52 -05:00
} else {
validff = false;
2025-02-19 19:17:31 -05:00
}
2025-03-01 20:38:52 -05:00
LogManager::instance().addLine();
if (validff) {
2025-02-19 19:17:31 -05:00
fastFile->SetCompany(company);
fastFile->SetStem(fastFileStem);
fastFile->Load(data);
return std::unique_ptr<FastFile>(fastFile);
}
2025-03-01 20:38:52 -05:00
2025-02-19 19:17:31 -05:00
// Open zone file after decompressing ff and writing
return nullptr;
}