2006-07-29 05:46:15 +00:00
# include <stdio.h>
# include <stdlib.h>
# include <string.h>
2008-05-22 07:43:48 +00:00
# include <assert.h>
2008-06-03 05:01:07 +00:00
# include <zlib.h>
2008-06-06 08:23:12 +00:00
# include <iomanip>
2008-06-17 06:55:07 +00:00
# include <fstream>
2008-06-17 08:32:24 +00:00
# include <limits.h>
2008-12-01 00:37:40 +00:00
# include <stdarg.h>
2010-05-17 21:02:38 +00:00
# include "emufile.h"
2009-10-08 13:48:15 +00:00
# include "version.h"
2006-07-29 05:46:15 +00:00
# include "types.h"
2006-08-01 05:50:19 +00:00
# include "utils/endian.h"
2006-07-29 05:46:15 +00:00
# include "palette.h"
# include "input.h"
# include "fceu.h"
2009-08-03 22:24:59 +00:00
# include "netplay.h"
2006-07-29 05:46:15 +00:00
# include "driver.h"
# include "state.h"
2006-08-01 05:50:19 +00:00
# include "file.h"
2006-07-29 05:46:15 +00:00
# include "video.h"
# include "movie.h"
2009-08-03 22:24:59 +00:00
# include "fds.h"
2008-09-24 06:06:31 +00:00
# ifdef _S9XLUA_H
2008-07-23 06:54:14 +00:00
# include "fceulua.h"
2008-09-24 06:06:31 +00:00
# endif
2008-06-17 06:55:07 +00:00
# include "utils/guid.h"
2008-05-23 09:58:38 +00:00
# include "utils/memory.h"
# include "utils/xstring.h"
2008-11-23 21:28:22 +00:00
# include <sstream>
2008-05-24 17:53:14 +00:00
2008-10-25 12:36:03 +00:00
# ifdef CREATE_AVI
# include "drivers/videolog/nesvideos-piece.h"
# endif
2008-08-18 20:34:59 +00:00
# ifdef WIN32
# include <windows.h>
2010-05-15 02:38:32 +00:00
# include "./drivers/win/common.h"
2010-08-21 03:59:17 +00:00
# include "./drivers/win/window.h"
2009-05-25 15:42:24 +00:00
extern void AddRecentMovieFile ( const char * filename ) ;
2011-09-30 19:49:21 +00:00
2012-07-01 14:11:24 +00:00
# include "./drivers/win/taseditor.h"
2012-03-09 15:20:51 +00:00
extern bool emulator_must_run_taseditor ;
2008-08-18 20:34:59 +00:00
# endif
2008-06-03 04:04:04 +00:00
using namespace std ;
2012-10-21 16:40:04 +00:00
# define MOVIE_VERSION 3
2006-07-29 05:46:15 +00:00
extern char FileBase [ ] ;
2008-12-24 21:26:09 +00:00
extern bool AutoSS ; //Declared in fceu.cpp, keeps track if a auto-savestate has been made
2008-05-26 01:10:13 +00:00
2008-11-24 00:45:25 +00:00
std : : vector < int > subtitleFrames ; //Frame numbers for subtitle messages
std : : vector < string > subtitleMessages ; //Messages of subtitles
bool subtitlesOnAVI = false ;
2008-12-23 01:07:55 +00:00
bool autoMovieBackup = false ; //Toggle that determines if movies should be backed up automatically before altering them
bool freshMovie = false ; //True when a movie loads, false when movie is altered. Used to determine if a movie has been altered since opening
2010-04-21 21:10:17 +00:00
bool movieFromPoweron = true ;
2008-12-23 01:07:55 +00:00
2012-04-14 14:06:34 +00:00
static int _currCommand = 0 ;
2008-12-23 01:07:55 +00:00
// Function declarations------------------------
2006-07-29 05:46:15 +00:00
2008-05-23 09:58:38 +00:00
//TODO - remove the synchack stuff from the replay gui and require it to be put into the fm2 file
//which the user would have already converted from fcm
//also cleanup the whole emulator version bullshit in replay. we dont support that old stuff anymore
2008-05-26 00:03:20 +00:00
//todo - better error handling for the compressed savestate
2006-07-29 05:46:15 +00:00
2008-05-26 00:03:20 +00:00
//todo - consider a MemoryBackedFile class..
//..a lot of work.. instead lets just read back from the current fcm
2006-07-29 05:46:15 +00:00
2008-06-20 06:52:20 +00:00
//todo - could we, given a field size, over-read from an inputstream and then parse out an integer?
//that would be faster than several reads, perhaps.
2008-05-23 09:58:38 +00:00
//sometimes we accidentally produce movie stop signals while we're trying to do other things with movies..
bool suppressMovieStop = false ;
2006-07-29 05:46:15 +00:00
2008-05-26 00:03:20 +00:00
//----movie engine main state
2008-06-05 06:49:11 +00:00
EMOVIEMODE movieMode = MOVIEMODE_INACTIVE ;
2006-07-29 05:46:15 +00:00
2008-05-23 09:58:38 +00:00
//this should not be set unless we are in MOVIEMODE_RECORD!
2008-06-06 08:23:12 +00:00
//FILE* fpRecordingMovie = 0;
2010-05-17 21:02:38 +00:00
EMUFILE * osRecordingMovie = NULL ;
2008-05-26 00:03:20 +00:00
2008-05-23 09:58:38 +00:00
int currFrameCounter ;
2008-05-26 00:03:20 +00:00
uint32 cur_input_display = 0 ;
2008-06-21 07:24:01 +00:00
int pauseframe = - 1 ;
2008-05-26 00:03:20 +00:00
bool movie_readonly = true ;
int input_display = 0 ;
int frame_display = 0 ;
2011-09-23 10:50:51 +00:00
int rerecord_display = 0 ;
2010-05-18 12:54:54 +00:00
bool fullSaveStateLoads = false ; //Option for loading a savestates full contents in read+write mode instead of up to the frame count in the savestate (useful as a recovery option)
2008-05-23 09:58:38 +00:00
2008-05-24 22:18:16 +00:00
SFORMAT FCEUMOV_STATEINFO [ ] = {
{ & currFrameCounter , 4 | FCEUSTATE_RLSB , " FCNT " } ,
{ 0 }
} ;
2008-05-26 07:34:40 +00:00
char curMovieFilename [ 512 ] = { 0 } ;
2008-05-28 07:41:49 +00:00
MovieData currMovieData ;
2010-04-28 19:25:52 +00:00
MovieData defaultMovieData ;
2008-06-27 04:39:14 +00:00
int currRerecordCount ;
2008-05-23 09:58:38 +00:00
2008-05-28 07:41:49 +00:00
void MovieData : : clearRecordRange ( int start , int len )
2006-07-29 05:46:15 +00:00
{
2008-05-28 07:41:49 +00:00
for ( int i = 0 ; i < len ; i + + )
2011-10-02 19:04:19 +00:00
{
2008-05-28 07:41:49 +00:00
records [ i + start ] . clear ( ) ;
2011-10-02 19:04:19 +00:00
}
2008-05-28 07:41:49 +00:00
}
2008-05-26 00:03:20 +00:00
2012-11-24 12:56:10 +00:00
void MovieData : : eraseRecords ( int at , int frames )
{
if ( at < ( int ) records . size ( ) )
{
if ( frames = = 1 )
{
// erase 1 frame
records . erase ( records . begin ( ) + at ) ;
} else
{
// erase many frames
if ( at + frames > ( int ) records . size ( ) )
frames = ( int ) records . size ( ) - at ;
records . erase ( records . begin ( ) + at , records . begin ( ) + ( at + frames ) ) ;
}
}
}
2008-06-27 07:16:08 +00:00
void MovieData : : insertEmpty ( int at , int frames )
{
2012-09-01 11:47:09 +00:00
if ( at = = - 1 )
2008-06-27 07:16:08 +00:00
{
2012-09-01 11:47:09 +00:00
records . resize ( records . size ( ) + frames ) ;
2011-10-15 20:20:22 +00:00
} else
2008-06-27 07:16:08 +00:00
{
2011-10-15 20:20:22 +00:00
records . insert ( records . begin ( ) + at , frames , MovieRecord ( ) ) ;
2008-06-27 07:16:08 +00:00
}
}
2011-10-03 17:29:33 +00:00
void MovieData : : cloneRegion ( int at , int frames )
{
2012-09-15 15:03:54 +00:00
if ( at < 0 ) return ;
2012-10-21 16:40:04 +00:00
2011-10-15 20:20:22 +00:00
records . insert ( records . begin ( ) + at , frames , MovieRecord ( ) ) ;
2011-10-03 17:29:33 +00:00
for ( int i = 0 ; i < frames ; i + + )
2012-09-15 15:03:54 +00:00
records [ i + at ] . Clone ( records [ i + at + frames ] ) ;
2011-10-03 17:29:33 +00:00
}
2012-09-01 11:47:09 +00:00
// ----------------------------------------------------------------------------
2010-05-15 01:05:42 +00:00
MovieRecord : : MovieRecord ( )
{
commands = 0 ;
2012-09-01 11:47:09 +00:00
* ( uint32 * ) & joysticks = 0 ;
memset ( zappers , 0 , sizeof ( zappers ) ) ;
2010-05-15 01:05:42 +00:00
}
2008-06-27 07:16:08 +00:00
void MovieRecord : : clear ( )
2012-10-21 16:40:04 +00:00
{
2008-06-29 16:24:37 +00:00
commands = 0 ;
2008-06-27 07:16:08 +00:00
* ( uint32 * ) & joysticks = 0 ;
2012-09-01 11:47:09 +00:00
memset ( zappers , 0 , sizeof ( zappers ) ) ;
2008-06-27 07:16:08 +00:00
}
2010-05-14 23:31:47 +00:00
bool MovieRecord : : Compare ( MovieRecord & compareRec )
{
//Joysticks, Zappers, and commands
2012-10-21 16:40:04 +00:00
2012-09-01 11:47:09 +00:00
if ( this - > commands ! = compareRec . commands )
return false ;
if ( ( * ( uint32 * ) & ( this - > joysticks ) ) ! = ( * ( uint32 * ) & ( compareRec . joysticks ) ) )
return false ;
if ( memcmp ( this - > zappers , compareRec . zappers , sizeof ( zappers ) ) )
return false ;
2010-05-14 23:31:47 +00:00
2012-09-01 11:47:09 +00:00
/*
2012-10-21 16:40:04 +00:00
if ( this - > joysticks ! = compareRec . joysticks )
2010-05-14 23:31:47 +00:00
return false ;
2010-05-14 23:51:42 +00:00
2010-05-15 01:05:42 +00:00
//if new commands are ever recordable, they need to be added here if we go with this method
2010-05-14 23:51:42 +00:00
if ( this - > command_reset ( ) ! = compareRec . command_reset ( ) ) return false ;
2010-05-15 01:40:39 +00:00
if ( this - > command_power ( ) ! = compareRec . command_power ( ) ) return false ;
2010-05-14 23:51:42 +00:00
if ( this - > command_fds_insert ( ) ! = compareRec . command_fds_insert ( ) ) return false ;
if ( this - > command_fds_select ( ) ! = compareRec . command_fds_select ( ) ) return false ;
2012-10-21 16:40:04 +00:00
2010-05-14 23:31:47 +00:00
if ( this - > zappers [ 0 ] . x ! = compareRec . zappers [ 0 ] . x ) return false ;
if ( this - > zappers [ 0 ] . y ! = compareRec . zappers [ 0 ] . y ) return false ;
if ( this - > zappers [ 0 ] . zaphit ! = compareRec . zappers [ 0 ] . zaphit ) return false ;
if ( this - > zappers [ 0 ] . b ! = compareRec . zappers [ 0 ] . b ) return false ;
if ( this - > zappers [ 0 ] . bogo ! = compareRec . zappers [ 0 ] . bogo ) return false ;
if ( this - > zappers [ 1 ] . x ! = compareRec . zappers [ 1 ] . x ) return false ;
if ( this - > zappers [ 1 ] . y ! = compareRec . zappers [ 1 ] . y ) return false ;
if ( this - > zappers [ 1 ] . zaphit ! = compareRec . zappers [ 1 ] . zaphit ) return false ;
if ( this - > zappers [ 1 ] . b ! = compareRec . zappers [ 1 ] . b ) return false ;
if ( this - > zappers [ 1 ] . bogo ! = compareRec . zappers [ 1 ] . bogo ) return false ;
2012-09-01 11:47:09 +00:00
*/
2010-05-14 23:31:47 +00:00
return true ;
}
2011-10-03 17:29:33 +00:00
void MovieRecord : : Clone ( MovieRecord & sourceRec )
{
2012-09-01 11:47:09 +00:00
* ( uint32 * ) & joysticks = * ( uint32 * ) ( & ( sourceRec . joysticks ) ) ;
memcpy ( this - > zappers , sourceRec . zappers , sizeof ( zappers ) ) ;
2011-10-03 17:29:33 +00:00
this - > commands = sourceRec . commands ;
}
2010-05-14 23:31:47 +00:00
2008-05-28 07:41:49 +00:00
const char MovieRecord : : mnemonics [ 8 ] = { ' A ' , ' B ' , ' S ' , ' T ' , ' U ' , ' D ' , ' L ' , ' R ' } ;
2012-09-01 11:47:09 +00:00
2010-05-17 21:02:38 +00:00
void MovieRecord : : dumpJoy ( EMUFILE * os , uint8 joystate )
2008-06-09 08:15:20 +00:00
{
//these are mnemonics for each joystick bit.
//since we usually use the regular joypad, these will be more helpful.
2008-07-04 21:09:49 +00:00
//but any character other than ' ' or '.' should count as a set bit
2008-06-09 08:15:20 +00:00
//maybe other input types will need to be encoded another way..
for ( int bit = 7 ; bit > = 0 ; bit - - )
{
int bitmask = ( 1 < < bit ) ;
char mnemonic = mnemonics [ bit ] ;
//if the bit is set write the mnemonic
if ( joystate & bitmask )
2010-05-17 21:02:38 +00:00
os - > fwrite ( & mnemonic , 1 ) ;
2008-07-04 21:09:49 +00:00
else //otherwise write an unset bit
2010-05-17 21:02:38 +00:00
write8le ( ' . ' , os ) ;
2008-06-09 08:15:20 +00:00
}
}
2010-05-17 21:02:38 +00:00
void MovieRecord : : parseJoy ( EMUFILE * is , uint8 & joystate )
2008-06-09 08:15:20 +00:00
{
2008-06-12 07:10:46 +00:00
char buf [ 8 ] ;
2010-05-17 21:02:38 +00:00
is - > fread ( buf , 8 ) ;
2008-06-09 08:15:20 +00:00
joystate = 0 ;
2008-06-12 07:10:46 +00:00
for ( int i = 0 ; i < 8 ; i + + )
2008-06-09 08:15:20 +00:00
{
2008-06-12 07:10:46 +00:00
joystate < < = 1 ;
2008-07-04 21:09:49 +00:00
joystate | = ( ( buf [ i ] = = ' . ' | | buf [ i ] = = ' ' ) ? 0 : 1 ) ;
2008-06-09 08:15:20 +00:00
}
}
2010-05-17 21:02:38 +00:00
void MovieRecord : : parse ( MovieData * md , EMUFILE * is )
2008-06-09 08:15:20 +00:00
{
//by the time we get in here, the initial pipe has already been extracted
2008-06-12 05:07:30 +00:00
//extract the commands
2008-06-20 06:52:20 +00:00
commands = uint32DecFromIstream ( is ) ;
2008-06-12 07:20:15 +00:00
//*is >> commands;
2010-05-17 21:02:38 +00:00
is - > fgetc ( ) ; //eat the pipe
2008-06-12 05:07:30 +00:00
2008-06-09 08:15:20 +00:00
//a special case: if fourscore is enabled, parse four gamepads
if ( md - > fourscore )
{
2010-05-17 21:02:38 +00:00
parseJoy ( is , joysticks [ 0 ] ) ; is - > fgetc ( ) ; //eat the pipe
parseJoy ( is , joysticks [ 1 ] ) ; is - > fgetc ( ) ; //eat the pipe
parseJoy ( is , joysticks [ 2 ] ) ; is - > fgetc ( ) ; //eat the pipe
parseJoy ( is , joysticks [ 3 ] ) ; is - > fgetc ( ) ; //eat the pipe
2008-06-09 08:15:20 +00:00
}
else
{
for ( int port = 0 ; port < 2 ; port + + )
{
if ( md - > ports [ port ] = = SI_GAMEPAD )
parseJoy ( is , joysticks [ port ] ) ;
else if ( md - > ports [ port ] = = SI_ZAPPER )
{
2008-06-20 06:52:20 +00:00
zappers [ port ] . x = uint32DecFromIstream ( is ) ;
zappers [ port ] . y = uint32DecFromIstream ( is ) ;
zappers [ port ] . b = uint32DecFromIstream ( is ) ;
zappers [ port ] . bogo = uint32DecFromIstream ( is ) ;
zappers [ port ] . zaphit = uint64DecFromIstream ( is ) ;
2008-06-09 08:15:20 +00:00
}
2012-10-21 16:40:04 +00:00
2010-05-17 21:02:38 +00:00
is - > fgetc ( ) ; //eat the pipe
2008-06-09 08:15:20 +00:00
}
}
//(no fcexp data is logged right now)
2010-05-17 21:02:38 +00:00
is - > fgetc ( ) ; //eat the pipe
2008-06-09 08:15:20 +00:00
//should be left at a newline
}
2008-06-20 22:15:30 +00:00
2010-05-17 21:02:38 +00:00
bool MovieRecord : : parseBinary ( MovieData * md , EMUFILE * is )
2008-06-20 22:15:30 +00:00
{
2010-05-17 21:02:38 +00:00
commands = ( uint8 ) is - > fgetc ( ) ;
2008-06-20 22:15:30 +00:00
//check for eof
2010-05-17 21:02:38 +00:00
if ( is - > eof ( ) ) return false ;
2008-06-20 22:15:30 +00:00
if ( md - > fourscore )
{
2010-05-17 21:02:38 +00:00
is - > fread ( ( char * ) & joysticks , 4 ) ;
2008-06-20 22:15:30 +00:00
}
else
{
for ( int port = 0 ; port < 2 ; port + + )
{
if ( md - > ports [ port ] = = SI_GAMEPAD )
2010-05-17 21:02:38 +00:00
joysticks [ port ] = ( uint8 ) is - > fgetc ( ) ;
2008-06-20 22:15:30 +00:00
else if ( md - > ports [ port ] = = SI_ZAPPER )
{
2010-05-17 21:02:38 +00:00
zappers [ port ] . x = ( uint8 ) is - > fgetc ( ) ;
zappers [ port ] . y = ( uint8 ) is - > fgetc ( ) ;
zappers [ port ] . b = ( uint8 ) is - > fgetc ( ) ;
zappers [ port ] . bogo = ( uint8 ) is - > fgetc ( ) ;
2008-06-28 20:20:40 +00:00
read64le ( & zappers [ port ] . zaphit , is ) ;
2008-06-20 22:15:30 +00:00
}
}
}
return true ;
}
2010-05-17 21:02:38 +00:00
void MovieRecord : : dumpBinary ( MovieData * md , EMUFILE * os , int index )
2008-06-20 22:15:30 +00:00
{
2010-05-17 21:02:38 +00:00
write8le ( commands , os ) ;
2008-06-20 22:15:30 +00:00
if ( md - > fourscore )
{
2010-05-17 21:02:38 +00:00
for ( int i = 0 ; i < 4 ; i + + )
os - > fwrite ( & joysticks [ i ] , sizeof ( joysticks [ i ] ) ) ;
2008-06-20 22:15:30 +00:00
}
else
{
for ( int port = 0 ; port < 2 ; port + + )
{
if ( md - > ports [ port ] = = SI_GAMEPAD )
2010-05-17 21:02:38 +00:00
os - > fwrite ( & joysticks [ port ] , sizeof ( joysticks [ port ] ) ) ;
2008-06-20 22:15:30 +00:00
else if ( md - > ports [ port ] = = SI_ZAPPER )
{
2010-05-17 21:02:38 +00:00
write8le ( zappers [ port ] . x , os ) ;
write8le ( zappers [ port ] . y , os ) ;
write8le ( zappers [ port ] . b , os ) ;
write8le ( zappers [ port ] . bogo , os ) ;
2008-06-20 22:15:30 +00:00
write64le ( zappers [ port ] . zaphit , os ) ;
}
}
}
}
2010-05-17 21:02:38 +00:00
void MovieRecord : : dump ( MovieData * md , EMUFILE * os , int index )
2008-06-03 04:04:04 +00:00
{
2008-06-12 05:07:30 +00:00
//dump the misc commands
2008-06-20 06:52:20 +00:00
//*os << '|' << setw(1) << (int)commands;
2010-05-17 21:02:38 +00:00
os - > fputc ( ' | ' ) ;
2008-06-20 06:52:20 +00:00
putdec < uint8 , 1 , true > ( os , commands ) ;
2008-06-12 05:07:30 +00:00
2008-06-09 08:15:20 +00:00
//a special case: if fourscore is enabled, dump four gamepads
if ( md - > fourscore )
{
2010-05-17 21:02:38 +00:00
os - > fputc ( ' | ' ) ;
dumpJoy ( os , joysticks [ 0 ] ) ; os - > fputc ( ' | ' ) ;
dumpJoy ( os , joysticks [ 1 ] ) ; os - > fputc ( ' | ' ) ;
dumpJoy ( os , joysticks [ 2 ] ) ; os - > fputc ( ' | ' ) ;
dumpJoy ( os , joysticks [ 3 ] ) ; os - > fputc ( ' | ' ) ;
2008-06-09 08:15:20 +00:00
}
else
2008-06-03 04:04:04 +00:00
{
2008-06-09 08:15:20 +00:00
for ( int port = 0 ; port < 2 ; port + + )
2008-06-03 04:04:04 +00:00
{
2010-05-17 21:02:38 +00:00
os - > fputc ( ' | ' ) ;
2008-06-09 08:15:20 +00:00
if ( md - > ports [ port ] = = SI_GAMEPAD )
dumpJoy ( os , joysticks [ port ] ) ;
else if ( md - > ports [ port ] = = SI_ZAPPER )
2008-06-20 06:52:20 +00:00
{
2010-05-17 21:02:38 +00:00
putdec < uint8 , 3 , true > ( os , zappers [ port ] . x ) ; os - > fputc ( ' ' ) ;
putdec < uint8 , 3 , true > ( os , zappers [ port ] . y ) ; os - > fputc ( ' ' ) ;
putdec < uint8 , 1 , true > ( os , zappers [ port ] . b ) ; os - > fputc ( ' ' ) ;
putdec < uint8 , 1 , true > ( os , zappers [ port ] . bogo ) ; os - > fputc ( ' ' ) ;
2008-06-20 06:52:20 +00:00
putdec < uint64 , 20 , false > ( os , zappers [ port ] . zaphit ) ;
}
2008-06-03 04:04:04 +00:00
}
2010-05-17 21:02:38 +00:00
os - > fputc ( ' | ' ) ;
2008-06-03 04:04:04 +00:00
}
2012-10-21 16:40:04 +00:00
2008-06-09 08:15:20 +00:00
//(no fcexp data is logged right now)
2010-05-17 21:02:38 +00:00
os - > fputc ( ' | ' ) ;
2006-07-29 05:46:15 +00:00
2008-05-28 07:41:49 +00:00
//each frame is on a new line
2010-05-17 21:02:38 +00:00
os - > fputc ( ' \n ' ) ;
2008-05-28 07:41:49 +00:00
}
2008-05-23 09:58:38 +00:00
2008-05-28 07:41:49 +00:00
MovieData : : MovieData ( )
: version ( MOVIE_VERSION )
, emuVersion ( FCEU_VERSION_NUMERIC )
, palFlag ( false )
2010-05-14 16:55:15 +00:00
, PPUflag ( false )
2009-10-01 14:59:41 +00:00
, rerecordCount ( 0 )
2008-10-25 12:36:03 +00:00
, binaryFlag ( false )
2012-01-08 18:19:49 +00:00
, loadFrameCount ( - 1 )
2010-04-28 19:25:52 +00:00
, microphone ( false )
2008-05-28 07:41:49 +00:00
{
memset ( & romChecksum , 0 , sizeof ( MD5DATA ) ) ;
}
2006-07-29 05:46:15 +00:00
2008-05-28 07:41:49 +00:00
void MovieData : : truncateAt ( int frame )
{
records . resize ( frame ) ;
}
2008-05-23 09:58:38 +00:00
2008-06-09 08:15:20 +00:00
void MovieData : : installValue ( std : : string & key , std : : string & val )
{
//todo - use another config system, or drive this from a little data structure. because this is gross
2010-03-02 04:59:45 +00:00
if ( key = = " FDS " )
installInt ( val , fds ) ;
2010-05-14 16:55:15 +00:00
else if ( key = = " NewPPU " )
2010-05-14 16:03:07 +00:00
installBool ( val , PPUflag ) ;
2010-03-04 02:03:33 +00:00
else if ( key = = " version " )
2008-06-09 08:15:20 +00:00
installInt ( val , version ) ;
else if ( key = = " emuVersion " )
installInt ( val , emuVersion ) ;
2008-06-22 06:53:49 +00:00
else if ( key = = " rerecordCount " )
installInt ( val , rerecordCount ) ;
2008-06-09 08:15:20 +00:00
else if ( key = = " palFlag " )
installBool ( val , palFlag ) ;
else if ( key = = " romFilename " )
romFilename = val ;
else if ( key = = " romChecksum " )
StringToBytes ( val , & romChecksum , MD5DATA : : size ) ;
else if ( key = = " guid " )
guid = FCEU_Guid : : fromString ( val ) ;
else if ( key = = " fourscore " )
installBool ( val , fourscore ) ;
2010-04-28 19:25:52 +00:00
else if ( key = = " microphone " )
installBool ( val , microphone ) ;
2008-06-09 08:15:20 +00:00
else if ( key = = " port0 " )
installInt ( val , ports [ 0 ] ) ;
else if ( key = = " port1 " )
installInt ( val , ports [ 1 ] ) ;
else if ( key = = " port2 " )
installInt ( val , ports [ 2 ] ) ;
2008-06-20 22:15:30 +00:00
else if ( key = = " binary " )
installBool ( val , binaryFlag ) ;
2008-06-22 23:41:24 +00:00
else if ( key = = " comment " )
2008-08-10 08:59:23 +00:00
comments . push_back ( mbstowcs ( val ) ) ;
2008-11-23 21:28:22 +00:00
else if ( key = = " subtitle " )
subtitles . push_back ( val ) ; //mbstowcs(val));
2008-06-09 08:15:20 +00:00
else if ( key = = " savestate " )
2008-05-23 09:58:38 +00:00
{
2008-07-24 03:01:39 +00:00
int len = Base64StringToBytesLength ( val ) ;
if ( len = = - 1 ) len = HexStringToBytesLength ( val ) ; // wasn't base64, try hex
2008-05-28 07:41:49 +00:00
if ( len > = 1 )
2008-05-23 09:58:38 +00:00
{
2008-05-28 07:41:49 +00:00
savestate . resize ( len ) ;
2008-07-24 03:01:39 +00:00
StringToBytes ( val , & savestate [ 0 ] , len ) ; // decodes either base64 or hex
2008-05-23 09:58:38 +00:00
}
2006-07-29 05:46:15 +00:00
}
2009-12-18 05:40:38 +00:00
else if ( key = = " length " )
{
installInt ( val , loadFrameCount ) ;
}
2008-05-28 07:41:49 +00:00
}
2006-07-29 05:46:15 +00:00
2010-05-17 21:02:38 +00:00
int MovieData : : dump ( EMUFILE * os , bool binary )
{
int start = os - > ftell ( ) ;
os - > fprintf ( " version %d \n " , version ) ;
os - > fprintf ( " emuVersion %d \n " , emuVersion ) ;
os - > fprintf ( " rerecordCount %d \n " , rerecordCount ) ;
os - > fprintf ( " palFlag %d \n " , ( palFlag ? 1 : 0 ) ) ;
os - > fprintf ( " romFilename %s \n " , romFilename . c_str ( ) ) ;
os - > fprintf ( " romChecksum %s \n " , BytesToString ( romChecksum . data , MD5DATA : : size ) . c_str ( ) ) ;
os - > fprintf ( " guid %s \n " , guid . toString ( ) . c_str ( ) ) ;
os - > fprintf ( " fourscore %d \n " , ( fourscore ? 1 : 0 ) ) ;
os - > fprintf ( " microphone %d \n " , ( microphone ? 1 : 0 ) ) ;
os - > fprintf ( " port0 %d \n " , ports [ 0 ] ) ;
os - > fprintf ( " port1 %d \n " , ports [ 1 ] ) ;
os - > fprintf ( " port2 %d \n " , ports [ 2 ] ) ;
2011-09-12 12:23:48 +00:00
os - > fprintf ( " FDS %d \n " , fds ? 1 : 0 ) ;
os - > fprintf ( " NewPPU %d \n " , PPUflag ? 1 : 0 ) ;
2008-06-22 22:25:30 +00:00
2008-06-22 23:41:24 +00:00
for ( uint32 i = 0 ; i < comments . size ( ) ; i + + )
2010-05-17 21:02:38 +00:00
os - > fprintf ( " comment %s \n " , wcstombs ( comments [ i ] ) . c_str ( ) ) ;
2008-11-23 21:28:22 +00:00
for ( uint32 i = 0 ; i < subtitles . size ( ) ; i + + )
2010-05-17 21:02:38 +00:00
os - > fprintf ( " subtitle %s \n " , subtitles [ i ] . c_str ( ) ) ;
2012-10-21 16:40:04 +00:00
2008-06-20 22:15:30 +00:00
if ( binary )
2010-05-17 21:02:38 +00:00
os - > fprintf ( " binary 1 \n " ) ;
2012-10-21 16:40:04 +00:00
2011-12-05 19:35:15 +00:00
if ( savestate . size ( ) )
2010-05-17 21:02:38 +00:00
os - > fprintf ( " savestate %s \n " , BytesToString ( & savestate [ 0 ] , savestate . size ( ) ) . c_str ( ) ) ;
2009-12-18 05:40:38 +00:00
2011-12-09 14:53:20 +00:00
if ( this - > loadFrameCount > = 0 )
os - > fprintf ( " length %d \n " , this - > loadFrameCount ) ;
2009-12-18 05:40:38 +00:00
2008-06-20 22:15:30 +00:00
if ( binary )
{
//put one | to start the binary dump
2010-05-17 21:02:38 +00:00
os - > fputc ( ' | ' ) ;
2008-06-20 22:15:30 +00:00
for ( int i = 0 ; i < ( int ) records . size ( ) ; i + + )
2011-12-05 19:35:15 +00:00
records [ i ] . dumpBinary ( this , os , i ) ;
} else
{
2008-06-20 22:15:30 +00:00
for ( int i = 0 ; i < ( int ) records . size ( ) ; i + + )
2011-12-05 19:35:15 +00:00
records [ i ] . dump ( this , os , i ) ;
}
2008-06-20 22:15:30 +00:00
2010-05-17 21:02:38 +00:00
int end = os - > ftell ( ) ;
2008-06-17 07:55:27 +00:00
return end - start ;
2008-06-02 14:02:02 +00:00
}
2008-05-26 00:03:20 +00:00
int FCEUMOV_GetFrame ( void )
{
return currFrameCounter ;
}
2008-05-24 22:18:16 +00:00
2009-03-07 19:26:07 +00:00
int FCEUI_GetLagCount ( void )
{
return lagCounter ;
}
2009-03-08 19:14:37 +00:00
bool FCEUI_GetLagged ( void )
{
2012-09-13 19:48:14 +00:00
if ( lagFlag )
return true ;
else
return false ;
}
void FCEUI_SetLagFlag ( bool value )
{
lagFlag = ( value ) ? 1 : 0 ;
2009-03-08 19:14:37 +00:00
}
2008-05-26 00:03:20 +00:00
bool FCEUMOV_ShouldPause ( void )
2008-05-23 09:58:38 +00:00
{
2011-09-25 18:17:26 +00:00
if ( pauseframe & & currFrameCounter + 1 = = pauseframe )
2008-05-26 00:03:20 +00:00
{
2011-09-25 18:17:26 +00:00
pauseframe = 0 ;
2008-05-26 00:03:20 +00:00
return true ;
}
else
{
return false ;
}
}
2008-06-05 06:49:11 +00:00
EMOVIEMODE FCEUMOV_Mode ( )
{
return movieMode ;
}
bool FCEUMOV_Mode ( EMOVIEMODE modemask )
2008-05-26 00:03:20 +00:00
{
2008-06-05 06:49:11 +00:00
return ( movieMode & modemask ) ! = 0 ;
2008-05-26 00:03:20 +00:00
}
2008-06-05 06:49:11 +00:00
bool FCEUMOV_Mode ( int modemask )
2008-05-26 00:03:20 +00:00
{
2008-06-05 06:49:11 +00:00
return FCEUMOV_Mode ( ( EMOVIEMODE ) modemask ) ;
2006-07-29 05:46:15 +00:00
}
2010-05-17 21:02:38 +00:00
static void LoadFM2_binarychunk ( MovieData & movieData , EMUFILE * fp , int size )
2008-06-20 22:15:30 +00:00
{
int recordsize = 1 ; //1 for the command
if ( movieData . fourscore )
recordsize + = 4 ; //4 joysticks
else
{
for ( int i = 0 ; i < 2 ; i + + )
{
switch ( movieData . ports [ i ] )
{
case SI_GAMEPAD : recordsize + + ; break ;
2008-06-28 20:20:40 +00:00
case SI_ZAPPER : recordsize + = 12 ; break ;
2008-06-20 22:15:30 +00:00
}
}
}
//find out how much remains in the file
2010-05-17 21:02:38 +00:00
int curr = fp - > ftell ( ) ;
fp - > fseek ( 0 , SEEK_END ) ;
int end = fp - > ftell ( ) ;
2008-06-20 22:15:30 +00:00
int flen = end - curr ;
2010-05-17 21:02:38 +00:00
fp - > fseek ( curr , SEEK_SET ) ;
2008-06-20 22:15:30 +00:00
//the amount todo is the min of the limiting size we received and the remaining contents of the file
int todo = std : : min ( size , flen ) ;
int numRecords = todo / recordsize ;
2009-12-18 05:40:38 +00:00
if ( movieData . loadFrameCount ! = - 1 & & movieData . loadFrameCount < numRecords )
numRecords = movieData . loadFrameCount ;
2008-06-20 22:15:30 +00:00
movieData . records . resize ( numRecords ) ;
for ( int i = 0 ; i < numRecords ; i + + )
{
movieData . records [ i ] . parseBinary ( & movieData , fp ) ;
}
}
2008-05-23 09:58:38 +00:00
//yuck... another custom text parser.
2010-05-17 21:02:38 +00:00
bool LoadFM2 ( MovieData & movieData , EMUFILE * fp , int size , bool stopAfterHeader )
2006-07-29 05:46:15 +00:00
{
2011-12-10 12:58:02 +00:00
// if there's no "binary" tag in the movie header, consider it as a movie in text format
movieData . binaryFlag = false ;
2011-12-09 14:53:20 +00:00
// Non-TASEditor projects consume until EOF
2011-12-10 12:58:02 +00:00
movieData . loadFrameCount = - 1 ;
2009-12-18 05:40:38 +00:00
2010-05-17 21:02:38 +00:00
std : : ios : : pos_type curr = fp - > ftell ( ) ;
2012-02-13 15:49:02 +00:00
if ( ! stopAfterHeader )
{
// first, look for an fcm signature
char fcmbuf [ 3 ] ;
fp - > fread ( fcmbuf , 3 ) ;
fp - > fseek ( curr , SEEK_SET ) ;
if ( ! strncmp ( fcmbuf , " FCM " , 3 ) ) {
FCEU_PrintError ( " FCM File format is no longer supported. Please use Tools > Convert FCM " ) ;
return false ;
}
2008-08-12 06:58:52 +00:00
}
2008-08-10 02:33:14 +00:00
//movie must start with "version 3"
char buf [ 9 ] ;
2010-05-17 21:02:38 +00:00
curr = fp - > ftell ( ) ;
fp - > fread ( buf , 9 ) ;
fp - > fseek ( curr , SEEK_SET ) ;
2008-08-10 02:33:14 +00:00
if ( fp - > fail ( ) ) return false ;
2012-10-21 16:40:04 +00:00
if ( memcmp ( buf , " version 3 " , 9 ) )
2008-08-10 02:33:14 +00:00
return false ;
2008-05-23 09:58:38 +00:00
std : : string key , value ;
enum {
2008-11-23 21:28:22 +00:00
NEWLINE , KEY , SEPARATOR , VALUE , RECORD , COMMENT , SUBTITLE
2008-05-23 09:58:38 +00:00
} state = NEWLINE ;
bool bail = false ;
2012-02-18 16:43:40 +00:00
bool iswhitespace , isrecchar , isnewline ;
int c ;
2008-05-23 09:58:38 +00:00
for ( ; ; )
2007-02-06 18:59:15 +00:00
{
2008-06-17 08:32:24 +00:00
if ( size - - < = 0 ) goto bail ;
2010-05-17 21:02:38 +00:00
c = fp - > fgetc ( ) ;
2008-05-23 09:58:38 +00:00
if ( c = = - 1 )
goto bail ;
2008-05-26 00:33:27 +00:00
iswhitespace = ( c = = ' ' | | c = = ' \t ' ) ;
isrecchar = ( c = = ' | ' ) ;
isnewline = ( c = = 10 | | c = = 13 ) ;
2008-06-20 22:15:30 +00:00
if ( isrecchar & & movieData . binaryFlag & & ! stopAfterHeader )
{
LoadFM2_binarychunk ( movieData , fp , size ) ;
2008-08-10 02:33:14 +00:00
return true ;
2011-12-10 12:58:02 +00:00
} else if ( isnewline & & movieData . loadFrameCount = = movieData . records . size ( ) )
// exit prematurely if loaded the specified amound of records
return true ;
2008-05-23 09:58:38 +00:00
switch ( state )
{
case NEWLINE :
if ( isnewline ) goto done ;
2008-05-24 22:18:16 +00:00
if ( iswhitespace ) goto done ;
2012-10-21 16:40:04 +00:00
if ( isrecchar )
2008-05-24 22:18:16 +00:00
goto dorecord ;
//must be a key
key = " " ;
value = " " ;
goto dokey ;
2008-05-23 09:58:38 +00:00
break ;
2008-06-06 08:23:12 +00:00
case RECORD :
{
2008-05-23 09:58:38 +00:00
dorecord :
2008-08-10 02:33:14 +00:00
if ( stopAfterHeader ) return true ;
2008-06-20 22:15:30 +00:00
int currcount = movieData . records . size ( ) ;
movieData . records . resize ( currcount + 1 ) ;
2010-05-17 21:02:38 +00:00
int preparse = fp - > ftell ( ) ;
2008-06-20 22:15:30 +00:00
movieData . records [ currcount ] . parse ( & movieData , fp ) ;
2010-05-17 21:02:38 +00:00
int postparse = fp - > ftell ( ) ;
2008-06-17 08:32:24 +00:00
size - = ( postparse - preparse ) ;
2008-05-23 09:58:38 +00:00
state = NEWLINE ;
break ;
}
2008-06-06 08:23:12 +00:00
2008-05-23 09:58:38 +00:00
case KEY :
dokey : //dookie
state = KEY ;
if ( iswhitespace ) goto doseparator ;
if ( isnewline ) goto commit ;
key + = c ;
break ;
case SEPARATOR :
doseparator :
state = SEPARATOR ;
if ( isnewline ) goto commit ;
if ( ! iswhitespace ) goto dovalue ;
break ;
case VALUE :
dovalue :
state = VALUE ;
if ( isnewline ) goto commit ;
value + = c ;
}
goto done ;
bail :
bail = true ;
if ( state = = VALUE ) goto commit ;
2008-06-19 16:19:48 +00:00
goto done ;
2008-05-23 09:58:38 +00:00
commit :
2008-06-09 08:15:20 +00:00
movieData . installValue ( key , value ) ;
2008-05-23 09:58:38 +00:00
state = NEWLINE ;
done : ;
2008-06-19 16:19:48 +00:00
if ( bail ) break ;
2007-02-06 18:59:15 +00:00
}
2008-08-10 02:33:14 +00:00
return true ;
2008-05-23 09:58:38 +00:00
}
2007-02-06 18:59:15 +00:00
2010-06-22 14:29:05 +00:00
/// Stop movie playback.
static void StopPlayback ( )
{
FCEU_DispMessageOnMovie ( " Movie playback stopped. " ) ;
movieMode = MOVIEMODE_INACTIVE ;
}
// Stop movie playback without closing the movie.
static void FinishPlayback ( )
{
extern int closeFinishedMovie ;
if ( closeFinishedMovie )
StopPlayback ( ) ;
else
{
FCEU_DispMessage ( " Movie finished playing. " , 0 ) ;
movieMode = MOVIEMODE_FINISHED ;
}
}
2008-06-22 06:53:49 +00:00
static void closeRecordingMovie ( )
{
if ( osRecordingMovie )
{
delete osRecordingMovie ;
osRecordingMovie = 0 ;
}
}
2007-02-06 18:59:15 +00:00
2008-05-23 09:58:38 +00:00
/// Stop movie recording
2008-06-22 06:53:49 +00:00
static void StopRecording ( )
2008-05-23 09:58:38 +00:00
{
2010-05-16 04:18:49 +00:00
FCEU_DispMessage ( " Movie recording stopped. " , 0 ) ;
2008-05-26 00:03:20 +00:00
movieMode = MOVIEMODE_INACTIVE ;
2012-10-21 16:40:04 +00:00
2008-06-22 06:53:49 +00:00
closeRecordingMovie ( ) ;
2006-07-29 05:46:15 +00:00
}
2007-02-06 18:59:15 +00:00
void FCEUI_StopMovie ( )
2006-07-29 05:46:15 +00:00
{
2008-05-23 09:58:38 +00:00
if ( suppressMovieStop )
return ;
2012-10-21 16:40:04 +00:00
2010-05-12 15:31:24 +00:00
if ( movieMode = = MOVIEMODE_PLAY | | movieMode = = MOVIEMODE_FINISHED )
2007-02-06 18:59:15 +00:00
StopPlayback ( ) ;
2008-05-23 09:58:38 +00:00
else if ( movieMode = = MOVIEMODE_RECORD )
2007-02-06 18:59:15 +00:00
StopRecording ( ) ;
2008-05-26 07:34:40 +00:00
2008-12-24 21:26:09 +00:00
curMovieFilename [ 0 ] = 0 ; //No longer a current movie filename
freshMovie = false ; //No longer a fresh movie loaded
if ( bindSavestate ) AutoSS = false ; //If bind movies to savestates is true, then there is no longer a valid auto-save to load
2010-08-21 03:59:17 +00:00
# ifdef WIN32
SetMainWindowText ( ) ;
# endif
2006-07-29 05:46:15 +00:00
}
2009-11-14 02:46:03 +00:00
void poweron ( bool shouldDisableBatteryLoading )
2008-06-05 06:49:11 +00:00
{
2008-06-25 06:34:08 +00:00
//// make a for-movie-recording power-on clear the game's save data, too
2008-08-12 06:52:48 +00:00
//extern char lastLoadedGameName [2048];
//extern int disableBatteryLoading, suppressAddPowerCommand;
//suppressAddPowerCommand=1;
//if(shouldDisableBatteryLoading) disableBatteryLoading=1;
//suppressMovieStop=true;
//{
// //we need to save the pause state through this process
// int oldPaused = EmulationPaused;
// // NOTE: this will NOT write an FCEUNPCMD_POWER into the movie file
// FCEUGI* gi = FCEUI_LoadGame(lastLoadedGameName, 0);
// assert(gi);
// PowerNES();
// EmulationPaused = oldPaused;
//}
//suppressMovieStop=false;
//if(shouldDisableBatteryLoading) disableBatteryLoading=0;
//suppressAddPowerCommand=0;
2008-06-28 23:31:02 +00:00
extern int disableBatteryLoading ;
2008-08-12 06:52:48 +00:00
disableBatteryLoading = 1 ;
2008-06-25 06:34:08 +00:00
PowerNES ( ) ;
2008-08-12 06:52:48 +00:00
disableBatteryLoading = 0 ;
2008-06-05 06:49:11 +00:00
}
2012-04-14 14:06:34 +00:00
void FCEUMOV_CreateCleanMovie ( )
2008-06-05 06:49:11 +00:00
{
currMovieData = MovieData ( ) ;
2011-09-12 12:23:48 +00:00
currMovieData . palFlag = FCEUI_GetCurrentVidSystem ( 0 , 0 ) ! = 0 ;
currMovieData . romFilename = FileBase ;
currMovieData . romChecksum = GameInfo - > MD5 ;
currMovieData . guid . newGuid ( ) ;
currMovieData . fourscore = FCEUI_GetInputFourscore ( ) ;
currMovieData . microphone = FCEUI_GetInputMicrophone ( ) ;
currMovieData . ports [ 0 ] = joyports [ 0 ] . type ;
currMovieData . ports [ 1 ] = joyports [ 1 ] . type ;
currMovieData . ports [ 2 ] = portFC . type ;
currMovieData . fds = isFDS ;
currMovieData . PPUflag = ( newppu ! = 0 ) ;
2008-06-05 06:49:11 +00:00
}
2012-04-14 14:06:34 +00:00
void FCEUMOV_ClearCommands ( )
{
_currCommand = 0 ;
}
2008-06-05 06:49:11 +00:00
2010-04-21 21:10:17 +00:00
bool FCEUMOV_FromPoweron ( )
{
return movieFromPoweron ;
}
2010-05-17 21:02:38 +00:00
bool MovieData : : loadSavestateFrom ( std : : vector < uint8 > * buf )
2008-05-28 07:41:49 +00:00
{
2010-05-17 21:02:38 +00:00
EMUFILE_MEMORY ms ( buf ) ;
2008-06-17 06:55:07 +00:00
return FCEUSS_LoadFP ( & ms , SSLOADPARAM_BACKUP ) ;
2008-05-28 07:41:49 +00:00
}
2010-05-17 21:02:38 +00:00
void MovieData : : dumpSavestateTo ( std : : vector < uint8 > * buf , int compressionLevel )
2008-05-28 07:41:49 +00:00
{
2010-05-17 21:02:38 +00:00
EMUFILE_MEMORY ms ( buf ) ;
2008-06-03 05:01:07 +00:00
FCEUSS_SaveMS ( & ms , compressionLevel ) ;
2008-06-03 05:50:58 +00:00
ms . trim ( ) ;
2008-05-28 07:41:49 +00:00
}
2006-07-29 05:46:15 +00:00
2008-05-23 09:58:38 +00:00
//begin playing an existing movie
2011-12-09 14:53:20 +00:00
bool FCEUI_LoadMovie ( const char * fname , bool _read_only , int _pauseframe )
2006-07-29 05:46:15 +00:00
{
2011-12-09 14:53:20 +00:00
if ( ! FCEU_IsValidUI ( FCEUI_PLAYMOVIE ) )
2009-05-25 18:17:19 +00:00
return true ; //adelikat: file did not fail to load, so let's return true here, just do nothing
2008-06-05 06:49:11 +00:00
2008-05-23 09:58:38 +00:00
assert ( fname ) ;
2008-05-22 07:43:48 +00:00
2008-06-11 04:54:33 +00:00
//mbg 6/10/08 - we used to call StopMovie here, but that cleared curMovieFilename and gave us crashes...
2010-05-12 15:31:24 +00:00
if ( movieMode = = MOVIEMODE_PLAY | | movieMode = = MOVIEMODE_FINISHED )
2008-06-11 04:54:33 +00:00
StopPlayback ( ) ;
else if ( movieMode = = MOVIEMODE_RECORD )
StopRecording ( ) ;
//--------------
2008-05-22 07:43:48 +00:00
2008-05-23 09:58:38 +00:00
currMovieData = MovieData ( ) ;
2012-10-21 16:40:04 +00:00
2008-05-26 00:03:20 +00:00
strcpy ( curMovieFilename , fname ) ;
2008-08-10 02:33:14 +00:00
FCEUFILE * fp = FCEU_fopen ( fname , 0 , " rb " , 0 ) ;
2009-05-25 18:17:19 +00:00
if ( ! fp ) return false ;
2008-08-10 02:33:14 +00:00
if ( fp - > isArchive ( ) & & ! _read_only ) {
FCEU_PrintError ( " Cannot open a movie in read+write from an archive. " ) ;
2009-05-25 18:17:19 +00:00
return true ; //adelikat: file did not fail to load, so return true (false is only for file not exist/unable to open errors
2008-08-10 02:33:14 +00:00
}
2009-05-25 15:42:24 +00:00
# ifdef WIN32
2010-05-15 02:06:15 +00:00
//Fix relative path if necessary and then add to the recent movie menu
extern std : : string BaseDirectory ;
2010-05-15 02:38:32 +00:00
2010-05-15 02:06:15 +00:00
string name = fname ;
2010-05-15 02:38:32 +00:00
if ( IsRelativePath ( fname ) )
2010-05-15 02:06:15 +00:00
{
2010-05-15 02:38:32 +00:00
name = ConvertRelativePath ( name ) ;
2010-05-15 02:06:15 +00:00
}
AddRecentMovieFile ( name . c_str ( ) ) ;
2009-05-25 15:42:24 +00:00
# endif
2010-06-07 21:39:10 +00:00
LoadFM2 ( currMovieData , fp - > stream , fp - > size , false ) ;
2009-04-23 21:34:30 +00:00
LoadSubtitles ( currMovieData ) ;
2008-06-09 08:15:20 +00:00
delete fp ;
2008-05-22 07:43:48 +00:00
2008-12-23 01:07:55 +00:00
freshMovie = true ; //Movie has been loaded, so it must be unaltered
2008-12-24 21:26:09 +00:00
if ( bindSavestate ) AutoSS = false ; //If bind savestate to movie is true, then their isn't a valid auto-save to load, so flag it
2008-06-22 06:53:49 +00:00
//fully reload the game to reinitialize everything before playing any movie
poweron ( true ) ;
2008-05-22 07:43:48 +00:00
2011-12-09 14:53:20 +00:00
if ( currMovieData . savestate . size ( ) )
2008-05-22 07:43:48 +00:00
{
2011-12-09 14:53:20 +00:00
//WE NEED TO LOAD A SAVESTATE
2010-04-21 21:10:17 +00:00
movieFromPoweron = false ;
2008-05-28 07:41:49 +00:00
bool success = MovieData : : loadSavestateFrom ( & currMovieData . savestate ) ;
2011-12-09 14:53:20 +00:00
if ( ! success ) return true ; //adelikat: I guess return true here? False is only for a bad movie filename, if it got this far the file was good?
2010-04-21 21:10:17 +00:00
} else {
movieFromPoweron = true ;
2008-05-22 07:43:48 +00:00
}
2008-05-26 07:34:40 +00:00
//if there is no savestate, we won't have this crucial piece of information at the start of the movie.
//so, we have to include it with the movie
2008-05-23 09:58:38 +00:00
if ( currMovieData . palFlag )
2008-05-22 07:43:48 +00:00
FCEUI_SetVidSystem ( 1 ) ;
else
FCEUI_SetVidSystem ( 0 ) ;
2008-06-09 08:15:20 +00:00
//force the input configuration stored in the movie to apply
2011-12-06 12:23:42 +00:00
FCEUD_SetInput ( currMovieData . fourscore , currMovieData . microphone , ( ESI ) currMovieData . ports [ 0 ] , ( ESI ) currMovieData . ports [ 1 ] , ( ESIFC ) currMovieData . ports [ 2 ] ) ;
2008-05-22 07:43:48 +00:00
2008-05-26 00:03:20 +00:00
//stuff that should only happen when we're ready to positively commit to the replay
2011-12-09 14:53:20 +00:00
currFrameCounter = 0 ;
pauseframe = _pauseframe ;
movie_readonly = _read_only ;
movieMode = MOVIEMODE_PLAY ;
currRerecordCount = currMovieData . rerecordCount ;
2008-06-21 07:24:01 +00:00
2011-12-09 14:53:20 +00:00
if ( movie_readonly )
FCEU_DispMessage ( " Replay started Read-Only. " , 0 ) ;
else
FCEU_DispMessage ( " Replay started Read+Write. " , 0 ) ;
2012-10-21 16:40:04 +00:00
2010-08-21 03:59:17 +00:00
# ifdef WIN32
SetMainWindowText ( ) ;
# endif
2008-10-25 12:36:03 +00:00
# ifdef CREATE_AVI
if ( LoggingEnabled )
{
2010-05-16 04:18:49 +00:00
FCEU_DispMessage ( " Video recording enabled. \n " , 0 ) ;
2008-10-25 12:36:03 +00:00
LoggingEnabled = 2 ;
}
# endif
2012-10-21 16:40:04 +00:00
2009-05-25 18:17:19 +00:00
return true ;
2008-05-23 09:58:38 +00:00
}
2008-05-22 07:43:48 +00:00
2008-05-26 00:03:20 +00:00
static void openRecordingMovie ( const char * fname )
{
2008-06-06 08:23:12 +00:00
osRecordingMovie = FCEUD_UTF8_fstream ( fname , " wb " ) ;
if ( ! osRecordingMovie )
2008-05-26 00:03:20 +00:00
FCEU_PrintError ( " Error opening movie output file: %s " , fname ) ;
strcpy ( curMovieFilename , fname ) ;
2010-05-15 01:48:53 +00:00
# ifdef WIN32
//Add to the recent movie menu
AddRecentMovieFile ( fname ) ;
# endif
2008-05-26 00:03:20 +00:00
}
2008-05-22 07:43:48 +00:00
2008-06-05 06:49:11 +00:00
2008-05-23 09:58:38 +00:00
//begin recording a new movie
2008-06-21 07:24:01 +00:00
//TODO - BUG - the record-from-another-savestate doesnt work.
2008-08-10 08:59:23 +00:00
void FCEUI_SaveMovie ( const char * fname , EMOVIE_FLAG flags , std : : wstring author )
2006-07-29 05:46:15 +00:00
{
2008-06-05 06:49:11 +00:00
if ( ! FCEU_IsValidUI ( FCEUI_RECORDMOVIE ) )
return ;
2008-05-23 09:58:38 +00:00
assert ( fname ) ;
2008-05-22 07:43:48 +00:00
FCEUI_StopMovie ( ) ;
2008-05-26 00:03:20 +00:00
openRecordingMovie ( fname ) ;
2008-05-22 07:43:48 +00:00
2008-05-26 02:20:56 +00:00
currFrameCounter = 0 ;
2008-08-07 04:56:40 +00:00
LagCounterReset ( ) ;
2012-04-14 14:06:34 +00:00
FCEUMOV_CreateCleanMovie ( ) ;
2008-08-10 08:59:23 +00:00
if ( author ! = L " " ) currMovieData . comments . push_back ( L " author " + author ) ;
2011-09-12 12:23:48 +00:00
2008-06-21 07:24:01 +00:00
if ( flags & MOVIE_FLAG_FROM_POWERON )
2008-05-22 07:43:48 +00:00
{
2010-04-21 21:10:17 +00:00
movieFromPoweron = true ;
2008-06-05 06:49:11 +00:00
poweron ( true ) ;
2008-05-22 07:43:48 +00:00
}
else
{
2010-04-21 21:10:17 +00:00
movieFromPoweron = false ;
2008-06-03 05:01:07 +00:00
MovieData : : dumpSavestateTo ( & currMovieData . savestate , Z_BEST_COMPRESSION ) ;
2008-05-22 07:43:48 +00:00
}
2012-04-14 14:06:34 +00:00
FCEUMOV_ClearCommands ( ) ;
2008-05-23 09:58:38 +00:00
//we are going to go ahead and dump the header. from now on we will only be appending frames
2008-06-20 22:15:30 +00:00
currMovieData . dump ( osRecordingMovie , false ) ;
2008-05-22 07:43:48 +00:00
2008-05-26 00:03:20 +00:00
movieMode = MOVIEMODE_RECORD ;
2008-05-26 03:13:16 +00:00
movie_readonly = false ;
2008-06-27 04:39:14 +00:00
currRerecordCount = 0 ;
2012-10-21 16:40:04 +00:00
2010-05-16 04:18:49 +00:00
FCEU_DispMessage ( " Movie recording started. " , 0 ) ;
2006-07-29 05:46:15 +00:00
}
2008-11-02 18:21:44 +00:00
2008-05-22 07:43:48 +00:00
//the main interaction point between the emulator and the movie system.
//either dumps the current joystick state or loads one state from the movie
2008-06-08 10:22:33 +00:00
void FCEUMOV_AddInputState ( )
2006-07-29 05:46:15 +00:00
{
2011-11-25 19:26:26 +00:00
# ifdef _WIN32
2012-11-09 18:33:07 +00:00
if ( movieMode = = MOVIEMODE_TASEDITOR )
2008-06-28 22:03:59 +00:00
{
2012-01-29 22:33:23 +00:00
// if movie length is less or equal to currFrame, pad it with empty frames
2012-11-09 18:33:07 +00:00
if ( ( ( int ) currMovieData . records . size ( ) - 1 ) < ( currFrameCounter + 1 ) )
currMovieData . insertEmpty ( - 1 , ( currFrameCounter + 1 ) - ( ( int ) currMovieData . records . size ( ) - 1 ) ) ;
2011-10-13 14:20:45 +00:00
2008-06-28 22:03:59 +00:00
MovieRecord * mr = & currMovieData . records [ currFrameCounter ] ;
2012-07-01 14:11:24 +00:00
if ( TaseditorIsRecording ( ) )
2008-06-28 22:03:59 +00:00
{
2012-02-06 16:57:46 +00:00
// record commands and buttons
mr - > commands | = _currCommand ;
2008-06-28 22:03:59 +00:00
joyports [ 0 ] . log ( mr ) ;
joyports [ 1 ] . log ( mr ) ;
2012-07-01 14:11:24 +00:00
Taseditor_RecordInput ( ) ;
2008-06-28 22:03:59 +00:00
}
2012-02-06 16:57:46 +00:00
// replay buttons
joyports [ 0 ] . load ( mr ) ;
joyports [ 1 ] . load ( mr ) ;
// replay commands
if ( mr - > command_power ( ) )
PowerNES ( ) ;
if ( mr - > command_reset ( ) )
ResetNES ( ) ;
if ( mr - > command_fds_insert ( ) )
FCEU_FDSInsert ( ) ;
if ( mr - > command_fds_select ( ) )
FCEU_FDSSelect ( ) ;
2011-11-25 19:26:26 +00:00
_currCommand = 0 ;
2011-09-27 13:43:30 +00:00
} else
2011-11-25 19:26:26 +00:00
# endif
2010-11-16 19:21:23 +00:00
if ( movieMode = = MOVIEMODE_PLAY )
2008-05-22 07:43:48 +00:00
{
2008-05-23 09:58:38 +00:00
//stop when we run out of frames
2010-05-15 21:22:16 +00:00
if ( currFrameCounter > = ( int ) currMovieData . records . size ( ) )
2008-05-22 07:43:48 +00:00
{
2010-05-12 15:31:24 +00:00
FinishPlayback ( ) ;
2012-09-25 20:28:14 +00:00
//tell all drivers to poll input and set up their logical states
for ( int port = 0 ; port < 2 ; port + + )
joyports [ port ] . driver - > Update ( port , joyports [ port ] . ptr , joyports [ port ] . attrib ) ;
portFC . driver - > Update ( portFC . ptr , portFC . attrib ) ;
2008-05-23 09:58:38 +00:00
}
else
{
2008-06-08 10:22:33 +00:00
MovieRecord * mr = & currMovieData . records [ currFrameCounter ] ;
2012-10-21 16:40:04 +00:00
2008-11-02 18:21:44 +00:00
//reset and power cycle if necessary
if ( mr - > command_power ( ) )
PowerNES ( ) ;
2008-06-12 05:07:30 +00:00
if ( mr - > command_reset ( ) )
ResetNES ( ) ;
2009-08-03 22:24:59 +00:00
if ( mr - > command_fds_insert ( ) )
FCEU_FDSInsert ( ) ;
if ( mr - > command_fds_select ( ) )
FCEU_FDSSelect ( ) ;
2008-06-08 10:22:33 +00:00
joyports [ 0 ] . load ( mr ) ;
joyports [ 1 ] . load ( mr ) ;
2008-05-23 09:58:38 +00:00
}
2008-05-26 22:45:17 +00:00
//if we are on the last frame, then pause the emulator if the player requested it
if ( currFrameCounter = = currMovieData . records . size ( ) - 1 )
{
if ( FCEUD_PauseAfterPlayback ( ) )
{
FCEUI_ToggleEmulationPause ( ) ;
}
}
2012-10-21 16:40:04 +00:00
//pause the movie at a specified frame
2008-05-23 09:58:38 +00:00
if ( FCEUMOV_ShouldPause ( ) & & FCEUI_EmulationPaused ( ) = = 0 )
{
FCEUI_ToggleEmulationPause ( ) ;
2010-05-16 04:18:49 +00:00
FCEU_DispMessage ( " Paused at specified movie frame " , 0 ) ;
2008-05-22 07:43:48 +00:00
}
2012-10-21 16:40:04 +00:00
2008-05-22 07:43:48 +00:00
}
2008-05-23 09:58:38 +00:00
else if ( movieMode = = MOVIEMODE_RECORD )
2008-05-22 07:43:48 +00:00
{
2008-05-26 00:03:20 +00:00
MovieRecord mr ;
2008-05-24 22:18:16 +00:00
2008-06-08 10:22:33 +00:00
joyports [ 0 ] . log ( & mr ) ;
joyports [ 1 ] . log ( & mr ) ;
2008-11-02 18:21:44 +00:00
mr . commands = _currCommand ;
_currCommand = 0 ;
2008-06-06 08:23:12 +00:00
2010-05-18 15:31:53 +00:00
//Adelikat: in normal mode, this is done at the time of loading a savestate in read+write mode
//If the user chooses it can be delayed to here
if ( fullSaveStateLoads & & ( currFrameCounter < ( int ) currMovieData . records . size ( ) ) )
currMovieData . truncateAt ( currFrameCounter ) ;
2011-12-09 14:53:20 +00:00
mr . dump ( & currMovieData , osRecordingMovie , currMovieData . records . size ( ) ) ; // to disk
2008-05-26 00:03:20 +00:00
currMovieData . records . push_back ( mr ) ;
2008-05-22 07:43:48 +00:00
}
2008-05-26 02:20:56 +00:00
currFrameCounter + + ;
2008-06-08 10:22:33 +00:00
extern uint8 joy [ 4 ] ;
memcpy ( & cur_input_display , joy , 4 ) ;
2006-07-29 05:46:15 +00:00
}
2008-06-08 10:22:33 +00:00
2012-10-21 16:40:04 +00:00
//TODO
2006-07-29 05:46:15 +00:00
void FCEUMOV_AddCommand ( int cmd )
{
2008-05-24 22:18:16 +00:00
// do nothing if not recording a movie
2012-02-06 16:57:46 +00:00
if ( movieMode ! = MOVIEMODE_RECORD & & movieMode ! = MOVIEMODE_TASEDITOR )
2008-05-24 22:18:16 +00:00
return ;
2006-07-29 05:46:15 +00:00
2008-11-02 18:21:44 +00:00
//NOTE: EMOVIECMD matches FCEUNPCMD_RESET and FCEUNPCMD_POWER
//we are lucky (well, I planned it that way)
2009-08-03 22:24:59 +00:00
switch ( cmd ) {
case FCEUNPCMD_FDSINSERT : cmd = MOVIECMD_FDS_INSERT ; break ;
case FCEUNPCMD_FDSSELECT : cmd = MOVIECMD_FDS_SELECT ; break ;
}
2008-11-02 18:21:44 +00:00
_currCommand | = cmd ;
2006-07-29 05:46:15 +00:00
}
void FCEU_DrawMovies ( uint8 * XBuf )
{
2011-09-03 14:33:51 +00:00
if ( frame_display )
2008-11-11 22:58:29 +00:00
{
2008-12-03 01:33:05 +00:00
char counterbuf [ 32 ] = { 0 } ;
2010-05-19 03:46:52 +00:00
int color = 0x20 ;
2008-12-03 01:33:05 +00:00
if ( movieMode = = MOVIEMODE_PLAY )
sprintf ( counterbuf , " %d/%d " , currFrameCounter , currMovieData . records . size ( ) ) ;
2012-10-21 16:40:04 +00:00
else if ( movieMode = = MOVIEMODE_RECORD )
2010-05-19 03:46:52 +00:00
sprintf ( counterbuf , " %d " , currFrameCounter ) ;
2010-05-12 15:31:24 +00:00
else if ( movieMode = = MOVIEMODE_FINISHED )
2010-05-19 03:46:52 +00:00
{
2010-05-12 15:31:24 +00:00
sprintf ( counterbuf , " %d/%d (finished) " , currFrameCounter , currMovieData . records . size ( ) ) ;
2010-05-19 03:46:52 +00:00
color = 0x17 ; //Show red to get attention
2012-01-12 19:56:17 +00:00
} else if ( movieMode = = MOVIEMODE_TASEDITOR )
2011-09-03 14:33:51 +00:00
{
sprintf ( counterbuf , " %d " , currFrameCounter ) ;
} else
2008-11-14 13:35:01 +00:00
sprintf ( counterbuf , " %d (no movie) " , currFrameCounter ) ;
2008-12-03 01:33:05 +00:00
if ( counterbuf [ 0 ] )
2010-05-19 03:46:52 +00:00
DrawTextTrans ( ClipSidesOffset + XBuf + FCEU_TextScanlineOffsetFromBottom ( 30 ) + 1 , 256 , ( uint8 * ) counterbuf , color + 0x80 ) ;
2008-05-26 01:55:45 +00:00
}
2011-09-23 18:09:40 +00:00
if ( rerecord_display & & movieMode ! = MOVIEMODE_INACTIVE )
2011-09-23 10:50:51 +00:00
{
2012-10-21 16:40:04 +00:00
char counterbuf [ 32 ] = { 0 } ;
2011-09-23 10:50:51 +00:00
sprintf ( counterbuf , " %d " , currMovieData . rerecordCount ) ;
2012-10-21 16:40:04 +00:00
2011-09-23 10:50:51 +00:00
if ( counterbuf [ 0 ] )
2011-09-23 18:09:40 +00:00
DrawTextTrans ( ClipSidesOffset + XBuf + FCEU_TextScanlineOffsetFromBottom ( 50 ) + 1 , 256 , ( uint8 * ) counterbuf , 0x28 + 0x80 ) ;
2011-09-23 10:50:51 +00:00
}
2006-07-29 05:46:15 +00:00
}
2008-07-10 02:17:42 +00:00
void FCEU_DrawLagCounter ( uint8 * XBuf )
{
2008-07-10 03:21:04 +00:00
uint8 color ;
2012-10-21 16:40:04 +00:00
2008-07-10 03:21:04 +00:00
if ( lagFlag ) color = 0x16 + 0x80 ; //If currently lagging display red
2008-12-03 01:33:05 +00:00
else color = 0x2A + 0x80 ; //else display green
2008-07-10 03:21:04 +00:00
2008-07-10 02:17:42 +00:00
if ( lagCounterDisplay )
{
2012-10-21 16:40:04 +00:00
char counterbuf [ 32 ] = { 0 } ;
2008-07-10 04:55:52 +00:00
sprintf ( counterbuf , " %d " , lagCounter ) ;
2012-10-21 16:40:04 +00:00
2008-07-10 02:17:42 +00:00
if ( counterbuf [ 0 ] )
2008-12-03 01:33:05 +00:00
DrawTextTrans ( ClipSidesOffset + XBuf + FCEU_TextScanlineOffsetFromBottom ( 40 ) + 1 , 256 , ( uint8 * ) counterbuf , color ) ; //0x20+0x80
2008-07-10 02:17:42 +00:00
}
}
2010-05-17 21:02:38 +00:00
int FCEUMOV_WriteState ( EMUFILE * os )
2008-06-02 14:02:02 +00:00
{
//we are supposed to dump the movie data into the savestate
2010-05-15 18:09:11 +00:00
if ( movieMode = = MOVIEMODE_RECORD | | movieMode = = MOVIEMODE_PLAY | | movieMode = = MOVIEMODE_FINISHED )
2008-06-20 22:15:30 +00:00
return currMovieData . dump ( os , true ) ;
2008-06-02 14:02:02 +00:00
else return 0 ;
}
2012-10-21 16:40:04 +00:00
// returns
2012-09-01 11:47:09 +00:00
int CheckTimelines ( MovieData & stateMovie , MovieData & currMovie )
2010-05-14 23:31:47 +00:00
{
2012-09-01 11:47:09 +00:00
// end_frame = min(urrMovie.records.size(), stateMovie.records.size(), currFrameCounter)
int end_frame = currMovie . records . size ( ) ;
if ( end_frame > ( int ) stateMovie . records . size ( ) )
end_frame = stateMovie . records . size ( ) ;
if ( end_frame > currFrameCounter )
end_frame = currFrameCounter ;
2012-09-03 19:02:32 +00:00
for ( int x = 0 ; x < end_frame ; x + + )
2010-05-14 23:31:47 +00:00
{
if ( ! stateMovie . records [ x ] . Compare ( currMovie . records [ x ] ) )
2012-09-03 19:02:32 +00:00
return x ;
2010-05-14 23:31:47 +00:00
}
2012-09-03 19:02:32 +00:00
// no mismatch found
return - 1 ;
2010-05-14 23:31:47 +00:00
}
2008-05-24 22:18:16 +00:00
static bool load_successful ;
2006-07-29 05:46:15 +00:00
2010-05-17 21:02:38 +00:00
bool FCEUMOV_ReadState ( EMUFILE * is , uint32 size )
2006-07-29 05:46:15 +00:00
{
2008-05-24 22:18:16 +00:00
load_successful = false ;
2011-12-09 14:53:20 +00:00
if ( ! movie_readonly )
{
if ( currMovieData . loadFrameCount > = 0 )
{
2012-03-09 15:20:51 +00:00
# ifdef WIN32
2011-12-09 14:53:20 +00:00
int result = MessageBox ( hAppWnd , " This movie is a TAS Editor project file. \n It can be modified in TAS Editor only. \n \n Open it in TAS Editor now? " , " Movie Replay " , MB_YESNO ) ;
if ( result = = IDYES )
2012-03-09 15:20:51 +00:00
emulator_must_run_taseditor = true ;
# else
2011-12-10 12:58:02 +00:00
FCEUI_printf ( " This movie is a TAS Editor project file! It can be modified in TAS Editor only. \n Movie is now Read-Only. \n " ) ;
2012-03-09 15:20:51 +00:00
# endif
2011-12-09 14:53:20 +00:00
movie_readonly = true ;
}
if ( FCEU_isFileInArchive ( curMovieFilename ) )
{
//a little rule: cant load states in read+write mode with a movie from an archive.
//so we are going to switch it to readonly mode in that case
FCEU_PrintError ( " Cannot loadstate in Read+Write with movie from archive. Movie is now Read-Only. " ) ;
movie_readonly = true ;
}
2008-08-10 02:33:14 +00:00
}
2008-05-26 00:03:20 +00:00
MovieData tempMovieData = MovieData ( ) ;
2010-05-17 21:02:38 +00:00
std : : ios : : pos_type curr = is - > ftell ( ) ;
2008-08-10 04:03:50 +00:00
if ( ! LoadFM2 ( tempMovieData , is , size , false ) ) {
2010-05-17 21:02:38 +00:00
is - > fseek ( ( uint32 ) curr + size , SEEK_SET ) ;
2008-08-10 04:03:50 +00:00
extern bool FCEU_state_loading_old_format ;
if ( FCEU_state_loading_old_format ) {
2010-05-12 15:31:24 +00:00
if ( movieMode = = MOVIEMODE_PLAY | | movieMode = = MOVIEMODE_RECORD | | movieMode = = MOVIEMODE_FINISHED ) {
2012-10-21 16:40:04 +00:00
//FCEUI_StopMovie(); //No reason to stop the movie, nothing destructive has happened yet.
2008-08-10 04:03:50 +00:00
FCEU_PrintError ( " You have tried to use an old savestate while playing a movie. This is unsupported (since the old savestate has old-format movie data in it which can't be converted on the fly) " ) ;
}
}
2008-08-10 03:19:59 +00:00
return false ;
2008-08-10 04:03:50 +00:00
}
2008-05-24 22:18:16 +00:00
2008-05-26 00:03:20 +00:00
//----------------
2010-05-30 23:45:55 +00:00
//complex TAS logic for loadstate
//fully conforms to the savestate logic documented in the Laws of TAS
//http://tasvideos.org/LawsOfTAS/OnSavestates.html
//----------------
2012-10-21 16:40:04 +00:00
2010-05-30 23:45:55 +00:00
/*
Playback or Recording + Read - only
* Check that GUID of movie and savestate - movie must match or else error
o on error : a message informing that the savestate doesn ' t belong to this movie . This is a GUID mismatch error . Give user a choice to load it anyway .
+ failstate : if use declines , loadstate attempt canceled , movie can resume as if not attempted if user has backup savstates enabled else stop movie
* Check that movie and savestate - movie are in same timeline . If not then this is a wrong timeline error .
o on error : a message informing that the savestate doesn ' t belong to this movie
+ failstate : loadstate attempt canceled , movie can resume as if not attempted if user has backup savestates enabled else stop movie
2012-09-01 11:47:09 +00:00
* Check that savestate - movie is not greater than movie . If it ' s greater then this is a future event error and is not allowed in read - only
2010-05-30 23:45:55 +00:00
o on error : message informing that the savestate is from a frame after the last frame of the movie
+ failstate - loadstate attempt cancelled , movie can resume if user has backup savesattes enabled , else stop movie
2012-09-01 11:47:09 +00:00
* Check that savestate framcount < = savestate movie length . If not this is a post - movie savestate and is not allowed in read - only
o on error : message informing that the savestate is from a frame after the last frame of the savestated movie
+ failstate - loadstate attempt cancelled , movie can resume if user has backup savesattes enabled , else stop movie
2010-05-30 23:45:55 +00:00
* All error checks have passed , state will be loaded
* Movie contained in the savestate will be discarded
2012-10-21 16:40:04 +00:00
* Movie is now in Playback mode
2010-05-30 23:45:55 +00:00
Playback , Recording + Read + write
* Check that GUID of movie and savestate - movie must match or else error
o on error : a message informing that the savestate doesn ' t belong to this movie . This is a GUID mismatch error . Give user a choice to load it anyway .
+ failstate : if use declines , loadstate attempt canceled , movie can resume as if not attempted ( stop movie if resume fails ) canceled , movie can resume if backup savestates enabled else stopmovie
* Check that savestate framcount < = savestate movie length . If not this is a post - movie savestate
2012-10-21 16:40:04 +00:00
o on post - movie : See post - movie event section .
2010-05-30 23:45:55 +00:00
* savestate passed all error checks and will now be loaded in its entirety and replace movie ( note : there will be no truncation yet )
* current framecount will be set to savestate framecount
* on the next frame of input , movie will be truncated to framecount
2012-10-21 16:40:04 +00:00
o ( note : savestate - movie can be a future event of the movie timeline , or a completely new timeline and it is still allowed )
2010-05-30 23:45:55 +00:00
* Rerecord count of movie will be incremented
2012-10-21 16:40:04 +00:00
* Movie is now in record mode
2010-05-30 23:45:55 +00:00
Post - movie savestate event
2012-10-21 16:40:04 +00:00
* Whan a savestate is loaded and is determined that the savestate - movie length is less than the savestate framecount then it is a post - movie savestate . These occur when a savestate was made during Movie Finished mode .
2010-05-30 23:45:55 +00:00
* If read + write , the entire savestate movie will be loaded and replace current movie .
* If read - only , the savestate movie will be discarded
* Mode will be switched to Move Finished
* Savestate will be loaded
* Current framecount changes to savestate framecount
2012-10-21 16:40:04 +00:00
* User will have control of input as if Movie inactive mode
2010-05-30 23:45:55 +00:00
*/
2010-06-07 21:39:10 +00:00
2010-05-12 15:31:24 +00:00
if ( movieMode = = MOVIEMODE_PLAY | | movieMode = = MOVIEMODE_RECORD | | movieMode = = MOVIEMODE_FINISHED )
2008-05-26 00:03:20 +00:00
{
//handle moviefile mismatch
if ( tempMovieData . guid ! = currMovieData . guid )
{
2008-08-18 20:34:59 +00:00
//mbg 8/18/08 - this code can be used to turn the error message into an OK/CANCEL
2008-08-19 04:48:45 +00:00
# ifdef WIN32
std : : string msg = " There is a mismatch between savestate's movie and current movie. \n current: " + currMovieData . guid . toString ( ) + " \n savestate: " + tempMovieData . guid . toString ( ) + " \n \n This means that you have loaded a savestate belonging to a different movie than the one you are playing now. \n \n Continue loading this savestate anyway? " ;
extern HWND pwindow ;
int result = MessageBox ( pwindow , msg . c_str ( ) , " Error loading savestate " , MB_OKCANCEL ) ;
if ( result = = IDCANCEL )
2010-05-28 19:39:16 +00:00
{
if ( ! backupSavestates ) //If backups are disabled we can just resume normally since we can't restore so stop movie and inform user
{
FCEU_PrintError ( " Unable to restore backup, movie playback stopped. " ) ;
FCEUI_StopMovie ( ) ;
}
2008-08-19 04:48:45 +00:00
return false ;
2010-05-28 19:39:16 +00:00
}
2008-08-19 04:48:45 +00:00
# else
2010-05-28 19:39:16 +00:00
if ( ! backupSavestates ) //If backups are disabled we can just resume normally since we can't restore so stop movie and inform user
{
FCEU_PrintError ( " Mismatch between savestate's movie and current movie. \n current: %s \n savestate: %s \n Unable to restore backup, movie playback stopped. \n " , currMovieData . guid . toString ( ) . c_str ( ) , tempMovieData . guid . toString ( ) . c_str ( ) ) ;
FCEUI_StopMovie ( ) ;
}
else
2010-06-07 21:39:10 +00:00
FCEU_PrintError ( " Mismatch between savestate's movie and current movie. \n current: %s \n savestate: %s \n " , currMovieData . guid . toString ( ) . c_str ( ) , tempMovieData . guid . toString ( ) . c_str ( ) ) ;
2010-05-28 19:39:16 +00:00
2008-08-18 20:34:59 +00:00
return false ;
2008-08-19 04:48:45 +00:00
# endif
2008-05-26 00:03:20 +00:00
}
2008-05-23 09:58:38 +00:00
2008-05-26 00:03:20 +00:00
closeRecordingMovie ( ) ;
2012-09-01 11:47:09 +00:00
if ( movie_readonly )
2008-05-26 00:03:20 +00:00
{
2012-09-01 11:47:09 +00:00
// currFrameCounter at this point represents the savestate framecount
int frame_of_mismatch = CheckTimelines ( tempMovieData , currMovieData ) ;
if ( frame_of_mismatch > = 0 )
2010-05-14 23:31:47 +00:00
{
2012-09-01 11:47:09 +00:00
// Wrong timeline, do apprioriate logic here
if ( ! backupSavestates ) //If backups are disabled we can just resume normally since we can't restore so stop movie and inform user
2010-05-16 01:23:01 +00:00
{
2012-09-01 11:47:09 +00:00
FCEU_PrintError ( " Error: Savestate not in the same timeline as movie! \n Frame %d branches from current timeline \n Unable to restore backup, movie playback stopped. " , frame_of_mismatch ) ;
FCEUI_StopMovie ( ) ;
} else
FCEU_PrintError ( " Error: Savestate not in the same timeline as movie! \n Frame %d branches from current timeline " , frame_of_mismatch ) ;
return false ;
} else if ( movieMode = = MOVIEMODE_FINISHED
& & currFrameCounter > ( int ) currMovieData . records . size ( )
2012-09-02 12:17:50 +00:00
& & currMovieData . records . size ( ) = = tempMovieData . records . size ( ) )
2012-09-01 11:47:09 +00:00
{
// special case (in MOVIEMODE_FINISHED mode)
// allow loading post-movie savestates that were made after finishing current movie
2010-05-16 01:23:01 +00:00
2012-09-01 11:47:09 +00:00
} else if ( currFrameCounter > ( int ) currMovieData . records . size ( ) )
2008-05-26 00:03:20 +00:00
{
2012-09-01 11:47:09 +00:00
// this is future event state, don't allow it
//TODO: turn frame counter to red to get attention
2010-05-28 19:25:42 +00:00
if ( ! backupSavestates ) //If backups are disabled we can just resume normally since we can't restore so stop movie and inform user
{
2012-09-01 11:47:09 +00:00
FCEU_PrintError ( " Error: Savestate is from a frame (%d) after the final frame in the movie (%d). This is not permitted. \n Unable to restore backup, movie playback stopped. " , currFrameCounter , currMovieData . records . size ( ) - 1 ) ;
2010-05-28 19:25:42 +00:00
FCEUI_StopMovie ( ) ;
2012-09-01 11:47:09 +00:00
} else
FCEU_PrintError ( " Savestate is from a frame (%d) after the final frame in the movie (%d). This is not permitted. " , currFrameCounter , currMovieData . records . size ( ) - 1 ) ;
2010-05-15 04:24:29 +00:00
return false ;
2012-09-01 11:47:09 +00:00
} else if ( currFrameCounter > ( int ) tempMovieData . records . size ( ) )
{
// this is post-movie savestate, don't allow it
//TODO: turn frame counter to red to get attention
if ( ! backupSavestates ) //If backups are disabled we can just resume normally since we can't restore so stop movie and inform user
{
FCEU_PrintError ( " Error: Savestate is from a frame (%d) after the final frame in the savestated movie (%d). This is not permitted. \n Unable to restore backup, movie playback stopped. " , currFrameCounter , tempMovieData . records . size ( ) - 1 ) ;
FCEUI_StopMovie ( ) ;
} else
FCEU_PrintError ( " Savestate is from a frame (%d) after the final frame in the savestated movie (%d). This is not permitted. " , currFrameCounter , tempMovieData . records . size ( ) - 1 ) ;
return false ;
} else
{
// Finally, this is a savestate file for this movie
movieMode = MOVIEMODE_PLAY ;
2008-05-26 00:03:20 +00:00
}
}
2010-05-15 21:22:16 +00:00
else //Read + write
2008-05-26 00:03:20 +00:00
{
2010-05-16 01:46:04 +00:00
if ( currFrameCounter > ( int ) tempMovieData . records . size ( ) )
{
//This is a post movie savestate, handle it differently
2011-12-05 19:35:15 +00:00
//Replace movie contents but then switch to movie finished mode
2012-10-21 16:40:04 +00:00
currMovieData = tempMovieData ;
2010-05-16 01:46:04 +00:00
openRecordingMovie ( curMovieFilename ) ;
currMovieData . dump ( osRecordingMovie , false /*currMovieData.binaryFlag*/ ) ;
FinishPlayback ( ) ;
}
else
{
2010-05-18 12:54:54 +00:00
//truncate before we copy, just to save some time, unless the user selects a full copy option
if ( ! fullSaveStateLoads )
tempMovieData . truncateAt ( currFrameCounter ) ; //we can only assume this here since we have checked that the frame counter is not greater than the movie data
2012-10-21 16:40:04 +00:00
currMovieData = tempMovieData ;
2010-05-16 01:46:04 +00:00
# ifdef _S9XLUA_H
if ( ! FCEU_LuaRerecordCountSkip ( ) )
currRerecordCount + + ;
# else
2008-07-23 06:54:14 +00:00
currRerecordCount + + ;
2010-05-16 01:46:04 +00:00
# endif
currMovieData . rerecordCount = currRerecordCount ;
openRecordingMovie ( curMovieFilename ) ;
currMovieData . dump ( osRecordingMovie , false /*currMovieData.binaryFlag*/ ) ;
movieMode = MOVIEMODE_RECORD ;
2010-04-28 04:50:32 +00:00
2010-05-16 01:46:04 +00:00
}
2008-05-26 00:03:20 +00:00
}
}
2012-10-21 16:40:04 +00:00
2008-05-26 00:03:20 +00:00
load_successful = true ;
2008-05-23 09:58:38 +00:00
2008-05-26 00:03:20 +00:00
return true ;
2006-07-29 05:46:15 +00:00
}
void FCEUMOV_PreLoad ( void )
{
load_successful = 0 ;
}
2008-06-05 06:49:11 +00:00
bool FCEUMOV_PostLoad ( void )
2006-07-29 05:46:15 +00:00
{
2012-01-12 19:56:17 +00:00
if ( movieMode = = MOVIEMODE_INACTIVE | | movieMode = = MOVIEMODE_TASEDITOR )
2008-06-05 06:49:11 +00:00
return true ;
2006-07-29 05:46:15 +00:00
else
2008-06-05 06:49:11 +00:00
return load_successful ;
}
2006-07-29 05:46:15 +00:00
void FCEUI_MovieToggleFrameDisplay ( void )
{
2008-05-26 01:55:45 +00:00
frame_display = ! frame_display ;
2006-07-29 05:46:15 +00:00
}
2011-10-26 23:10:53 +00:00
void FCEUI_MovieToggleRerecordDisplay ( )
{
rerecord_display ^ = 1 ;
}
2006-07-29 05:46:15 +00:00
void FCEUI_ToggleInputDisplay ( void )
{
2008-05-26 02:05:01 +00:00
switch ( input_display )
{
case 0 :
input_display = 1 ;
break ;
case 1 :
input_display = 2 ;
break ;
case 2 :
input_display = 4 ;
break ;
default :
input_display = 0 ;
break ;
}
2006-07-29 05:46:15 +00:00
}
2009-03-15 21:12:15 +00:00
int FCEUI_GetMovieLength ( )
{
return currMovieData . records . size ( ) ;
}
int FCEUI_GetMovieRerecordCount ( )
{
return currMovieData . rerecordCount ;
}
2008-05-26 00:03:20 +00:00
bool FCEUI_GetMovieToggleReadOnly ( )
2006-07-29 05:46:15 +00:00
{
2008-05-26 00:03:20 +00:00
return movie_readonly ;
}
2009-03-15 23:21:52 +00:00
void FCEUI_SetMovieToggleReadOnly ( bool which )
{
if ( which ) //If set to readonly
{
if ( ! movie_readonly ) //If not already set
{
movie_readonly = true ;
2010-05-16 04:18:49 +00:00
FCEU_DispMessage ( " Movie is now Read-Only. " , 0 ) ;
2009-03-15 23:21:52 +00:00
}
else //Else restate message
2010-05-16 04:18:49 +00:00
FCEU_DispMessage ( " Movie is Read-Only. " , 0 ) ;
2009-03-15 23:21:52 +00:00
}
else //If set to read+write
{
if ( movie_readonly ) //If not already set
{
movie_readonly = false ;
2010-05-16 04:18:49 +00:00
FCEU_DispMessage ( " Movie is now Read+Write. " , 0 ) ;
2009-03-15 23:21:52 +00:00
}
else //Else restate message
2010-05-16 04:18:49 +00:00
FCEU_DispMessage ( " Movie is Read+Write. " , 0 ) ;
2009-03-15 23:21:52 +00:00
}
}
2008-05-26 00:03:20 +00:00
void FCEUI_MovieToggleReadOnly ( )
{
2010-05-12 16:05:36 +00:00
char message [ 260 ] ;
2012-10-21 16:40:04 +00:00
2008-05-26 00:03:20 +00:00
if ( movie_readonly )
2010-05-12 16:05:36 +00:00
strcpy ( message , " Movie is now Read+Write " ) ;
2006-07-29 05:46:15 +00:00
else
2010-05-12 16:05:36 +00:00
{
strcpy ( message , " Movie is now Read-Only " ) ;
}
2012-10-21 16:40:04 +00:00
2010-05-12 16:05:36 +00:00
if ( movieMode = = MOVIEMODE_INACTIVE )
strcat ( message , " (no movie) " ) ;
else if ( movieMode = = MOVIEMODE_FINISHED )
strcat ( message , " (finished) " ) ;
2012-10-21 16:40:04 +00:00
2010-05-16 04:18:49 +00:00
FCEU_DispMessage ( message , 0 ) ;
2008-05-26 00:03:20 +00:00
movie_readonly = ! movie_readonly ;
2006-07-29 05:46:15 +00:00
}
2008-06-04 00:53:09 +00:00
void FCEUI_MoviePlayFromBeginning ( void )
{
2012-01-12 19:56:17 +00:00
if ( movieMode = = MOVIEMODE_TASEDITOR )
2011-10-26 23:10:53 +00:00
{
# ifdef WIN32
2012-07-01 14:11:24 +00:00
Taseditor_EMUCMD ( EMUCMD_MOVIE_PLAY_FROM_BEGINNING ) ;
2011-10-26 23:10:53 +00:00
# endif
} else if ( movieMode ! = MOVIEMODE_INACTIVE )
2008-06-04 00:53:09 +00:00
{
2010-05-25 00:50:27 +00:00
if ( currMovieData . savestate . empty ( ) )
{
movie_readonly = true ;
2011-09-27 13:43:30 +00:00
movieMode = MOVIEMODE_PLAY ;
2012-10-21 16:40:04 +00:00
poweron ( true ) ;
2010-05-25 00:50:27 +00:00
currFrameCounter = 0 ;
FCEU_DispMessage ( " Movie is now Read-Only. Playing from beginning. " , 0 ) ;
}
2010-05-25 02:34:10 +00:00
else
{
2011-12-09 14:53:20 +00:00
// movie starting from savestate - reload movie file
2010-05-25 02:34:10 +00:00
string str = curMovieFilename ;
FCEUI_StopMovie ( ) ;
2011-12-09 14:53:20 +00:00
if ( FCEUI_LoadMovie ( str . c_str ( ) , 1 , 0 ) )
2010-05-25 02:34:10 +00:00
{
movieMode = MOVIEMODE_PLAY ;
2011-12-09 14:53:20 +00:00
movie_readonly = true ;
2010-05-25 02:34:10 +00:00
FCEU_DispMessage ( " Movie is now Read-Only. Playing from beginning. " , 0 ) ;
}
2011-09-27 13:43:30 +00:00
//currMovieData.loadSavestateFrom(&currMovieData.savestate); //TODO: make something like this work instead so it doesn't have to reload
2010-05-25 02:34:10 +00:00
}
2008-06-04 00:53:09 +00:00
}
2010-08-21 03:59:17 +00:00
# ifdef WIN32
SetMainWindowText ( ) ;
# endif
2008-06-04 00:53:09 +00:00
}
2009-03-15 23:46:57 +00:00
string FCEUI_GetMovieName ( void )
{
return curMovieFilename ;
}
2008-06-04 00:53:09 +00:00
2008-11-11 00:02:22 +00:00
bool FCEUI_MovieGetInfo ( FCEUFILE * fp , MOVIE_INFO & info , bool skipFrameCount )
2006-07-29 05:46:15 +00:00
{
2008-05-23 09:58:38 +00:00
MovieData md ;
2010-06-07 21:39:10 +00:00
if ( ! LoadFM2 ( md , fp - > stream , fp - > size , skipFrameCount ) )
2008-08-10 02:33:14 +00:00
return false ;
2012-10-21 16:40:04 +00:00
2008-11-11 00:02:22 +00:00
info . movie_version = md . version ;
info . poweron = md . savestate . size ( ) = = 0 ;
2010-05-25 03:09:53 +00:00
info . reset = false ; //Soft-reset isn't used from starting movies anymore, so this will be false, better for FCEUFILE to have that info (as |1| on the first frame indicates it
2008-11-11 00:02:22 +00:00
info . pal = md . palFlag ;
2010-05-14 16:55:15 +00:00
info . ppuflag = md . PPUflag ;
2008-11-11 00:02:22 +00:00
info . nosynchack = true ;
info . num_frames = md . records . size ( ) ;
info . md5_of_rom_used = md . romChecksum ;
info . emu_version_used = md . emuVersion ;
info . name_of_rom_used = md . romFilename ;
info . rerecord_count = md . rerecordCount ;
info . comments = md . comments ;
2008-11-23 21:28:22 +00:00
info . subtitles = md . subtitles ;
2008-05-22 07:43:48 +00:00
2008-06-09 08:15:20 +00:00
return true ;
2008-05-23 09:58:38 +00:00
}
2008-11-23 21:28:22 +00:00
//This function creates an array of frame numbers and corresponding strings for displaying subtitles
2010-04-02 04:34:12 +00:00
void LoadSubtitles ( MovieData & moviedata )
2008-11-23 21:28:22 +00:00
{
2011-12-25 21:54:31 +00:00
subtitleFrames . resize ( 0 ) ;
subtitleMessages . resize ( 0 ) ;
2008-11-23 21:28:22 +00:00
extern std : : vector < string > subtitles ;
2009-04-23 21:34:30 +00:00
for ( uint32 i = 0 ; i < moviedata . subtitles . size ( ) ; i + + )
2008-11-23 21:28:22 +00:00
{
2009-04-23 21:34:30 +00:00
std : : string & subtitle = moviedata . subtitles [ i ] ;
2008-11-23 21:28:22 +00:00
size_t splitat = subtitle . find_first_of ( ' ' ) ;
std : : string key , value ;
2012-10-21 16:40:04 +00:00
2008-11-23 21:28:22 +00:00
//If we can't split them, then don't process this one
if ( splitat = = std : : string : : npos )
2011-12-25 21:54:31 +00:00
{
}
2008-11-23 21:28:22 +00:00
//Else split the subtitle into the int and string arrays
else
2011-12-25 21:54:31 +00:00
{
key = subtitle . substr ( 0 , splitat ) ;
value = subtitle . substr ( splitat + 1 ) ;
subtitleFrames . push_back ( atoi ( key . c_str ( ) ) ) ;
subtitleMessages . push_back ( value ) ;
}
2008-11-23 21:28:22 +00:00
}
}
//Every frame, this will be called to determine if a subtitle should be displayed, which one, and then to display it
void ProcessSubtitles ( void )
{
if ( movieMode = = MOVIEMODE_INACTIVE ) return ;
for ( uint32 i = 0 ; i < currMovieData . subtitles . size ( ) ; i + + )
{
if ( currFrameCounter = = subtitleFrames [ i ] )
2008-11-23 23:49:15 +00:00
FCEU_DisplaySubtitles ( " %s " , subtitleMessages [ i ] . c_str ( ) ) ;
2008-11-23 21:28:22 +00:00
}
2008-11-23 23:49:15 +00:00
}
void FCEU_DisplaySubtitles ( char * format , . . . )
{
va_list ap ;
va_start ( ap , format ) ;
vsnprintf ( subtitleMessage . errmsg , sizeof ( subtitleMessage . errmsg ) , format , ap ) ;
va_end ( ap ) ;
2011-12-25 21:54:31 +00:00
subtitleMessage . howlong = 400 ;
2008-11-24 00:45:25 +00:00
subtitleMessage . isMovieMessage = subtitlesOnAVI ;
2010-05-16 04:18:49 +00:00
subtitleMessage . linesFromBottom = 0 ;
2008-12-01 00:37:40 +00:00
}
2008-12-23 01:07:55 +00:00
2009-05-25 20:30:32 +00:00
void FCEUI_CreateMovieFile ( std : : string fn )
{
MovieData md = currMovieData ; //Get current movie data
2010-05-17 21:02:38 +00:00
EMUFILE * outf = FCEUD_UTF8_fstream ( fn , " wb " ) ; //open/create file
2009-05-25 20:30:32 +00:00
md . dump ( outf , false ) ; //dump movie data
delete outf ; //clean up, delete file object
}
2008-12-23 01:07:55 +00:00
void FCEUI_MakeBackupMovie ( bool dispMessage )
{
//This function generates backup movie files
string currentFn ; //Current movie fillename
string backupFn ; //Target backup filename
string tempFn ; //temp used in back filename creation
stringstream stream ;
int x ; //Temp variable for string manip
bool exist = false ; //Used to test if filename exists
bool overflow = false ; //Used for special situation when backup numbering exceeds limit
currentFn = curMovieFilename ; //Get current moviefilename
backupFn = curMovieFilename ; //Make backup filename the same as current moviefilename
x = backupFn . find_last_of ( " . " ) ; //Find file extension
backupFn = backupFn . substr ( 0 , x ) ; //Remove extension
tempFn = backupFn ; //Store the filename at this point
for ( unsigned int backNum = 0 ; backNum < 999 ; backNum + + ) //999 = arbituary limit to backup files
{
stream . str ( " " ) ; //Clear stream
if ( backNum > 99 )
stream < < " - " < < backNum ; //assign backNum to stream
else if ( backNum < = 99 & & backNum > = 10 )
stream < < " -0 " ; //Make it 010, etc if two digits
else
stream < < " -00 " < < backNum ; //Make it 001, etc if single digit
backupFn . append ( stream . str ( ) ) ; //add number to bak filename
backupFn . append ( " .bak " ) ; //add extension
exist = CheckFileExists ( backupFn . c_str ( ) ) ; //Check if file exists
2012-10-21 16:40:04 +00:00
if ( ! exist )
2008-12-23 01:07:55 +00:00
break ; //Yeah yeah, I should use a do loop or something
else
{
backupFn = tempFn ; //Before we loop again, reset the filename
2012-10-21 16:40:04 +00:00
2008-12-23 01:07:55 +00:00
if ( backNum = = 999 ) //If 999 exists, we have overflowed, let's handle that
{
backupFn . append ( " -001.bak " ) ; //We are going to simply overwrite 001.bak
overflow = true ; //Flag that we have exceeded limit
break ; //Just in case
}
}
}
2009-05-25 20:30:32 +00:00
FCEUI_CreateMovieFile ( backupFn ) ;
2012-10-21 16:40:04 +00:00
2008-12-23 01:07:55 +00:00
//TODO, decide if fstream successfully opened the file and print error message if it doesn't
2012-10-21 16:40:04 +00:00
if ( dispMessage ) //If we should inform the user
2008-12-23 01:07:55 +00:00
{
if ( overflow )
2010-05-16 04:18:49 +00:00
FCEUI_DispMessage ( " Backup overflow, overwriting %s " , 0 , backupFn . c_str ( ) ) ; //Inform user of overflow
2008-12-23 01:07:55 +00:00
else
2010-05-16 04:18:49 +00:00
FCEUI_DispMessage ( " %s created " , 0 , backupFn . c_str ( ) ) ; //Inform user of backup filename
2008-12-23 01:07:55 +00:00
}
}