#include #include #include #include #include #include #include "main.h" #include "a5200utils.h" #include "atari.h" #include "global.h" #include "cartridge.h" #include "input.h" #include "emu/pia.h" #include "clickNoQuit_wav.h" #include "bgBottom.h" #include "bgTop.h" #include "bgFileSel.h" #include "bgCardSel.h" #include "altirra_5200_os.h" FICA5200 a5200romlist[1024]; unsigned int counta5200=0, countfiles=0, ucFicAct=0; int gTotalAtariFrames = 0; int bg0, bg1, bg0b, bg1b, bg2, bg3; unsigned int etatEmu; gamecfg GameConf; // Game Config svg #define cxBG (myCart.offset_x<<8) #define cyBG (myCart.offset_y<<8) #define xdxBG (((320 / myCart.scale_x) << 8) | (320 % myCart.scale_x)) #define ydyBG (((256 / myCart.scale_y) << 8) | (256 % myCart.scale_y)) unsigned int atari_pal16[256] = {0}; unsigned char *filebuffer; unsigned char sound_buffer[SNDLENGTH]; unsigned char *psound_buffer; #define MAX_DEBUG 5 int debug[MAX_DEBUG]={0}; //#define DEBUG_DUMP static void DumpDebugData(void) { #ifdef DEBUG_DUMP char dbgbuf[32]; for (int i=0; ivalEnd;ucFade--) { if (ucScr & 0x01) REG_BLDY=ucFade; if (ucScr & 0x02) REG_BLDY_SUB=ucFade; for (ucBcl=0;ucBcl0 ? "<" : " ")); dsPrintValue(31,22,0,(char *) (NoDebGame+14" : " ")); sprintf(szName,"%s","A TO SELECT A GAME, B TO GO BACK"); dsPrintValue(16-strlen(szName)/2,23,0,szName); for (ucBcl=0;ucBcl<17; ucBcl++) { ucGame= ucBcl+NoDebGame; if (ucGame < counta5200) { char szName2[256]; maxLen=strlen(a5200romlist[ucGame].filename); strcpy(szName,a5200romlist[ucGame].filename); if (maxLen>29) szName[29]='\0'; if (a5200romlist[ucGame].directory) { a5200romlist[ucGame].filename[29] = 0; sprintf(szName,"[%s]",a5200romlist[ucGame].filename); sprintf(szName2,"%-29s",szName); dsPrintValue(0,5+ucBcl,(ucSel == ucBcl ? 1 : 0),szName2); } else { sprintf(szName2,"%-29s",strupr(szName)); dsPrintValue(1,5+ucBcl,(ucSel == ucBcl ? 1 : 0),szName2); } } } } unsigned int dsWaitForRom(void) { bool bDone=false, bRet=false; u32 ucHaut=0x00, ucBas=0x00,ucSHaut=0x00, ucSBas=0x00,romSelected= 0, firstRomDisplay=0,nbRomPerPage, uNbRSPage, uLenFic=0,ucFlip=0, ucFlop=0; char szName[64]; decompress(bgFileSelTiles, bgGetGfxPtr(bg0b), LZ77Vram); decompress(bgFileSelMap, (void*) bgGetMapPtr(bg0b), LZ77Vram); dmaCopy((void *) bgFileSelPal,(u16*) BG_PALETTE_SUB,256*2); unsigned short dmaVal = *(bgGetMapPtr(bg1b) +31*32); dmaFillWords(dmaVal | (dmaVal<<16),(void*) bgGetMapPtr(bg1b),32*24*2); nbRomPerPage = (counta5200>=17 ? 17 : counta5200); uNbRSPage = (counta5200>=5 ? 5 : counta5200); if (ucFicAct>counta5200-nbRomPerPage) { firstRomDisplay=counta5200-nbRomPerPage; romSelected=ucFicAct-counta5200+nbRomPerPage; } else { firstRomDisplay=ucFicAct; romSelected=0; } dsDisplayFiles(firstRomDisplay,romSelected); while (!bDone) { if (keysCurrent() & KEY_UP) { if (!ucHaut) { ucFicAct = (ucFicAct>0 ? ucFicAct-1 : counta5200-1); if (romSelected>uNbRSPage) { romSelected -= 1; } else { if (firstRomDisplay>0) { firstRomDisplay -= 1; } else { if (romSelected>0) { romSelected -= 1; } else { firstRomDisplay=counta5200-nbRomPerPage; romSelected=nbRomPerPage-1; } } } ucHaut=0x01; dsDisplayFiles(firstRomDisplay,romSelected); } else { ucHaut++; if (ucHaut>10) ucHaut=0; } } else { ucHaut = 0; } if (keysCurrent() & KEY_DOWN) { if (!ucBas) { ucFicAct = (ucFicAct< counta5200-1 ? ucFicAct+1 : 0); if (romSelected10) ucBas=0; } } else { ucBas = 0; } if ((keysCurrent() & KEY_R) || (keysCurrent() & KEY_RIGHT)) { if (!ucSBas) { ucFicAct = (ucFicAct< counta5200-nbRomPerPage ? ucFicAct+nbRomPerPage : counta5200-nbRomPerPage); if (firstRomDisplay10) ucSBas=0; } } else { ucSBas = 0; } if ((keysCurrent() & KEY_L) || (keysCurrent() & KEY_LEFT)) { if (!ucSHaut) { ucFicAct = (ucFicAct> nbRomPerPage ? ucFicAct-nbRomPerPage : 0); if (firstRomDisplay>nbRomPerPage) { firstRomDisplay -= nbRomPerPage; } else { firstRomDisplay = 0; } if (ucFicAct == 0) romSelected = 0; ucSHaut=0x01; dsDisplayFiles(firstRomDisplay,romSelected); } else { ucSHaut++; if (ucSHaut>10) ucSHaut=0; } } else { ucSHaut = 0; } if ( keysCurrent() & KEY_B ) { bDone=true; while (keysCurrent() & KEY_B); } if (keysCurrent() & KEY_A) { if (!a5200romlist[ucFicAct].directory) { bRet=true; bDone=true; } else { chdir(a5200romlist[ucFicAct].filename); a52FindFiles(); ucFicAct = 0; nbRomPerPage = (counta5200>=16 ? 16 : counta5200); uNbRSPage = (counta5200>=5 ? 5 : counta5200); if (ucFicAct>counta5200-nbRomPerPage) { firstRomDisplay=counta5200-nbRomPerPage; romSelected=ucFicAct-counta5200+nbRomPerPage; } else { firstRomDisplay=ucFicAct; romSelected=0; } dsDisplayFiles(firstRomDisplay,romSelected); while (keysCurrent() & KEY_A); } } // Scroll la selection courante if (strlen(a5200romlist[ucFicAct].filename) > 29) { ucFlip++; if (ucFlip >= 8) { ucFlip = 0; uLenFic++; if ((uLenFic+29)>strlen(a5200romlist[ucFicAct].filename)) { ucFlop++; if (ucFlop >= 8) { uLenFic=0; ucFlop = 0; } else uLenFic--; } strncpy(szName,a5200romlist[ucFicAct].filename+uLenFic,29); szName[29] = '\0'; dsPrintValue(1,5+romSelected,1,szName); } } swiWaitForVBlank(); } decompress(bgBottomTiles, bgGetGfxPtr(bg0b), LZ77Vram); decompress(bgBottomMap, (void*) bgGetMapPtr(bg0b), LZ77Vram); dmaCopy((void *) bgBottomPal,(u16*) BG_PALETTE_SUB,256*2); dmaVal = *(bgGetMapPtr(bg1b) +31*32); dmaFillWords(dmaVal | (dmaVal<<16),(void*) bgGetMapPtr(bg1b),32*24*2); return bRet; } unsigned int dsWaitOnMenu(unsigned int actState) { unsigned int uState=A5200_PLAYINIT; unsigned int keys_pressed; bool bDone=false, romSel; int iTx,iTy; while (!bDone) { // wait for stylus keys_pressed = keysCurrent(); if (keys_pressed & KEY_TOUCH) { touchPosition touch; touchRead(&touch); iTx = touch.px; iTy = touch.py; if ((iTx>206) && (iTx<250) && (iTy>110) && (iTy<129)) { // 207,111 -> 249,128 quit soundPlaySample(clickNoQuit_wav, SoundFormat_16Bit, clickNoQuit_wav_size, 22050, 127, 64, false, 0); bDone=dsWaitOnQuit(); if (bDone) uState=A5200_QUITSTDS; } if ((iTx>71) && (iTx<183) && (iTy>7) && (iTy<43)) { // 72,8 -> 182,42 cartridge slot bDone=true; // Find files in current directory and show it a52FindFiles(); romSel=dsWaitForRom(); if (romSel) { uState=A5200_PLAYINIT; dsLoadGame(a5200romlist[ucFicAct].filename); } else { uState=actState; } } } swiWaitForVBlank(); } return uState; } void dsPrintValue(int x, int y, unsigned int isSelect, char *pchStr) { u16 *pusEcran,*pusMap; u16 usCharac; char *pTrTxt=pchStr; char ch; pusEcran=(u16*) (bgGetMapPtr(bg1b))+x+(y<<5); pusMap=(u16*) (bgGetMapPtr(bg0b)+(2*isSelect+24)*32); while((*pTrTxt)!='\0' ) { ch = *pTrTxt; if (ch >= 'a' && ch <= 'z') ch -= 32; // Faster than strcpy/strtoupper usCharac=0x0000; if ((ch) == '|') usCharac=*(pusMap); else if (((ch)<' ') || ((ch)>'_')) usCharac=*(pusMap); else if((ch)<'@') usCharac=*(pusMap+(ch)-' '); else usCharac=*(pusMap+32+(ch)-'@'); *pusEcran++=usCharac; pTrTxt++; } } //--------------------------------------------------------------------------------- void dsInstallSoundEmuFIFO(void) { FifoMessage msg; msg.SoundPlay.data = &sound_buffer; msg.SoundPlay.freq = SOUND_FREQ; msg.SoundPlay.volume = 127; msg.SoundPlay.pan = 64; msg.SoundPlay.loop = 1; msg.SoundPlay.format = ((1)<<4) | SoundFormat_8Bit; msg.SoundPlay.loopPoint = 0; msg.SoundPlay.dataSize = SNDLENGTH >> 2; msg.type = EMUARM7_PLAY_SND; fifoSendDatamsg(FIFO_USER_01, sizeof(msg), (u8*)&msg); } extern u32 trig0, trig1; extern u32 stick0; extern u32 stick1; int full_speed = 0; void dsMainLoop(void) { char fpsbuf[32]; unsigned int keys_pressed,keys_touch=0, romSel; int iTx,iTy, shiftctrl; bool showFps=false; // Timers are fed with 33.513982 MHz clock. // With DIV_1024 the clock is 32,728.5 ticks per sec... TIMER0_DATA=0; TIMER0_CR=TIMER_ENABLE|TIMER_DIV_1024; TIMER1_DATA=0; TIMER1_CR=TIMER_ENABLE | TIMER_DIV_1024; while(etatEmu != A5200_QUITSTDS) { switch (etatEmu) { case A5200_MENUINIT: dsShowScreenMain(); etatEmu = A5200_MENUSHOW; break; case A5200_MENUSHOW: etatEmu = dsWaitOnMenu(A5200_MENUSHOW); Atari800_Initialise(); break; case A5200_PLAYINIT: dsShowScreenEmu(); irqEnable(IRQ_TIMER2); etatEmu = A5200_PLAYGAME; break; case A5200_PLAYGAME: // 65535 = 1 frame // 1 frame = 1/50 ou 1/60 (0.02 or 0.016) // 656 -> 50 fps et 546 -> 60 fps if (!full_speed) { while(TIMER0_DATA < 546) ; } TIMER0_CR=0; TIMER0_DATA=0; TIMER0_CR=TIMER_ENABLE|TIMER_DIV_1024; // Execute one frame Atari800_Frame(1); // ------------------------------------------------------------- // Stuff to do once/second such as FPS display and Debug Data // ------------------------------------------------------------- if (TIMER1_DATA >= 32728) // 1000MS (1 sec) { TIMER1_CR = 0; TIMER1_DATA = 0; TIMER1_CR=TIMER_ENABLE | TIMER_DIV_1024; if (showFps) { siprintf(fpsbuf,"%03d",gTotalAtariFrames); dsPrintValue(0,0,0, fpsbuf); } // Show FPS DumpDebugData(); gTotalAtariFrames = 0; } // Read keys keys_pressed=keysCurrent(); key_consol = CONSOL_NONE; //|= (CONSOL_OPTION | CONSOL_SELECT | CONSOL_START); /* OPTION/START/SELECT key OFF */ shiftctrl = 0; key_shift = 0; trig0 = ((keys_pressed & KEY_A) || (keys_pressed & KEY_Y)) ? 0 : 1; stick0 = STICK_CENTRE; stick1 = STICK_CENTRE; if (keys_pressed & KEY_B) { shiftctrl ^= AKEY_SHFT; key_shift = 1; } key_code = shiftctrl ? 0x40 : 0x00; // if touch screen pressed if (keys_pressed & KEY_TOUCH) { touchPosition touch; touchRead(&touch); iTx = touch.px; iTy = touch.py; if ((iTx>206) && (iTx<250) && (iTy>112) && (iTy<130)) { //quit irqDisable(IRQ_TIMER2); fifoSendValue32(FIFO_USER_01,(1<<16) | (0) | SOUND_SET_VOLUME); soundPlaySample(clickNoQuit_wav, SoundFormat_16Bit, clickNoQuit_wav_size, 22050, 127, 64, false, 0); if (dsWaitOnQuit()) etatEmu=A5200_QUITSTDS; else { irqEnable(IRQ_TIMER2); fifoSendValue32(FIFO_USER_01,(1<<16) | (127) | SOUND_SET_VOLUME); } } else if ((iTx>240) && (iTx<256) && (iTy>0) && (iTy<20)) { // Full Speed Toggle ... upper corner... if (keys_touch == 0) { full_speed = 1-full_speed; if (full_speed) dsPrintValue(30,0,0,"FS"); else dsPrintValue(30,0,0," "); keys_touch = 1; } } else if ((iTx>120) && (iTx<160) && (iTy>112) && (iTy<130)) { //pause if (!keys_touch) soundPlaySample(clickNoQuit_wav, SoundFormat_16Bit, clickNoQuit_wav_size, 22050, 127, 64, false, 0); key_code = AKEY_5200_PAUSE + key_code; keys_touch = 1; } else if ((iTx>65) && (iTx<108) && (iTy>112) && (iTy<130)) { //reset if (!keys_touch) soundPlaySample(clickNoQuit_wav, SoundFormat_16Bit, clickNoQuit_wav_size, 22050, 127, 64, false, 0); key_code = AKEY_5200_RESET + key_code; keys_touch = 1; } else if ((iTx>10) && (iTx<50) && (iTy>112) && (iTy<130)) { //start if (!keys_touch) soundPlaySample(clickNoQuit_wav, SoundFormat_16Bit, clickNoQuit_wav_size, 22050, 127, 64, false, 0); key_code = AKEY_5200_START + key_code; keys_touch = 1; } else if ((iTy>155) && (iTy<185)) { char padKey[] = {AKEY_5200_0,AKEY_5200_1,AKEY_5200_2,AKEY_5200_3,AKEY_5200_4,AKEY_5200_5,AKEY_5200_6,AKEY_5200_7,AKEY_5200_8,AKEY_5200_9,AKEY_5200_HASH,AKEY_5200_ASTERISK}; if (!keys_touch) soundPlaySample(clickNoQuit_wav, SoundFormat_16Bit, clickNoQuit_wav_size, 22050, 127, 64, false, 0); if (iTx > 0) iTx--; if (iTx > 0) iTx--; key_code = padKey[iTx / 21] + key_code; keys_touch = 1; } else if ((iTx>71) && (iTx<183) && (iTy>7) && (iTy<43)) { // 72,8 -> 182,42 cartridge slot irqDisable(IRQ_TIMER2); fifoSendValue32(FIFO_USER_01,(1<<16) | (0) | SOUND_SET_VOLUME); // Find files in current directory and show it a52FindFiles(); romSel=dsWaitForRom(); if (romSel) { etatEmu=A5200_PLAYINIT; dsLoadGame(a5200romlist[ucFicAct].filename); if (full_speed) dsPrintValue(30,0,0,"FS"); else dsPrintValue(30,0,0," ");} else { irqEnable(IRQ_TIMER2); } fifoSendValue32(FIFO_USER_01,(1<<16) | (127) | SOUND_SET_VOLUME); } } else { keys_touch = 0; } if (myCart.control == CTRL_JOY) { if (keys_pressed & KEY_UP) stick0 = STICK_FORWARD; if (keys_pressed & KEY_LEFT) stick0 = STICK_LEFT; if (keys_pressed & KEY_RIGHT) stick0 = STICK_RIGHT; if (keys_pressed & KEY_DOWN) stick0 = STICK_BACK; if ((keys_pressed & KEY_UP) && (keys_pressed & KEY_LEFT)) stick0 = STICK_UL; if ((keys_pressed & KEY_UP) && (keys_pressed & KEY_RIGHT)) stick0 = STICK_UR; if ((keys_pressed & KEY_DOWN) && (keys_pressed & KEY_LEFT)) stick0 = STICK_LL; if ((keys_pressed & KEY_DOWN) && (keys_pressed & KEY_RIGHT)) stick0 = STICK_LR; } else if (myCart.control == CTRL_SWAP) { trig1 = (keys_pressed & KEY_A) ? 0 : 1; if (keys_pressed & KEY_UP) stick1 = STICK_FORWARD; if (keys_pressed & KEY_LEFT) stick1 = STICK_LEFT; if (keys_pressed & KEY_RIGHT) stick1 = STICK_RIGHT; if (keys_pressed & KEY_DOWN) stick1 = STICK_BACK; if ((keys_pressed & KEY_UP) && (keys_pressed & KEY_LEFT)) stick1 = STICK_UL; if ((keys_pressed & KEY_UP) && (keys_pressed & KEY_RIGHT)) stick1 = STICK_UR; if ((keys_pressed & KEY_DOWN) && (keys_pressed & KEY_LEFT)) stick1 = STICK_LL; if ((keys_pressed & KEY_DOWN) && (keys_pressed & KEY_RIGHT)) stick1 = STICK_LR; } else if (myCart.control == CTRL_ROBO) { if (keys_pressed & KEY_UP) stick0 = STICK_FORWARD; if (keys_pressed & KEY_LEFT) stick0 = STICK_LEFT; if (keys_pressed & KEY_RIGHT) stick0 = STICK_RIGHT; if (keys_pressed & KEY_DOWN) stick0 = STICK_BACK; if ((keys_pressed & KEY_UP) && (keys_pressed & KEY_LEFT)) stick0 = STICK_UL; if ((keys_pressed & KEY_UP) && (keys_pressed & KEY_RIGHT)) stick0 = STICK_UR; if ((keys_pressed & KEY_DOWN) && (keys_pressed & KEY_LEFT)) stick0 = STICK_LL; if ((keys_pressed & KEY_DOWN) && (keys_pressed & KEY_RIGHT)) stick0 = STICK_LR; if (keys_pressed & KEY_X) stick1 = STICK_FORWARD; if (keys_pressed & KEY_Y) stick1 = STICK_LEFT; if (keys_pressed & KEY_A) stick1 = STICK_RIGHT; if (keys_pressed & KEY_B) stick1 = STICK_BACK; if ((keys_pressed & KEY_X) && (keys_pressed & KEY_Y)) stick1 = STICK_UL; if ((keys_pressed & KEY_X) && (keys_pressed & KEY_A)) stick1 = STICK_UR; if ((keys_pressed & KEY_B) && (keys_pressed & KEY_Y)) stick1 = STICK_LL; if ((keys_pressed & KEY_B) && (keys_pressed & KEY_A)) stick1 = STICK_LR; } else if (myCart.control == CTRL_FROG) { trig0=0; if (keys_pressed & KEY_UP) {stick0 = STICK_FORWARD; trig0=1;} if (keys_pressed & KEY_LEFT) {stick0 = STICK_LEFT; trig0=1;} if (keys_pressed & KEY_RIGHT) {stick0 = STICK_RIGHT;trig0=1;} if (keys_pressed & KEY_DOWN) {stick0 = STICK_BACK; trig0=1;} } else if (myCart.control == CTRL_QBERT) { if (keys_pressed & KEY_UP) {stick0 = STICK_UR; } if (keys_pressed & KEY_LEFT) {stick0 = STICK_UL;} if (keys_pressed & KEY_RIGHT) {stick0 = STICK_LR;} if (keys_pressed & KEY_DOWN) {stick0 = STICK_LL; } trig0=0; } if (keys_pressed & KEY_START) key_code = AKEY_5200_START + key_code; if (keys_pressed & KEY_SELECT) key_code = AKEY_5200_PAUSE + key_code; if (keys_pressed & KEY_R) key_code = AKEY_5200_ASTERISK + key_code; if (keys_pressed & KEY_L) key_code = AKEY_5200_HASH + key_code; static int last_keys = 99; if (keys_pressed != last_keys) { last_keys = keys_pressed; if (myCart.control != CTRL_ROBO) { if (keys_pressed & KEY_X) {showFps = 1-showFps;dsPrintValue(0,0,0, " ");} } #if 0 if (keys_pressed & KEY_R) myCart.offset_y++; if (keys_pressed & KEY_L) myCart.offset_y--; if (keys_pressed & KEY_X) if (myCart.scale_y <= 256) myCart.scale_y++; if (keys_pressed & KEY_Y) if (myCart.scale_y >= 192) myCart.scale_y--; debug[0] = myCart.offset_y; debug[1] = myCart.scale_y; #endif } break; } } } //---------------------------------------------------------------------------------- // Find files (a78 / bin) available int a52Filescmp (const void *c1, const void *c2) { FICA5200 *p1 = (FICA5200 *) c1; FICA5200 *p2 = (FICA5200 *) c2; return strcmp (p1->filename, p2->filename); } void a52FindFiles(void) { struct stat statbuf; DIR *pdir; struct dirent *pent; char filenametmp[255]; counta5200 = countfiles= 0; pdir = opendir("."); if (pdir) { while (((pent=readdir(pdir))!=NULL)) { stat(pent->d_name,&statbuf); strcpy(filenametmp,pent->d_name); if(S_ISDIR(statbuf.st_mode)) { if (!( (filenametmp[0] == '.') && (strlen(filenametmp) == 1))) { a5200romlist[counta5200].directory = true; strcpy(a5200romlist[counta5200].filename,filenametmp); counta5200++; } } else { if (strlen(filenametmp)>4) { if ( (strcasecmp(strrchr(filenametmp, '.'), ".a52") == 0) ) { a5200romlist[counta5200].directory = false; strcpy(a5200romlist[counta5200].filename,filenametmp); counta5200++;countfiles++; } if ( (strcasecmp(strrchr(filenametmp, '.'), ".bin") == 0) ) { a5200romlist[counta5200].directory = false; strcpy(a5200romlist[counta5200].filename,filenametmp); counta5200++;countfiles++; } } } } closedir(pdir); } if (counta5200) qsort (a5200romlist, counta5200, sizeof (FICA5200), a52Filescmp); }