2013-10-06 21:26:08 -07:00
// Copyright (C) 2003 Dolphin Project.
// 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, version 2.0 or later versions.
// 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 2.0 for more details.
// A copy of the GPL 2.0 should have been included with the program.
// If not, see http://www.gnu.org/licenses/
// Official SVN repository and contact information can be found at
// http://code.google.com/p/dolphin-emu/
2013-12-30 21:37:19 -08:00
# include <cstring>
2016-12-18 19:01:05 +01:00
# include <snappy-c.h>
2013-12-30 21:37:19 -08:00
2013-10-06 21:26:08 -07:00
# include "ChunkFile.h"
2017-02-24 18:59:41 +01:00
# include "StringUtils.h"
2013-10-06 21:26:08 -07:00
PointerWrapSection PointerWrap : : Section ( const char * title , int ver ) {
return Section ( title , ver , ver ) ;
}
PointerWrapSection PointerWrap : : Section ( const char * title , int minVer , int ver ) {
char marker [ 16 ] = { 0 } ;
int foundVersion = ver ;
2017-03-13 10:41:06 +01:00
// This is strncpy because we rely on its weird non-null-terminating truncation behaviour.
// Can't replace it with the more sensible truncate_cpy because that would break savestates.
2017-02-28 00:03:32 +01:00
strncpy ( marker , title , sizeof ( marker ) ) ;
if ( ! ExpectVoid ( marker , sizeof ( marker ) ) )
{
2013-10-06 21:26:08 -07:00
// Might be before we added name markers for safety.
if ( foundVersion = = 1 & & ExpectVoid ( & foundVersion , sizeof ( foundVersion ) ) )
DoMarker ( title ) ;
// Wasn't found, but maybe we can still load the state.
else
foundVersion = 0 ;
}
else
Do ( foundVersion ) ;
if ( error = = ERROR_FAILURE | | foundVersion < minVer | | foundVersion > ver ) {
2017-03-06 13:10:23 +01:00
WARN_LOG ( SAVESTATE , " Savestate failure: wrong version %d found for %s " , foundVersion , title ) ;
2013-10-06 21:26:08 -07:00
SetError ( ERROR_FAILURE ) ;
return PointerWrapSection ( * this , - 1 , title ) ;
}
return PointerWrapSection ( * this , foundVersion , title ) ;
}
void PointerWrap : : SetError ( Error error_ ) {
if ( error < error_ ) {
error = error_ ;
}
if ( error > ERROR_WARNING ) {
mode = PointerWrap : : MODE_MEASURE ;
}
}
bool PointerWrap : : ExpectVoid ( void * data , int size ) {
switch ( mode ) {
case MODE_READ : if ( memcmp ( data , * ptr , size ) ! = 0 ) return false ; break ;
case MODE_WRITE : memcpy ( * ptr , data , size ) ; break ;
case MODE_MEASURE : break ; // MODE_MEASURE - don't need to do anything
case MODE_VERIFY :
for ( int i = 0 ; i < size ; i + + )
_dbg_assert_msg_ ( COMMON , ( ( u8 * ) data ) [ i ] = = ( * ptr ) [ i ] , " Savestate verification failure: %d (0x%X) (at %p) != %d (0x%X) (at %p). \n " , ( ( u8 * ) data ) [ i ] , ( ( u8 * ) data ) [ i ] , & ( ( u8 * ) data ) [ i ] , ( * ptr ) [ i ] , ( * ptr ) [ i ] , & ( * ptr ) [ i ] ) ;
break ;
default : break ; // throw an error?
}
( * ptr ) + = size ;
return true ;
}
void PointerWrap : : DoVoid ( void * data , int size ) {
switch ( mode ) {
case MODE_READ : memcpy ( data , * ptr , size ) ; break ;
case MODE_WRITE : memcpy ( * ptr , data , size ) ; break ;
case MODE_MEASURE : break ; // MODE_MEASURE - don't need to do anything
2013-10-06 21:42:51 -07:00
case MODE_VERIFY :
for ( int i = 0 ; i < size ; i + + )
_dbg_assert_msg_ ( COMMON , ( ( u8 * ) data ) [ i ] = = ( * ptr ) [ i ] , " Savestate verification failure: %d (0x%X) (at %p) != %d (0x%X) (at %p). \n " , ( ( u8 * ) data ) [ i ] , ( ( u8 * ) data ) [ i ] , & ( ( u8 * ) data ) [ i ] , ( * ptr ) [ i ] , ( * ptr ) [ i ] , & ( * ptr ) [ i ] ) ;
break ;
2013-10-06 21:26:08 -07:00
default : break ; // throw an error?
}
( * ptr ) + = size ;
}
void PointerWrap : : Do ( std : : string & x ) {
int stringLen = ( int ) x . length ( ) + 1 ;
Do ( stringLen ) ;
switch ( mode ) {
case MODE_READ : x = ( char * ) * ptr ; break ;
case MODE_WRITE : memcpy ( * ptr , x . c_str ( ) , stringLen ) ; break ;
case MODE_MEASURE : break ;
case MODE_VERIFY : _dbg_assert_msg_ ( COMMON , ! strcmp ( x . c_str ( ) , ( char * ) * ptr ) , " Savestate verification failure: \" %s \" != \" %s \" (at %p). \n " , x . c_str ( ) , ( char * ) * ptr , ptr ) ; break ;
}
( * ptr ) + = stringLen ;
}
void PointerWrap : : Do ( std : : wstring & x ) {
int stringLen = sizeof ( wchar_t ) * ( ( int ) x . length ( ) + 1 ) ;
Do ( stringLen ) ;
switch ( mode ) {
case MODE_READ : x = ( wchar_t * ) * ptr ; break ;
case MODE_WRITE : memcpy ( * ptr , x . c_str ( ) , stringLen ) ; break ;
case MODE_MEASURE : break ;
case MODE_VERIFY : _dbg_assert_msg_ ( COMMON , x = = ( wchar_t * ) * ptr , " Savestate verification failure: \" %ls \" != \" %ls \" (at %p). \n " , x . c_str ( ) , ( wchar_t * ) * ptr , ptr ) ; break ;
}
( * ptr ) + = stringLen ;
}
2013-10-06 21:42:51 -07:00
struct standard_tm {
int tm_sec ;
int tm_min ;
int tm_hour ;
int tm_mday ;
int tm_mon ;
int tm_year ;
int tm_wday ;
int tm_yday ;
int tm_isdst ;
} ;
void PointerWrap : : Do ( tm & t ) {
// We savestate this separately because some platforms use extra data at the end.
// However, old files may have the native tm in them.
// Since the first value in the struct is 0-59, we save a funny value and check for it.
// If our funny value ('tm' 0x1337) exists, it's a new version savestate.
int funnyValue = 0x13376D74 ;
if ( ExpectVoid ( & funnyValue , sizeof ( funnyValue ) ) ) {
standard_tm stm ;
if ( mode = = MODE_READ ) {
// Null out the extra members, e.g. tm_gmtoff or tm_zone.
memset ( & t , 0 , sizeof ( t ) ) ;
} else {
memcpy ( & stm , & t , sizeof ( stm ) ) ;
}
DoVoid ( ( void * ) & stm , sizeof ( stm ) ) ;
memcpy ( & t , & stm , sizeof ( stm ) ) ;
} else {
DoVoid ( ( void * ) & t , sizeof ( t ) ) ;
}
}
2013-10-06 21:26:08 -07:00
void PointerWrap : : DoMarker ( const char * prevName , u32 arbitraryNumber ) {
u32 cookie = arbitraryNumber ;
Do ( cookie ) ;
if ( mode = = PointerWrap : : MODE_READ & & cookie ! = arbitraryNumber ) {
2014-09-06 10:47:25 +02:00
PanicAlert ( " Error: After \" %s \" , found %d (0x%X) instead of save marker %d (0x%X). Aborting savestate load... " , prevName , cookie , cookie , arbitraryNumber , arbitraryNumber ) ;
2013-10-06 21:26:08 -07:00
SetError ( ERROR_FAILURE ) ;
}
}
PointerWrapSection : : ~ PointerWrapSection ( ) {
if ( ver_ > 0 ) {
p_ . DoMarker ( title_ ) ;
}
}
2016-01-23 12:53:03 -08:00
CChunkFileReader : : Error CChunkFileReader : : LoadFileHeader ( File : : IOFile & pFile , SChunkHeader & header , std : : string * title ) {
if ( ! pFile ) {
2017-03-06 13:10:23 +01:00
ERROR_LOG ( SAVESTATE , " ChunkReader: Can't open file for reading " ) ;
2013-10-06 21:26:08 -07:00
return ERROR_BAD_FILE ;
}
2016-01-23 12:53:03 -08:00
const u64 fileSize = pFile . GetSize ( ) ;
u64 headerSize = sizeof ( SChunkHeader ) ;
if ( fileSize < headerSize ) {
2017-03-06 13:10:23 +01:00
ERROR_LOG ( SAVESTATE , " ChunkReader: File too small " ) ;
2013-10-06 21:26:08 -07:00
return ERROR_BAD_FILE ;
}
2016-01-23 12:53:03 -08:00
if ( ! pFile . ReadArray ( & header , 1 ) ) {
2017-03-06 13:10:23 +01:00
ERROR_LOG ( SAVESTATE , " ChunkReader: Bad header size " ) ;
2013-10-06 21:26:08 -07:00
return ERROR_BAD_FILE ;
}
2016-01-23 12:53:03 -08:00
if ( header . Revision < REVISION_MIN ) {
2017-03-06 13:10:23 +01:00
ERROR_LOG ( SAVESTATE , " ChunkReader: Wrong file revision, got %d expected >= %d " , header . Revision , REVISION_MIN ) ;
2013-10-06 21:26:08 -07:00
return ERROR_BAD_FILE ;
}
2016-01-23 12:53:03 -08:00
if ( header . Revision > = REVISION_TITLE ) {
char titleFixed [ 128 ] ;
if ( ! pFile . ReadArray ( titleFixed , sizeof ( titleFixed ) ) ) {
2017-03-06 13:10:23 +01:00
ERROR_LOG ( SAVESTATE , " ChunkReader: Unable to read title " ) ;
2016-01-23 12:53:03 -08:00
return ERROR_BAD_FILE ;
}
if ( title ) {
* title = titleFixed ;
}
headerSize + = 128 ;
} else if ( title ) {
title - > clear ( ) ;
}
u32 sz = ( u32 ) ( fileSize - headerSize ) ;
if ( header . ExpectedSize ! = sz ) {
2017-03-06 13:10:23 +01:00
ERROR_LOG ( SAVESTATE , " ChunkReader: Bad file size, got %u expected %u " , sz , header . ExpectedSize ) ;
2013-10-06 21:26:08 -07:00
return ERROR_BAD_FILE ;
}
2016-01-23 12:53:03 -08:00
return ERROR_NONE ;
}
CChunkFileReader : : Error CChunkFileReader : : GetFileTitle ( const std : : string & filename , std : : string * title ) {
if ( ! File : : Exists ( filename ) ) {
2017-03-06 13:10:23 +01:00
ERROR_LOG ( SAVESTATE , " ChunkReader: File doesn't exist " ) ;
2013-10-06 21:26:08 -07:00
return ERROR_BAD_FILE ;
}
2016-01-23 12:53:03 -08:00
File : : IOFile pFile ( filename , " rb " ) ;
SChunkHeader header ;
return LoadFileHeader ( pFile , header , title ) ;
}
CChunkFileReader : : Error CChunkFileReader : : LoadFile ( const std : : string & filename , const char * gitVersion , u8 * & _buffer , size_t & sz , std : : string * failureReason ) {
if ( ! File : : Exists ( filename ) ) {
* failureReason = " LoadStateDoesntExist " ;
2017-03-06 13:10:23 +01:00
ERROR_LOG ( SAVESTATE , " ChunkReader: File doesn't exist " ) ;
2016-01-23 12:53:03 -08:00
return ERROR_BAD_FILE ;
}
File : : IOFile pFile ( filename , " rb " ) ;
SChunkHeader header ;
Error err = LoadFileHeader ( pFile , header , nullptr ) ;
if ( err ! = ERROR_NONE ) {
return err ;
}
2013-10-06 21:26:08 -07:00
// read the state
2016-01-23 12:53:03 -08:00
sz = header . ExpectedSize ;
2013-10-06 21:26:08 -07:00
u8 * buffer = new u8 [ sz ] ;
if ( ! pFile . ReadBytes ( buffer , sz ) )
{
2017-03-06 13:10:23 +01:00
ERROR_LOG ( SAVESTATE , " ChunkReader: Error reading file " ) ;
2015-01-17 11:56:19 -08:00
delete [ ] buffer ;
2013-10-06 21:26:08 -07:00
return ERROR_BAD_FILE ;
}
_buffer = buffer ;
if ( header . Compress ) {
u8 * uncomp_buffer = new u8 [ header . UncompressedSize ] ;
size_t uncomp_size = header . UncompressedSize ;
snappy_uncompress ( ( const char * ) buffer , sz , ( char * ) uncomp_buffer , & uncomp_size ) ;
2013-10-19 14:16:07 -07:00
if ( ( u32 ) uncomp_size ! = header . UncompressedSize ) {
2017-03-06 13:10:23 +01:00
ERROR_LOG ( SAVESTATE , " Size mismatch: file: %u calc: %u " , header . UncompressedSize , ( u32 ) uncomp_size ) ;
2015-10-04 12:09:05 +02:00
delete [ ] uncomp_buffer ;
2013-10-06 21:26:08 -07:00
return ERROR_BAD_FILE ;
}
_buffer = uncomp_buffer ;
sz = uncomp_size ;
delete [ ] buffer ;
}
return ERROR_NONE ;
}
2015-10-04 12:09:05 +02:00
// Takes ownership of buffer.
2016-01-23 12:53:03 -08:00
CChunkFileReader : : Error CChunkFileReader : : SaveFile ( const std : : string & filename , const std : : string & title , const char * gitVersion , u8 * buffer , size_t sz ) {
2017-03-06 13:10:23 +01:00
INFO_LOG ( SAVESTATE , " ChunkReader: Writing %s " , filename . c_str ( ) ) ;
2013-10-06 21:26:08 -07:00
2016-01-23 12:53:03 -08:00
File : : IOFile pFile ( filename , " wb " ) ;
2013-10-06 21:26:08 -07:00
if ( ! pFile )
{
2017-03-06 13:10:23 +01:00
ERROR_LOG ( SAVESTATE , " ChunkReader: Error opening file for write " ) ;
2015-10-04 12:09:05 +02:00
delete [ ] buffer ;
2013-10-06 21:26:08 -07:00
return ERROR_BAD_FILE ;
}
bool compress = true ;
// Create header
SChunkHeader header ;
header . Compress = compress ? 1 : 0 ;
2016-01-23 12:53:03 -08:00
header . Revision = REVISION_CURRENT ;
2013-10-19 14:16:07 -07:00
header . ExpectedSize = ( u32 ) sz ;
header . UncompressedSize = ( u32 ) sz ;
2017-03-13 10:41:06 +01:00
truncate_cpy ( header . GitVersion , gitVersion ) ;
2013-10-06 21:26:08 -07:00
2016-01-23 12:53:03 -08:00
// Setup the fixed-length title.
char titleFixed [ 128 ] ;
2017-03-13 10:41:06 +01:00
truncate_cpy ( titleFixed , title . c_str ( ) ) ;
2016-01-23 12:53:03 -08:00
2013-10-06 21:26:08 -07:00
// Write to file
if ( compress ) {
size_t comp_len = snappy_max_compressed_length ( sz ) ;
u8 * compressed_buffer = new u8 [ comp_len ] ;
snappy_compress ( ( const char * ) buffer , sz , ( char * ) compressed_buffer , & comp_len ) ;
delete [ ] buffer ;
2013-10-19 14:16:07 -07:00
header . ExpectedSize = ( u32 ) comp_len ;
2016-01-23 12:53:03 -08:00
if ( ! pFile . WriteArray ( & header , 1 ) ) {
2017-03-06 13:10:23 +01:00
ERROR_LOG ( SAVESTATE , " ChunkReader: Failed writing header " ) ;
2013-10-06 21:26:08 -07:00
return ERROR_BAD_FILE ;
}
2016-01-23 12:53:03 -08:00
if ( ! pFile . WriteArray ( titleFixed , sizeof ( titleFixed ) ) ) {
2017-03-06 13:10:23 +01:00
ERROR_LOG ( SAVESTATE , " ChunkReader: Failed writing title " ) ;
2016-01-23 12:53:03 -08:00
return ERROR_BAD_FILE ;
}
2013-10-06 21:26:08 -07:00
if ( ! pFile . WriteBytes ( & compressed_buffer [ 0 ] , comp_len ) ) {
2017-03-06 13:10:23 +01:00
ERROR_LOG ( SAVESTATE , " ChunkReader: Failed writing compressed data " ) ;
2013-10-06 21:26:08 -07:00
return ERROR_BAD_FILE ;
} else {
2017-03-06 13:10:23 +01:00
INFO_LOG ( SAVESTATE , " Savestate: Compressed %i bytes into %i " , ( int ) sz , ( int ) comp_len ) ;
2013-10-06 21:26:08 -07:00
}
delete [ ] compressed_buffer ;
} else {
if ( ! pFile . WriteArray ( & header , 1 ) )
{
2017-03-06 13:10:23 +01:00
ERROR_LOG ( SAVESTATE , " ChunkReader: Failed writing header " ) ;
2015-10-04 12:09:05 +02:00
delete [ ] buffer ;
2013-10-06 21:26:08 -07:00
return ERROR_BAD_FILE ;
}
if ( ! pFile . WriteBytes ( & buffer [ 0 ] , sz ) )
{
2017-03-06 13:10:23 +01:00
ERROR_LOG ( SAVESTATE , " ChunkReader: Failed writing data " ) ;
2015-10-04 12:09:05 +02:00
delete [ ] buffer ;
2013-10-06 21:26:08 -07:00
return ERROR_BAD_FILE ;
}
delete [ ] buffer ;
}
2017-03-06 13:10:23 +01:00
INFO_LOG ( SAVESTATE , " ChunkReader: Done writing %s " , filename . c_str ( ) ) ;
2013-10-06 21:26:08 -07:00
return ERROR_NONE ;
}