home *** CD-ROM | disk | FTP | other *** search
- // VisualBoyAdvance - Nintendo Gameboy/GameboyAdvance (TM) emulator.
- // Copyright (C) 1999-2003 Forgotten
- // Copyright (C) 2004 Forgotten and the VBA development 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, 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., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
-
- #include <math.h>
- #include <memory.h>
- #include <stdlib.h>
-
- #include "GBA.h"
- #include "bios.h"
- #include "GBAinline.h"
- #include "Globals.h"
-
- s16 sineTable[256] = {
- (s16)0x0000, (s16)0x0192, (s16)0x0323, (s16)0x04B5, (s16)0x0645, (s16)0x07D5, (s16)0x0964, (s16)0x0AF1,
- (s16)0x0C7C, (s16)0x0E05, (s16)0x0F8C, (s16)0x1111, (s16)0x1294, (s16)0x1413, (s16)0x158F, (s16)0x1708,
- (s16)0x187D, (s16)0x19EF, (s16)0x1B5D, (s16)0x1CC6, (s16)0x1E2B, (s16)0x1F8B, (s16)0x20E7, (s16)0x223D,
- (s16)0x238E, (s16)0x24DA, (s16)0x261F, (s16)0x275F, (s16)0x2899, (s16)0x29CD, (s16)0x2AFA, (s16)0x2C21,
- (s16)0x2D41, (s16)0x2E5A, (s16)0x2F6B, (s16)0x3076, (s16)0x3179, (s16)0x3274, (s16)0x3367, (s16)0x3453,
- (s16)0x3536, (s16)0x3612, (s16)0x36E5, (s16)0x37AF, (s16)0x3871, (s16)0x392A, (s16)0x39DA, (s16)0x3A82,
- (s16)0x3B20, (s16)0x3BB6, (s16)0x3C42, (s16)0x3CC5, (s16)0x3D3E, (s16)0x3DAE, (s16)0x3E14, (s16)0x3E71,
- (s16)0x3EC5, (s16)0x3F0E, (s16)0x3F4E, (s16)0x3F84, (s16)0x3FB1, (s16)0x3FD3, (s16)0x3FEC, (s16)0x3FFB,
- (s16)0x4000, (s16)0x3FFB, (s16)0x3FEC, (s16)0x3FD3, (s16)0x3FB1, (s16)0x3F84, (s16)0x3F4E, (s16)0x3F0E,
- (s16)0x3EC5, (s16)0x3E71, (s16)0x3E14, (s16)0x3DAE, (s16)0x3D3E, (s16)0x3CC5, (s16)0x3C42, (s16)0x3BB6,
- (s16)0x3B20, (s16)0x3A82, (s16)0x39DA, (s16)0x392A, (s16)0x3871, (s16)0x37AF, (s16)0x36E5, (s16)0x3612,
- (s16)0x3536, (s16)0x3453, (s16)0x3367, (s16)0x3274, (s16)0x3179, (s16)0x3076, (s16)0x2F6B, (s16)0x2E5A,
- (s16)0x2D41, (s16)0x2C21, (s16)0x2AFA, (s16)0x29CD, (s16)0x2899, (s16)0x275F, (s16)0x261F, (s16)0x24DA,
- (s16)0x238E, (s16)0x223D, (s16)0x20E7, (s16)0x1F8B, (s16)0x1E2B, (s16)0x1CC6, (s16)0x1B5D, (s16)0x19EF,
- (s16)0x187D, (s16)0x1708, (s16)0x158F, (s16)0x1413, (s16)0x1294, (s16)0x1111, (s16)0x0F8C, (s16)0x0E05,
- (s16)0x0C7C, (s16)0x0AF1, (s16)0x0964, (s16)0x07D5, (s16)0x0645, (s16)0x04B5, (s16)0x0323, (s16)0x0192,
- (s16)0x0000, (s16)0xFE6E, (s16)0xFCDD, (s16)0xFB4B, (s16)0xF9BB, (s16)0xF82B, (s16)0xF69C, (s16)0xF50F,
- (s16)0xF384, (s16)0xF1FB, (s16)0xF074, (s16)0xEEEF, (s16)0xED6C, (s16)0xEBED, (s16)0xEA71, (s16)0xE8F8,
- (s16)0xE783, (s16)0xE611, (s16)0xE4A3, (s16)0xE33A, (s16)0xE1D5, (s16)0xE075, (s16)0xDF19, (s16)0xDDC3,
- (s16)0xDC72, (s16)0xDB26, (s16)0xD9E1, (s16)0xD8A1, (s16)0xD767, (s16)0xD633, (s16)0xD506, (s16)0xD3DF,
- (s16)0xD2BF, (s16)0xD1A6, (s16)0xD095, (s16)0xCF8A, (s16)0xCE87, (s16)0xCD8C, (s16)0xCC99, (s16)0xCBAD,
- (s16)0xCACA, (s16)0xC9EE, (s16)0xC91B, (s16)0xC851, (s16)0xC78F, (s16)0xC6D6, (s16)0xC626, (s16)0xC57E,
- (s16)0xC4E0, (s16)0xC44A, (s16)0xC3BE, (s16)0xC33B, (s16)0xC2C2, (s16)0xC252, (s16)0xC1EC, (s16)0xC18F,
- (s16)0xC13B, (s16)0xC0F2, (s16)0xC0B2, (s16)0xC07C, (s16)0xC04F, (s16)0xC02D, (s16)0xC014, (s16)0xC005,
- (s16)0xC000, (s16)0xC005, (s16)0xC014, (s16)0xC02D, (s16)0xC04F, (s16)0xC07C, (s16)0xC0B2, (s16)0xC0F2,
- (s16)0xC13B, (s16)0xC18F, (s16)0xC1EC, (s16)0xC252, (s16)0xC2C2, (s16)0xC33B, (s16)0xC3BE, (s16)0xC44A,
- (s16)0xC4E0, (s16)0xC57E, (s16)0xC626, (s16)0xC6D6, (s16)0xC78F, (s16)0xC851, (s16)0xC91B, (s16)0xC9EE,
- (s16)0xCACA, (s16)0xCBAD, (s16)0xCC99, (s16)0xCD8C, (s16)0xCE87, (s16)0xCF8A, (s16)0xD095, (s16)0xD1A6,
- (s16)0xD2BF, (s16)0xD3DF, (s16)0xD506, (s16)0xD633, (s16)0xD767, (s16)0xD8A1, (s16)0xD9E1, (s16)0xDB26,
- (s16)0xDC72, (s16)0xDDC3, (s16)0xDF19, (s16)0xE075, (s16)0xE1D5, (s16)0xE33A, (s16)0xE4A3, (s16)0xE611,
- (s16)0xE783, (s16)0xE8F8, (s16)0xEA71, (s16)0xEBED, (s16)0xED6C, (s16)0xEEEF, (s16)0xF074, (s16)0xF1FB,
- (s16)0xF384, (s16)0xF50F, (s16)0xF69C, (s16)0xF82B, (s16)0xF9BB, (s16)0xFB4B, (s16)0xFCDD, (s16)0xFE6E
- };
-
- void BIOS_ArcTan()
- {
- #ifdef DEV_VERSION
- if(systemVerbose & VERBOSE_SWI) {
- log("ArcTan: %08x (VCOUNT=%2d)\n",
- reg[0].I,
- VCOUNT);
- }
- #endif
-
- s32 a = -((s32)(reg[0].I * reg[0].I)) >> 14;
- s32 b = ((0xA9 * a) >> 14) + 0x390;
- b = ((b * a) >> 14) + 0x91C;
- b = ((b * a) >> 14) + 0xFB6;
- b = ((b * a) >> 14) + 0x16AA;
- b = ((b * a) >> 14) + 0x2081;
- b = ((b * a) >> 14) + 0x3651;
- b = ((b * a) >> 14) + 0xA2F9;
- reg[0].I = (reg[0].I * b) >> 16;
-
- #ifdef DEV_VERSION
- if(systemVerbose & VERBOSE_SWI) {
- log("ArcTan: return=%08x\n",
- reg[0].I);
- }
- #endif
- }
-
- void BIOS_ArcTan2()
- {
- #ifdef DEV_VERSION
- if(systemVerbose & VERBOSE_SWI) {
- log("ArcTan2: %08x,%08x (VCOUNT=%2d)\n",
- reg[0].I,
- reg[1].I,
- VCOUNT);
- }
- #endif
-
- s16 x = reg[0].I;
- s16 y = reg[1].I;
-
- if (y == 0) {
- reg[0].I = 0x8000 & x;
- reg[3].I = 0x170;
- } else {
- if (x == 0) {
- reg[0].I = (0x8000 & y) + 0x4000;
- reg[3].I = 0x170;
- } else {
- if (abs(x) > abs(y)) {
- reg[1].I = x;
- reg[0].I = y << 14;
- BIOS_Div();
- BIOS_ArcTan();
- if (x < 0)
- reg[0].I = 0x8000 + reg[0].I;
- else
- reg[0].I = ((y & 0x8000) << 1 ) + reg[0].I;
- reg[3].I = 0x170;
- } else {
- reg[0].I = x << 14;
- BIOS_Div();
- BIOS_ArcTan();
- reg[0].I = (0x4000 + (y & 0x8000)) - reg[0].I;
- reg[3].I = 0x170;
- }
- }
- }
-
- #ifdef DEV_VERSION
- if(systemVerbose & VERBOSE_SWI) {
- log("ArcTan2: return=%08x\n",
- reg[0].I);
- }
- #endif
- }
-
- void BIOS_BitUnPack()
- {
- #ifdef DEV_VERSION
- if(systemVerbose & VERBOSE_SWI) {
- log("BitUnPack: %08x,%08x,%08x (VCOUNT=%2d)\n",
- reg[0].I,
- reg[1].I,
- reg[2].I,
- VCOUNT);
- }
- #endif
-
- u32 source = reg[0].I;
- u32 dest = reg[1].I;
- u32 header = reg[2].I;
-
- int len = CPUReadHalfWord(header);
- // check address
- int bits = CPUReadByte(header+2);
- int revbits = 8 - bits;
- // u32 value = 0;
- u32 base = CPUReadMemory(header+4);
- bool addBase = (base & 0x80000000) ? true : false;
- base &= 0x7fffffff;
- int dataSize = CPUReadByte(header+3);
-
- int data = 0;
- int bitwritecount = 0;
- while(1) {
- len -= 1;
- if(len < 0)
- break;
- int mask = 0xff >> revbits;
- u8 b = CPUReadByte(source);
- source++;
- int bitcount = 0;
- while(1) {
- if(bitcount >= 8)
- break;
- u32 d = b & mask;
- u32 temp = d >> bitcount;
- if(!temp && addBase) {
- temp += base;
- }
- data |= temp << bitwritecount;
- bitwritecount += dataSize;
- if(bitwritecount >= 32) {
- CPUWriteMemory(dest, data);
- dest += 4;
- data = 0;
- bitwritecount = 0;
- }
- mask <<= bits;
- bitcount += bits;
- }
- }
- }
-
- void BIOS_BgAffineSet()
- {
- #ifdef DEV_VERSION
- if(systemVerbose & VERBOSE_SWI) {
- log("BgAffineSet: %08x,%08x,%08x (VCOUNT=%2d)\n",
- reg[0].I,
- reg[1].I,
- reg[2].I,
- VCOUNT);
- }
- #endif
-
- u32 src = reg[0].I;
- u32 dest = reg[1].I;
- int num = reg[2].I;
-
- for(int i = 0; i < num; i++) {
- s32 cx = CPUReadMemory(src);
- src+=4;
- s32 cy = CPUReadMemory(src);
- src+=4;
- s16 dispx = CPUReadHalfWord(src);
- src+=2;
- s16 dispy = CPUReadHalfWord(src);
- src+=2;
- s16 rx = CPUReadHalfWord(src);
- src+=2;
- s16 ry = CPUReadHalfWord(src);
- src+=2;
- u16 theta = CPUReadHalfWord(src)>>8;
- src+=4; // keep structure alignment
- s32 a = sineTable[(theta+0x40)&255];
- s32 b = sineTable[theta];
-
- s16 dx = (rx * a)>>14;
- s16 dmx = (rx * b)>>14;
- s16 dy = (ry * b)>>14;
- s16 dmy = (ry * a)>>14;
-
- CPUWriteHalfWord(dest, dx);
- dest += 2;
- CPUWriteHalfWord(dest, -dmx);
- dest += 2;
- CPUWriteHalfWord(dest, dy);
- dest += 2;
- CPUWriteHalfWord(dest, dmy);
- dest += 2;
-
- s32 startx = cx - dx * dispx + dmx * dispy;
- s32 starty = cy - dy * dispx - dmy * dispy;
-
- CPUWriteMemory(dest, startx);
- dest += 4;
- CPUWriteMemory(dest, starty);
- dest += 4;
- }
- }
-
- void BIOS_CpuSet()
- {
- #ifdef DEV_VERSION
- if(systemVerbose & VERBOSE_SWI) {
- log("CpuSet: 0x%08x,0x%08x,0x%08x (VCOUNT=%d)\n", reg[0].I, reg[1].I,
- reg[2].I, VCOUNT);
- }
- #endif
-
- u32 source = reg[0].I;
- u32 dest = reg[1].I;
- u32 cnt = reg[2].I;
-
- if(((source & 0xe000000) == 0) ||
- ((source + (((cnt << 11)>>9) & 0x1fffff)) & 0xe000000) == 0)
- return;
-
- int count = cnt & 0x1FFFFF;
-
- // 32-bit ?
- if((cnt >> 26) & 1) {
- // needed for 32-bit mode!
- source &= 0xFFFFFFFC;
- dest &= 0xFFFFFFFC;
- // fill ?
- if((cnt >> 24) & 1) {
- u32 value = CPUReadMemory(source);
- while(count) {
- CPUWriteMemory(dest, value);
- dest += 4;
- count--;
- }
- } else {
- // copy
- while(count) {
- CPUWriteMemory(dest, CPUReadMemory(source));
- source += 4;
- dest += 4;
- count--;
- }
- }
- } else {
- // 16-bit fill?
- if((cnt >> 24) & 1) {
- u16 value = CPUReadHalfWord(source);
- while(count) {
- CPUWriteHalfWord(dest, value);
- dest += 2;
- count--;
- }
- } else {
- // copy
- while(count) {
- CPUWriteHalfWord(dest, CPUReadHalfWord(source));
- source += 2;
- dest += 2;
- count--;
- }
- }
- }
- }
-
- void BIOS_CpuFastSet()
- {
- #ifdef DEV_VERSION
- if(systemVerbose & VERBOSE_SWI) {
- log("CpuFastSet: 0x%08x,0x%08x,0x%08x (VCOUNT=%d)\n", reg[0].I, reg[1].I,
- reg[2].I, VCOUNT);
- }
- #endif
-
- u32 source = reg[0].I;
- u32 dest = reg[1].I;
- u32 cnt = reg[2].I;
-
- if(((source & 0xe000000) == 0) ||
- ((source + (((cnt << 11)>>9) & 0x1fffff)) & 0xe000000) == 0)
- return;
-
- // needed for 32-bit mode!
- source &= 0xFFFFFFFC;
- dest &= 0xFFFFFFFC;
-
- int count = cnt & 0x1FFFFF;
-
- // fill?
- if((cnt >> 24) & 1) {
- while(count > 0) {
- // BIOS always transfers 32 bytes at a time
- u32 value = CPUReadMemory(source);
- for(int i = 0; i < 8; i++) {
- CPUWriteMemory(dest, value);
- dest += 4;
- }
- count -= 8;
- }
- } else {
- // copy
- while(count > 0) {
- // BIOS always transfers 32 bytes at a time
- for(int i = 0; i < 8; i++) {
- CPUWriteMemory(dest, CPUReadMemory(source));
- source += 4;
- dest += 4;
- }
- count -= 8;
- }
- }
- }
-
- void BIOS_Diff8bitUnFilterWram()
- {
- #ifdef DEV_VERSION
- if(systemVerbose & VERBOSE_SWI) {
- log("Diff8bitUnFilterWram: 0x%08x,0x%08x (VCOUNT=%d)\n", reg[0].I,
- reg[1].I, VCOUNT);
- }
- #endif
-
- u32 source = reg[0].I;
- u32 dest = reg[1].I;
-
- u32 header = CPUReadMemory(source);
- source += 4;
-
- if(((source & 0xe000000) == 0) ||
- ((source + ((header >> 8) & 0x1fffff) & 0xe000000) == 0))
- return;
-
- int len = header >> 8;
-
- u8 data = CPUReadByte(source++);
- CPUWriteByte(dest++, data);
- len--;
-
- while(len > 0) {
- u8 diff = CPUReadByte(source++);
- data += diff;
- CPUWriteByte(dest++, data);
- len--;
- }
- }
-
- void BIOS_Diff8bitUnFilterVram()
- {
- #ifdef DEV_VERSION
- if(systemVerbose & VERBOSE_SWI) {
- log("Diff8bitUnFilterVram: 0x%08x,0x%08x (VCOUNT=%d)\n", reg[0].I,
- reg[1].I, VCOUNT);
- }
- #endif
-
- u32 source = reg[0].I;
- u32 dest = reg[1].I;
-
- u32 header = CPUReadMemory(source);
- source += 4;
-
- if(((source & 0xe000000) == 0) ||
- ((source + ((header >> 8) & 0x1fffff)) & 0xe000000) == 0)
- return;
-
- int len = header >> 8;
-
- u8 data = CPUReadByte(source++);
- u16 writeData = data;
- int shift = 8;
- int bytes = 1;
-
- while(len >= 2) {
- u8 diff = CPUReadByte(source++);
- data += diff;
- writeData |= (data << shift);
- bytes++;
- shift += 8;
- if(bytes == 2) {
- CPUWriteHalfWord(dest, writeData);
- dest += 2;
- len -= 2;
- bytes = 0;
- writeData = 0;
- shift = 0;
- }
- }
- }
-
- void BIOS_Diff16bitUnFilter()
- {
- #ifdef DEV_VERSION
- if(systemVerbose & VERBOSE_SWI) {
- log("Diff16bitUnFilter: 0x%08x,0x%08x (VCOUNT=%d)\n", reg[0].I,
- reg[1].I, VCOUNT);
- }
- #endif
-
- u32 source = reg[0].I;
- u32 dest = reg[1].I;
-
- u32 header = CPUReadMemory(source);
- source += 4;
-
- if(((source & 0xe000000) == 0) ||
- ((source + ((header >> 8) & 0x1fffff)) & 0xe000000) == 0)
- return;
-
- int len = header >> 8;
-
- u16 data = CPUReadHalfWord(source);
- source += 2;
- CPUWriteHalfWord(dest, data);
- dest += 2;
- len -= 2;
-
- while(len >= 2) {
- u16 diff = CPUReadHalfWord(source);
- source += 2;
- data += diff;
- CPUWriteHalfWord(dest, data);
- dest += 2;
- len -= 2;
- }
- }
-
- void BIOS_Div()
- {
- #ifdef DEV_VERSION
- if(systemVerbose & VERBOSE_SWI) {
- log("Div: 0x%08x,0x%08x (VCOUNT=%d)\n",
- reg[0].I,
- reg[1].I,
- VCOUNT);
- }
- #endif
-
- int number = reg[0].I;
- int denom = reg[1].I;
-
- if(denom != 0) {
- reg[0].I = number / denom;
- reg[1].I = number % denom;
- s32 temp = (s32)reg[0].I;
- reg[3].I = temp < 0 ? (u32)-temp : (u32)temp;
- }
- #ifdef DEV_VERSION
- if(systemVerbose & VERBOSE_SWI) {
- log("Div: return=0x%08x,0x%08x,0x%08x\n",
- reg[0].I,
- reg[1].I,
- reg[3].I);
- }
- #endif
- }
-
- void BIOS_DivARM()
- {
- #ifdef DEV_VERSION
- if(systemVerbose & VERBOSE_SWI) {
- log("DivARM: 0x%08x, (VCOUNT=%d)\n",
- reg[0].I,
- VCOUNT);
- }
- #endif
-
- u32 temp = reg[0].I;
- reg[0].I = reg[1].I;
- reg[1].I = temp;
- BIOS_Div();
- }
-
- void BIOS_HuffUnComp()
- {
- #ifdef DEV_VERSION
- if(systemVerbose & VERBOSE_SWI) {
- log("HuffUnComp: 0x%08x,0x%08x (VCOUNT=%d)\n",
- reg[0].I,
- reg[1].I,
- VCOUNT);
- }
- #endif
-
- u32 source = reg[0].I;
- u32 dest = reg[1].I;
-
- u32 header = CPUReadMemory(source);
- source += 4;
-
- if(((source & 0xe000000) == 0) ||
- ((source + ((header >> 8) & 0x1fffff)) & 0xe000000) == 0)
- return;
-
- u8 treeSize = CPUReadByte(source++);
-
- u32 treeStart = source;
-
- source += (treeSize<<1) + 1;
-
- int len = header >> 8;
-
- u32 mask = 0x80000000;
- u32 data = CPUReadMemory(source);
- source += 4;
-
- int pos = 0;
- u8 rootNode = CPUReadByte(treeStart);
- u8 currentNode = rootNode;
- bool writeData = false;
- int byteShift = 0;
- int byteCount = 0;
- u32 writeValue = 0;
-
- if((header & 0x0F) == 8) {
- while(len > 0) {
- // take left
- if(pos == 0)
- pos++;
- else
- pos += (((currentNode & 0x3F)+1)<<1);
-
- if(data & mask) {
- // right
- if(currentNode & 0x40)
- writeData = true;
- currentNode = CPUReadByte(treeStart+pos+1);
- } else {
- // left
- if(currentNode & 0x80)
- writeData = true;
- currentNode = CPUReadByte(treeStart+pos);
- }
-
- if(writeData) {
- writeValue |= (currentNode << byteShift);
- byteCount++;
- byteShift += 8;
-
- pos = 0;
- currentNode = rootNode;
- writeData = false;
-
- if(byteCount == 4) {
- byteCount = 0;
- byteShift = 0;
- CPUWriteMemory(dest, writeValue);
- writeValue = 0;
- dest += 4;
- len -= 4;
- }
- }
- mask >>= 1;
- if(mask == 0) {
- mask = 0x80000000;
- data = CPUReadMemory(source);
- source += 4;
- }
- }
- } else {
- int halfLen = 0;
- int value = 0;
- while(len > 0) {
- // take left
- if(pos == 0)
- pos++;
- else
- pos += (((currentNode & 0x3F)+1)<<1);
-
- if((data & mask)) {
- // right
- if(currentNode & 0x40)
- writeData = true;
- currentNode = CPUReadByte(treeStart+pos+1);
- } else {
- // left
- if(currentNode & 0x80)
- writeData = true;
- currentNode = CPUReadByte(treeStart+pos);
- }
-
- if(writeData) {
- if(halfLen == 0)
- value |= currentNode;
- else
- value |= (currentNode<<4);
-
- halfLen += 4;
- if(halfLen == 8) {
- writeValue |= (value << byteShift);
- byteCount++;
- byteShift += 8;
-
- halfLen = 0;
- value = 0;
-
- if(byteCount == 4) {
- byteCount = 0;
- byteShift = 0;
- CPUWriteMemory(dest, writeValue);
- dest += 4;
- writeValue = 0;
- len -= 4;
- }
- }
- pos = 0;
- currentNode = rootNode;
- writeData = false;
- }
- mask >>= 1;
- if(mask == 0) {
- mask = 0x80000000;
- data = CPUReadMemory(source);
- source += 4;
- }
- }
- }
- }
-
- void BIOS_LZ77UnCompVram()
- {
- #ifdef DEV_VERSION
- if(systemVerbose & VERBOSE_SWI) {
- log("LZ77UnCompVram: 0x%08x,0x%08x (VCOUNT=%d)\n",
- reg[0].I,
- reg[1].I,
- VCOUNT);
- }
- #endif
-
- u32 source = reg[0].I;
- u32 dest = reg[1].I;
-
- u32 header = CPUReadMemory(source);
- source += 4;
-
- if(((source & 0xe000000) == 0) ||
- ((source + ((header >> 8) & 0x1fffff)) & 0xe000000) == 0)
- return;
-
- int byteCount = 0;
- int byteShift = 0;
- u32 writeValue = 0;
-
- int len = header >> 8;
-
- while(len > 0) {
- u8 d = CPUReadByte(source++);
-
- if(d) {
- for(int i = 0; i < 8; i++) {
- if(d & 0x80) {
- u16 data = CPUReadByte(source++) << 8;
- data |= CPUReadByte(source++);
- int length = (data >> 12) + 3;
- int offset = (data & 0x0FFF);
- u32 windowOffset = dest + byteCount - offset - 1;
- for(int i = 0; i < length; i++) {
- writeValue |= (CPUReadByte(windowOffset++) << byteShift);
- byteShift += 8;
- byteCount++;
-
- if(byteCount == 2) {
- CPUWriteHalfWord(dest, writeValue);
- dest += 2;
- byteCount = 0;
- byteShift = 0;
- writeValue = 0;
- }
- len--;
- if(len == 0)
- return;
- }
- } else {
- writeValue |= (CPUReadByte(source++) << byteShift);
- byteShift += 8;
- byteCount++;
- if(byteCount == 2) {
- CPUWriteHalfWord(dest, writeValue);
- dest += 2;
- byteCount = 0;
- byteShift = 0;
- writeValue = 0;
- }
- len--;
- if(len == 0)
- return;
- }
- d <<= 1;
- }
- } else {
- for(int i = 0; i < 8; i++) {
- writeValue |= (CPUReadByte(source++) << byteShift);
- byteShift += 8;
- byteCount++;
- if(byteCount == 2) {
- CPUWriteHalfWord(dest, writeValue);
- dest += 2;
- byteShift = 0;
- byteCount = 0;
- writeValue = 0;
- }
- len--;
- if(len == 0)
- return;
- }
- }
- }
- }
-
- void BIOS_LZ77UnCompWram()
- {
- #ifdef DEV_VERSION
- if(systemVerbose & VERBOSE_SWI) {
- log("LZ77UnCompWram: 0x%08x,0x%08x (VCOUNT=%d)\n", reg[0].I, reg[1].I,
- VCOUNT);
- }
- #endif
-
- u32 source = reg[0].I;
- u32 dest = reg[1].I;
-
- u32 header = CPUReadMemory(source);
- source += 4;
-
- if(((source & 0xe000000) == 0) ||
- ((source + ((header >> 8) & 0x1fffff)) & 0xe000000) == 0)
- return;
-
- int len = header >> 8;
-
- while(len > 0) {
- u8 d = CPUReadByte(source++);
-
- if(d) {
- for(int i = 0; i < 8; i++) {
- if(d & 0x80) {
- u16 data = CPUReadByte(source++) << 8;
- data |= CPUReadByte(source++);
- int length = (data >> 12) + 3;
- int offset = (data & 0x0FFF);
- u32 windowOffset = dest - offset - 1;
- for(int i = 0; i < length; i++) {
- CPUWriteByte(dest++, CPUReadByte(windowOffset++));
- len--;
- if(len == 0)
- return;
- }
- } else {
- CPUWriteByte(dest++, CPUReadByte(source++));
- len--;
- if(len == 0)
- return;
- }
- d <<= 1;
- }
- } else {
- for(int i = 0; i < 8; i++) {
- CPUWriteByte(dest++, CPUReadByte(source++));
- len--;
- if(len == 0)
- return;
- }
- }
- }
- }
-
- void BIOS_ObjAffineSet()
- {
- #ifdef DEV_VERSION
- if(systemVerbose & VERBOSE_SWI) {
- log("ObjAffineSet: 0x%08x,0x%08x,0x%08x,0x%08x (VCOUNT=%d)\n",
- reg[0].I,
- reg[1].I,
- reg[2].I,
- reg[3].I,
- VCOUNT);
- }
- #endif
-
- u32 src = reg[0].I;
- u32 dest = reg[1].I;
- int num = reg[2].I;
- int offset = reg[3].I;
-
- for(int i = 0; i < num; i++) {
- s16 rx = CPUReadHalfWord(src);
- src+=2;
- s16 ry = CPUReadHalfWord(src);
- src+=2;
- u16 theta = CPUReadHalfWord(src)>>8;
- src+=4; // keep structure alignment
-
- s32 a = (s32)sineTable[(theta+0x40)&255];
- s32 b = (s32)sineTable[theta];
-
- s16 dx = ((s32)rx * a)>>14;
- s16 dmx = ((s32)rx * b)>>14;
- s16 dy = ((s32)ry * b)>>14;
- s16 dmy = ((s32)ry * a)>>14;
-
- CPUWriteHalfWord(dest, dx);
- dest += offset;
- CPUWriteHalfWord(dest, -dmx);
- dest += offset;
- CPUWriteHalfWord(dest, dy);
- dest += offset;
- CPUWriteHalfWord(dest, dmy);
- dest += offset;
- }
- }
-
- void BIOS_RegisterRamReset(u32 flags)
- {
- // no need to trace here. this is only called directly from GBA.cpp
- // to emulate bios initialization
-
- if(flags) {
- if(flags & 0x01) {
- // clear work RAM
- memset(workRAM, 0, 0x40000);
- }
- if(flags & 0x02) {
- // clear internal RAM
- memset(internalRAM, 0, 0x7e00); // don't clear 0x7e00-0x7fff
- }
- if(flags & 0x04) {
- // clear palette RAM
- memset(paletteRAM, 0, 0x400);
- }
- if(flags & 0x08) {
- // clear VRAM
- memset(vram, 0, 0x18000);
- }
- if(flags & 0x10) {
- // clean OAM
- memset(oam, 0, 0x400);
- }
-
- if(flags & 0x80) {
- int i;
- for(i = 0; i < 8; i++)
- CPUUpdateRegister(0x200+i*2, 0);
-
- CPUUpdateRegister(0x202, 0xFFFF);
-
- for(i = 0; i < 8; i++)
- CPUUpdateRegister(0x4+i*2, 0);
-
- for(i = 0; i < 16; i++)
- CPUUpdateRegister(0x20+i*2, 0);
-
- for(i = 0; i < 24; i++)
- CPUUpdateRegister(0xb0+i*2, 0);
-
- CPUUpdateRegister(0x130, 0);
- CPUUpdateRegister(0x20, 0x100);
- CPUUpdateRegister(0x30, 0x100);
- CPUUpdateRegister(0x26, 0x100);
- CPUUpdateRegister(0x36, 0x100);
- }
-
- if(flags & 0x20) {
- int i;
- for(i = 0; i < 8; i++)
- CPUUpdateRegister(0x110+i*2, 0);
- CPUUpdateRegister(0x134, 0x8000);
- for(i = 0; i < 7; i++)
- CPUUpdateRegister(0x140+i*2, 0);
- }
-
- if(flags & 0x40) {
- int i;
- CPUWriteByte(0x4000084, 0);
- CPUWriteByte(0x4000084, 0x80);
- CPUWriteMemory(0x4000080, 0x880e0000);
- CPUUpdateRegister(0x88, CPUReadHalfWord(0x4000088)&0x3ff);
- CPUWriteByte(0x4000070, 0x70);
- for(i = 0; i < 8; i++)
- CPUUpdateRegister(0x90+i*2, 0);
- CPUWriteByte(0x4000070, 0);
- for(i = 0; i < 8; i++)
- CPUUpdateRegister(0x90+i*2, 0);
- CPUWriteByte(0x4000084, 0);
- }
- }
- }
-
- void BIOS_RegisterRamReset()
- {
- #ifdef DEV_VERSION
- if(systemVerbose & VERBOSE_SWI) {
- log("RegisterRamReset: 0x%08x (VCOUNT=%d)\n",
- reg[0].I,
- VCOUNT);
- }
- #endif
-
- BIOS_RegisterRamReset(reg[0].I);
- }
-
- void BIOS_RLUnCompVram()
- {
- #ifdef DEV_VERSION
- if(systemVerbose & VERBOSE_SWI) {
- log("RLUnCompVram: 0x%08x,0x%08x (VCOUNT=%d)\n",
- reg[0].I,
- reg[1].I,
- VCOUNT);
- }
- #endif
-
- u32 source = reg[0].I;
- u32 dest = reg[1].I;
-
- u32 header = CPUReadMemory(source);
- source += 4;
-
- if(((source & 0xe000000) == 0) ||
- ((source + ((header >> 8) & 0x1fffff)) & 0xe000000) == 0)
- return;
-
- int len = header >> 8;
- int byteCount = 0;
- int byteShift = 0;
- u32 writeValue = 0;
-
- while(len > 0) {
- u8 d = CPUReadByte(source++);
- int l = d & 0x7F;
- if(d & 0x80) {
- u8 data = CPUReadByte(source++);
- l += 3;
- for(int i = 0;i < l; i++) {
- writeValue |= (data << byteShift);
- byteShift += 8;
- byteCount++;
-
- if(byteCount == 2) {
- CPUWriteHalfWord(dest, writeValue);
- dest += 2;
- byteCount = 0;
- byteShift = 0;
- writeValue = 0;
- }
- len--;
- if(len == 0)
- return;
- }
- } else {
- l++;
- for(int i = 0; i < l; i++) {
- writeValue |= (CPUReadByte(source++) << byteShift);
- byteShift += 8;
- byteCount++;
- if(byteCount == 2) {
- CPUWriteHalfWord(dest, writeValue);
- dest += 2;
- byteCount = 0;
- byteShift = 0;
- writeValue = 0;
- }
- len--;
- if(len == 0)
- return;
- }
- }
- }
- }
-
- void BIOS_RLUnCompWram()
- {
- #ifdef DEV_VERSION
- if(systemVerbose & VERBOSE_SWI) {
- log("RLUnCompWram: 0x%08x,0x%08x (VCOUNT=%d)\n",
- reg[0].I,
- reg[1].I,
- VCOUNT);
- }
- #endif
-
- u32 source = reg[0].I;
- u32 dest = reg[1].I;
-
- u32 header = CPUReadMemory(source);
- source += 4;
-
- if(((source & 0xe000000) == 0) ||
- ((source + ((header >> 8) & 0x1fffff)) & 0xe000000) == 0)
- return;
-
- int len = header >> 8;
-
- while(len > 0) {
- u8 d = CPUReadByte(source++);
- int l = d & 0x7F;
- if(d & 0x80) {
- u8 data = CPUReadByte(source++);
- l += 3;
- for(int i = 0;i < l; i++) {
- CPUWriteByte(dest++, data);
- len--;
- if(len == 0)
- return;
- }
- } else {
- l++;
- for(int i = 0; i < l; i++) {
- CPUWriteByte(dest++, CPUReadByte(source++));
- len--;
- if(len == 0)
- return;
- }
- }
- }
- }
-
- void BIOS_SoftReset()
- {
- #ifdef DEV_VERSION
- if(systemVerbose & VERBOSE_SWI) {
- log("SoftReset: (VCOUNT=%d)\n", VCOUNT);
- }
- #endif
-
- armState = true;
- armMode = 0x1F;
- armIrqEnable = false;
- C_FLAG = V_FLAG = N_FLAG = Z_FLAG = false;
- reg[13].I = 0x03007F00;
- reg[14].I = 0x00000000;
- reg[16].I = 0x00000000;
- reg[R13_IRQ].I = 0x03007FA0;
- reg[R14_IRQ].I = 0x00000000;
- reg[SPSR_IRQ].I = 0x00000000;
- reg[R13_SVC].I = 0x03007FE0;
- reg[R14_SVC].I = 0x00000000;
- reg[SPSR_SVC].I = 0x00000000;
- u8 b = internalRAM[0x7ffa];
-
- memset(&internalRAM[0x7e00], 0, 0x200);
-
- if(b) {
- armNextPC = 0x02000000;
- reg[15].I = 0x02000004;
- } else {
- armNextPC = 0x08000000;
- reg[15].I = 0x08000004;
- }
- }
-
- void BIOS_Sqrt()
- {
- #ifdef DEV_VERSION
- if(systemVerbose & VERBOSE_SWI) {
- log("Sqrt: %08x (VCOUNT=%2d)\n",
- reg[0].I,
- VCOUNT);
- }
- #endif
- reg[0].I = (u32)sqrt((double)reg[0].I);
- #ifdef DEV_VERSION
- if(systemVerbose & VERBOSE_SWI) {
- log("Sqrt: return=%08x\n",
- reg[0].I);
- }
- #endif
- }
-
- void BIOS_MidiKey2Freq()
- {
- #ifdef DEV_VERSION
- if(systemVerbose & VERBOSE_SWI) {
- log("MidiKey2Freq: WaveData=%08x mk=%08x fp=%08x\n",
- reg[0].I,
- reg[1].I,
- reg[2].I);
- }
- #endif
- int freq = CPUReadMemory(reg[0].I+4);
- double tmp;
- tmp = ((double)(180 - reg[1].I)) - ((double)reg[2].I / 256.f);
- tmp = pow((double)2.f, tmp / 12.f);
- reg[0].I = (int)((double)freq / tmp);
-
- #ifdef DEV_VERSION
- if(systemVerbose & VERBOSE_SWI) {
- log("MidiKey2Freq: return %08x\n",
- reg[0].I);
- }
- #endif
- }
-
- void BIOS_SndDriverJmpTableCopy()
- {
- #ifdef DEV_VERSION
- if(systemVerbose & VERBOSE_SWI) {
- log("SndDriverJmpTableCopy: dest=%08x\n",
- reg[0].I);
- }
- #endif
- for(int i = 0; i < 0x24; i++) {
- CPUWriteMemory(reg[0].I, 0x9c);
- reg[0].I += 4;
- }
- }
-