Mercurial > audlegacy-plugins
view src/sexypsf/PsxMem.c @ 3183:19e3ec80dac9
alsa-ng: Another arithmetic overflow (hopefully the last one).
author | John Lindgren <john.lindgren@tds.net> |
---|---|
date | Fri, 12 Jun 2009 17:15:10 -0400 |
parents | 8a4fbe599b05 |
children |
line wrap: on
line source
/* Pcsx - Pc Psx Emulator * Copyright (C) 1999-2002 Pcsx Team * * 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; either version 2 of the License, or * (at your option) any later version. * * 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 for more details. * * You should have received a copy of the GNU General Public License * along with this program; if not, write to the Free Software * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA */ #include <string.h> #include <stdlib.h> #include "PsxCommon.h" char *psxM; char *psxP; char *psxR; char *psxH; char **psxMemLUT; void LoadPSXMem(u32 address, s32 length, unsigned char *data) { //printf("%08x %08x\n",address,length); while(length>0) { if(address&65535) { u32 tmplen; //puts("Squishy"); tmplen=((65536-(address&65535))>(u32)length)?(u32)length:65536-(address&65535); if(psxMemLUT[address>>16]) memcpy((char *)(psxMemLUT[address>>16]+(address&65535)),data,tmplen); address+=tmplen; data+=tmplen; length-=tmplen; //printf("%08x %08x\n",address,tmplen); continue; } if(psxMemLUT[address>>16]) { memcpy((char *)(psxMemLUT[address>>16]),data,(length<65536)?length:65536); } data+=65536; address+=65536; length-=65536; } } static int writeok; int psxMemInit() { int i; writeok=1; psxMemLUT = malloc(0x10000 * sizeof *psxMemLUT); memset(psxMemLUT, 0, 0x10000 * sizeof *psxMemLUT); psxM = (char*)malloc(0x00200000); psxP = (char*)malloc(0x00010000); psxH = (char*)malloc(0x00010000); psxR = (char*)malloc(0x00080000); if (psxMemLUT == NULL || psxM == NULL || psxP == NULL || psxH == NULL || psxR == NULL) { printf("Error allocating memory"); return -1; } for (i=0; i<0x80; i++) psxMemLUT[i + 0x0000] = &psxM[(i & 0x1f) << 16]; memcpy(psxMemLUT + 0x8000, psxMemLUT, 0x80 * sizeof *psxMemLUT); memcpy(psxMemLUT + 0xa000, psxMemLUT, 0x80 * sizeof *psxMemLUT); for (i=0; i<0x01; i++) psxMemLUT[i + 0x1f00] = &psxP[i << 16]; for (i=0; i<0x01; i++) psxMemLUT[i + 0x1f80] = &psxH[i << 16]; for (i=0; i<0x08; i++) psxMemLUT[i + 0xbfc0] = &psxR[i << 16]; return 0; } void psxMemReset() { memset(psxM, 0, 0x00200000); memset(psxP, 0, 0x00010000); } void psxMemShutdown() { if (psxM) free(psxM); if (psxP) free(psxP); if (psxH) free(psxH); if (psxR) free(psxR); if (psxMemLUT) free(psxMemLUT); psxM = psxP = psxH = psxR = NULL; psxMemLUT = NULL; } u8 psxMemRead8(u32 mem) { char *p; u32 t; t = mem >> 16; if (t == 0x1f80) { if (mem < 0x1f801000) return psxHu8(mem); else return psxHwRead8(mem); } else { p = (char *)(psxMemLUT[t]); if (p != NULL) { return *(u8 *)(p + (mem & 0xffff)); } else { return 0; } } } u16 psxMemRead16(u32 mem) { char *p; u32 t; t = mem >> 16; if (t == 0x1f80) { if (mem < 0x1f801000) return BFLIP16(psxHu16(mem)); else return psxHwRead16(mem); } else { p = (char *)(psxMemLUT[t]); if (p != NULL) { return BFLIP16(*(u16 *)(p + (mem & 0xffff))); } else { return 0; } } } u32 psxMemRead32(u32 mem) { char *p; u32 t; t = mem >> 16; if (t == 0x1f80) { if (mem < 0x1f801000) return BFLIP32(psxHu32(mem)); else return psxHwRead32(mem); } else { p = (char *)(psxMemLUT[t]); if (p != NULL) { return BFLIP32(*(u32 *)(p + (mem & 0xffff))); } else { return 0; } } } void psxMemWrite8(u32 mem, u8 value) { char *p; u32 t; t = mem >> 16; if (t == 0x1f80) { if (mem < 0x1f801000) psxHu8(mem) = value; else psxHwWrite8(mem, value); } else { p = (char *)(psxMemLUT[t]); if (p != NULL) { *(u8 *)(p + (mem & 0xffff)) = value; } } } void psxMemWrite16(u32 mem, u16 value) { char *p; u32 t; t = mem >> 16; if (t == 0x1f80) { if (mem < 0x1f801000) psxHu16(mem) = BFLIP16(value); else psxHwWrite16(mem, value); } else { p = (char *)(psxMemLUT[t]); if (p != NULL) { *(u16 *)(p + (mem & 0xffff)) = BFLIP16(value); } } } void psxMemWrite32(u32 mem, u32 value) { char *p; u32 t; // if ((mem&0x1fffff) == 0x71E18 || value == 0x48088800) SysPrintf("t2fix!!\n"); t = mem >> 16; if (t == 0x1f80) { if (mem < 0x1f801000) psxHu32(mem) = BFLIP32(value); else psxHwWrite32(mem, value); } else { p = (char *)(psxMemLUT[t]); if (p != NULL) { *(u32 *)(p + (mem & 0xffff)) = BFLIP32(value); } else { if (mem != 0xfffe0130) { } else { int i; switch (value) { case 0x800: case 0x804: if (writeok == 0) break; writeok = 0; memset(psxMemLUT + 0x0000, 0, 0x80 * sizeof *psxMemLUT); memset(psxMemLUT + 0x8000, 0, 0x80 * sizeof *psxMemLUT); memset(psxMemLUT + 0xa000, 0, 0x80 * sizeof *psxMemLUT); break; case 0x1e988: if (writeok == 1) break; writeok = 1; for (i=0; i<0x80; i++) psxMemLUT[i + 0x0000] = &psxM[(i & 0x1f) << 16]; memcpy(psxMemLUT + 0x8000, psxMemLUT, 0x80 * sizeof *psxMemLUT); memcpy(psxMemLUT + 0xa000, psxMemLUT, 0x80 * sizeof *psxMemLUT); break; default: break; } } } } }