/* 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ #ifndef WIN32 #include #endif #include #include #include #include #include #ifdef _USE_SHARED_MEMORY_ #include #endif #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 CREATE_AVI #include "drivers/videolog/nesvideos-piece.h" #endif uint8 *XBuf=NULL; uint8 *XBackBuf=NULL; int ClipSidesOffset=0; //Used to move displayed messages when Clips left and right sides is checked static uint8 *xbsave=NULL; GUIMESSAGE guiMessage; GUIMESSAGE subtitleMessage; //for input display extern int input_display; extern uint32 cur_input_display; #ifdef _USE_SHARED_MEMORY_ HANDLE mapXBuf; #endif void FCEU_KillVirtualVideo(void) { //mbg merge TODO 7/17/06 temporarily removed //if(xbsave) //{ // free(xbsave); // xbsave=0; //} //if(XBuf) //{ //UnmapViewOfFile(XBuf); //CloseHandle(mapXBuf); //mapXBuf=NULL; //} //if(XBackBuf) //{ // free(XBackBuf); // XBackBuf=0; //} } /** * 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) { if(!XBuf) /* Some driver code may allocate XBuf externally. */ /* 256 bytes per scanline, * 240 scanline maximum, +16 for alignment, */ #ifdef _USE_SHARED_MEMORY_ mapXBuf = CreateFileMapping((HANDLE)0xFFFFFFFF,NULL,PAGE_READWRITE, 0, 256 * 256 + 16, "fceu.XBuf"); if(mapXBuf == NULL || GetLastError() == ERROR_ALREADY_EXISTS) { CloseHandle(mapXBuf); mapXBuf = NULL; XBuf = (uint8*) (FCEU_malloc(256 * 256 + 16)); XBackBuf = (uint8*) (FCEU_malloc(256 * 256 + 16)); } else { XBuf = (uint8 *)MapViewOfFile(mapXBuf, FILE_MAP_WRITE, 0, 0, 0); XBackBuf = (uint8*) (FCEU_malloc(256 * 256 + 16)); } if (!XBuf || !XBackBuf) { return 0; } #else if(!(XBuf= (uint8*) (FCEU_malloc(256 * 256 + 16))) || !(XBackBuf= (uint8*) (FCEU_malloc(256 * 256 + 16)))) { return 0; } #endif //_USE_SHARED_MEMORY_ xbsave = XBuf; if( sizeof(uint8*) == 4 ) { uintptr_t m = (uintptr_t)XBuf; m = ( 8 - m) & 7; XBuf+=m; } memset(XBuf,128,256*256); //*240); memset(XBackBuf,128,256*256); return 1; } #ifdef FRAMESKIP //#define SHOWFPS void ShowFPS(void); void FCEU_PutImageDummy(void) { #ifdef SHOWFPS ShowFPS(); #endif 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; } static void ReallySnap(void) { int x=SaveSnapshot(); if(!x) FCEU_DispMessage("Error saving screen snapshot."); else FCEU_DispMessage("Screen snapshot %d saved.",x-1); } void FCEU_PutImage(void) { #ifdef SHOWFPS ShowFPS(); #endif if(GameInfo->type==GIT_NSF) { DrawNSF(XBuf); //Save snapshot after NSF screen is drawn. Why would we want to do it before? if(dosnapsave) { 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 //Update AVI before overlay stuff is written if(!FCEUI_EmulationPaused()) FCEUI_AviVideoUpdate(XBuf); //Save snapshot before overlay stuff is written. if(dosnapsave) { ReallySnap(); dosnapsave=0; } if(GameInfo->type==GIT_VSUNI) FCEU_VSUniDraw(XBuf); FCEU_DrawSaveStates(XBuf); FCEU_DrawMovies(XBuf); FCEU_DrawLagCounter(XBuf); FCEU_DrawNTSCControlBars(XBuf); FCEU_DrawRecordingStatus(XBuf); } DrawMessage(false); if(FCEUD_ShouldDrawInputAids()) FCEU_DrawInput(XBuf); //Fancy input display code if(input_display) { int controller, c, color; 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(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; c = cur_input_display >> (controller * 8); c &= 255; //A color = c&1?0xA7:0xCF; 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 = c&2?0xA7:0xCF; 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 = c&4?0xA7:0xCF; for(i = 0; i < 4; i++) { t[11+5*256+i] = color; t[11+6*256+i] = color; } //Start color = c&8?0xA7:0xCF; for(i = 0; i < 4; i++) { t[17+5*256+i] = color; t[17+6*256+i] = color; } //Up color = c&16?0xA7:0xCF; for(i = 0; i < 3; i++) { for(j = 0; j < 3; j++) { t[3+i+256*j] = color; } } //Down color = c&32?0xA7:0xCF; for(i = 0; i < 3; i++) { for(j = 0; j < 3; j++) { t[3+i+256*j+6*256] = color; } } //Left color = c&64?0xA7:0xCF; for(i = 0; i < 3; i++) { for(j = 0; j < 3; j++) { t[3*256+i+256*j] = color; } } //Right color = c&128?0xA7:0xCF; for(i = 0; i < 3; i++) { for(j = 0; j < 3; j++) { t[6+3*256+i+256*j] = color; } } } } } void FCEU_DispMessageOnMovie(char *format, ...) { va_list ap; va_start(ap,format); vsnprintf(guiMessage.errmsg,sizeof(guiMessage.errmsg),format,ap); va_end(ap); guiMessage.howlong = 180; guiMessage.isMovieMessage = true; } void FCEU_DispMessage(char *format, ...) { va_list ap; va_start(ap,format); vsnprintf(guiMessage.errmsg,sizeof(guiMessage.errmsg),format,ap); va_end(ap); guiMessage.howlong = 180; guiMessage.isMovieMessage = false; #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; } static int WritePNGChunk(FILE *fp, uint32 size, 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; } int SaveSnapshot(void) { unsigned int lastu=0; char *fn=0; int totallines=FSettings.LastSLine-FSettings.FirstSLine+1; int x,u,y; FILE *pp=NULL; uint8 *compmem=NULL; uLongf compmemsize=totallines*263+12; if(!(compmem=(uint8 *)FCEU_malloc(compmemsize))) return 0; for(u=lastu;u<99999;u++) { pp=FCEUD_UTF8fopen((fn=strdup(FCEU_MakeFName(FCEUMKF_SNAP,u,"png").c_str())),"rb"); if(pp==NULL) break; fclose(pp); } lastu=u; if(!(pp=FCEUD_UTF8fopen(fn,"wb"))) { free(fn); return 0; } free(fn); { 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 *)malloc((totallines<<8)+totallines))) goto PNGerr; // mork=dest=XBuf; for(y=0;y