mirror of
https://github.com/pspdev/pspsdk.git
synced 2025-10-03 16:51:27 +00:00
206 lines
5.1 KiB
C
206 lines
5.1 KiB
C
/*
|
|
# _____ ___ ____ ___ ____
|
|
# ____| | ____| | | |____|
|
|
# | ___| | ___| ____| | \ PSPDEV Open Source Project.
|
|
#-----------------------------------------------------------------------
|
|
# (c) 2005 Dan Peori <peori@oopo.net>
|
|
# Licenced under Academic Free License version 2.0
|
|
# Review pspsdk README & LICENSE files for further details.
|
|
#
|
|
# 2006-12-26 - Andrew Whatson <whatson@gmail.com>
|
|
# - rewrote for easier reading
|
|
# - gave "correct" names to UNKNOWN.* files
|
|
# - improved memory efficiency with large PBPs
|
|
# - no longer outputs empty files
|
|
#
|
|
# $Id: unpack-pbp.c 2379 2008-04-08 21:18:58Z jim $
|
|
*/
|
|
|
|
#ifdef HAVE_CONFIG_H
|
|
#include "config.h"
|
|
#endif
|
|
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
|
|
#ifdef HAVE_MALLOC_H
|
|
#include <malloc.h>
|
|
#endif
|
|
|
|
#ifdef WORDS_BIGENDIAN
|
|
|
|
// Swap the bytes of an int for big-endian machines
|
|
static int swap_int(int n)
|
|
{
|
|
return ((n >> 24) & 0xff) | ((n >> 8) & 0xff00) | ((n << 8) & 0xff0000) | ((n << 24) & 0xff000000);
|
|
}
|
|
|
|
#endif
|
|
|
|
// Struct to describe the header of a PBP file
|
|
typedef struct {
|
|
char signature[4];
|
|
int version;
|
|
int offset[8];
|
|
} HEADER;
|
|
|
|
// Correct PBP signature
|
|
char correct_sig[4] = {
|
|
0x00,
|
|
0x50, // P
|
|
0x42, // B
|
|
0x50 // P
|
|
};
|
|
|
|
// Names of files included in a PBP
|
|
char *filename[8] = {
|
|
"PARAM.SFO",
|
|
"ICON0.PNG",
|
|
"ICON1.PMF",
|
|
"PIC0.PNG",
|
|
"PIC1.PNG",
|
|
"SND0.AT3",
|
|
"DATA.PSP",
|
|
"DATA.PSAR"
|
|
};
|
|
|
|
// Maximum size of read buffer
|
|
int maxbuffer = 16 * 1024 * 1024;
|
|
|
|
int main(int argc, char *argv[]) {
|
|
FILE *infile;
|
|
FILE *outfile;
|
|
HEADER header;
|
|
int loop0;
|
|
int total_size;
|
|
|
|
// Check for the correct number of arguments
|
|
if (argc != 2) {
|
|
printf("USAGE: %s <filename>\n", argv[0]);
|
|
return -1;
|
|
}
|
|
|
|
// Open the specified PBP
|
|
infile = fopen(argv[1], "rb");
|
|
if (infile == NULL) {
|
|
printf("ERROR: Could not open the input file. (%s)\n", argv[1]);
|
|
return -1;
|
|
}
|
|
|
|
// Get the size of the PBP
|
|
fseek(infile, 0, SEEK_END);
|
|
total_size = ftell(infile);
|
|
fseek(infile, 0, SEEK_SET);
|
|
if (total_size < 0) {
|
|
printf("ERROR: Could not get the input file size.\n");
|
|
return -1;
|
|
}
|
|
|
|
// Read in the header
|
|
if (fread(&header, sizeof(HEADER), 1, infile) < 0) {
|
|
printf("ERROR: Could not read the input file header.\n");
|
|
return -1;
|
|
}
|
|
|
|
// Check the signature
|
|
for (loop0 = 0; loop0 < sizeof(correct_sig); loop0++) {
|
|
if (header.signature[loop0] != correct_sig[loop0]) {
|
|
printf("ERROR: Input file is not a PBP file.\n");
|
|
return -1;
|
|
}
|
|
}
|
|
|
|
#ifdef WORDS_BIGENDIAN
|
|
|
|
// Swap the bytes of the offsets for big-endian machines
|
|
for (loop0 = 0; loop0 < 8; loop0++) {
|
|
header.offset[loop0] = swap_int(header.offset[loop0]);
|
|
}
|
|
|
|
#endif
|
|
|
|
// For each file in the PBP
|
|
for (loop0 = 0; loop0 < 8; loop0++) {
|
|
void *buffer;
|
|
int size;
|
|
|
|
// Get the size of this file
|
|
if (loop0 == 7) {
|
|
size = total_size - header.offset[loop0];
|
|
} else {
|
|
size = header.offset[loop0 + 1] - header.offset[loop0];
|
|
}
|
|
|
|
// Print out the file details
|
|
printf("[%d] %10d bytes | %s\n", loop0, size, filename[loop0]);
|
|
|
|
// Skip the file if empty
|
|
if (!size) continue;
|
|
|
|
// Seek to the proper position in the file
|
|
if (fseek(infile, header.offset[loop0], SEEK_SET) != 0) {
|
|
printf("ERROR: Could not seek in the input file.\n");
|
|
return -1;
|
|
}
|
|
|
|
// Open the output file
|
|
outfile = fopen(filename[loop0], "wb");
|
|
if (outfile == NULL) {
|
|
printf("ERROR: Could not open the output file. (%s)\n", filename[loop0]);
|
|
return -1;
|
|
}
|
|
|
|
do {
|
|
int readsize;
|
|
|
|
// Make sure we don't exceed the maximum buffer size
|
|
if (size > maxbuffer) {
|
|
readsize = maxbuffer;
|
|
} else {
|
|
readsize = size;
|
|
}
|
|
size -= readsize;
|
|
|
|
// Create the read buffer
|
|
buffer = malloc(readsize);
|
|
if (buffer == NULL) {
|
|
printf("ERROR: Could not allocate the section data buffer. (%d)\n", readsize);
|
|
return -1;
|
|
}
|
|
|
|
// Read in the data from the PBP
|
|
if (fread(buffer, readsize, 1, infile) < 0) {
|
|
printf("ERROR: Could not read in the section data.\n");
|
|
return -1;
|
|
}
|
|
|
|
// Write the contents of the buffer to the output file
|
|
if (fwrite(buffer, readsize, 1, outfile) < 0) {
|
|
printf("ERROR: Could not write out the section data.\n");
|
|
return -1;
|
|
}
|
|
|
|
// Clean up the buffer
|
|
free(buffer);
|
|
|
|
// Repeat if we haven't finished writing the file
|
|
} while (size);
|
|
|
|
// Close the output file
|
|
if (fclose(outfile) < 0) {
|
|
printf("ERROR: Could not close the output file.\n");
|
|
return -1;
|
|
}
|
|
|
|
}
|
|
|
|
// Close the PBP
|
|
if (fclose(infile) < 0) {
|
|
printf("ERROR: Could not close the input file.\n");
|
|
return -1;
|
|
}
|
|
|
|
// Exit successful
|
|
return 0;
|
|
}
|