/* FCE Ultra - NES/Famicom Emulator * * Copyright notice for this file: * Copyright (C) 2002 Xodnizel * * 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 "types.h" #include "video.h" #include "fceu.h" #include "file.h" #include "utils/memory.h" #include "utils/crc32.h" #include "state.h" #include "movie.h" #include "palette.h" #include "nsf.h" #include "input.h" #include "vsuni.h" #include "drawing.h" #include "driver.h" #ifdef _S9XLUA_H #include "fceulua.h" #endif #ifdef __WIN_DRIVER__ #include "drivers/win/common.h" //For DirectX constants #include "drivers/win/input.h" #endif #ifdef CREATE_AVI #include "drivers/videolog/nesvideos-piece.h" #endif //no stdint in win32 (but we could add it if we needed to) #ifndef WIN32 #include #endif #include #include #include #include #include //XBuf: //0-63 is reserved for 7 special colours used by FCEUX (overlay, etc.) //64-127 is the most-used emphasis setting per frame //128-195 is the palette with no emphasis //196-255 is the palette with all emphasis bits on u8 *XBuf=NULL; //used for current display u8 *XBackBuf=NULL; //ppu output is stashed here before drawing happens u8 *XDBuf=NULL; //corresponding to XBuf but with deemph bits u8 *XDBackBuf=NULL; //corresponding to XBackBuf but with deemph bits int ClipSidesOffset=0; //Used to move displayed messages when Clips left and right sides is checked static u8 *xbsave=NULL; GUIMESSAGE guiMessage; GUIMESSAGE subtitleMessage; bool vidGuiMsgEna = true; //for input display extern int input_display; extern uint32 cur_input_display; bool oldInputDisplay = false; unsigned int lastu = 0; std::string AsSnapshotName =""; //adelikat:this will set the snapshot name when for s savesnapshot as function void FCEUI_SetSnapshotAsName(std::string name) { AsSnapshotName = name; } std::string FCEUI_GetSnapshotAsName() { return AsSnapshotName; } void FCEU_KillVirtualVideo(void) { if ( XBuf ) { FCEU_free(XBuf); XBuf = NULL; } if ( XBackBuf ) { FCEU_free(XBackBuf); XBackBuf = NULL; } if ( XDBuf ) { FCEU_free(XDBuf); XDBuf = NULL; } if ( XDBackBuf ) { FCEU_free(XDBackBuf); XDBackBuf = NULL; } //printf("Video Core Cleanup\n"); } /** * Return: Flag that indicates whether the function was succesful or not. * * TODO: This function is Windows-only. It should probably be moved. **/ int FCEU_InitVirtualVideo(void) { //Some driver code may allocate XBuf externally. //256 bytes per scanline, * 240 scanline maximum, +16 for alignment, if(XBuf) return 1; XBuf = (u8*)FCEU_malloc(256 * 256 + 16); XBackBuf = (u8*)FCEU_malloc(256 * 256 + 16); XDBuf = (u8*)FCEU_malloc(256 * 256 + 16); XDBackBuf = (u8*)FCEU_malloc(256 * 256 + 16); if(!XBuf || !XBackBuf || !XDBuf || !XDBackBuf) { return 0; } xbsave = XBuf; if( sizeof(uint8*) == 4 ) { uintptr_t m = (uintptr_t)XBuf; m = ( 8 - m) & 7; XBuf+=m; } memset(XBuf,128,256*256); memset(XBackBuf,128,256*256); memset(XBuf,128,256*256); memset(XBackBuf,128,256*256); return 1; } #ifdef FRAMESKIP void FCEU_PutImageDummy(void) { ShowFPS(); if(GameInfo->type!=GIT_NSF) { FCEU_DrawNTSCControlBars(XBuf); FCEU_DrawSaveStates(XBuf); FCEU_DrawMovies(XBuf); } if(guiMessage.howlong) guiMessage.howlong--; /* DrawMessage() */ } #endif static int dosnapsave=0; void FCEUI_SaveSnapshot(void) { dosnapsave=1; } void FCEUI_SaveSnapshotAs(void) { dosnapsave=2; } static void ReallySnap(void) { int x=SaveSnapshot(); if(!x) FCEU_DispMessage("Error saving screen snapshot.",0); else FCEU_DispMessage("Screen snapshot %d saved.",0,x-1); } static uint32 GetButtonColor(uint32 held, uint32 c, uint32 ci, int bit) { uint32 on = FCEUMOV_Mode(MOVIEMODE_PLAY) ? 0x90 : 0xA7; //Standard, or Gray depending on movie mode uint32 oni = 0xA0; //Color for immediate keyboard buttons uint32 blend = 0xB6; //Blend of immiate and last held buttons uint32 ahold = 0x87; //Auto hold uint32 off = 0xCF; uint32 color; uint8 mask = 1u << bit; if (held & mask) { //If auto-hold if (!(ci & mask)) color = ahold; else color = (c & mask) ? on : off; //If the button is pressed down (immediate) that negates auto hold, however it is only off if the previous frame the button wasn't pressed! } else { if (c & mask) color = (ci & mask) ? blend : on; //If immedaite buttons are pressed and they match the previous frame, blend the colors else color = (ci & mask) ? oni : off; } return color; } void FCEU_PutImage(void) { if(dosnapsave==2) //Save screenshot as, currently only flagged & run by the Win32 build. //TODO SDL: implement this? { char nameo[512]; strcpy(nameo,FCEUI_GetSnapshotAsName().c_str()); if (nameo[0]) { SaveSnapshot(nameo); FCEU_DispMessage("Snapshot Saved.",0); } dosnapsave=0; } if(GameInfo->type==GIT_NSF) { DrawNSF(XBuf); #ifdef _S9XLUA_H FCEU_LuaGui(XBuf); #endif //Save snapshot after NSF screen is drawn. Why would we want to do it before? if(dosnapsave==1) { ReallySnap(); dosnapsave=0; } } else { //Save backbuffer before overlay stuff is written. if(!FCEUI_EmulationPaused()) memcpy(XBackBuf, XBuf, 256*256); //Some messages need to be displayed before the avi is dumped DrawMessage(true); #ifdef _S9XLUA_H // Lua gui should draw before the avi is dumped. FCEU_LuaGui(XBuf); #endif //Save snapshot if(dosnapsave==1) { ReallySnap(); dosnapsave=0; } if (!FCEUI_AviEnableHUDrecording()) snapAVI(); if(GameInfo->type==GIT_VSUNI) FCEU_VSUniDraw(XBuf); FCEU_DrawSaveStates(XBuf); FCEU_DrawMovies(XBuf); FCEU_DrawLagCounter(XBuf); FCEU_DrawNTSCControlBars(XBuf); FCEU_DrawRecordingStatus(XBuf); ShowFPS(); } if(FCEUD_ShouldDrawInputAids()) FCEU_DrawInput(XBuf); //Fancy input display code if(input_display) { int i, j; uint8 *t = XBuf+(FSettings.LastSLine-9)*256 + 20; //mbg merge 7/17/06 changed t to uint8* if(input_display > 4) input_display = 4; for(int controller = 0; controller < input_display; controller++, t += 56) { for(i = 0; i < 34;i++) for(j = 0; j <9 ; j++) t[i+j*256] = (t[i+j*256] & 0x30) | 0xC1; for(i = 3; i < 6; i++) for(j = 3; j< 6; j++) t[i+j*256] = 0xCF; uint32 held = 0; uint32 c = cur_input_display >> (controller * 8); uint32 ci = 0; uint32 color; #ifdef __WIN_DRIVER__ extern uint32 JSAutoHeld; // This doesn't work in anything except windows for now. // It doesn't get set anywhere in other ports. if (!oldInputDisplay) ci = FCEUMOV_Mode(MOVIEMODE_PLAY) ? 0 : GetGamepadPressedImmediate() >> (controller * 8); if (!oldInputDisplay && !FCEUMOV_Mode(MOVIEMODE_PLAY)) held = (JSAutoHeld >> (controller * 8)); #else // Put other port info here #endif //adelikat: I apologize to anyone who ever sifts through this color assignment //A color = GetButtonColor(held, c, ci, 0); for(i=0; i < 4; i++) { for(j = 0; j < 4; j++) { if(i%3==0 && j %3 == 0) continue; t[30+4*256+i+j*256] = color; } } //B color = GetButtonColor(held, c, ci, 1); for(i=0; i < 4; i++) { for(j = 0; j < 4; j++) { if(i%3==0 && j %3 == 0) continue; t[24+4*256+i+j*256] = color; } } //Select color = GetButtonColor(held, c, ci, 2); for(i = 0; i < 4; i++) { t[11+5*256+i] = color; t[11+6*256+i] = color; } //Start color = GetButtonColor(held, c, ci, 3); for(i = 0; i < 4; i++) { t[17+5*256+i] = color; t[17+6*256+i] = color; } //Up color = GetButtonColor(held, c, ci, 4); for(i = 0; i < 3; i++) { for(j = 0; j < 3; j++) { t[3+i+256*j] = color; } } //Down color = GetButtonColor(held, c, ci, 5); for(i = 0; i < 3; i++) { for(j = 0; j < 3; j++) { t[3+i+256*j+6*256] = color; } } //Left color = GetButtonColor(held, c, ci, 6); for(i = 0; i < 3; i++) { for(j = 0; j < 3; j++) { t[3*256+i+256*j] = color; } } //Right color = GetButtonColor(held, c, ci, 7); for(i = 0; i < 3; i++) { for(j = 0; j < 3; j++) { t[6+3*256+i+256*j] = color; } } } } if (FCEUI_AviEnableHUDrecording()) { if (FCEUI_AviDisableMovieMessages()) { snapAVI(); DrawMessage(false); } else { DrawMessage(false); snapAVI(); } } else DrawMessage(false); } void snapAVI() { //Update AVI if(!FCEUI_EmulationPaused()) FCEUI_AviVideoUpdate(XBuf); } void FCEU_DispMessageOnMovie(const char *format, ...) { va_list ap; va_start(ap,format); vsnprintf(guiMessage.errmsg,sizeof(guiMessage.errmsg),format,ap); va_end(ap); if ( vidGuiMsgEna ) { guiMessage.howlong = 180; } guiMessage.isMovieMessage = true; guiMessage.linesFromBottom = 0; if (FCEUI_AviIsRecording() && FCEUI_AviDisableMovieMessages()) guiMessage.howlong = 0; } void FCEU_DispMessage(const char *format, int disppos=0, ...) { va_list ap; va_start(ap,disppos); vsnprintf(guiMessage.errmsg,sizeof(guiMessage.errmsg),format,ap); va_end(ap); // also log messages char temp[2048]; va_start(ap,disppos); vsnprintf(temp,sizeof(temp),format,ap); va_end(ap); strcat(temp, "\n"); FCEU_printf(temp); if ( vidGuiMsgEna ) { guiMessage.howlong = 180; } guiMessage.isMovieMessage = false; guiMessage.linesFromBottom = disppos; //adelikat: Pretty sure this code fails, Movie playback stopped is done with FCEU_DispMessageOnMovie() #ifdef CREATE_AVI if(LoggingEnabled == 2) { /* While in AVI recording mode, only display bare minimum * of messages */ if(strcmp(guiMessage.errmsg, "Movie playback stopped.") != 0) guiMessage.howlong = 0; } #endif } void FCEU_ResetMessages() { guiMessage.howlong = 0; guiMessage.isMovieMessage = false; guiMessage.linesFromBottom = 0; } static int WritePNGChunk(FILE *fp, uint32 size, const char *type, uint8 *data) { uint32 crc; uint8 tempo[4]; tempo[0]=size>>24; tempo[1]=size>>16; tempo[2]=size>>8; tempo[3]=size; if(fwrite(tempo,4,1,fp)!=1) return 0; if(fwrite(type,4,1,fp)!=1) return 0; if(size) if(fwrite(data,1,size,fp)!=size) return 0; crc=CalcCRC32(0,(uint8 *)type,4); if(size) crc=CalcCRC32(crc,data,size); tempo[0]=crc>>24; tempo[1]=crc>>16; tempo[2]=crc>>8; tempo[3]=crc; if(fwrite(tempo,4,1,fp)!=1) return 0; return 1; } uint32 GetScreenPixel(int x, int y, bool usebackup) { uint8 r,g,b; if (((x < 0) || (x > 255)) || ((y < 0) || (y > 255))) return -1; if (usebackup) FCEUD_GetPalette(XBackBuf[(y*256)+x],&r,&g,&b); else FCEUD_GetPalette(XBuf[(y*256)+x],&r,&g,&b); return ((int) (r) << 16) | ((int) (g) << 8) | (int) (b); } int GetScreenPixelPalette(int x, int y, bool usebackup) { if (((x < 0) || (x > 255)) || ((y < 0) || (y > 255))) return -1; if (usebackup) return XBackBuf[(y*256)+x] & 0x3f; else return XBuf[(y*256)+x] & 0x3f; } int SaveSnapshot(void) { #ifdef GEKKO return 0; #else int totallines=FSettings.LastSLine-FSettings.FirstSLine+1; int x,u,y; FILE *pp=NULL; uint8 *compmem=NULL; uLongf compmemsize=(totallines*263+12)*3; if(!(compmem=(uint8 *)FCEU_malloc(compmemsize))) return 0; for (u = lastu; u < 99999; ++u) { pp=FCEUD_UTF8fopen(FCEU_MakeFName(FCEUMKF_SNAP,u,"png").c_str(),"rb"); if(pp==NULL) break; fclose(pp); } lastu = u; if(!(pp=FCEUD_UTF8fopen(FCEU_MakeFName(FCEUMKF_SNAP,u,"png").c_str(),"wb"))) { free(compmem); return 0; } { static const uint8 header[8]={137,80,78,71,13,10,26,10}; if(fwrite(header,8,1,pp)!=1) goto PNGerr; } { uint8 chunko[13]; chunko[0]=chunko[1]=chunko[3]=0; chunko[2]=0x1; // Width of 256 chunko[4]=chunko[5]=chunko[6]=0; chunko[7]=totallines; // Height chunko[8]=8; // 8 bits per sample(24 bits per pixel) chunko[9]=2; // Color type; RGB triplet chunko[10]=0; // compression: deflate chunko[11]=0; // Basic adapative filter set(though none are used). chunko[12]=0; // No interlace. if(!WritePNGChunk(pp,13,"IHDR",chunko)) goto PNGerr; } { uint8 *tmp=XBuf+FSettings.FirstSLine*256; uint8 *dest,*mal,*mork; int bufsize = (256*3+1)*totallines; if(!(mal=mork=dest=(uint8 *)FCEU_dmalloc(bufsize))) goto PNGerr; // mork=dest=XBuf; for(y=0;y>0x10)&0xFF; *dest++=(color>>0x08)&0xFF; *dest++=(color>>0x00)&0xFF; tmp++; } } if(compress(compmem,&compmemsize,mork,bufsize)!=Z_OK) { if(mal) free(mal); goto PNGerr; } if(mal) free(mal); if(!WritePNGChunk(pp,compmemsize,"IDAT",compmem)) goto PNGerr; } if(!WritePNGChunk(pp,0,"IEND",0)) goto PNGerr; free(compmem); fclose(pp); return u+1; PNGerr: if(compmem) free(compmem); if(pp) fclose(pp); return(0); #endif } //overloaded SaveSnapshot for "Savesnapshot As" function int SaveSnapshot(char fileName[512]) { int totallines=FSettings.LastSLine-FSettings.FirstSLine+1; int x,y; FILE *pp=NULL; uint8 *compmem=NULL; uLongf compmemsize=totallines*263+12; if(!(compmem=(uint8 *)FCEU_malloc(compmemsize))) return 0; if(!(pp=FCEUD_UTF8fopen(fileName,"wb"))) { free(compmem); return 0; } { static uint8 header[8]={137,80,78,71,13,10,26,10}; if(fwrite(header,8,1,pp)!=1) goto PNGerr; } { uint8 chunko[13]; chunko[0]=chunko[1]=chunko[3]=0; chunko[2]=0x1; // Width of 256 chunko[4]=chunko[5]=chunko[6]=0; chunko[7]=totallines; // Height chunko[8]=8; // bit depth chunko[9]=3; // Color type; indexed 8-bit chunko[10]=0; // compression: deflate chunko[11]=0; // Basic adapative filter set(though none are used). chunko[12]=0; // No interlace. if(!WritePNGChunk(pp,13,"IHDR",chunko)) goto PNGerr; } { uint8 pdata[256*3]; for(x=0;x<256;x++) FCEUD_GetPalette(x,pdata+x*3,pdata+x*3+1,pdata+x*3+2); if(!WritePNGChunk(pp,256*3,"PLTE",pdata)) goto PNGerr; } { uint8 *tmp=XBuf+FSettings.FirstSLine*256; uint8 *dest,*mal,*mork; if(!(mal=mork=dest=(uint8 *)FCEU_dmalloc((totallines<<8)+totallines))) goto PNGerr; // mork=dest=XBuf; for(y=0;y FCEUD_GetTimeFreq() ) { sprintf(fpsmsg, "%.1f", (double)boopcount / ((double)da / FCEUD_GetTimeFreq())); boopcount = 0; boop_ts = ts; } boopcount++; DrawTextTrans(XBuf + ((256 - ClipSidesOffset) - 40) + (FSettings.FirstSLine + 4) * 256, 256, (uint8*)fpsmsg, 0xA0); #endif }