INL-retro-progdump/host/source/file.c

81 lines
2.6 KiB
C

#include "file.h"
#define SIZE_NES_HEADER 16
#define SIZE_PRG_BANK 16384
#define SIZE_CHR_BANK 8192
//Need to pass in pointer to a filepointer to properly pass by reference
//the OS/stdio creates a FILE struct and returns the address of that struct
//so when this function opens a file it's setting the value of a pointer
//for the calling function. To set a pointer we must have a pointer to that pointer..
int open_file( FILE **fptr, char *filename )
{
//first open file
*fptr = fopen( filename, "rb");
//returns file ptr on success, NULL on fail
check( *fptr, "Unable to open file: %s in read binary mode", filename);
return SUCCESS;
error:
return -1;
}
int detect_file( rom_image *rom )
{
int rv = 0;
uint8_t header[SIZE_NES_HEADER];
//size_t fread(void *ptr, size_t size, size_t nmemb, FILE *stream);
rv = fread( header, sizeof(header[0]), (sizeof(header)/sizeof(header[0])), rom->fileptr);
check( rv = sizeof(header), "Unable to read NES header");
//0-3: Constant $4E $45 $53 $1A ("NES" followed by MS-DOS end-of-file)
if ( (header[0]=='N') && (header[1]=='E') && (header[2]=='S') && (header[3]==0x1A) ) {
debug("detected ines header");
rom->console = NES_CART;
} else {
debug("only ines files currently accepted as input");
goto error;
}
//4: Size of PRG ROM in 16 KB units
rom->prg_size = header[4] * SIZE_PRG_BANK;
debug("PRG ROM size= %d", rom->prg_size);
//5: Size of CHR ROM in 8 KB units (Value 0 means the board uses CHR RAM)
rom->chr_size = header[5] * SIZE_CHR_BANK;
debug("CHR ROM size= %d", rom->chr_size);
//6: Flags 6
// 76543210
// ||||||||
// ||||+||+- 0xx0: vertical arrangement/horizontal mirroring (CIRAM A10 = PPU A11)
// |||| || 0xx1: horizontal arrangement/vertical mirroring (CIRAM A10 = PPU A10)
// |||| || 1xxx: four-screen VRAM
// |||| |+-- 1: Cartridge contains battery-backed PRG RAM ($6000-7FFF) or other persistent memory
// |||| +--- 1: 512-byte trainer at $7000-$71FF (stored before PRG data)
// ++++----- Lower nybble of mapper number
rom->mapper = (header[6]>>4);
rom->mapper |= (header[7] & 0xF0);
debug("mapper #%d", rom->mapper);
rom->mirroring = header[6] & 0x09; //0b0000 1001
debug("mirroring:%x", rom->mirroring);
//7: Flags 7
// 76543210
// ||||||||
// |||||||+- VS Unisystem
// ||||||+-- PlayChoice-10 (8KB of Hint Screen data stored after CHR data)
// ||||++--- If equal to 2, flags 8-15 are in NES 2.0 format
// ++++----- Upper nybble of mapper number
//8: Size of PRG RAM in 8 KB units (Value 0 infers 8 KB for compatibility; see PRG RAM circuit)
//9: Flags 9
//10: Flags 10 (unofficial)
//11-15: Zero filled
return SUCCESS;
error:
return -1;
}