pcsxr/macosx/Source/PcsxrMemCardArray.m

346 lines
8.6 KiB
Objective-C

//
// PcsxrMemCardArray.m
// Pcsxr
//
// Created by C.W. Betts on 7/6/13.
//
//
#import "PcsxrMemCardArray.h"
#import "ConfigurationController.h"
#include "sio.h"
#define MAX_MEMCARD_BLOCKS 15
static inline void CopyMemcardData(char *from, char *to, int srci, int dsti, char *str)
{
// header
memmove(to + (dsti + 1) * 128, from + (srci + 1) * 128, 128);
SaveMcd(str, to, (dsti + 1) * 128, 128);
// data
memmove(to + (dsti + 1) * 1024 * 8, from + (srci+1) * 1024 * 8, 1024 * 8);
SaveMcd(str, to, (dsti + 1) * 1024 * 8, 1024 * 8);
}
static inline char* BlankHeader()
{
struct PSXMemHeader {
unsigned int allocState;
unsigned int fileSize;
unsigned short nextBlock;
char fileName[21];
unsigned char garbage[96];
unsigned char checksum;
};
static struct PSXMemHeader *toReturn = NULL;
if (!toReturn) {
toReturn = calloc(sizeof(struct PSXMemHeader), 1);
//FIXME: Which value is right?
toReturn->allocState = 0x000000a0;
//toReturn->allocState = 0xa0000000;
toReturn->nextBlock = 0xFFFF;
unsigned char *bytePtr = (unsigned char*)toReturn;
for (int i = 0; i < sizeof(struct PSXMemHeader) - sizeof(unsigned char); i++) {
toReturn->checksum ^= bytePtr[i];
}
}
return (char*)toReturn;
}
static inline void ClearMemcardData(char *to, int dsti, char *str)
{
// header
char *header = BlankHeader();
memcpy(to + (dsti + 1) * 128, header, 128);
SaveMcd(str, to, (dsti + 1) * 128, 128);
// data
memset(to + (dsti + 1) * 1024 * 8, 0, 1024 * 8);
SaveMcd(str, to, (dsti + 1) * 1024 * 8, 1024 * 8);
}
@interface PcsxrMemCardArray ()
@property (strong) NSArray *rawArray;
@property (readonly) char* memDataPtr;
@property int cardNumber;
@end
@implementation PcsxrMemCardArray
@synthesize rawArray;
@synthesize cardNumber;
- (char*)memDataPtr
{
if (cardNumber == 1) {
return Mcd1Data;
} else {
return Mcd2Data;
}
}
- (const char *)memCardCPath
{
if (cardNumber == 1) {
return Config.Mcd1;
} else {
return Config.Mcd2;
}
}
- (instancetype)initWithMemoryCardNumber:(int)carNum
{
NSParameterAssert(carNum == 1 || carNum == 2);
if (self = [super init]) {
NSMutableArray *tmpMemArray = [[NSMutableArray alloc] initWithCapacity:MAX_MEMCARD_BLOCKS];
cardNumber = carNum;
int i = 0, x;
while (i < MAX_MEMCARD_BLOCKS) {
x = 1;
McdBlock memBlock;
GetMcdBlockInfo(carNum, i + 1, &memBlock);
if ([PcsxrMemoryObject memFlagsFromBlockFlags:memBlock.Flags] == PCSXRMemFlagFree) {
//Free space: ignore
i++;
continue;
}
while (i + x < MAX_MEMCARD_BLOCKS) {
McdBlock tmpBlock;
GetMcdBlockInfo(carNum, i + x + 1, &tmpBlock);
if ((tmpBlock.Flags & 0x3) == 0x3) {
x++;
break;
} else if ((tmpBlock.Flags & 0x2) == 0x2) {
x++;
} else {
break;
}
};
@autoreleasepool {
PcsxrMemoryObject *obj = [[PcsxrMemoryObject alloc] initWithMcdBlock:&memBlock startingIndex:i size:x];
[tmpMemArray addObject:obj];
}
i += x;
}
self.rawArray = [[NSArray alloc] initWithArray:tmpMemArray];
}
return self;
}
- (int)indexOfFreeBlocksWithSize:(int)asize
{
int foundcount = 0, i = 0;
McdBlock obj;
// search for empty (formatted) blocks first
while (i < MAX_MEMCARD_BLOCKS && foundcount < asize) {
GetMcdBlockInfo(cardNumber, 1 + i++, &obj);
//&Blocks[target_card][++i];
if ((obj.Flags & 0xFF) == 0xA0) { // if A0 but not A1
foundcount++;
} else if (foundcount >= 1) { // need to find n count consecutive blocks
foundcount = 0;
} else {
//i++;
}
//printf("formatstatus=%x\n", Info->Flags);
}
if (foundcount == asize)
return (i-foundcount);
// no free formatted slots, try to find a deleted one
foundcount = i = 0;
while (i < MAX_MEMCARD_BLOCKS && foundcount < asize) {
GetMcdBlockInfo(cardNumber, 1 + i++, &obj);
if ((obj.Flags & 0xF0) == 0xA0) { // A2 or A6 f.e.
foundcount++;
} else if (foundcount >= 1) { // need to find n count consecutive blocks
foundcount = 0;
} else {
//i++;
}
//printf("delstatus=%x\n", Info->Flags);
}
if (foundcount == asize)
return (i-foundcount);
return -1;
}
- (BOOL)moveBlockAtIndex:(int)idx toMemoryCard:(PcsxrMemCardArray*)otherCard
{
if (idx == [rawArray count]) {
#ifdef DEBUG
NSLog(@"Trying to get an object one more than the length of the raw array. Perhaps you were trying to \"move\" the free blocks. We don't want to do this.");
#endif
return NO;
}
PcsxrMemoryObject *tmpObj = rawArray[idx];
int memSize = tmpObj.blockSize;
if ([otherCard availableBlocks] < memSize) {
NSLog(@"Failing because the other card does not have enough space!");
return NO;
}
int toCopy = [otherCard indexOfFreeBlocksWithSize:memSize];
if (toCopy == -1) {
NSLog(@"Not enough consecutive blocks. Compacting the other card.");
[otherCard compactMemory];
//Since we're accessing the mem card data directly (instead of via PcsxrMemoryObject objects) using the following calls, we don't need to reload the data.
toCopy = [otherCard indexOfFreeBlocksWithSize:memSize];
NSAssert(toCopy != -1, @"Compacting the card should have made space!");
}
int memIdx = tmpObj.startingIndex;
for (int i = 0; i < memSize; i++) {
CopyMemcardData([self memDataPtr], [otherCard memDataPtr], memIdx + i, toCopy + i, (char*)otherCard.memCardCPath);
}
return YES;
}
- (int)freeBlocks
{
int memSize = MAX_MEMCARD_BLOCKS;
for (PcsxrMemoryObject *memObj in rawArray) {
memSize -= memObj.blockSize;
}
return memSize;
}
- (int)availableBlocks
{
int memSize = MAX_MEMCARD_BLOCKS;
for (PcsxrMemoryObject *memObj in rawArray) {
if (memObj.flag != PCSXRMemFlagDeleted) {
memSize -= memObj.blockSize;
}
}
return memSize;
}
- (NSArray*)memoryArray
{
int freeSize = [self freeBlocks];
if (freeSize) {
McdBlock theBlock;
//Create a blank "block" that will be used to show the amount of free blocks
theBlock.Flags = 0xA0;
theBlock.IconCount = 0;
PcsxrMemoryObject *freeObj = [[PcsxrMemoryObject alloc] initWithMcdBlock:&theBlock startingIndex:MAX_MEMCARD_BLOCKS - 1 - freeSize size:freeSize];
return [rawArray arrayByAddingObject:freeObj];
} else
return rawArray;
}
- (NSURL*)memCardURL
{
if (cardNumber == 1) {
return [[NSUserDefaults standardUserDefaults] URLForKey:@"Mcd1"];
} else {
return [[NSUserDefaults standardUserDefaults] URLForKey:@"Mcd2"];
}
}
- (int)memorySizeAtIndex:(int)idx
{
if (idx == [rawArray count]) {
#ifdef DEBUG
NSLog(@"Trying to get an object one more than the length of the raw array. Perhaps you were trying to \"count\" the free blocks?");
#endif
return [self freeBlocks];
}
return [rawArray[idx] blockSize];
}
- (void)compactMemory
{
int i = 0, x = 1;
while (i < MAX_MEMCARD_BLOCKS && x < MAX_MEMCARD_BLOCKS) {
x = i;
McdBlock baseBlock;
GetMcdBlockInfo(cardNumber, i+1, &baseBlock);
PCSXRMemFlag theFlags = [PcsxrMemoryObject memFlagsFromBlockFlags:baseBlock.Flags];
if (theFlags == PCSXRMemFlagDeleted || theFlags == PCSXRMemFlagFree) {
PCSXRMemFlag up1Flags = theFlags;
while ((up1Flags == PCSXRMemFlagDeleted || up1Flags == PCSXRMemFlagFree) && x < MAX_MEMCARD_BLOCKS) {
x++;
McdBlock up1Block;
GetMcdBlockInfo(cardNumber, x+1, &up1Block);
up1Flags = [PcsxrMemoryObject memFlagsFromBlockFlags:up1Block.Flags];
}
if (x >= MAX_MEMCARD_BLOCKS) {
break;
}
CopyMemcardData(self.memDataPtr, self.memDataPtr, x, i, (char*)[[self.memCardURL path] fileSystemRepresentation]);
ClearMemcardData(self.memDataPtr, x, (char*)self.memCardCPath);
}
i++;
}
while (i < MAX_MEMCARD_BLOCKS) {
ClearMemcardData(self.memDataPtr, i, (char*)self.memCardCPath);
i++;
}
LoadMcd(cardNumber, (char*)self.memCardCPath);
}
- (void)deleteMemoryBlocksAtIndex:(int)slotnum
{
char *data, *ptr, *filename;
if (cardNumber == 1) {
filename = Config.Mcd1;
data = Mcd1Data;
} else {
filename = Config.Mcd2;
data = Mcd2Data;
}
if (slotnum == [rawArray count]) {
#ifdef DEBUG
NSLog(@"Trying to get an object one more than the length of the raw array. Perhaps you were trying to \"delete\" the free blocks?");
#endif
return;
}
PcsxrMemoryObject *theObj = rawArray[slotnum];
McdBlock flagBlock;
for(int i = theObj.startingIndex + 1; i < (theObj.startingIndex + theObj.blockSize + 1); i++)
{
char xor = 0;
GetMcdBlockInfo(cardNumber, i, &flagBlock);
ptr = data + i * 128;
if ((flagBlock.Flags & 0xF0) == 0xA0) {
if ((flagBlock.Flags & 0xF) >= 1 &&
(flagBlock.Flags & 0xF) <= 3) { // deleted
*ptr = 0x50 | (flagBlock.Flags & 0xF);
} else return;
} else if ((flagBlock.Flags & 0xF0) == 0x50) { // used
*ptr = 0xA0 | (flagBlock.Flags & 0xF);
} else { continue; }
for (unsigned char j = 0; j < 127; j++) xor ^= *ptr++;
*ptr = xor;
SaveMcd(filename, data, i * 128, 128);
}
}
@end