2013-04-08 19:46:41 +00:00
|
|
|
// Copyright (c) 2013- PPSSPP Project.
|
|
|
|
|
|
|
|
// This program is free software: you can redistribute it and/or modify
|
|
|
|
// it under the terms of the GNU General Public License as published by
|
|
|
|
// the Free Software Foundation, version 2.0 or later versions.
|
|
|
|
|
|
|
|
// This program is distributed in the hope that it will be useful,
|
|
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
// GNU General Public License 2.0 for more details.
|
|
|
|
|
|
|
|
// A copy of the GPL 2.0 should have been included with the program.
|
|
|
|
// If not, see http://www.gnu.org/licenses/
|
|
|
|
|
|
|
|
// Official git repository and contact information can be found at
|
|
|
|
// https://github.com/hrydgard/ppsspp and http://www.ppsspp.org/.
|
|
|
|
|
|
|
|
#include <fstream>
|
|
|
|
#include <string>
|
2013-04-09 07:05:05 +00:00
|
|
|
#include <cstring>
|
2013-04-08 19:46:41 +00:00
|
|
|
|
|
|
|
#include "Common/Log.h"
|
2013-08-27 07:28:46 +00:00
|
|
|
#include "Common/FileUtil.h"
|
2013-04-08 19:46:41 +00:00
|
|
|
#include "Core/ELF/PBPReader.h"
|
|
|
|
|
2013-12-11 13:22:25 +00:00
|
|
|
PBPReader::PBPReader(const char *filename) : header_(), isELF_(false) {
|
2013-08-27 07:28:46 +00:00
|
|
|
file_ = File::OpenCFile(filename, "rb");
|
2013-04-08 19:46:41 +00:00
|
|
|
if (!file_) {
|
2013-12-11 13:22:25 +00:00
|
|
|
ERROR_LOG(LOADER, "Failed to open PBP file %s", filename);
|
2013-04-08 19:46:41 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
fseek(file_, 0, SEEK_END);
|
|
|
|
fileSize_ = ftell(file_);
|
|
|
|
fseek(file_, 0, SEEK_SET);
|
2015-01-19 16:31:41 +00:00
|
|
|
if (fread((char *)&header_, 1, sizeof(header_), file_) != sizeof(header_)) {
|
|
|
|
ERROR_LOG(LOADER, "PBP is too small to be valid: %s", filename);
|
|
|
|
fclose(file_);
|
|
|
|
file_ = nullptr;
|
|
|
|
return;
|
|
|
|
}
|
2013-04-08 19:46:41 +00:00
|
|
|
if (memcmp(header_.magic, "\0PBP", 4) != 0) {
|
2013-12-11 13:22:25 +00:00
|
|
|
if (memcmp(header_.magic, "\nFLE", 4) != 0) {
|
|
|
|
DEBUG_LOG(LOADER, "%s: File actually an ELF, not a PBP", filename);
|
|
|
|
isELF_ = true;
|
|
|
|
} else {
|
|
|
|
ERROR_LOG(LOADER, "Magic number in %s indicated no PBP: %s", filename, header_.magic);
|
|
|
|
}
|
2013-04-08 19:46:41 +00:00
|
|
|
fclose(file_);
|
2015-01-19 16:31:41 +00:00
|
|
|
file_ = nullptr;
|
2013-04-08 19:46:41 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2013-08-10 21:04:23 +00:00
|
|
|
DEBUG_LOG(LOADER, "Loading PBP, version = %08x", header_.version);
|
2013-04-08 19:46:41 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
u8 *PBPReader::GetSubFile(PBPSubFile file, size_t *outSize) {
|
2015-01-19 16:31:41 +00:00
|
|
|
if (!file_) {
|
|
|
|
*outSize = 0;
|
|
|
|
return new u8[0];
|
|
|
|
}
|
|
|
|
|
|
|
|
const size_t expected = GetSubFileSize(file);
|
|
|
|
const u32 off = header_.offsets[(int)file];
|
|
|
|
|
|
|
|
*outSize = expected;
|
|
|
|
if (fseek(file_, off, SEEK_SET) != 0) {
|
|
|
|
ERROR_LOG(LOADER, "PBP file offset invalid: %d", off);
|
|
|
|
*outSize = 0;
|
|
|
|
return new u8[0];
|
|
|
|
} else {
|
|
|
|
u8 *buffer = new u8[expected];
|
|
|
|
size_t bytes = fread(buffer, 1, expected, file_);
|
|
|
|
if (bytes != expected) {
|
|
|
|
ERROR_LOG(LOADER, "PBP file read truncated: %d -> %d", (int)expected, (int)bytes);
|
|
|
|
if (bytes < expected) {
|
|
|
|
*outSize = bytes;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return buffer;
|
|
|
|
}
|
2013-04-08 19:46:41 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void PBPReader::GetSubFileAsString(PBPSubFile file, std::string *out) {
|
2015-01-19 16:31:41 +00:00
|
|
|
if (!file_) {
|
|
|
|
out->clear();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
const size_t expected = GetSubFileSize(file);
|
|
|
|
const u32 off = header_.offsets[(int)file];
|
|
|
|
|
|
|
|
out->resize(expected);
|
|
|
|
if (fseek(file_, off, SEEK_SET) != 0) {
|
|
|
|
ERROR_LOG(LOADER, "PBP file offset invalid: %d", off);
|
|
|
|
out->clear();
|
|
|
|
} else {
|
|
|
|
size_t bytes = fread((void *)out->data(), 1, expected, file_);
|
|
|
|
if (bytes != expected) {
|
|
|
|
ERROR_LOG(LOADER, "PBP file read truncated: %d -> %d", (int)expected, (int)bytes);
|
|
|
|
if (bytes < expected) {
|
|
|
|
out->resize(bytes);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2013-04-08 19:46:41 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
PBPReader::~PBPReader() {
|
|
|
|
if (file_)
|
|
|
|
fclose(file_);
|
|
|
|
}
|