Mercurial > vba-clojure
changeset 33:44974c3e093b
found source of problem for video recording
author | Robert McIntyre <rlm@mit.edu> |
---|---|
date | Mon, 05 Mar 2012 01:25:11 -0600 (2012-03-05) |
parents | 0dc331ec7f27 |
children | 6bde5b67a2b8 |
files | commands.sh src/common/movie.cpp src/gb/GB.cpp src/lua/loadlib.c src/sdl/SDL.cpp |
diffstat | 5 files changed, 4723 insertions(+), 4684 deletions(-) [+] |
line wrap: on
line diff
1.1 --- a/commands.sh Sun Mar 04 22:44:42 2012 -0600 1.2 +++ b/commands.sh Mon Mar 05 01:25:11 2012 -0600 1.3 @@ -1,12 +1,21 @@ 1.4 + 1.5 # This buffer is for notes you don't want to save, and for Lisp evaluation. 1.6 # If you want to create a file, visit that file with C-x C-f, 1.7 # then enter the text in that file's own buffer. 1.8 1.9 1.10 -./VisualBoyAdvance ../../../pokemon-escape/roms/Pokemon\ Yellow\ \(U\)\ \[C\]\[\!\].gbc 1.11 +#./VisualBoyAdvance ../../../pokemon-escape/roms/Pokemon\ Yellow\ \(U\)\ \[C\]\[\!\].gbc 1.12 1.13 1.14 -./VisualBoyAdvance --recordmovie=./rlm.vbm ../../../pokemon-escape/roms/Pokemon\ Yellow\ \(U\)\ \[C\]\[\!\].gbc 1.15 +#./VisualBoyAdvance --recordmovie=./rlm.vbm ../../../pokemon-escape/roms/Pokemon\ Yellow\ \(U\)\ \[C\]\[\!\].gbc 1.16 1.17 +rm -f /home/r/proj/vba-restore/build/movie.vba 1.18 1.19 -./VisualBoyAdvance --recordmovie=/home/r/proj/vba-restore/build/ /home/r/proj/pokemon-escape/roms/Pokemon\ Yellow\ \(U\)\ \[C\]\[\!\].gbc 1.20 \ No newline at end of file 1.21 +cd /home/r/proj/vba-restore/build 1.22 +make install 1.23 + 1.24 +/home/r/proj/vba-restore/build/artifacts/bin/VisualBoyAdvance --recordmovie=/home/r/proj/vba-restore/build/movie.vba /home/r/proj/pokemon-escape/roms/yellow.gbc 1.25 + 1.26 +#.//VisualBoyAdvance --playmovie=/home/r/proj/vba-restore/build/movie.vba /home/r/proj/pokemon-escape/roms/Pokemon\ Yellow\ \(U\)\ \[C\]\[\!\].gbc 1.27 + 1.28 +hexdump -C /home/r/proj/vba-restore/build/movie.vba
2.1 --- a/src/common/movie.cpp Sun Mar 04 22:44:42 2012 -0600 2.2 +++ b/src/common/movie.cpp Mon Mar 05 01:25:11 2012 -0600 2.3 @@ -73,1613 +73,1628 @@ 2.4 // little-endian integer pop/push functions: 2.5 static inline uint32 Pop32(const uint8 * &ptr) 2.6 { 2.7 - uint32 v = (ptr[0] | (ptr[1] << 8) | (ptr[2] << 16) | (ptr[3] << 24)); 2.8 - ptr += 4; 2.9 - return v; 2.10 + uint32 v = (ptr[0] | (ptr[1] << 8) | (ptr[2] << 16) | (ptr[3] << 24)); 2.11 + ptr += 4; 2.12 + return v; 2.13 } 2.14 2.15 static inline uint16 Pop16(const uint8 * &ptr) /* const version */ 2.16 { 2.17 - uint16 v = (ptr[0] | (ptr[1] << 8)); 2.18 - ptr += 2; 2.19 - return v; 2.20 + uint16 v = (ptr[0] | (ptr[1] << 8)); 2.21 + ptr += 2; 2.22 + return v; 2.23 } 2.24 2.25 static inline uint16 Pop16(uint8 * &ptr) /* non-const version */ 2.26 { 2.27 - uint16 v = (ptr[0] | (ptr[1] << 8)); 2.28 - ptr += 2; 2.29 - return v; 2.30 + uint16 v = (ptr[0] | (ptr[1] << 8)); 2.31 + ptr += 2; 2.32 + return v; 2.33 } 2.34 2.35 static inline uint8 Pop8(const uint8 * &ptr) 2.36 { 2.37 - return *(ptr)++; 2.38 + return *(ptr)++; 2.39 } 2.40 2.41 static inline void Push32(uint32 v, uint8 * &ptr) 2.42 { 2.43 - ptr[0] = (uint8)(v & 0xff); 2.44 - ptr[1] = (uint8)((v >> 8) & 0xff); 2.45 - ptr[2] = (uint8)((v >> 16) & 0xff); 2.46 - ptr[3] = (uint8)((v >> 24) & 0xff); 2.47 - ptr += 4; 2.48 + ptr[0] = (uint8)(v & 0xff); 2.49 + ptr[1] = (uint8)((v >> 8) & 0xff); 2.50 + ptr[2] = (uint8)((v >> 16) & 0xff); 2.51 + ptr[3] = (uint8)((v >> 24) & 0xff); 2.52 + ptr += 4; 2.53 } 2.54 2.55 static inline void Push16(uint16 v, uint8 * &ptr) 2.56 { 2.57 - ptr[0] = (uint8)(v & 0xff); 2.58 - ptr[1] = (uint8)((v >> 8) & 0xff); 2.59 - ptr += 2; 2.60 + ptr[0] = (uint8)(v & 0xff); 2.61 + ptr[1] = (uint8)((v >> 8) & 0xff); 2.62 + ptr += 2; 2.63 } 2.64 2.65 static inline void Push8(uint8 v, uint8 * &ptr) 2.66 { 2.67 - *ptr++ = v; 2.68 + *ptr++ = v; 2.69 } 2.70 2.71 // little-endian integer read/write functions: 2.72 static inline uint16 Read16(const uint8 *ptr) 2.73 { 2.74 - return ptr[0] | (ptr[1] << 8); 2.75 + return ptr[0] | (ptr[1] << 8); 2.76 } 2.77 2.78 static inline void Write16(uint16 v, uint8 *ptr) 2.79 { 2.80 - ptr[0] = uint8(v & 0xff); 2.81 - ptr[1] = uint8((v >> 8) & 0xff); 2.82 + ptr[0] = uint8(v & 0xff); 2.83 + ptr[1] = uint8((v >> 8) & 0xff); 2.84 } 2.85 2.86 static long file_length(FILE *fp) 2.87 { 2.88 - long cur_pos = ftell(fp); 2.89 - fseek(fp, 0, SEEK_END); 2.90 - long length = ftell(fp); 2.91 - fseek(fp, cur_pos, SEEK_SET); 2.92 - return length; 2.93 + long cur_pos = ftell(fp); 2.94 + fseek(fp, 0, SEEK_END); 2.95 + long length = ftell(fp); 2.96 + fseek(fp, cur_pos, SEEK_SET); 2.97 + return length; 2.98 } 2.99 2.100 static int bytes_per_frame(SMovie &mov) 2.101 { 2.102 - int num_controllers = 0; 2.103 + int num_controllers = 0; 2.104 2.105 - for (int i = 0; i < MOVIE_NUM_OF_POSSIBLE_CONTROLLERS; ++i) 2.106 - if (mov.header.controllerFlags & MOVIE_CONTROLLER(i)) 2.107 - ++num_controllers; 2.108 + for (int i = 0; i < MOVIE_NUM_OF_POSSIBLE_CONTROLLERS; ++i) 2.109 + if (mov.header.controllerFlags & MOVIE_CONTROLLER(i)) 2.110 + ++num_controllers; 2.111 2.112 - return CONTROLLER_DATA_SIZE * num_controllers; 2.113 + return CONTROLLER_DATA_SIZE * num_controllers; 2.114 } 2.115 2.116 static void reserve_buffer_space(uint32 space_needed) 2.117 { 2.118 - if (space_needed > Movie.inputBufferSize) 2.119 - { 2.120 - uint32 ptr_offset = Movie.inputBufferPtr - Movie.inputBuffer; 2.121 - uint32 alloc_chunks = (space_needed - 1) / BUFFER_GROWTH_SIZE + 1; 2.122 - uint32 old_size = Movie.inputBufferSize; 2.123 - Movie.inputBufferSize = BUFFER_GROWTH_SIZE * alloc_chunks; 2.124 - Movie.inputBuffer = (uint8 *)realloc(Movie.inputBuffer, Movie.inputBufferSize); 2.125 - // FIXME: this only fixes the random input problem during dma-frame-skip, but not the skip 2.126 - memset(Movie.inputBuffer + old_size, 0, Movie.inputBufferSize - old_size); 2.127 - Movie.inputBufferPtr = Movie.inputBuffer + ptr_offset; 2.128 - } 2.129 + if (space_needed > Movie.inputBufferSize) 2.130 + { 2.131 + uint32 ptr_offset = Movie.inputBufferPtr - Movie.inputBuffer; 2.132 + uint32 alloc_chunks = (space_needed - 1) / BUFFER_GROWTH_SIZE + 1; 2.133 + uint32 old_size = Movie.inputBufferSize; 2.134 + Movie.inputBufferSize = BUFFER_GROWTH_SIZE * alloc_chunks; 2.135 + Movie.inputBuffer = (uint8 *)realloc(Movie.inputBuffer, Movie.inputBufferSize); 2.136 + // FIXME: this only fixes the random input problem during dma-frame-skip, but not the skip 2.137 + memset(Movie.inputBuffer + old_size, 0, Movie.inputBufferSize - old_size); 2.138 + Movie.inputBufferPtr = Movie.inputBuffer + ptr_offset; 2.139 + } 2.140 } 2.141 2.142 static int read_movie_header(FILE *file, SMovie &movie) 2.143 { 2.144 - assert(file != NULL); 2.145 - assert(VBM_HEADER_SIZE == sizeof(SMovieFileHeader)); // sanity check on the header type definition 2.146 + assert(file != NULL); 2.147 + assert(VBM_HEADER_SIZE == sizeof(SMovieFileHeader)); // sanity check on the header type definition 2.148 2.149 - uint8 headerData [VBM_HEADER_SIZE]; 2.150 + uint8 headerData [VBM_HEADER_SIZE]; 2.151 2.152 - if (fread(headerData, 1, VBM_HEADER_SIZE, file) != VBM_HEADER_SIZE) 2.153 - return MOVIE_WRONG_FORMAT; // if we failed to read in all VBM_HEADER_SIZE bytes of the header 2.154 + if (fread(headerData, 1, VBM_HEADER_SIZE, file) != VBM_HEADER_SIZE) 2.155 + return MOVIE_WRONG_FORMAT; // if we failed to read in all VBM_HEADER_SIZE bytes of the header 2.156 2.157 - const uint8 * ptr = headerData; 2.158 - SMovieFileHeader &header = movie.header; 2.159 + const uint8 * ptr = headerData; 2.160 + SMovieFileHeader &header = movie.header; 2.161 2.162 - header.magic = Pop32(ptr); 2.163 - if (header.magic != VBM_MAGIC) 2.164 - return MOVIE_WRONG_FORMAT; 2.165 + header.magic = Pop32(ptr); 2.166 + if (header.magic != VBM_MAGIC) 2.167 + return MOVIE_WRONG_FORMAT; 2.168 2.169 - header.version = Pop32(ptr); 2.170 - if (header.version != VBM_VERSION) 2.171 - return MOVIE_WRONG_VERSION; 2.172 + header.version = Pop32(ptr); 2.173 + if (header.version != VBM_VERSION) 2.174 + return MOVIE_WRONG_VERSION; 2.175 2.176 - header.uid = Pop32(ptr); 2.177 - header.length_frames = Pop32(ptr) + 1; // HACK: add 1 to the length for compatibility 2.178 - header.rerecord_count = Pop32(ptr); 2.179 + header.uid = Pop32(ptr); 2.180 + header.length_frames = Pop32(ptr) + 1; // HACK: add 1 to the length for compatibility 2.181 + header.rerecord_count = Pop32(ptr); 2.182 2.183 - header.startFlags = Pop8(ptr); 2.184 - header.controllerFlags = Pop8(ptr); 2.185 - header.typeFlags = Pop8(ptr); 2.186 - header.optionFlags = Pop8(ptr); 2.187 + header.startFlags = Pop8(ptr); 2.188 + header.controllerFlags = Pop8(ptr); 2.189 + header.typeFlags = Pop8(ptr); 2.190 + header.optionFlags = Pop8(ptr); 2.191 2.192 - header.saveType = Pop32(ptr); 2.193 - header.flashSize = Pop32(ptr); 2.194 - header.gbEmulatorType = Pop32(ptr); 2.195 + header.saveType = Pop32(ptr); 2.196 + header.flashSize = Pop32(ptr); 2.197 + header.gbEmulatorType = Pop32(ptr); 2.198 2.199 - for (int i = 0; i < 12; i++) 2.200 - header.romTitle[i] = Pop8(ptr); 2.201 + for (int i = 0; i < 12; i++) 2.202 + header.romTitle[i] = Pop8(ptr); 2.203 2.204 - header.minorVersion = Pop8(ptr); 2.205 + header.minorVersion = Pop8(ptr); 2.206 2.207 - header.romCRC = Pop8(ptr); 2.208 - header.romOrBiosChecksum = Pop16(ptr); 2.209 - header.romGameCode = Pop32(ptr); 2.210 + header.romCRC = Pop8(ptr); 2.211 + header.romOrBiosChecksum = Pop16(ptr); 2.212 + header.romGameCode = Pop32(ptr); 2.213 2.214 - header.offset_to_savestate = Pop32(ptr); 2.215 - header.offset_to_controller_data = Pop32(ptr); 2.216 + header.offset_to_savestate = Pop32(ptr); 2.217 + header.offset_to_controller_data = Pop32(ptr); 2.218 2.219 - return MOVIE_SUCCESS; 2.220 + return MOVIE_SUCCESS; 2.221 } 2.222 2.223 static void write_movie_header(FILE *file, const SMovie &movie) 2.224 { 2.225 - assert(ftell(file) == 0); // we assume file points to beginning of movie file 2.226 + assert(ftell(file) == 0); // we assume file points to beginning of movie file 2.227 2.228 - uint8 headerData [VBM_HEADER_SIZE]; 2.229 - uint8 *ptr = headerData; 2.230 - const SMovieFileHeader &header = movie.header; 2.231 + uint8 headerData [VBM_HEADER_SIZE]; 2.232 + uint8 *ptr = headerData; 2.233 + const SMovieFileHeader &header = movie.header; 2.234 2.235 - Push32(header.magic, ptr); 2.236 - Push32(header.version, ptr); 2.237 + Push32(header.magic, ptr); 2.238 + Push32(header.version, ptr); 2.239 2.240 - Push32(header.uid, ptr); 2.241 - Push32(header.length_frames - 1, ptr); // HACK: reduce the length by 1 for compatibility with certain faulty old tools 2.242 - // like TME 2.243 - Push32(header.rerecord_count, ptr); 2.244 + Push32(header.uid, ptr); 2.245 + Push32(header.length_frames - 1, ptr); // HACK: reduce the length by 1 for compatibility with certain faulty old tools 2.246 + // like TME 2.247 + Push32(header.rerecord_count, ptr); 2.248 2.249 - Push8(header.startFlags, ptr); 2.250 - Push8(header.controllerFlags, ptr); 2.251 - Push8(header.typeFlags, ptr); 2.252 - Push8(header.optionFlags, ptr); 2.253 + Push8(header.startFlags, ptr); 2.254 + Push8(header.controllerFlags, ptr); 2.255 + Push8(header.typeFlags, ptr); 2.256 + Push8(header.optionFlags, ptr); 2.257 2.258 - Push32(header.saveType, ptr); 2.259 - Push32(header.flashSize, ptr); 2.260 - Push32(header.gbEmulatorType, ptr); 2.261 + Push32(header.saveType, ptr); 2.262 + Push32(header.flashSize, ptr); 2.263 + Push32(header.gbEmulatorType, ptr); 2.264 2.265 - for (int i = 0; i < 12; ++i) 2.266 - Push8(header.romTitle[i], ptr); 2.267 + for (int i = 0; i < 12; ++i) 2.268 + Push8(header.romTitle[i], ptr); 2.269 2.270 - Push8(header.minorVersion, ptr); 2.271 + Push8(header.minorVersion, ptr); 2.272 2.273 - Push8(header.romCRC, ptr); 2.274 - Push16(header.romOrBiosChecksum, ptr); 2.275 - Push32(header.romGameCode, ptr); 2.276 + Push8(header.romCRC, ptr); 2.277 + Push16(header.romOrBiosChecksum, ptr); 2.278 + Push32(header.romGameCode, ptr); 2.279 2.280 - Push32(header.offset_to_savestate, ptr); 2.281 - Push32(header.offset_to_controller_data, ptr); 2.282 + Push32(header.offset_to_savestate, ptr); 2.283 + Push32(header.offset_to_controller_data, ptr); 2.284 2.285 - fwrite(headerData, 1, VBM_HEADER_SIZE, file); 2.286 + fwrite(headerData, 1, VBM_HEADER_SIZE, file); 2.287 } 2.288 2.289 static void flush_movie_header() 2.290 { 2.291 - assert(Movie.file != 0 && "logical error!"); 2.292 - if (!Movie.file) 2.293 - return; 2.294 + assert(Movie.file != 0 && "logical error!"); 2.295 + if (!Movie.file) 2.296 + return; 2.297 2.298 - long originalPos = ftell(Movie.file); 2.299 + long originalPos = ftell(Movie.file); 2.300 2.301 - // (over-)write the header 2.302 - fseek(Movie.file, 0, SEEK_SET); 2.303 - write_movie_header(Movie.file, Movie); 2.304 + // (over-)write the header 2.305 + fseek(Movie.file, 0, SEEK_SET); 2.306 + write_movie_header(Movie.file, Movie); 2.307 2.308 - fflush(Movie.file); 2.309 + fflush(Movie.file); 2.310 2.311 - fseek(Movie.file, originalPos, SEEK_SET); 2.312 + fseek(Movie.file, originalPos, SEEK_SET); 2.313 } 2.314 2.315 static void flush_movie_frames() 2.316 { 2.317 - assert(Movie.file && "logical error!"); 2.318 - if (!Movie.file) 2.319 - return; 2.320 + assert(Movie.file && "logical error!"); 2.321 + if (!Movie.file) 2.322 + return; 2.323 2.324 - long originalPos = ftell(Movie.file); 2.325 + long originalPos = ftell(Movie.file); 2.326 2.327 - // overwrite the controller data 2.328 - fseek(Movie.file, Movie.header.offset_to_controller_data, SEEK_SET); 2.329 - fwrite(Movie.inputBuffer, 1, Movie.bytesPerFrame * Movie.header.length_frames, Movie.file); 2.330 + // overwrite the controller data 2.331 + fseek(Movie.file, Movie.header.offset_to_controller_data, SEEK_SET); 2.332 + fwrite(Movie.inputBuffer, 1, Movie.bytesPerFrame * Movie.header.length_frames, Movie.file); 2.333 2.334 - fflush(Movie.file); 2.335 + fflush(Movie.file); 2.336 2.337 - fseek(Movie.file, originalPos, SEEK_SET); 2.338 + fseek(Movie.file, originalPos, SEEK_SET); 2.339 } 2.340 2.341 static void truncate_movie(long length) 2.342 { 2.343 - // truncate movie to length 2.344 - // NOTE: it's certain that the savestate block is never after the 2.345 - // controller data block, because the VBM format decrees it. 2.346 + // truncate movie to length 2.347 + // NOTE: it's certain that the savestate block is never after the 2.348 + // controller data block, because the VBM format decrees it. 2.349 2.350 - assert(Movie.file && length >= 0); 2.351 - if (!Movie.file || length < 0) 2.352 - return; 2.353 + assert(Movie.file && length >= 0); 2.354 + if (!Movie.file || length < 0) 2.355 + return; 2.356 2.357 - assert(Movie.header.offset_to_savestate <= Movie.header.offset_to_controller_data); 2.358 - if (Movie.header.offset_to_savestate > Movie.header.offset_to_controller_data) 2.359 - return; 2.360 + assert(Movie.header.offset_to_savestate <= Movie.header.offset_to_controller_data); 2.361 + if (Movie.header.offset_to_savestate > Movie.header.offset_to_controller_data) 2.362 + return; 2.363 2.364 - Movie.header.length_frames = length; 2.365 - flush_movie_header(); 2.366 - const long truncLen = long(Movie.header.offset_to_controller_data + Movie.bytesPerFrame * length); 2.367 - if (file_length(Movie.file) != truncLen) 2.368 - { 2.369 - ftruncate(fileno(Movie.file), truncLen); 2.370 - } 2.371 + Movie.header.length_frames = length; 2.372 + flush_movie_header(); 2.373 + const long truncLen = long(Movie.header.offset_to_controller_data + Movie.bytesPerFrame * length); 2.374 + if (file_length(Movie.file) != truncLen) 2.375 + { 2.376 + ftruncate(fileno(Movie.file), truncLen); 2.377 + } 2.378 } 2.379 2.380 static void remember_input_state() 2.381 { 2.382 - for (int i = 0; i < MOVIE_NUM_OF_POSSIBLE_CONTROLLERS; ++i) 2.383 + for (int i = 0; i < MOVIE_NUM_OF_POSSIBLE_CONTROLLERS; ++i) 2.384 + { 2.385 + if (systemCartridgeType == 0) 2.386 { 2.387 - if (systemCartridgeType == 0) 2.388 - { 2.389 - initialInputs[i] = u16(~P1 & 0x03FF); 2.390 - } 2.391 - else 2.392 - { 2.393 - extern int32 gbJoymask[4]; 2.394 - for (int i = 0; i < 4; ++i) 2.395 - initialInputs[i] = u16(gbJoymask[i] & 0xFFFF); 2.396 - } 2.397 + initialInputs[i] = u16(~P1 & 0x03FF); 2.398 } 2.399 + else 2.400 + { 2.401 + extern int32 gbJoymask[4]; 2.402 + for (int i = 0; i < 4; ++i) 2.403 + initialInputs[i] = u16(gbJoymask[i] & 0xFFFF); 2.404 + } 2.405 + } 2.406 } 2.407 2.408 static void change_state(MovieState new_state) 2.409 { 2.410 #if (defined(WIN32) && !defined(SDL)) 2.411 - theApp.frameSearching = false; 2.412 - theApp.frameSearchSkipping = false; 2.413 + theApp.frameSearching = false; 2.414 + theApp.frameSearchSkipping = false; 2.415 #endif 2.416 2.417 - if (new_state == MOVIE_STATE_NONE) 2.418 + if (new_state == MOVIE_STATE_NONE) 2.419 + { 2.420 + Movie.pauseFrame = -1; 2.421 + 2.422 + if (Movie.state == MOVIE_STATE_NONE) 2.423 + return; 2.424 + 2.425 + truncate_movie(Movie.header.length_frames); 2.426 + 2.427 + fclose(Movie.file); 2.428 + Movie.file = NULL; 2.429 + Movie.currentFrame = 0; 2.430 +#if (defined(WIN32) && !defined(SDL)) 2.431 + // undo changes to border settings 2.432 + { 2.433 + gbBorderOn = prevBorder; 2.434 + theApp.winGbBorderOn = prevWinBorder; 2.435 + gbBorderAutomatic = prevBorderAuto; 2.436 + systemGbBorderOn(); 2.437 + } 2.438 +#endif 2.439 + gbEmulatorType = prevEmulatorType; 2.440 + 2.441 + extern int32 gbDMASpeedVersion; 2.442 + gbDMASpeedVersion = 1; 2.443 + 2.444 + extern int32 gbEchoRAMFixOn; 2.445 + gbEchoRAMFixOn = 1; 2.446 + 2.447 + gbNullInputHackTempEnabled = gbNullInputHackEnabled; 2.448 + 2.449 + if (Movie.inputBuffer) 2.450 { 2.451 - Movie.pauseFrame = -1; 2.452 + free(Movie.inputBuffer); 2.453 + Movie.inputBuffer = NULL; 2.454 + } 2.455 + } 2.456 + else if (new_state == MOVIE_STATE_PLAY) 2.457 + { 2.458 + assert(Movie.file); 2.459 2.460 - if (Movie.state == MOVIE_STATE_NONE) 2.461 - return; 2.462 + // this would cause problems if not dealt with 2.463 + if (Movie.currentFrame >= Movie.header.length_frames) 2.464 + { 2.465 + new_state = MOVIE_STATE_END; 2.466 + Movie.inputBufferPtr = Movie.inputBuffer + Movie.bytesPerFrame * Movie.header.length_frames; 2.467 + } 2.468 + } 2.469 + else if (new_state == MOVIE_STATE_RECORD) 2.470 + { 2.471 + assert(Movie.file); 2.472 2.473 - truncate_movie(Movie.header.length_frames); 2.474 - 2.475 - fclose(Movie.file); 2.476 - Movie.file = NULL; 2.477 - Movie.currentFrame = 0; 2.478 -#if (defined(WIN32) && !defined(SDL)) 2.479 - // undo changes to border settings 2.480 - { 2.481 - gbBorderOn = prevBorder; 2.482 - theApp.winGbBorderOn = prevWinBorder; 2.483 - gbBorderAutomatic = prevBorderAuto; 2.484 - systemGbBorderOn(); 2.485 - } 2.486 -#endif 2.487 - gbEmulatorType = prevEmulatorType; 2.488 - 2.489 - extern int32 gbDMASpeedVersion; 2.490 - gbDMASpeedVersion = 1; 2.491 - 2.492 - extern int32 gbEchoRAMFixOn; 2.493 - gbEchoRAMFixOn = 1; 2.494 - 2.495 - gbNullInputHackTempEnabled = gbNullInputHackEnabled; 2.496 - 2.497 - if (Movie.inputBuffer) 2.498 - { 2.499 - free(Movie.inputBuffer); 2.500 - Movie.inputBuffer = NULL; 2.501 - } 2.502 - } 2.503 - else if (new_state == MOVIE_STATE_PLAY) 2.504 + // this would cause problems if not dealt with 2.505 + if (Movie.currentFrame > Movie.header.length_frames) 2.506 { 2.507 - assert(Movie.file); 2.508 - 2.509 - // this would cause problems if not dealt with 2.510 - if (Movie.currentFrame >= Movie.header.length_frames) 2.511 - { 2.512 - new_state = MOVIE_STATE_END; 2.513 - Movie.inputBufferPtr = Movie.inputBuffer + Movie.bytesPerFrame * Movie.header.length_frames; 2.514 - } 2.515 - } 2.516 - else if (new_state == MOVIE_STATE_RECORD) 2.517 - { 2.518 - assert(Movie.file); 2.519 - 2.520 - // this would cause problems if not dealt with 2.521 - if (Movie.currentFrame > Movie.header.length_frames) 2.522 - { 2.523 - new_state = MOVIE_STATE_END; 2.524 - Movie.inputBufferPtr = Movie.inputBuffer + Movie.bytesPerFrame * Movie.header.length_frames; 2.525 - } 2.526 - 2.527 - fseek(Movie.file, Movie.header.offset_to_controller_data + Movie.bytesPerFrame * Movie.currentFrame, SEEK_SET); 2.528 + new_state = MOVIE_STATE_END; 2.529 + Movie.inputBufferPtr = Movie.inputBuffer + Movie.bytesPerFrame * Movie.header.length_frames; 2.530 } 2.531 2.532 - if (new_state == MOVIE_STATE_END && Movie.state != MOVIE_STATE_END) 2.533 + fseek(Movie.file, Movie.header.offset_to_controller_data + Movie.bytesPerFrame * Movie.currentFrame, SEEK_SET); 2.534 + } 2.535 + 2.536 + if (new_state == MOVIE_STATE_END && Movie.state != MOVIE_STATE_END) 2.537 + { 2.538 +#if defined(SDL) 2.539 + systemClearJoypads(); 2.540 +#endif 2.541 + systemScreenMessage("Movie end"); 2.542 + } 2.543 + 2.544 + Movie.state = new_state; 2.545 + 2.546 + // checking for movie end 2.547 + bool willPause = false; 2.548 + 2.549 + // if the movie's been set to pause at a certain frame 2.550 + if (Movie.state != MOVIE_STATE_NONE && Movie.pauseFrame >= 0 && Movie.currentFrame == (uint32)Movie.pauseFrame) 2.551 + { 2.552 + Movie.pauseFrame = -1; 2.553 + willPause = true; 2.554 + } 2.555 + 2.556 + if (Movie.state == MOVIE_STATE_END) 2.557 + { 2.558 + if (Movie.currentFrame == Movie.header.length_frames) 2.559 { 2.560 -#if defined(SDL) 2.561 - systemClearJoypads(); 2.562 -#endif 2.563 - systemScreenMessage("Movie end"); 2.564 - } 2.565 - 2.566 - Movie.state = new_state; 2.567 - 2.568 - // checking for movie end 2.569 - bool willPause = false; 2.570 - 2.571 - // if the movie's been set to pause at a certain frame 2.572 - if (Movie.state != MOVIE_STATE_NONE && Movie.pauseFrame >= 0 && Movie.currentFrame == (uint32)Movie.pauseFrame) 2.573 - { 2.574 - Movie.pauseFrame = -1; 2.575 - willPause = true; 2.576 - } 2.577 - 2.578 - if (Movie.state == MOVIE_STATE_END) 2.579 - { 2.580 - if (Movie.currentFrame == Movie.header.length_frames) 2.581 - { 2.582 #if (defined(WIN32) && !defined(SDL)) 2.583 - if (theApp.movieOnEndPause) 2.584 - { 2.585 - willPause = true; 2.586 - } 2.587 + if (theApp.movieOnEndPause) 2.588 + { 2.589 + willPause = true; 2.590 + } 2.591 #else 2.592 - // SDL FIXME 2.593 + // SDL FIXME 2.594 #endif 2.595 2.596 #if (defined(WIN32) && !defined(SDL)) 2.597 - switch (theApp.movieOnEndBehavior) 2.598 - { 2.599 - case 1: 2.600 - // the old behavior 2.601 - //VBAMovieRestart(); 2.602 - break; 2.603 - case 2: 2.604 + switch (theApp.movieOnEndBehavior) 2.605 + { 2.606 + case 1: 2.607 + // the old behavior 2.608 + //VBAMovieRestart(); 2.609 + break; 2.610 + case 2: 2.611 #else 2.612 - // SDL FIXME 2.613 + // SDL FIXME 2.614 #endif 2.615 - if (Movie.RecordedThisSession) 2.616 - { 2.617 - // if user has been recording this movie since the last time it started playing, 2.618 - // they probably don't want the movie to end now during playback, 2.619 - // so switch back to recording when it reaches the end 2.620 - VBAMovieSwitchToRecording(); 2.621 - systemScreenMessage("Recording resumed"); 2.622 - willPause = true; 2.623 - } 2.624 + if (Movie.RecordedThisSession) 2.625 + { 2.626 + // if user has been recording this movie since the last time it started playing, 2.627 + // they probably don't want the movie to end now during playback, 2.628 + // so switch back to recording when it reaches the end 2.629 + VBAMovieSwitchToRecording(); 2.630 + systemScreenMessage("Recording resumed"); 2.631 + willPause = true; 2.632 + } 2.633 #if (defined(WIN32) && !defined(SDL)) 2.634 - break; 2.635 - case 3: 2.636 - // keep open 2.637 - break; 2.638 - case 0: 2.639 - // fall through 2.640 - default: 2.641 - // close movie 2.642 - //VBAMovieStop(false); 2.643 - break; 2.644 - } 2.645 + break; 2.646 + case 3: 2.647 + // keep open 2.648 + break; 2.649 + case 0: 2.650 + // fall through 2.651 + default: 2.652 + // close movie 2.653 + //VBAMovieStop(false); 2.654 + break; 2.655 + } 2.656 #else 2.657 - // SDL FIXME 2.658 + // SDL FIXME 2.659 #endif 2.660 - } 2.661 + } 2.662 #if 1 2.663 - else if (Movie.currentFrame > Movie.header.length_frames) 2.664 - { 2.665 + else if (Movie.currentFrame > Movie.header.length_frames) 2.666 + { 2.667 #if (defined(WIN32) && !defined(SDL)) 2.668 - switch (theApp.movieOnEndBehavior) 2.669 - { 2.670 - case 1: 2.671 - // FIXME: this should be delayed till the current frame ends 2.672 - VBAMovieRestart(); 2.673 - break; 2.674 - case 2: 2.675 - // nothing 2.676 - break; 2.677 - case 3: 2.678 - // keep open 2.679 - break; 2.680 - case 0: 2.681 - // fall through 2.682 - default: 2.683 - // close movie 2.684 - VBAMovieStop(false); 2.685 - break; 2.686 - } 2.687 + switch (theApp.movieOnEndBehavior) 2.688 + { 2.689 + case 1: 2.690 + // FIXME: this should be delayed till the current frame ends 2.691 + VBAMovieRestart(); 2.692 + break; 2.693 + case 2: 2.694 + // nothing 2.695 + break; 2.696 + case 3: 2.697 + // keep open 2.698 + break; 2.699 + case 0: 2.700 + // fall through 2.701 + default: 2.702 + // close movie 2.703 + VBAMovieStop(false); 2.704 + break; 2.705 + } 2.706 #else 2.707 - // SDLFIXME 2.708 + // SDLFIXME 2.709 #endif 2.710 - } 2.711 + } 2.712 #endif 2.713 - } // end if (Movie.state == MOVIE_STATE_END) 2.714 + } // end if (Movie.state == MOVIE_STATE_END) 2.715 2.716 - if (willPause) 2.717 - { 2.718 - systemSetPause(true); 2.719 - } 2.720 + if (willPause) 2.721 + { 2.722 + systemSetPause(true); 2.723 + } 2.724 } 2.725 2.726 void VBAMovieInit() 2.727 { 2.728 - memset(&Movie, 0, sizeof(Movie)); 2.729 - Movie.state = MOVIE_STATE_NONE; 2.730 - Movie.pauseFrame = -1; 2.731 + memset(&Movie, 0, sizeof(Movie)); 2.732 + Movie.state = MOVIE_STATE_NONE; 2.733 + Movie.pauseFrame = -1; 2.734 2.735 - resetSignaled = false; 2.736 - resetSignaledLast = false; 2.737 + resetSignaled = false; 2.738 + resetSignaledLast = false; 2.739 } 2.740 2.741 void VBAMovieGetRomInfo(const SMovie &movieInfo, char romTitle [12], uint32 &romGameCode, uint16 &checksum, uint8 &crc) 2.742 { 2.743 - if (systemCartridgeType == 0) // GBA 2.744 - { 2.745 - extern u8 *bios, *rom; 2.746 - memcpy(romTitle, &rom[0xa0], 12); // GBA TITLE 2.747 - memcpy(&romGameCode, &rom[0xac], 4); // GBA ROM GAME CODE 2.748 - if ((movieInfo.header.optionFlags & MOVIE_SETTING_USEBIOSFILE) != 0) 2.749 - checksum = utilCalcBIOSChecksum(bios, 4); // GBA BIOS CHECKSUM 2.750 - else 2.751 - checksum = 0; 2.752 - crc = rom[0xbd]; // GBA ROM CRC 2.753 - } 2.754 - else // non-GBA 2.755 - { 2.756 - extern u8 *gbRom; 2.757 - memcpy(romTitle, &gbRom[0x134], 12); // GB TITLE (note this can be 15 but is truncated to 12) 2.758 - romGameCode = (uint32)gbRom[0x146]; // GB ROM UNIT CODE 2.759 + if (systemCartridgeType == 0) // GBA 2.760 + { 2.761 + extern u8 *bios, *rom; 2.762 + memcpy(romTitle, &rom[0xa0], 12); // GBA TITLE 2.763 + memcpy(&romGameCode, &rom[0xac], 4); // GBA ROM GAME CODE 2.764 + if ((movieInfo.header.optionFlags & MOVIE_SETTING_USEBIOSFILE) != 0) 2.765 + checksum = utilCalcBIOSChecksum(bios, 4); // GBA BIOS CHECKSUM 2.766 + else 2.767 + checksum = 0; 2.768 + crc = rom[0xbd]; // GBA ROM CRC 2.769 + } 2.770 + else // non-GBA 2.771 + { 2.772 + extern u8 *gbRom; 2.773 + memcpy(romTitle, &gbRom[0x134], 12); // GB TITLE (note this can be 15 but is truncated to 12) 2.774 + romGameCode = (uint32)gbRom[0x146]; // GB ROM UNIT CODE 2.775 2.776 - checksum = (gbRom[0x14e] << 8) | gbRom[0x14f]; // GB ROM CHECKSUM, read from big-endian 2.777 - crc = gbRom[0x14d]; // GB ROM CRC 2.778 - } 2.779 + checksum = (gbRom[0x14e] << 8) | gbRom[0x14f]; // GB ROM CHECKSUM, read from big-endian 2.780 + crc = gbRom[0x14d]; // GB ROM CRC 2.781 + } 2.782 } 2.783 2.784 #ifdef SDL 2.785 static void GetBatterySaveName(char *buffer) 2.786 { 2.787 - extern char batteryDir[2048], filename[2048]; // from SDL.cpp 2.788 - extern char *sdlGetFilename(char *name); // from SDL.cpp 2.789 - if (batteryDir[0]) 2.790 - sprintf(buffer, "%s/%s.sav", batteryDir, sdlGetFilename(filename)); 2.791 - else 2.792 - sprintf(buffer, "%s.sav", filename); 2.793 + extern char batteryDir[2048], filename[2048]; // from SDL.cpp 2.794 + extern char *sdlGetFilename(char *name); // from SDL.cpp 2.795 + if (batteryDir[0]) 2.796 + sprintf(buffer, "%s/%s.sav", batteryDir, sdlGetFilename(filename)); 2.797 + else 2.798 + sprintf(buffer, "%s.sav", filename); 2.799 } 2.800 2.801 #endif 2.802 2.803 static void SetPlayEmuSettings() 2.804 { 2.805 - prevEmulatorType = gbEmulatorType; 2.806 - gbEmulatorType = Movie.header.gbEmulatorType; 2.807 + prevEmulatorType = gbEmulatorType; 2.808 + gbEmulatorType = Movie.header.gbEmulatorType; 2.809 2.810 #if (defined(WIN32) && !defined(SDL)) 2.811 -// theApp.removeIntros = false; 2.812 - theApp.skipBiosFile = (Movie.header.optionFlags & MOVIE_SETTING_SKIPBIOSFILE) != 0; 2.813 - theApp.useBiosFile = (Movie.header.optionFlags & MOVIE_SETTING_USEBIOSFILE) != 0; 2.814 + // theApp.removeIntros = false; 2.815 + theApp.skipBiosFile = (Movie.header.optionFlags & MOVIE_SETTING_SKIPBIOSFILE) != 0; 2.816 + theApp.useBiosFile = (Movie.header.optionFlags & MOVIE_SETTING_USEBIOSFILE) != 0; 2.817 #else 2.818 - extern int saveType, sdlRtcEnable, sdlFlashSize; // from SDL.cpp 2.819 - extern bool8 useBios, skipBios, removeIntros; // from SDL.cpp 2.820 - useBios = (Movie.header.optionFlags & MOVIE_SETTING_USEBIOSFILE) != 0; 2.821 - skipBios = (Movie.header.optionFlags & MOVIE_SETTING_SKIPBIOSFILE) != 0; 2.822 - removeIntros = false /*(Movie.header.optionFlags & MOVIE_SETTING_REMOVEINTROS) != 0*/; 2.823 + extern int saveType, sdlRtcEnable, sdlFlashSize; // from SDL.cpp 2.824 + extern bool8 useBios, skipBios, removeIntros; // from SDL.cpp 2.825 + useBios = (Movie.header.optionFlags & MOVIE_SETTING_USEBIOSFILE) != 0; 2.826 + skipBios = (Movie.header.optionFlags & MOVIE_SETTING_SKIPBIOSFILE) != 0; 2.827 + removeIntros = false /*(Movie.header.optionFlags & MOVIE_SETTING_REMOVEINTROS) != 0*/; 2.828 #endif 2.829 2.830 - extern void SetPrefetchHack(bool); 2.831 - if (systemCartridgeType == 0) // lag disablement applies only to GBA 2.832 - SetPrefetchHack((Movie.header.optionFlags & MOVIE_SETTING_LAGHACK) != 0); 2.833 + extern void SetPrefetchHack(bool); 2.834 + if (systemCartridgeType == 0) // lag disablement applies only to GBA 2.835 + SetPrefetchHack((Movie.header.optionFlags & MOVIE_SETTING_LAGHACK) != 0); 2.836 2.837 - gbNullInputHackTempEnabled = ((Movie.header.optionFlags & MOVIE_SETTING_GBINPUTHACK) != 0); 2.838 + gbNullInputHackTempEnabled = ((Movie.header.optionFlags & MOVIE_SETTING_GBINPUTHACK) != 0); 2.839 2.840 - // some GB/GBC games depend on the sound rate, so just use the highest one 2.841 - systemSoundSetQuality(1); 2.842 - useOldFrameTiming = false; 2.843 + // some GB/GBC games depend on the sound rate, so just use the highest one 2.844 + systemSoundSetQuality(1); 2.845 + useOldFrameTiming = false; 2.846 2.847 - extern int32 gbDMASpeedVersion; 2.848 - if ((Movie.header.optionFlags & MOVIE_SETTING_GBCFF55FIX) != 0) 2.849 - gbDMASpeedVersion = 1; 2.850 - else 2.851 - gbDMASpeedVersion = 0; // old CGB HDMA5 timing was used 2.852 + extern int32 gbDMASpeedVersion; 2.853 + if ((Movie.header.optionFlags & MOVIE_SETTING_GBCFF55FIX) != 0) 2.854 + gbDMASpeedVersion = 1; 2.855 + else 2.856 + gbDMASpeedVersion = 0; // old CGB HDMA5 timing was used 2.857 2.858 - extern int32 gbEchoRAMFixOn; 2.859 - if ((Movie.header.optionFlags & MOVIE_SETTING_GBECHORAMFIX) != 0) 2.860 - gbEchoRAMFixOn = 1; 2.861 - else 2.862 - gbEchoRAMFixOn = 0; 2.863 + extern int32 gbEchoRAMFixOn; 2.864 + if ((Movie.header.optionFlags & MOVIE_SETTING_GBECHORAMFIX) != 0) 2.865 + gbEchoRAMFixOn = 1; 2.866 + else 2.867 + gbEchoRAMFixOn = 0; 2.868 2.869 #if (defined(WIN32) && !defined(SDL)) 2.870 - rtcEnable((Movie.header.optionFlags & MOVIE_SETTING_RTCENABLE) != 0); 2.871 - theApp.winSaveType = Movie.header.saveType; 2.872 - theApp.winFlashSize = Movie.header.flashSize; 2.873 + rtcEnable((Movie.header.optionFlags & MOVIE_SETTING_RTCENABLE) != 0); 2.874 + theApp.winSaveType = Movie.header.saveType; 2.875 + theApp.winFlashSize = Movie.header.flashSize; 2.876 2.877 - prevBorder = gbBorderOn; 2.878 - prevWinBorder = theApp.winGbBorderOn; 2.879 - prevBorderAuto = gbBorderAutomatic; 2.880 - if ((gbEmulatorType == 2 || gbEmulatorType == 5) 2.881 - && !theApp.hideMovieBorder) // games played in SGB mode can have a border 2.882 + prevBorder = gbBorderOn; 2.883 + prevWinBorder = theApp.winGbBorderOn; 2.884 + prevBorderAuto = gbBorderAutomatic; 2.885 + if ((gbEmulatorType == 2 || gbEmulatorType == 5) 2.886 + && !theApp.hideMovieBorder) // games played in SGB mode can have a border 2.887 + { 2.888 + gbBorderOn = true; 2.889 + theApp.winGbBorderOn = true; 2.890 + gbBorderAutomatic = false; 2.891 + } 2.892 + else 2.893 + { 2.894 + gbBorderOn = false; 2.895 + theApp.winGbBorderOn = false; 2.896 + gbBorderAutomatic = false; 2.897 + if (theApp.hideMovieBorder) 2.898 { 2.899 - gbBorderOn = true; 2.900 - theApp.winGbBorderOn = true; 2.901 - gbBorderAutomatic = false; 2.902 + theApp.hideMovieBorder = false; 2.903 + prevBorder = false; // it might be expected behaviour that it stays hidden after the movie 2.904 } 2.905 - else 2.906 - { 2.907 - gbBorderOn = false; 2.908 - theApp.winGbBorderOn = false; 2.909 - gbBorderAutomatic = false; 2.910 - if (theApp.hideMovieBorder) 2.911 - { 2.912 - theApp.hideMovieBorder = false; 2.913 - prevBorder = false; // it might be expected behaviour that it stays hidden after the movie 2.914 - } 2.915 - } 2.916 - systemGbBorderOn(); 2.917 + } 2.918 + systemGbBorderOn(); 2.919 #else 2.920 - sdlRtcEnable = (Movie.header.optionFlags & MOVIE_SETTING_RTCENABLE) != 0; 2.921 - saveType = Movie.header.saveType; 2.922 - sdlFlashSize = Movie.header.flashSize; 2.923 + sdlRtcEnable = (Movie.header.optionFlags & MOVIE_SETTING_RTCENABLE) != 0; 2.924 + saveType = Movie.header.saveType; 2.925 + sdlFlashSize = Movie.header.flashSize; 2.926 #endif 2.927 } 2.928 2.929 static void HardResetAndSRAMClear() 2.930 { 2.931 #if (defined(WIN32) && !defined(SDL)) 2.932 - winEraseBatteryFile(); // delete the damn SRAM file and keep it from being resurrected from RAM 2.933 - MainWnd *temp = ((MainWnd *)theApp.m_pMainWnd); 2.934 - if (!temp->winFileRun(true)) // restart running the game 2.935 - { 2.936 - temp->winFileClose(); 2.937 - } 2.938 + winEraseBatteryFile(); // delete the damn SRAM file and keep it from being resurrected from RAM 2.939 + MainWnd *temp = ((MainWnd *)theApp.m_pMainWnd); 2.940 + if (!temp->winFileRun(true)) // restart running the game 2.941 + { 2.942 + temp->winFileClose(); 2.943 + } 2.944 #else 2.945 - char fname [1024]; 2.946 - GetBatterySaveName(fname); 2.947 - remove(fname); // delete the damn SRAM file 2.948 + char fname [1024]; 2.949 + GetBatterySaveName(fname); 2.950 + remove(fname); // delete the damn SRAM file 2.951 2.952 - // Henceforth, emuCleanUp means "clear out SRAM" 2.953 - //theEmulator.emuCleanUp(); // keep it from being resurrected from RAM <--This is wrong, it'll deallocate all variables --Felipe 2.954 + // Henceforth, emuCleanUp means "clear out SRAM" 2.955 + //theEmulator.emuCleanUp(); // keep it from being resurrected from RAM <--This is wrong, it'll deallocate all variables --Felipe 2.956 2.957 - /// FIXME the correct SDL code to call for a full restart isn't in a function yet 2.958 - theEmulator.emuReset(false); 2.959 + /// FIXME the correct SDL code to call for a full restart isn't in a function yet 2.960 + theEmulator.emuReset(false); 2.961 #endif 2.962 } 2.963 2.964 int VBAMovieOpen(const char *filename, bool8 read_only) 2.965 { 2.966 - loadingMovie = true; 2.967 - uint8 movieReadOnly = read_only ? 1 : 0; 2.968 + loadingMovie = true; 2.969 + uint8 movieReadOnly = read_only ? 1 : 0; 2.970 2.971 - FILE * file; 2.972 - STREAM stream; 2.973 - int result; 2.974 - int fn; 2.975 + FILE * file; 2.976 + STREAM stream; 2.977 + int result; 2.978 + int fn; 2.979 2.980 - char movie_filename[_MAX_PATH]; 2.981 + char movie_filename[_MAX_PATH]; 2.982 #ifdef WIN32 2.983 - _fullpath(movie_filename, filename, _MAX_PATH); 2.984 + _fullpath(movie_filename, filename, _MAX_PATH); 2.985 #else 2.986 - // SDL FIXME: convert to fullpath 2.987 - strncpy(movie_filename, filename, _MAX_PATH); 2.988 - movie_filename[_MAX_PATH - 1] = '\0'; 2.989 + // SDL FIXME: convert to fullpath 2.990 + strncpy(movie_filename, filename, _MAX_PATH); 2.991 + movie_filename[_MAX_PATH - 1] = '\0'; 2.992 #endif 2.993 2.994 - if (movie_filename[0] == '\0') 2.995 - { loadingMovie = false; return MOVIE_FILE_NOT_FOUND; } 2.996 + if (movie_filename[0] == '\0') 2.997 + { loadingMovie = false; return MOVIE_FILE_NOT_FOUND; } 2.998 2.999 - if (!emulating) 2.1000 - { loadingMovie = false; return MOVIE_UNKNOWN_ERROR; } 2.1001 + if (!emulating) 2.1002 + { loadingMovie = false; return MOVIE_UNKNOWN_ERROR; } 2.1003 2.1004 -// bool alreadyOpen = (Movie.file != NULL && _stricmp(movie_filename, Movie.filename) == 0); 2.1005 + // bool alreadyOpen = (Movie.file != NULL && _stricmp(movie_filename, Movie.filename) == 0); 2.1006 2.1007 -// if (alreadyOpen) 2.1008 - change_state(MOVIE_STATE_NONE); // have to stop current movie before trying to re-open it 2.1009 + // if (alreadyOpen) 2.1010 + change_state(MOVIE_STATE_NONE); // have to stop current movie before trying to re-open it 2.1011 2.1012 - if (!(file = fopen(movie_filename, "rb+"))) 2.1013 - if (!(file = fopen(movie_filename, "rb"))) 2.1014 - { loadingMovie = false; return MOVIE_FILE_NOT_FOUND; } 2.1015 - //else 2.1016 - // movieReadOnly = 2; // we have to open the movie twice, no need to do this both times 2.1017 + if (!(file = fopen(movie_filename, "rb+"))) 2.1018 + if (!(file = fopen(movie_filename, "rb"))) 2.1019 + { loadingMovie = false; return MOVIE_FILE_NOT_FOUND; } 2.1020 + //else 2.1021 + // movieReadOnly = 2; // we have to open the movie twice, no need to do this both times 2.1022 2.1023 -// if (!alreadyOpen) 2.1024 -// change_state(MOVIE_STATE_NONE); // stop current movie when we're able to open the other one 2.1025 -// 2.1026 -// if (!(file = fopen(movie_filename, "rb+"))) 2.1027 -// if(!(file = fopen(movie_filename, "rb"))) 2.1028 -// {loadingMovie = false; return MOVIE_FILE_NOT_FOUND;} 2.1029 -// else 2.1030 -// movieReadOnly = 2; 2.1031 + // if (!alreadyOpen) 2.1032 + // change_state(MOVIE_STATE_NONE); // stop current movie when we're able to open the other one 2.1033 + // 2.1034 + // if (!(file = fopen(movie_filename, "rb+"))) 2.1035 + // if(!(file = fopen(movie_filename, "rb"))) 2.1036 + // {loadingMovie = false; return MOVIE_FILE_NOT_FOUND;} 2.1037 + // else 2.1038 + // movieReadOnly = 2; 2.1039 2.1040 - // clear out the current movie 2.1041 - VBAMovieInit(); 2.1042 + // clear out the current movie 2.1043 + VBAMovieInit(); 2.1044 2.1045 - // read header 2.1046 - if ((result = read_movie_header(file, Movie)) != MOVIE_SUCCESS) 2.1047 - { 2.1048 - fclose(file); 2.1049 - { loadingMovie = false; return result; } 2.1050 - } 2.1051 + // read header 2.1052 + if ((result = read_movie_header(file, Movie)) != MOVIE_SUCCESS) 2.1053 + { 2.1054 + fclose(file); 2.1055 + { loadingMovie = false; return result; } 2.1056 + } 2.1057 2.1058 - // set emulator settings that make the movie more likely to stay synchronized 2.1059 - SetPlayEmuSettings(); 2.1060 + // set emulator settings that make the movie more likely to stay synchronized 2.1061 + SetPlayEmuSettings(); 2.1062 2.1063 -// extern bool systemLoadBIOS(); 2.1064 -// if (!systemLoadBIOS()) 2.1065 -// { loadingMovie = false; return MOVIE_UNKNOWN_ERROR; } 2.1066 + // extern bool systemLoadBIOS(); 2.1067 + // if (!systemLoadBIOS()) 2.1068 + // { loadingMovie = false; return MOVIE_UNKNOWN_ERROR; } 2.1069 2.1070 - // read the metadata / author info from file 2.1071 - fread(Movie.authorInfo, 1, MOVIE_METADATA_SIZE, file); 2.1072 - fn = dup(fileno(file)); // XXX: why does this fail?? it returns -1 but errno == 0 2.1073 - fclose(file); 2.1074 + // read the metadata / author info from file 2.1075 + fread(Movie.authorInfo, 1, MOVIE_METADATA_SIZE, file); 2.1076 + fn = dup(fileno(file)); // XXX: why does this fail?? it returns -1 but errno == 0 2.1077 + fclose(file); 2.1078 2.1079 - // apparently this lseek is necessary 2.1080 - lseek(fn, Movie.header.offset_to_savestate, SEEK_SET); 2.1081 - if (!(stream = utilGzReopen(fn, "rb"))) 2.1082 - if (!(stream = utilGzOpen(movie_filename, "rb"))) 2.1083 - { loadingMovie = false; return MOVIE_FILE_NOT_FOUND; } 2.1084 - else 2.1085 - fn = dup(fileno(file)); 2.1086 - // in case the above dup failed but opening the file normally doesn't fail 2.1087 + // apparently this lseek is necessary 2.1088 + lseek(fn, Movie.header.offset_to_savestate, SEEK_SET); 2.1089 + if (!(stream = utilGzReopen(fn, "rb"))) 2.1090 + if (!(stream = utilGzOpen(movie_filename, "rb"))) 2.1091 + { loadingMovie = false; return MOVIE_FILE_NOT_FOUND; } 2.1092 + else 2.1093 + fn = dup(fileno(file)); 2.1094 + // in case the above dup failed but opening the file normally doesn't fail 2.1095 2.1096 - if (Movie.header.startFlags & MOVIE_START_FROM_SNAPSHOT) 2.1097 - { 2.1098 - // load the snapshot 2.1099 - result = theEmulator.emuReadStateFromStream(stream) ? MOVIE_SUCCESS : MOVIE_WRONG_FORMAT; 2.1100 + if (Movie.header.startFlags & MOVIE_START_FROM_SNAPSHOT) 2.1101 + { 2.1102 + // load the snapshot 2.1103 + result = theEmulator.emuReadStateFromStream(stream) ? MOVIE_SUCCESS : MOVIE_WRONG_FORMAT; 2.1104 2.1105 - // FIXME: Kludge for conversion 2.1106 - remember_input_state(); 2.1107 - } 2.1108 - else if (Movie.header.startFlags & MOVIE_START_FROM_SRAM) 2.1109 - { 2.1110 - // 'soft' reset: 2.1111 - theEmulator.emuReset(false); 2.1112 + // FIXME: Kludge for conversion 2.1113 + remember_input_state(); 2.1114 + } 2.1115 + else if (Movie.header.startFlags & MOVIE_START_FROM_SRAM) 2.1116 + { 2.1117 + // 'soft' reset: 2.1118 + theEmulator.emuReset(false); 2.1119 2.1120 - // load the SRAM 2.1121 - result = theEmulator.emuReadBatteryFromStream(stream) ? MOVIE_SUCCESS : MOVIE_WRONG_FORMAT; 2.1122 - } 2.1123 - else 2.1124 - { 2.1125 - HardResetAndSRAMClear(); 2.1126 - } 2.1127 + // load the SRAM 2.1128 + result = theEmulator.emuReadBatteryFromStream(stream) ? MOVIE_SUCCESS : MOVIE_WRONG_FORMAT; 2.1129 + } 2.1130 + else 2.1131 + { 2.1132 + HardResetAndSRAMClear(); 2.1133 + } 2.1134 2.1135 - utilGzClose(stream); 2.1136 + utilGzClose(stream); 2.1137 2.1138 - if (result != MOVIE_SUCCESS) 2.1139 - { loadingMovie = false; return result; } 2.1140 + if (result != MOVIE_SUCCESS) 2.1141 + { loadingMovie = false; return result; } 2.1142 2.1143 -// if (!(file = fopen(movie_filename, /*read_only ? "rb" :*/ "rb+"))) // want to be able to switch out of read-only later 2.1144 -// { 2.1145 -// if(!Movie.readOnly || !(file = fopen(movie_filename, "rb"))) // try read-only if failed 2.1146 -// return MOVIE_FILE_NOT_FOUND; 2.1147 -// } 2.1148 - if (!(file = fopen(movie_filename, "rb+"))) 2.1149 - if (!(file = fopen(movie_filename, "rb"))) 2.1150 - { loadingMovie = false; return MOVIE_FILE_NOT_FOUND; } 2.1151 - else 2.1152 - movieReadOnly = 2; 2.1153 + // if (!(file = fopen(movie_filename, /*read_only ? "rb" :*/ "rb+"))) // want to be able to switch out of read-only later 2.1154 + // { 2.1155 + // if(!Movie.readOnly || !(file = fopen(movie_filename, "rb"))) // try read-only if failed 2.1156 + // return MOVIE_FILE_NOT_FOUND; 2.1157 + // } 2.1158 + if (!(file = fopen(movie_filename, "rb+"))) 2.1159 + if (!(file = fopen(movie_filename, "rb"))) 2.1160 + { loadingMovie = false; return MOVIE_FILE_NOT_FOUND; } 2.1161 + else 2.1162 + movieReadOnly = 2; 2.1163 2.1164 - // recalculate length of movie from the file size 2.1165 - Movie.bytesPerFrame = bytes_per_frame(Movie); 2.1166 - fseek(file, 0, SEEK_END); 2.1167 - long fileSize = ftell(file); 2.1168 - Movie.header.length_frames = (fileSize - Movie.header.offset_to_controller_data) / Movie.bytesPerFrame; 2.1169 + // recalculate length of movie from the file size 2.1170 + Movie.bytesPerFrame = bytes_per_frame(Movie); 2.1171 + fseek(file, 0, SEEK_END); 2.1172 + long fileSize = ftell(file); 2.1173 + Movie.header.length_frames = (fileSize - Movie.header.offset_to_controller_data) / Movie.bytesPerFrame; 2.1174 2.1175 - if (fseek(file, Movie.header.offset_to_controller_data, SEEK_SET)) 2.1176 - { fclose(file); loadingMovie = false; return MOVIE_WRONG_FORMAT; } 2.1177 + if (fseek(file, Movie.header.offset_to_controller_data, SEEK_SET)) 2.1178 + { fclose(file); loadingMovie = false; return MOVIE_WRONG_FORMAT; } 2.1179 2.1180 - strcpy(Movie.filename, movie_filename); 2.1181 - Movie.file = file; 2.1182 - Movie.inputBufferPtr = Movie.inputBuffer; 2.1183 - Movie.currentFrame = 0; 2.1184 - Movie.readOnly = movieReadOnly; 2.1185 - Movie.RecordedThisSession = false; 2.1186 + strcpy(Movie.filename, movie_filename); 2.1187 + Movie.file = file; 2.1188 + Movie.inputBufferPtr = Movie.inputBuffer; 2.1189 + Movie.currentFrame = 0; 2.1190 + Movie.readOnly = movieReadOnly; 2.1191 + Movie.RecordedThisSession = false; 2.1192 2.1193 - // read controller data 2.1194 - uint32 to_read = Movie.bytesPerFrame * Movie.header.length_frames; 2.1195 - reserve_buffer_space(to_read); 2.1196 - fread(Movie.inputBuffer, 1, to_read, file); 2.1197 + // read controller data 2.1198 + uint32 to_read = Movie.bytesPerFrame * Movie.header.length_frames; 2.1199 + reserve_buffer_space(to_read); 2.1200 + fread(Movie.inputBuffer, 1, to_read, file); 2.1201 2.1202 - change_state(MOVIE_STATE_PLAY); 2.1203 + change_state(MOVIE_STATE_PLAY); 2.1204 2.1205 - char messageString[64] = "Movie "; 2.1206 - bool converted = false; 2.1207 - if (autoConvertMovieWhenPlaying) 2.1208 - { 2.1209 - int result = VBAMovieConvertCurrent(); 2.1210 - if (result == MOVIE_SUCCESS) 2.1211 - strcat(messageString, "converted and "); 2.1212 - else if (result == MOVIE_WRONG_VERSION) 2.1213 - strcat(messageString, "higher revision "); 2.1214 - } 2.1215 + char messageString[64] = "Movie "; 2.1216 + bool converted = false; 2.1217 + if (autoConvertMovieWhenPlaying) 2.1218 + { 2.1219 + int result = VBAMovieConvertCurrent(); 2.1220 + if (result == MOVIE_SUCCESS) 2.1221 + strcat(messageString, "converted and "); 2.1222 + else if (result == MOVIE_WRONG_VERSION) 2.1223 + strcat(messageString, "higher revision "); 2.1224 + } 2.1225 2.1226 - if (Movie.state == MOVIE_STATE_PLAY) 2.1227 - strcat(messageString, "replaying "); 2.1228 - else 2.1229 - strcat(messageString, "finished "); 2.1230 - if (Movie.readOnly) 2.1231 - strcat(messageString, "(read)"); 2.1232 - else 2.1233 - strcat(messageString, "(edit)"); 2.1234 - systemScreenMessage(messageString); 2.1235 + if (Movie.state == MOVIE_STATE_PLAY) 2.1236 + strcat(messageString, "replaying "); 2.1237 + else 2.1238 + strcat(messageString, "finished "); 2.1239 + if (Movie.readOnly) 2.1240 + strcat(messageString, "(read)"); 2.1241 + else 2.1242 + strcat(messageString, "(edit)"); 2.1243 + systemScreenMessage(messageString); 2.1244 2.1245 - VBAUpdateButtonPressDisplay(); 2.1246 - VBAUpdateFrameCountDisplay(); 2.1247 - systemRefreshScreen(); 2.1248 + VBAUpdateButtonPressDisplay(); 2.1249 + VBAUpdateFrameCountDisplay(); 2.1250 + systemRefreshScreen(); 2.1251 2.1252 - { loadingMovie = false; return MOVIE_SUCCESS; } 2.1253 + { loadingMovie = false; return MOVIE_SUCCESS; } 2.1254 } 2.1255 2.1256 static void SetRecordEmuSettings() 2.1257 { 2.1258 - Movie.header.optionFlags = 0; 2.1259 + Movie.header.optionFlags = 0; 2.1260 #if (defined(WIN32) && !defined(SDL)) 2.1261 - if (theApp.useBiosFile) 2.1262 - Movie.header.optionFlags |= MOVIE_SETTING_USEBIOSFILE; 2.1263 - if (theApp.skipBiosFile) 2.1264 - Movie.header.optionFlags |= MOVIE_SETTING_SKIPBIOSFILE; 2.1265 - if (rtcIsEnabled()) 2.1266 - Movie.header.optionFlags |= MOVIE_SETTING_RTCENABLE; 2.1267 - Movie.header.saveType = theApp.winSaveType; 2.1268 - Movie.header.flashSize = theApp.winFlashSize; 2.1269 + if (theApp.useBiosFile) 2.1270 + Movie.header.optionFlags |= MOVIE_SETTING_USEBIOSFILE; 2.1271 + if (theApp.skipBiosFile) 2.1272 + Movie.header.optionFlags |= MOVIE_SETTING_SKIPBIOSFILE; 2.1273 + if (rtcIsEnabled()) 2.1274 + Movie.header.optionFlags |= MOVIE_SETTING_RTCENABLE; 2.1275 + Movie.header.saveType = theApp.winSaveType; 2.1276 + Movie.header.flashSize = theApp.winFlashSize; 2.1277 #else 2.1278 - extern int saveType, sdlRtcEnable, sdlFlashSize; // from SDL.cpp 2.1279 - extern bool8 useBios, skipBios; // from SDL.cpp 2.1280 - if (useBios) 2.1281 - Movie.header.optionFlags |= MOVIE_SETTING_USEBIOSFILE; 2.1282 - if (skipBios) 2.1283 - Movie.header.optionFlags |= MOVIE_SETTING_SKIPBIOSFILE; 2.1284 - if (sdlRtcEnable) 2.1285 - Movie.header.optionFlags |= MOVIE_SETTING_RTCENABLE; 2.1286 - Movie.header.saveType = saveType; 2.1287 - Movie.header.flashSize = sdlFlashSize; 2.1288 + extern int saveType, sdlRtcEnable, sdlFlashSize; // from SDL.cpp 2.1289 + extern bool8 useBios, skipBios; // from SDL.cpp 2.1290 + if (useBios) 2.1291 + Movie.header.optionFlags |= MOVIE_SETTING_USEBIOSFILE; 2.1292 + if (skipBios) 2.1293 + Movie.header.optionFlags |= MOVIE_SETTING_SKIPBIOSFILE; 2.1294 + if (sdlRtcEnable) 2.1295 + Movie.header.optionFlags |= MOVIE_SETTING_RTCENABLE; 2.1296 + Movie.header.saveType = saveType; 2.1297 + Movie.header.flashSize = sdlFlashSize; 2.1298 #endif 2.1299 - prevEmulatorType = Movie.header.gbEmulatorType = gbEmulatorType; 2.1300 + prevEmulatorType = Movie.header.gbEmulatorType = gbEmulatorType; 2.1301 2.1302 - if (!memLagTempEnabled) 2.1303 - Movie.header.optionFlags |= MOVIE_SETTING_LAGHACK; 2.1304 + if (!memLagTempEnabled) 2.1305 + Movie.header.optionFlags |= MOVIE_SETTING_LAGHACK; 2.1306 2.1307 - if (gbNullInputHackTempEnabled) 2.1308 - Movie.header.optionFlags |= MOVIE_SETTING_GBINPUTHACK; 2.1309 + if (gbNullInputHackTempEnabled) 2.1310 + Movie.header.optionFlags |= MOVIE_SETTING_GBINPUTHACK; 2.1311 2.1312 - Movie.header.optionFlags |= MOVIE_SETTING_GBCFF55FIX; 2.1313 - extern int32 gbDMASpeedVersion; 2.1314 - gbDMASpeedVersion = 1; 2.1315 + Movie.header.optionFlags |= MOVIE_SETTING_GBCFF55FIX; 2.1316 + extern int32 gbDMASpeedVersion; 2.1317 + gbDMASpeedVersion = 1; 2.1318 2.1319 - Movie.header.optionFlags |= MOVIE_SETTING_GBECHORAMFIX; 2.1320 - extern int32 gbEchoRAMFixOn; 2.1321 - gbEchoRAMFixOn = 1; 2.1322 + Movie.header.optionFlags |= MOVIE_SETTING_GBECHORAMFIX; 2.1323 + extern int32 gbEchoRAMFixOn; 2.1324 + gbEchoRAMFixOn = 1; 2.1325 2.1326 - // some GB/GBC games depend on the sound rate, so just use the highest one 2.1327 - systemSoundSetQuality(1); 2.1328 + // some GB/GBC games depend on the sound rate, so just use the highest one 2.1329 + systemSoundSetQuality(1); 2.1330 2.1331 - useOldFrameTiming = false; 2.1332 + useOldFrameTiming = false; 2.1333 2.1334 #if (defined(WIN32) && !defined(SDL)) 2.1335 -// theApp.removeIntros = false; 2.1336 + // theApp.removeIntros = false; 2.1337 2.1338 - prevBorder = gbBorderOn; 2.1339 - prevWinBorder = theApp.winGbBorderOn; 2.1340 - prevBorderAuto = gbBorderAutomatic; 2.1341 - if (gbEmulatorType == 2 || gbEmulatorType == 5) // only games played in SGB mode will have a border 2.1342 - { 2.1343 - gbBorderOn = true; 2.1344 - theApp.winGbBorderOn = true; 2.1345 - gbBorderAutomatic = false; 2.1346 - } 2.1347 - else 2.1348 - { 2.1349 - gbBorderOn = false; 2.1350 - theApp.winGbBorderOn = false; 2.1351 - gbBorderAutomatic = false; 2.1352 - } 2.1353 - systemGbBorderOn(); 2.1354 + prevBorder = gbBorderOn; 2.1355 + prevWinBorder = theApp.winGbBorderOn; 2.1356 + prevBorderAuto = gbBorderAutomatic; 2.1357 + if (gbEmulatorType == 2 || gbEmulatorType == 5) // only games played in SGB mode will have a border 2.1358 + { 2.1359 + gbBorderOn = true; 2.1360 + theApp.winGbBorderOn = true; 2.1361 + gbBorderAutomatic = false; 2.1362 + } 2.1363 + else 2.1364 + { 2.1365 + gbBorderOn = false; 2.1366 + theApp.winGbBorderOn = false; 2.1367 + gbBorderAutomatic = false; 2.1368 + } 2.1369 + systemGbBorderOn(); 2.1370 #else 2.1371 - /// SDLFIXME 2.1372 + /// SDLFIXME 2.1373 #endif 2.1374 } 2.1375 2.1376 uint16 VBAMovieGetCurrentInputOf(int controllerNum, bool normalOnly) 2.1377 { 2.1378 - if (controllerNum < 0 || controllerNum >= MOVIE_NUM_OF_POSSIBLE_CONTROLLERS) 2.1379 - return 0; 2.1380 + if (controllerNum < 0 || controllerNum >= MOVIE_NUM_OF_POSSIBLE_CONTROLLERS) 2.1381 + return 0; 2.1382 2.1383 - return normalOnly ? (currentButtons[controllerNum] & BUTTON_REGULAR_MASK) : currentButtons[controllerNum]; 2.1384 + return normalOnly ? (currentButtons[controllerNum] & BUTTON_REGULAR_MASK) : currentButtons[controllerNum]; 2.1385 } 2.1386 2.1387 int VBAMovieCreate(const char *filename, const char *authorInfo, uint8 startFlags, uint8 controllerFlags, uint8 typeFlags) 2.1388 { 2.1389 - // make sure at least one controller is enabled 2.1390 - if ((controllerFlags & MOVIE_CONTROLLERS_ANY_MASK) == 0) 2.1391 - return MOVIE_WRONG_FORMAT; 2.1392 + // make sure at least one controller is enabled 2.1393 + if ((controllerFlags & MOVIE_CONTROLLERS_ANY_MASK) == 0) 2.1394 + return MOVIE_WRONG_FORMAT; 2.1395 2.1396 - if (!emulating) 2.1397 - return MOVIE_UNKNOWN_ERROR; 2.1398 + if (!emulating) 2.1399 + return MOVIE_UNKNOWN_ERROR; 2.1400 2.1401 - loadingMovie = true; 2.1402 + loadingMovie = true; 2.1403 2.1404 - FILE * file; 2.1405 - STREAM stream; 2.1406 - int fn; 2.1407 + FILE * file; 2.1408 + STREAM stream; 2.1409 + int fn; 2.1410 2.1411 - char movie_filename [_MAX_PATH]; 2.1412 + char movie_filename [_MAX_PATH]; 2.1413 #ifdef WIN32 2.1414 - _fullpath(movie_filename, filename, _MAX_PATH); 2.1415 + _fullpath(movie_filename, filename, _MAX_PATH); 2.1416 #else 2.1417 - // FIXME: convert to fullpath 2.1418 - strncpy(movie_filename, filename, _MAX_PATH); 2.1419 - movie_filename[_MAX_PATH - 1] = '\0'; 2.1420 + // FIXME: convert to fullpath 2.1421 + strncpy(movie_filename, filename, _MAX_PATH); 2.1422 + movie_filename[_MAX_PATH - 1] = '\0'; 2.1423 #endif 2.1424 2.1425 - bool alreadyOpen = (Movie.file != NULL && stricmp(movie_filename, Movie.filename) == 0); 2.1426 + bool alreadyOpen = (Movie.file != NULL && stricmp(movie_filename, Movie.filename) == 0); 2.1427 2.1428 - if (alreadyOpen) 2.1429 - change_state(MOVIE_STATE_NONE); // have to stop current movie before trying to re-open it 2.1430 + if (alreadyOpen) 2.1431 + change_state(MOVIE_STATE_NONE); // have to stop current movie before trying to re-open it 2.1432 2.1433 - if (movie_filename[0] == '\0') 2.1434 + if (movie_filename[0] == '\0') 2.1435 + { loadingMovie = false; return MOVIE_FILE_NOT_FOUND; } 2.1436 + 2.1437 + if (!(file = fopen(movie_filename, "wb"))) 2.1438 + { loadingMovie = false; return MOVIE_FILE_NOT_FOUND; } 2.1439 + 2.1440 + if (!alreadyOpen) 2.1441 + change_state(MOVIE_STATE_NONE); // stop current movie when we're able to open the other one 2.1442 + 2.1443 + // clear out the current movie 2.1444 + printf("RLM: movie init\n"); 2.1445 + 2.1446 + VBAMovieInit(); 2.1447 + 2.1448 + // fill in the movie's header 2.1449 + Movie.header.uid = (uint32)time(NULL); 2.1450 + Movie.header.magic = VBM_MAGIC; 2.1451 + Movie.header.version = VBM_VERSION; 2.1452 + Movie.header.rerecord_count = 0; 2.1453 + Movie.header.length_frames = 0; 2.1454 + Movie.header.startFlags = startFlags; 2.1455 + Movie.header.controllerFlags = controllerFlags; 2.1456 + Movie.header.typeFlags = typeFlags; 2.1457 + Movie.header.minorVersion = VBM_REVISION; 2.1458 + 2.1459 + // set emulator settings that make the movie more likely to stay synchronized when it's later played back 2.1460 + SetRecordEmuSettings(); 2.1461 + 2.1462 + // set ROM and BIOS checksums and stuff 2.1463 + VBAMovieGetRomInfo(Movie, Movie.header.romTitle, Movie.header.romGameCode, Movie.header.romOrBiosChecksum, Movie.header.romCRC); 2.1464 + 2.1465 + printf("RLM: Writing movie header\n"); 2.1466 + // write the header to file 2.1467 + write_movie_header(file, Movie); 2.1468 + 2.1469 + printf("RLM: setting metadata\n"); 2.1470 + 2.1471 + // copy over the metadata / author info 2.1472 + VBAMovieSetMetadata("________________Robert McIntyre______________________________________________________________________________________________________________________________________________________________________________________________________________________"); 2.1473 + 2.1474 + printf("RLM: writing metadata\n"); 2.1475 + 2.1476 + // write the metadata / author info to file 2.1477 + 2.1478 + 2.1479 + fwrite(Movie.authorInfo, 1, sizeof(char) * MOVIE_METADATA_SIZE, file); 2.1480 + 2.1481 + // write snapshot or SRAM if applicable 2.1482 + if (Movie.header.startFlags & MOVIE_START_FROM_SNAPSHOT 2.1483 + || Movie.header.startFlags & MOVIE_START_FROM_SRAM) 2.1484 + { 2.1485 + Movie.header.offset_to_savestate = (uint32)ftell(file); 2.1486 + 2.1487 + // close the file and reopen it as a stream: 2.1488 + 2.1489 + fn = dup(fileno(file)); 2.1490 + fclose(file); 2.1491 + 2.1492 + if (!(stream = utilGzReopen(fn, "ab"))) // append mode to start at end, no seek necessary 2.1493 { loadingMovie = false; return MOVIE_FILE_NOT_FOUND; } 2.1494 2.1495 - if (!(file = fopen(movie_filename, "wb"))) 2.1496 + // write the save data: 2.1497 + if (Movie.header.startFlags & MOVIE_START_FROM_SNAPSHOT) 2.1498 + { 2.1499 + // save snapshot 2.1500 + if (!theEmulator.emuWriteStateToStream(stream)) 2.1501 + { 2.1502 + utilGzClose(stream); 2.1503 + { loadingMovie = false; return MOVIE_UNKNOWN_ERROR; } 2.1504 + } 2.1505 + } 2.1506 + else if (Movie.header.startFlags & MOVIE_START_FROM_SRAM) 2.1507 + { 2.1508 + // save SRAM 2.1509 + if (!theEmulator.emuWriteBatteryToStream(stream)) 2.1510 + { 2.1511 + utilGzClose(stream); 2.1512 + { loadingMovie = false; return MOVIE_UNKNOWN_ERROR; } 2.1513 + } 2.1514 + 2.1515 + // 'soft' reset: 2.1516 + theEmulator.emuReset(false); 2.1517 + } 2.1518 + 2.1519 + utilGzClose(stream); 2.1520 + 2.1521 + // reopen the file and seek back to the end 2.1522 + 2.1523 + if (!(file = fopen(movie_filename, "rb+"))) 2.1524 { loadingMovie = false; return MOVIE_FILE_NOT_FOUND; } 2.1525 2.1526 - if (!alreadyOpen) 2.1527 - change_state(MOVIE_STATE_NONE); // stop current movie when we're able to open the other one 2.1528 + fseek(file, 0, SEEK_END); 2.1529 + } 2.1530 + else // no snapshot or SRAM 2.1531 + { 2.1532 + HardResetAndSRAMClear(); 2.1533 + } 2.1534 2.1535 - // clear out the current movie 2.1536 - VBAMovieInit(); 2.1537 + Movie.header.offset_to_controller_data = (uint32)ftell(file); 2.1538 2.1539 - // fill in the movie's header 2.1540 - Movie.header.uid = (uint32)time(NULL); 2.1541 - Movie.header.magic = VBM_MAGIC; 2.1542 - Movie.header.version = VBM_VERSION; 2.1543 - Movie.header.rerecord_count = 0; 2.1544 - Movie.header.length_frames = 0; 2.1545 - Movie.header.startFlags = startFlags; 2.1546 - Movie.header.controllerFlags = controllerFlags; 2.1547 - Movie.header.typeFlags = typeFlags; 2.1548 - Movie.header.minorVersion = VBM_REVISION; 2.1549 + strcpy(Movie.filename, movie_filename); 2.1550 + Movie.file = file; 2.1551 + Movie.bytesPerFrame = bytes_per_frame(Movie); 2.1552 + Movie.inputBufferPtr = Movie.inputBuffer; 2.1553 + Movie.currentFrame = 0; 2.1554 + Movie.readOnly = false; 2.1555 + Movie.RecordedThisSession = true; 2.1556 2.1557 - // set emulator settings that make the movie more likely to stay synchronized when it's later played back 2.1558 - SetRecordEmuSettings(); 2.1559 + change_state(MOVIE_STATE_RECORD); 2.1560 2.1561 - // set ROM and BIOS checksums and stuff 2.1562 - VBAMovieGetRomInfo(Movie, Movie.header.romTitle, Movie.header.romGameCode, Movie.header.romOrBiosChecksum, Movie.header.romCRC); 2.1563 - 2.1564 - // write the header to file 2.1565 - write_movie_header(file, Movie); 2.1566 - 2.1567 - // copy over the metadata / author info 2.1568 - VBAMovieSetMetadata(authorInfo); 2.1569 - 2.1570 - // write the metadata / author info to file 2.1571 - fwrite(Movie.authorInfo, 1, sizeof(char) * MOVIE_METADATA_SIZE, file); 2.1572 - 2.1573 - // write snapshot or SRAM if applicable 2.1574 - if (Movie.header.startFlags & MOVIE_START_FROM_SNAPSHOT 2.1575 - || Movie.header.startFlags & MOVIE_START_FROM_SRAM) 2.1576 - { 2.1577 - Movie.header.offset_to_savestate = (uint32)ftell(file); 2.1578 - 2.1579 - // close the file and reopen it as a stream: 2.1580 - 2.1581 - fn = dup(fileno(file)); 2.1582 - fclose(file); 2.1583 - 2.1584 - if (!(stream = utilGzReopen(fn, "ab"))) // append mode to start at end, no seek necessary 2.1585 - { loadingMovie = false; return MOVIE_FILE_NOT_FOUND; } 2.1586 - 2.1587 - // write the save data: 2.1588 - if (Movie.header.startFlags & MOVIE_START_FROM_SNAPSHOT) 2.1589 - { 2.1590 - // save snapshot 2.1591 - if (!theEmulator.emuWriteStateToStream(stream)) 2.1592 - { 2.1593 - utilGzClose(stream); 2.1594 - { loadingMovie = false; return MOVIE_UNKNOWN_ERROR; } 2.1595 - } 2.1596 - } 2.1597 - else if (Movie.header.startFlags & MOVIE_START_FROM_SRAM) 2.1598 - { 2.1599 - // save SRAM 2.1600 - if (!theEmulator.emuWriteBatteryToStream(stream)) 2.1601 - { 2.1602 - utilGzClose(stream); 2.1603 - { loadingMovie = false; return MOVIE_UNKNOWN_ERROR; } 2.1604 - } 2.1605 - 2.1606 - // 'soft' reset: 2.1607 - theEmulator.emuReset(false); 2.1608 - } 2.1609 - 2.1610 - utilGzClose(stream); 2.1611 - 2.1612 - // reopen the file and seek back to the end 2.1613 - 2.1614 - if (!(file = fopen(movie_filename, "rb+"))) 2.1615 - { loadingMovie = false; return MOVIE_FILE_NOT_FOUND; } 2.1616 - 2.1617 - fseek(file, 0, SEEK_END); 2.1618 - } 2.1619 - else // no snapshot or SRAM 2.1620 - { 2.1621 - HardResetAndSRAMClear(); 2.1622 - } 2.1623 - 2.1624 - Movie.header.offset_to_controller_data = (uint32)ftell(file); 2.1625 - 2.1626 - strcpy(Movie.filename, movie_filename); 2.1627 - Movie.file = file; 2.1628 - Movie.bytesPerFrame = bytes_per_frame(Movie); 2.1629 - Movie.inputBufferPtr = Movie.inputBuffer; 2.1630 - Movie.currentFrame = 0; 2.1631 - Movie.readOnly = false; 2.1632 - Movie.RecordedThisSession = true; 2.1633 - 2.1634 - change_state(MOVIE_STATE_RECORD); 2.1635 - 2.1636 - systemScreenMessage("Recording movie..."); 2.1637 - { loadingMovie = false; return MOVIE_SUCCESS; } 2.1638 + systemScreenMessage("Recording movie..."); 2.1639 + { loadingMovie = false; return MOVIE_SUCCESS; } 2.1640 } 2.1641 2.1642 void VBAUpdateButtonPressDisplay() 2.1643 { 2.1644 - uint32 keys = currentButtons[0] & BUTTON_REGULAR_RECORDING_MASK; 2.1645 + uint32 keys = currentButtons[0] & BUTTON_REGULAR_RECORDING_MASK; 2.1646 2.1647 - const static char KeyMap[] = { 'A', 'B', 's', 'S', '>', '<', '^', 'v', 'R', 'L', '!', '?', '{', '}', 'v', '^' }; 2.1648 - const static int KeyOrder[] = { 5, 6, 4, 7, 0, 1, 9, 8, 3, 2, 12, 15, 13, 14, 11, 10 }; // < ^ > v A B L R S s { = } _ 2.1649 - // ? ! 2.1650 - char buffer[256]; 2.1651 - sprintf(buffer, " "); 2.1652 + const static char KeyMap[] = { 'A', 'B', 's', 'S', '>', '<', '^', 'v', 'R', 'L', '!', '?', '{', '}', 'v', '^' }; 2.1653 + const static int KeyOrder[] = { 5, 6, 4, 7, 0, 1, 9, 8, 3, 2, 12, 15, 13, 14, 11, 10 }; // < ^ > v A B L R S s { = } _ 2.1654 + // ? ! 2.1655 + char buffer[256]; 2.1656 + sprintf(buffer, " "); 2.1657 2.1658 #ifndef WIN32 2.1659 - // don't bother color-coding autofire and such 2.1660 - int i; 2.1661 - for (i = 0; i < 15; i++) 2.1662 + // don't bother color-coding autofire and such 2.1663 + int i; 2.1664 + for (i = 0; i < 15; i++) 2.1665 + { 2.1666 + int j = KeyOrder[i]; 2.1667 + int mask = (1 << (j)); 2.1668 + buffer[strlen(" ") + i] = ((keys & mask) != 0) ? KeyMap[j] : ' '; 2.1669 + } 2.1670 + 2.1671 + systemScreenMessage(buffer, 2, -1); 2.1672 +#else 2.1673 + const bool eraseAll = !theApp.inputDisplay; 2.1674 + uint32 autoHeldKeys = eraseAll ? 0 : theApp.autoHold & BUTTON_REGULAR_RECORDING_MASK; 2.1675 + uint32 autoFireKeys = eraseAll ? 0 : (theApp.autoFire | theApp.autoFire2) & BUTTON_REGULAR_RECORDING_MASK; 2.1676 + uint32 pressedKeys = eraseAll ? 0 : keys; 2.1677 + 2.1678 + char colorList[64]; 2.1679 + memset(colorList, 1, strlen(buffer)); 2.1680 + 2.1681 + if (!eraseAll) 2.1682 + { 2.1683 + for (int i = 0; i < 15; i++) 2.1684 { 2.1685 - int j = KeyOrder[i]; 2.1686 - int mask = (1 << (j)); 2.1687 - buffer[strlen(" ") + i] = ((keys & mask) != 0) ? KeyMap[j] : ' '; 2.1688 + const int j = KeyOrder[i]; 2.1689 + const int mask = (1 << (j)); 2.1690 + bool pressed = (pressedKeys & mask) != 0; 2.1691 + const bool autoHeld = (autoHeldKeys & mask) != 0; 2.1692 + const bool autoFired = (autoFireKeys & mask) != 0; 2.1693 + const bool erased = (lastKeys & mask) != 0 && (!pressed && !autoHeld && !autoFired); 2.1694 + extern int textMethod; 2.1695 + if (textMethod != 2 && (autoHeld || (autoFired && !pressed) || erased)) 2.1696 + { 2.1697 + int colorNum = 1; // default is white 2.1698 + if (autoHeld) 2.1699 + colorNum += (pressed ? 2 : 1); // yellow if pressed, red if not 2.1700 + else if (autoFired) 2.1701 + colorNum += 5; // blue if autofired and not currently pressed 2.1702 + else if (erased) 2.1703 + colorNum += 8; // black on black 2.1704 + 2.1705 + colorList[strlen(" ") + i] = colorNum; 2.1706 + pressed = true; 2.1707 + } 2.1708 + buffer[strlen(" ") + i] = pressed ? KeyMap[j] : ' '; 2.1709 } 2.1710 + } 2.1711 2.1712 - systemScreenMessage(buffer, 2, -1); 2.1713 -#else 2.1714 - const bool eraseAll = !theApp.inputDisplay; 2.1715 - uint32 autoHeldKeys = eraseAll ? 0 : theApp.autoHold & BUTTON_REGULAR_RECORDING_MASK; 2.1716 - uint32 autoFireKeys = eraseAll ? 0 : (theApp.autoFire | theApp.autoFire2) & BUTTON_REGULAR_RECORDING_MASK; 2.1717 - uint32 pressedKeys = eraseAll ? 0 : keys; 2.1718 + lastKeys = currentButtons[0]; 2.1719 + lastKeys |= theApp.autoHold & BUTTON_REGULAR_RECORDING_MASK; 2.1720 + lastKeys |= (theApp.autoFire | theApp.autoFire2) & BUTTON_REGULAR_RECORDING_MASK; 2.1721 2.1722 - char colorList[64]; 2.1723 - memset(colorList, 1, strlen(buffer)); 2.1724 - 2.1725 - if (!eraseAll) 2.1726 - { 2.1727 - for (int i = 0; i < 15; i++) 2.1728 - { 2.1729 - const int j = KeyOrder[i]; 2.1730 - const int mask = (1 << (j)); 2.1731 - bool pressed = (pressedKeys & mask) != 0; 2.1732 - const bool autoHeld = (autoHeldKeys & mask) != 0; 2.1733 - const bool autoFired = (autoFireKeys & mask) != 0; 2.1734 - const bool erased = (lastKeys & mask) != 0 && (!pressed && !autoHeld && !autoFired); 2.1735 - extern int textMethod; 2.1736 - if (textMethod != 2 && (autoHeld || (autoFired && !pressed) || erased)) 2.1737 - { 2.1738 - int colorNum = 1; // default is white 2.1739 - if (autoHeld) 2.1740 - colorNum += (pressed ? 2 : 1); // yellow if pressed, red if not 2.1741 - else if (autoFired) 2.1742 - colorNum += 5; // blue if autofired and not currently pressed 2.1743 - else if (erased) 2.1744 - colorNum += 8; // black on black 2.1745 - 2.1746 - colorList[strlen(" ") + i] = colorNum; 2.1747 - pressed = true; 2.1748 - } 2.1749 - buffer[strlen(" ") + i] = pressed ? KeyMap[j] : ' '; 2.1750 - } 2.1751 - } 2.1752 - 2.1753 - lastKeys = currentButtons[0]; 2.1754 - lastKeys |= theApp.autoHold & BUTTON_REGULAR_RECORDING_MASK; 2.1755 - lastKeys |= (theApp.autoFire | theApp.autoFire2) & BUTTON_REGULAR_RECORDING_MASK; 2.1756 - 2.1757 - systemScreenMessage(buffer, 2, -1, colorList); 2.1758 + systemScreenMessage(buffer, 2, -1, colorList); 2.1759 #endif 2.1760 } 2.1761 2.1762 void VBAUpdateFrameCountDisplay() 2.1763 { 2.1764 - const int MAGICAL_NUMBER = 64; // FIXME: this won't do any better, but only to remind you of sz issues 2.1765 - char frameDisplayString[MAGICAL_NUMBER]; 2.1766 - char lagFrameDisplayString[MAGICAL_NUMBER]; 2.1767 - char extraCountDisplayString[MAGICAL_NUMBER]; 2.1768 + const int MAGICAL_NUMBER = 64; // FIXME: this won't do any better, but only to remind you of sz issues 2.1769 + char frameDisplayString[MAGICAL_NUMBER]; 2.1770 + char lagFrameDisplayString[MAGICAL_NUMBER]; 2.1771 + char extraCountDisplayString[MAGICAL_NUMBER]; 2.1772 2.1773 #if (defined(WIN32) && !defined(SDL)) 2.1774 - if (theApp.frameCounter) 2.1775 + if (theApp.frameCounter) 2.1776 +#else 2.1777 + /// SDL FIXME 2.1778 +#endif 2.1779 + { 2.1780 + switch (Movie.state) 2.1781 + { 2.1782 + case MOVIE_STATE_PLAY: 2.1783 + case MOVIE_STATE_END: 2.1784 + { 2.1785 + sprintf(frameDisplayString, "%d / %d", Movie.currentFrame, Movie.header.length_frames); 2.1786 + if (!Movie.readOnly) 2.1787 + strcat(frameDisplayString, " (edit)"); 2.1788 + break; 2.1789 + } 2.1790 + case MOVIE_STATE_RECORD: 2.1791 + { 2.1792 + sprintf(frameDisplayString, "%d (record)", Movie.currentFrame); 2.1793 + break; 2.1794 + } 2.1795 + default: 2.1796 + { 2.1797 + sprintf(frameDisplayString, "%d (no movie)", systemCounters.frameCount); 2.1798 + break; 2.1799 + } 2.1800 + } 2.1801 + 2.1802 +#if (defined(WIN32) && !defined(SDL)) 2.1803 + if (theApp.lagCounter) 2.1804 #else 2.1805 /// SDL FIXME 2.1806 #endif 2.1807 { 2.1808 - switch (Movie.state) 2.1809 - { 2.1810 - case MOVIE_STATE_PLAY: 2.1811 - case MOVIE_STATE_END: 2.1812 - { 2.1813 - sprintf(frameDisplayString, "%d / %d", Movie.currentFrame, Movie.header.length_frames); 2.1814 - if (!Movie.readOnly) 2.1815 - strcat(frameDisplayString, " (edit)"); 2.1816 - break; 2.1817 - } 2.1818 - case MOVIE_STATE_RECORD: 2.1819 - { 2.1820 - sprintf(frameDisplayString, "%d (record)", Movie.currentFrame); 2.1821 - break; 2.1822 - } 2.1823 - default: 2.1824 - { 2.1825 - sprintf(frameDisplayString, "%d (no movie)", systemCounters.frameCount); 2.1826 - break; 2.1827 - } 2.1828 - } 2.1829 + // sprintf(lagFrameDisplayString, " %c %d", systemCounters.laggedLast ? '*' : '|', systemCounters.lagCount); 2.1830 + sprintf(lagFrameDisplayString, " | %d%s", systemCounters.lagCount, systemCounters.laggedLast ? " *" : ""); 2.1831 + strcat(frameDisplayString, lagFrameDisplayString); 2.1832 + } 2.1833 2.1834 #if (defined(WIN32) && !defined(SDL)) 2.1835 - if (theApp.lagCounter) 2.1836 -#else 2.1837 - /// SDL FIXME 2.1838 -#endif 2.1839 - { 2.1840 -// sprintf(lagFrameDisplayString, " %c %d", systemCounters.laggedLast ? '*' : '|', systemCounters.lagCount); 2.1841 - sprintf(lagFrameDisplayString, " | %d%s", systemCounters.lagCount, systemCounters.laggedLast ? " *" : ""); 2.1842 - strcat(frameDisplayString, lagFrameDisplayString); 2.1843 - } 2.1844 - 2.1845 -#if (defined(WIN32) && !defined(SDL)) 2.1846 - if (theApp.extraCounter) 2.1847 -#else 2.1848 - /// SDL FIXME 2.1849 -#endif 2.1850 - { 2.1851 - sprintf(extraCountDisplayString, " | %d", systemCounters.frameCount - systemCounters.extraCount); 2.1852 - strcat(frameDisplayString, extraCountDisplayString); 2.1853 - } 2.1854 - } 2.1855 -#if (defined(WIN32) && !defined(SDL)) 2.1856 - else 2.1857 - { 2.1858 - frameDisplayString[0] = '\0'; 2.1859 - } 2.1860 + if (theApp.extraCounter) 2.1861 #else 2.1862 /// SDL FIXME 2.1863 #endif 2.1864 - systemScreenMessage(frameDisplayString, 1, -1); 2.1865 + { 2.1866 + sprintf(extraCountDisplayString, " | %d", systemCounters.frameCount - systemCounters.extraCount); 2.1867 + strcat(frameDisplayString, extraCountDisplayString); 2.1868 + } 2.1869 + } 2.1870 +#if (defined(WIN32) && !defined(SDL)) 2.1871 + else 2.1872 + { 2.1873 + frameDisplayString[0] = '\0'; 2.1874 + } 2.1875 +#else 2.1876 + /// SDL FIXME 2.1877 +#endif 2.1878 + systemScreenMessage(frameDisplayString, 1, -1); 2.1879 } 2.1880 2.1881 // this function should only be called once every frame 2.1882 void VBAMovieUpdateState() 2.1883 { 2.1884 - ++Movie.currentFrame; 2.1885 - 2.1886 - if (Movie.state == MOVIE_STATE_PLAY) 2.1887 + ++Movie.currentFrame; 2.1888 + printf("RLM: inside updateState\n"); 2.1889 + if (Movie.state == MOVIE_STATE_PLAY) 2.1890 + { 2.1891 + Movie.inputBufferPtr += Movie.bytesPerFrame; 2.1892 + if (Movie.currentFrame >= Movie.header.length_frames) 2.1893 { 2.1894 - Movie.inputBufferPtr += Movie.bytesPerFrame; 2.1895 - if (Movie.currentFrame >= Movie.header.length_frames) 2.1896 - { 2.1897 - // the movie ends anyway; what to do next depends on the settings 2.1898 - change_state(MOVIE_STATE_END); 2.1899 - } 2.1900 + // the movie ends anyway; what to do next depends on the settings 2.1901 + change_state(MOVIE_STATE_END); 2.1902 } 2.1903 - else if (Movie.state == MOVIE_STATE_RECORD) 2.1904 - { 2.1905 - // use first fseek? 2.1906 - fwrite(Movie.inputBufferPtr, 1, Movie.bytesPerFrame, Movie.file); 2.1907 - Movie.header.length_frames = Movie.currentFrame; 2.1908 - Movie.inputBufferPtr += Movie.bytesPerFrame; 2.1909 - Movie.RecordedThisSession = true; 2.1910 - flush_movie_header(); 2.1911 - } 2.1912 - else if (Movie.state == MOVIE_STATE_END) 2.1913 - { 2.1914 - change_state(MOVIE_STATE_END); 2.1915 - } 2.1916 + } 2.1917 + else if (Movie.state == MOVIE_STATE_RECORD) 2.1918 + { 2.1919 + printf("RLM: Movie_STATE_RECORD\n"); 2.1920 + // use first fseek? 2.1921 + //TODO: THis is the problem. 2.1922 + //fwrite(Movie.inputBufferPtr, 1, Movie.bytesPerFrame, Movie.file); 2.1923 + printf("RLM: fuck.\n"); 2.1924 + Movie.header.length_frames = Movie.currentFrame; 2.1925 + Movie.inputBufferPtr += Movie.bytesPerFrame; 2.1926 + Movie.RecordedThisSession = true; 2.1927 + flush_movie_header(); 2.1928 + } 2.1929 + else if (Movie.state == MOVIE_STATE_END) 2.1930 + { 2.1931 + change_state(MOVIE_STATE_END); 2.1932 + } 2.1933 } 2.1934 2.1935 void VBAMovieRead(int i, bool /*sensor*/) 2.1936 { 2.1937 - if (Movie.state != MOVIE_STATE_PLAY) 2.1938 - return; 2.1939 + if (Movie.state != MOVIE_STATE_PLAY) 2.1940 + return; 2.1941 2.1942 - if (i < 0 || i >= MOVIE_NUM_OF_POSSIBLE_CONTROLLERS) 2.1943 - return; // not a controller we're recognizing 2.1944 + if (i < 0 || i >= MOVIE_NUM_OF_POSSIBLE_CONTROLLERS) 2.1945 + return; // not a controller we're recognizing 2.1946 2.1947 - if (Movie.header.controllerFlags & MOVIE_CONTROLLER(i)) 2.1948 - { 2.1949 - currentButtons[i] = Read16(Movie.inputBufferPtr + CONTROLLER_DATA_SIZE * i); 2.1950 - } 2.1951 - else 2.1952 - { 2.1953 - currentButtons[i] = 0; // pretend the controller is disconnected 2.1954 - } 2.1955 + if (Movie.header.controllerFlags & MOVIE_CONTROLLER(i)) 2.1956 + { 2.1957 + currentButtons[i] = Read16(Movie.inputBufferPtr + CONTROLLER_DATA_SIZE * i); 2.1958 + } 2.1959 + else 2.1960 + { 2.1961 + currentButtons[i] = 0; // pretend the controller is disconnected 2.1962 + } 2.1963 2.1964 - if ((currentButtons[i] & BUTTON_MASK_NEW_RESET) != 0) 2.1965 - resetSignaled = true; 2.1966 + if ((currentButtons[i] & BUTTON_MASK_NEW_RESET) != 0) 2.1967 + resetSignaled = true; 2.1968 } 2.1969 2.1970 void VBAMovieWrite(int i, bool /*sensor*/) 2.1971 { 2.1972 - if (Movie.state != MOVIE_STATE_RECORD) 2.1973 - return; 2.1974 + if (Movie.state != MOVIE_STATE_RECORD) 2.1975 + return; 2.1976 2.1977 - if (i < 0 || i >= MOVIE_NUM_OF_POSSIBLE_CONTROLLERS) 2.1978 - return; // not a controller we're recognizing 2.1979 + if (i < 0 || i >= MOVIE_NUM_OF_POSSIBLE_CONTROLLERS) 2.1980 + return; // not a controller we're recognizing 2.1981 2.1982 - reserve_buffer_space((uint32)((Movie.inputBufferPtr - Movie.inputBuffer) + Movie.bytesPerFrame)); 2.1983 + reserve_buffer_space((uint32)((Movie.inputBufferPtr - Movie.inputBuffer) + Movie.bytesPerFrame)); 2.1984 2.1985 - if (Movie.header.controllerFlags & MOVIE_CONTROLLER(i)) 2.1986 + if (Movie.header.controllerFlags & MOVIE_CONTROLLER(i)) 2.1987 + { 2.1988 + // get the current controller data 2.1989 + uint16 buttonData = currentButtons[i]; 2.1990 + 2.1991 + // mask away the irrelevent bits 2.1992 + buttonData &= BUTTON_REGULAR_MASK | BUTTON_MOTION_MASK; 2.1993 + 2.1994 + // soft-reset "button" for 1 frame if the game is reset while recording 2.1995 + if (resetSignaled) 2.1996 { 2.1997 - // get the current controller data 2.1998 - uint16 buttonData = currentButtons[i]; 2.1999 + buttonData |= BUTTON_MASK_NEW_RESET; 2.2000 + } 2.2001 2.2002 - // mask away the irrelevent bits 2.2003 - buttonData &= BUTTON_REGULAR_MASK | BUTTON_MOTION_MASK; 2.2004 + // backward compatibility kludge 2.2005 + if (resetSignaledLast) 2.2006 + { 2.2007 + buttonData |= BUTTON_MASK_OLD_RESET; 2.2008 + } 2.2009 2.2010 - // soft-reset "button" for 1 frame if the game is reset while recording 2.2011 - if (resetSignaled) 2.2012 - { 2.2013 - buttonData |= BUTTON_MASK_NEW_RESET; 2.2014 - } 2.2015 + Write16(buttonData, Movie.inputBufferPtr + CONTROLLER_DATA_SIZE * i); 2.2016 2.2017 - // backward compatibility kludge 2.2018 - if (resetSignaledLast) 2.2019 - { 2.2020 - buttonData |= BUTTON_MASK_OLD_RESET; 2.2021 - } 2.2022 - 2.2023 - Write16(buttonData, Movie.inputBufferPtr + CONTROLLER_DATA_SIZE * i); 2.2024 - 2.2025 - // and for display 2.2026 - currentButtons[i] = buttonData; 2.2027 - } 2.2028 - else 2.2029 - { 2.2030 - // pretend the controller is disconnected (otherwise input it gives could cause desync since we're not writing it to the 2.2031 - // movie) 2.2032 - currentButtons[i] = 0; 2.2033 - } 2.2034 + // and for display 2.2035 + currentButtons[i] = buttonData; 2.2036 + } 2.2037 + else 2.2038 + { 2.2039 + // pretend the controller is disconnected (otherwise input it gives could cause desync since we're not writing it to the 2.2040 + // movie) 2.2041 + currentButtons[i] = 0; 2.2042 + } 2.2043 } 2.2044 2.2045 void VBAMovieStop(bool8 suppress_message) 2.2046 { 2.2047 - if (Movie.state != MOVIE_STATE_NONE) 2.2048 - { 2.2049 - change_state(MOVIE_STATE_NONE); 2.2050 - if (!suppress_message) 2.2051 - systemScreenMessage("Movie stop"); 2.2052 - } 2.2053 + if (Movie.state != MOVIE_STATE_NONE) 2.2054 + { 2.2055 + change_state(MOVIE_STATE_NONE); 2.2056 + if (!suppress_message) 2.2057 + systemScreenMessage("Movie stop"); 2.2058 + } 2.2059 } 2.2060 2.2061 int VBAMovieGetInfo(const char *filename, SMovie *info) 2.2062 { 2.2063 - assert(info != NULL); 2.2064 - if (info == NULL) 2.2065 - return -1; 2.2066 + assert(info != NULL); 2.2067 + if (info == NULL) 2.2068 + return -1; 2.2069 2.2070 - FILE * file; 2.2071 - int result; 2.2072 - SMovie &local_movie = *info; 2.2073 + FILE * file; 2.2074 + int result; 2.2075 + SMovie &local_movie = *info; 2.2076 2.2077 - memset(info, 0, sizeof(*info)); 2.2078 - if (filename[0] == '\0') 2.2079 - return MOVIE_FILE_NOT_FOUND; 2.2080 - if (!(file = fopen(filename, "rb"))) 2.2081 - return MOVIE_FILE_NOT_FOUND; 2.2082 + memset(info, 0, sizeof(*info)); 2.2083 + if (filename[0] == '\0') 2.2084 + return MOVIE_FILE_NOT_FOUND; 2.2085 + if (!(file = fopen(filename, "rb"))) 2.2086 + return MOVIE_FILE_NOT_FOUND; 2.2087 2.2088 - // read header 2.2089 - if ((result = (read_movie_header(file, local_movie))) != MOVIE_SUCCESS) 2.2090 - { 2.2091 - fclose(file); 2.2092 - return result; 2.2093 - } 2.2094 + // read header 2.2095 + if ((result = (read_movie_header(file, local_movie))) != MOVIE_SUCCESS) 2.2096 + { 2.2097 + fclose(file); 2.2098 + return result; 2.2099 + } 2.2100 2.2101 - // read the metadata / author info from file 2.2102 - fread(local_movie.authorInfo, 1, sizeof(char) * MOVIE_METADATA_SIZE, file); 2.2103 + // read the metadata / author info from file 2.2104 + fread(local_movie.authorInfo, 1, sizeof(char) * MOVIE_METADATA_SIZE, file); 2.2105 2.2106 - strncpy(local_movie.filename, filename, _MAX_PATH); 2.2107 - local_movie.filename[_MAX_PATH - 1] = '\0'; 2.2108 + strncpy(local_movie.filename, filename, _MAX_PATH); 2.2109 + local_movie.filename[_MAX_PATH - 1] = '\0'; 2.2110 2.2111 - if (Movie.file != NULL && stricmp(local_movie.filename, Movie.filename) == 0) // alreadyOpen 2.2112 - { 2.2113 - local_movie.bytesPerFrame = Movie.bytesPerFrame; 2.2114 - local_movie.header.length_frames = Movie.header.length_frames; 2.2115 - } 2.2116 - else 2.2117 - { 2.2118 - // recalculate length of movie from the file size 2.2119 - local_movie.bytesPerFrame = bytes_per_frame(local_movie); 2.2120 - fseek(file, 0, SEEK_END); 2.2121 - int fileSize = ftell(file); 2.2122 - local_movie.header.length_frames = 2.2123 - (fileSize - local_movie.header.offset_to_controller_data) / local_movie.bytesPerFrame; 2.2124 - } 2.2125 + if (Movie.file != NULL && stricmp(local_movie.filename, Movie.filename) == 0) // alreadyOpen 2.2126 + { 2.2127 + local_movie.bytesPerFrame = Movie.bytesPerFrame; 2.2128 + local_movie.header.length_frames = Movie.header.length_frames; 2.2129 + } 2.2130 + else 2.2131 + { 2.2132 + // recalculate length of movie from the file size 2.2133 + local_movie.bytesPerFrame = bytes_per_frame(local_movie); 2.2134 + fseek(file, 0, SEEK_END); 2.2135 + int fileSize = ftell(file); 2.2136 + local_movie.header.length_frames = 2.2137 + (fileSize - local_movie.header.offset_to_controller_data) / local_movie.bytesPerFrame; 2.2138 + } 2.2139 2.2140 - fclose(file); 2.2141 + fclose(file); 2.2142 2.2143 - if (access(filename, W_OK)) 2.2144 - info->readOnly = true; 2.2145 + if (access(filename, W_OK)) 2.2146 + info->readOnly = true; 2.2147 2.2148 - return MOVIE_SUCCESS; 2.2149 + return MOVIE_SUCCESS; 2.2150 } 2.2151 2.2152 bool8 VBAMovieActive() 2.2153 { 2.2154 - return (Movie.state != MOVIE_STATE_NONE); 2.2155 + return (Movie.state != MOVIE_STATE_NONE); 2.2156 } 2.2157 2.2158 bool8 VBAMovieLoading() 2.2159 { 2.2160 - return loadingMovie; 2.2161 + return loadingMovie; 2.2162 } 2.2163 2.2164 bool8 VBAMoviePlaying() 2.2165 { 2.2166 - return (Movie.state == MOVIE_STATE_PLAY); 2.2167 + return (Movie.state == MOVIE_STATE_PLAY); 2.2168 } 2.2169 2.2170 bool8 VBAMovieRecording() 2.2171 { 2.2172 - return (Movie.state == MOVIE_STATE_RECORD); 2.2173 + return (Movie.state == MOVIE_STATE_RECORD); 2.2174 } 2.2175 2.2176 bool8 VBAMovieReadOnly() 2.2177 { 2.2178 - if (!VBAMovieActive()) 2.2179 - return false; 2.2180 + if (!VBAMovieActive()) 2.2181 + return false; 2.2182 2.2183 - return Movie.readOnly; 2.2184 + return Movie.readOnly; 2.2185 } 2.2186 2.2187 void VBAMovieToggleReadOnly() 2.2188 { 2.2189 - if (!VBAMovieActive()) 2.2190 - return; 2.2191 + if (!VBAMovieActive()) 2.2192 + return; 2.2193 2.2194 - if (Movie.readOnly != 2) 2.2195 - { 2.2196 - Movie.readOnly = !Movie.readOnly; 2.2197 + if (Movie.readOnly != 2) 2.2198 + { 2.2199 + Movie.readOnly = !Movie.readOnly; 2.2200 2.2201 - systemScreenMessage(Movie.readOnly ? "Movie now read-only" : "Movie now editable"); 2.2202 - } 2.2203 - else 2.2204 - { 2.2205 - systemScreenMessage("Can't toggle read-only movie"); 2.2206 - } 2.2207 + systemScreenMessage(Movie.readOnly ? "Movie now read-only" : "Movie now editable"); 2.2208 + } 2.2209 + else 2.2210 + { 2.2211 + systemScreenMessage("Can't toggle read-only movie"); 2.2212 + } 2.2213 } 2.2214 2.2215 uint32 VBAMovieGetVersion() 2.2216 { 2.2217 - if (!VBAMovieActive()) 2.2218 - return 0; 2.2219 + if (!VBAMovieActive()) 2.2220 + return 0; 2.2221 2.2222 - return Movie.header.version; 2.2223 + return Movie.header.version; 2.2224 } 2.2225 2.2226 uint32 VBAMovieGetMinorVersion() 2.2227 { 2.2228 - if (!VBAMovieActive()) 2.2229 - return 0; 2.2230 + if (!VBAMovieActive()) 2.2231 + return 0; 2.2232 2.2233 - return Movie.header.minorVersion; 2.2234 + return Movie.header.minorVersion; 2.2235 } 2.2236 2.2237 uint32 VBAMovieGetId() 2.2238 { 2.2239 - if (!VBAMovieActive()) 2.2240 - return 0; 2.2241 + if (!VBAMovieActive()) 2.2242 + return 0; 2.2243 2.2244 - return Movie.header.uid; 2.2245 + return Movie.header.uid; 2.2246 } 2.2247 2.2248 uint32 VBAMovieGetLength() 2.2249 { 2.2250 - if (!VBAMovieActive()) 2.2251 - return 0; 2.2252 + if (!VBAMovieActive()) 2.2253 + return 0; 2.2254 2.2255 - return Movie.header.length_frames; 2.2256 + return Movie.header.length_frames; 2.2257 } 2.2258 2.2259 uint32 VBAMovieGetFrameCounter() 2.2260 { 2.2261 - if (!VBAMovieActive()) 2.2262 - return 0; 2.2263 + if (!VBAMovieActive()) 2.2264 + return 0; 2.2265 2.2266 - return Movie.currentFrame; 2.2267 + return Movie.currentFrame; 2.2268 } 2.2269 2.2270 uint32 VBAMovieGetRerecordCount() 2.2271 { 2.2272 - if (!VBAMovieActive()) 2.2273 - return 0; 2.2274 + if (!VBAMovieActive()) 2.2275 + return 0; 2.2276 2.2277 - return Movie.header.rerecord_count; 2.2278 + return Movie.header.rerecord_count; 2.2279 } 2.2280 2.2281 uint32 VBAMovieSetRerecordCount(uint32 newRerecordCount) 2.2282 { 2.2283 - uint32 oldRerecordCount = 0; 2.2284 - if (!VBAMovieActive()) 2.2285 - return 0; 2.2286 + uint32 oldRerecordCount = 0; 2.2287 + if (!VBAMovieActive()) 2.2288 + return 0; 2.2289 2.2290 - oldRerecordCount = Movie.header.rerecord_count; 2.2291 - Movie.header.rerecord_count = newRerecordCount; 2.2292 - return oldRerecordCount; 2.2293 + oldRerecordCount = Movie.header.rerecord_count; 2.2294 + Movie.header.rerecord_count = newRerecordCount; 2.2295 + return oldRerecordCount; 2.2296 } 2.2297 2.2298 std::string VBAMovieGetAuthorInfo() 2.2299 { 2.2300 - if (!VBAMovieActive()) 2.2301 - return ""; 2.2302 + if (!VBAMovieActive()) 2.2303 + return ""; 2.2304 2.2305 - return Movie.authorInfo; 2.2306 + return Movie.authorInfo; 2.2307 } 2.2308 2.2309 std::string VBAMovieGetFilename() 2.2310 { 2.2311 - if (!VBAMovieActive()) 2.2312 - return ""; 2.2313 + if (!VBAMovieActive()) 2.2314 + return ""; 2.2315 2.2316 - return Movie.filename; 2.2317 + return Movie.filename; 2.2318 } 2.2319 2.2320 void VBAMovieFreeze(uint8 * *buf, uint32 *size) 2.2321 { 2.2322 - // sanity check 2.2323 - if (!VBAMovieActive()) 2.2324 - { 2.2325 - return; 2.2326 - } 2.2327 + // sanity check 2.2328 + if (!VBAMovieActive()) 2.2329 + { 2.2330 + return; 2.2331 + } 2.2332 2.2333 - *buf = NULL; 2.2334 - *size = 0; 2.2335 + *buf = NULL; 2.2336 + *size = 0; 2.2337 2.2338 - // compute size needed for the buffer 2.2339 - // room for header.uid, currentFrame, and header.length_frames 2.2340 - uint32 size_needed = sizeof(Movie.header.uid) + sizeof(Movie.currentFrame) + sizeof(Movie.header.length_frames); 2.2341 - size_needed += (uint32)(Movie.bytesPerFrame * Movie.header.length_frames); 2.2342 - *buf = new uint8[size_needed]; 2.2343 - *size = size_needed; 2.2344 + // compute size needed for the buffer 2.2345 + // room for header.uid, currentFrame, and header.length_frames 2.2346 + uint32 size_needed = sizeof(Movie.header.uid) + sizeof(Movie.currentFrame) + sizeof(Movie.header.length_frames); 2.2347 + size_needed += (uint32)(Movie.bytesPerFrame * Movie.header.length_frames); 2.2348 + *buf = new uint8[size_needed]; 2.2349 + *size = size_needed; 2.2350 2.2351 - uint8 *ptr = *buf; 2.2352 - if (!ptr) 2.2353 - { 2.2354 - return; 2.2355 - } 2.2356 + uint8 *ptr = *buf; 2.2357 + if (!ptr) 2.2358 + { 2.2359 + return; 2.2360 + } 2.2361 2.2362 - Push32(Movie.header.uid, ptr); 2.2363 - Push32(Movie.currentFrame, ptr); 2.2364 - Push32(Movie.header.length_frames - 1, ptr); // HACK: shorten the length by 1 for backward compatibility 2.2365 + Push32(Movie.header.uid, ptr); 2.2366 + Push32(Movie.currentFrame, ptr); 2.2367 + Push32(Movie.header.length_frames - 1, ptr); // HACK: shorten the length by 1 for backward compatibility 2.2368 2.2369 - memcpy(ptr, Movie.inputBuffer, Movie.bytesPerFrame * Movie.header.length_frames); 2.2370 + memcpy(ptr, Movie.inputBuffer, Movie.bytesPerFrame * Movie.header.length_frames); 2.2371 } 2.2372 2.2373 int VBAMovieUnfreeze(const uint8 *buf, uint32 size) 2.2374 { 2.2375 - // sanity check 2.2376 - if (!VBAMovieActive()) 2.2377 + // sanity check 2.2378 + if (!VBAMovieActive()) 2.2379 + { 2.2380 + return MOVIE_NOT_FROM_A_MOVIE; 2.2381 + } 2.2382 + 2.2383 + const uint8 *ptr = buf; 2.2384 + if (size < sizeof(Movie.header.uid) + sizeof(Movie.currentFrame) + sizeof(Movie.header.length_frames)) 2.2385 + { 2.2386 + return MOVIE_WRONG_FORMAT; 2.2387 + } 2.2388 + 2.2389 + uint32 movie_id = Pop32(ptr); 2.2390 + uint32 current_frame = Pop32(ptr); 2.2391 + uint32 end_frame = Pop32(ptr) + 1; // HACK: restore the length for backward compatibility 2.2392 + uint32 space_needed = Movie.bytesPerFrame * end_frame; 2.2393 + 2.2394 + if (movie_id != Movie.header.uid) 2.2395 + return MOVIE_NOT_FROM_THIS_MOVIE; 2.2396 + 2.2397 + if (space_needed > size) 2.2398 + return MOVIE_WRONG_FORMAT; 2.2399 + 2.2400 + if (Movie.readOnly) 2.2401 + { 2.2402 + // here, we are going to keep the input data from the movie file 2.2403 + // and simply rewind to the currentFrame pointer 2.2404 + // this will cause a desync if the savestate is not in sync // <-- NOT ANYMORE 2.2405 + // with the on-disk recording data, but it's easily solved 2.2406 + // by loading another savestate or playing the movie from the beginning 2.2407 + 2.2408 + // don't allow loading a state inconsistent with the current movie 2.2409 + uint32 length_history = min(current_frame, Movie.header.length_frames); 2.2410 + if (end_frame < length_history) 2.2411 + return MOVIE_SNAPSHOT_INCONSISTENT; 2.2412 + 2.2413 + uint32 space_shared = Movie.bytesPerFrame * length_history; 2.2414 + if (memcmp(Movie.inputBuffer, ptr, space_shared)) 2.2415 + return MOVIE_SNAPSHOT_INCONSISTENT; 2.2416 + 2.2417 + Movie.currentFrame = current_frame; 2.2418 + Movie.inputBufferPtr = Movie.inputBuffer + Movie.bytesPerFrame * min(current_frame, Movie.header.length_frames); 2.2419 + } 2.2420 + else 2.2421 + { 2.2422 + // here, we are going to take the input data from the savestate 2.2423 + // and make it the input data for the current movie, then continue 2.2424 + // writing new input data at the currentFrame pointer 2.2425 + Movie.currentFrame = current_frame; 2.2426 + Movie.header.length_frames = end_frame; 2.2427 + if (!VBALuaRerecordCountSkip()) 2.2428 + ++Movie.header.rerecord_count; 2.2429 + 2.2430 + Movie.RecordedThisSession = true; 2.2431 + 2.2432 + // do this before calling reserve_buffer_space() 2.2433 + Movie.inputBufferPtr = Movie.inputBuffer + Movie.bytesPerFrame * min(current_frame, Movie.header.length_frames); 2.2434 + reserve_buffer_space(space_needed); 2.2435 + memcpy(Movie.inputBuffer, ptr, space_needed); 2.2436 + 2.2437 + // for consistency, no auto movie conversion here since we don't auto convert the corresponding savestate 2.2438 + flush_movie_header(); 2.2439 + flush_movie_frames(); 2.2440 + } 2.2441 + 2.2442 + change_state(MOVIE_STATE_PLAY); // check for movie end 2.2443 + 2.2444 + // necessary! 2.2445 + resetSignaled = false; 2.2446 + resetSignaledLast = false; 2.2447 + 2.2448 + // necessary to check if there's a reset signal at the previous frame 2.2449 + if (current_frame > 0) 2.2450 + { 2.2451 + const u8 NEW_RESET = u8(BUTTON_MASK_NEW_RESET >> 8); 2.2452 + for (int i = 0; i < MOVIE_NUM_OF_POSSIBLE_CONTROLLERS; ++i) 2.2453 { 2.2454 - return MOVIE_NOT_FROM_A_MOVIE; 2.2455 + if ((Movie.header.controllerFlags & MOVIE_CONTROLLER(i)) && (*(Movie.inputBufferPtr+1- Movie.bytesPerFrame) & NEW_RESET)) 2.2456 + { 2.2457 + resetSignaledLast = true; 2.2458 + break; 2.2459 + } 2.2460 } 2.2461 + } 2.2462 2.2463 - const uint8 *ptr = buf; 2.2464 - if (size < sizeof(Movie.header.uid) + sizeof(Movie.currentFrame) + sizeof(Movie.header.length_frames)) 2.2465 - { 2.2466 - return MOVIE_WRONG_FORMAT; 2.2467 - } 2.2468 - 2.2469 - uint32 movie_id = Pop32(ptr); 2.2470 - uint32 current_frame = Pop32(ptr); 2.2471 - uint32 end_frame = Pop32(ptr) + 1; // HACK: restore the length for backward compatibility 2.2472 - uint32 space_needed = Movie.bytesPerFrame * end_frame; 2.2473 - 2.2474 - if (movie_id != Movie.header.uid) 2.2475 - return MOVIE_NOT_FROM_THIS_MOVIE; 2.2476 - 2.2477 - if (space_needed > size) 2.2478 - return MOVIE_WRONG_FORMAT; 2.2479 - 2.2480 - if (Movie.readOnly) 2.2481 - { 2.2482 - // here, we are going to keep the input data from the movie file 2.2483 - // and simply rewind to the currentFrame pointer 2.2484 - // this will cause a desync if the savestate is not in sync // <-- NOT ANYMORE 2.2485 - // with the on-disk recording data, but it's easily solved 2.2486 - // by loading another savestate or playing the movie from the beginning 2.2487 - 2.2488 - // don't allow loading a state inconsistent with the current movie 2.2489 - uint32 length_history = min(current_frame, Movie.header.length_frames); 2.2490 - if (end_frame < length_history) 2.2491 - return MOVIE_SNAPSHOT_INCONSISTENT; 2.2492 - 2.2493 - uint32 space_shared = Movie.bytesPerFrame * length_history; 2.2494 - if (memcmp(Movie.inputBuffer, ptr, space_shared)) 2.2495 - return MOVIE_SNAPSHOT_INCONSISTENT; 2.2496 - 2.2497 - Movie.currentFrame = current_frame; 2.2498 - Movie.inputBufferPtr = Movie.inputBuffer + Movie.bytesPerFrame * min(current_frame, Movie.header.length_frames); 2.2499 - } 2.2500 - else 2.2501 - { 2.2502 - // here, we are going to take the input data from the savestate 2.2503 - // and make it the input data for the current movie, then continue 2.2504 - // writing new input data at the currentFrame pointer 2.2505 - Movie.currentFrame = current_frame; 2.2506 - Movie.header.length_frames = end_frame; 2.2507 - if (!VBALuaRerecordCountSkip()) 2.2508 - ++Movie.header.rerecord_count; 2.2509 - 2.2510 - Movie.RecordedThisSession = true; 2.2511 - 2.2512 - // do this before calling reserve_buffer_space() 2.2513 - Movie.inputBufferPtr = Movie.inputBuffer + Movie.bytesPerFrame * min(current_frame, Movie.header.length_frames); 2.2514 - reserve_buffer_space(space_needed); 2.2515 - memcpy(Movie.inputBuffer, ptr, space_needed); 2.2516 - 2.2517 - // for consistency, no auto movie conversion here since we don't auto convert the corresponding savestate 2.2518 - flush_movie_header(); 2.2519 - flush_movie_frames(); 2.2520 - } 2.2521 - 2.2522 - change_state(MOVIE_STATE_PLAY); // check for movie end 2.2523 - 2.2524 - // necessary! 2.2525 - resetSignaled = false; 2.2526 - resetSignaledLast = false; 2.2527 - 2.2528 - // necessary to check if there's a reset signal at the previous frame 2.2529 - if (current_frame > 0) 2.2530 - { 2.2531 - const u8 NEW_RESET = u8(BUTTON_MASK_NEW_RESET >> 8); 2.2532 - for (int i = 0; i < MOVIE_NUM_OF_POSSIBLE_CONTROLLERS; ++i) 2.2533 - { 2.2534 - if ((Movie.header.controllerFlags & MOVIE_CONTROLLER(i)) && (*(Movie.inputBufferPtr+1- Movie.bytesPerFrame) & NEW_RESET)) 2.2535 - { 2.2536 - resetSignaledLast = true; 2.2537 - break; 2.2538 - } 2.2539 - } 2.2540 - } 2.2541 - 2.2542 - return MOVIE_SUCCESS; 2.2543 + return MOVIE_SUCCESS; 2.2544 } 2.2545 2.2546 bool VBAMovieEnded() 2.2547 { 2.2548 - return (Movie.state == MOVIE_STATE_END); 2.2549 -// return (Movie.state != MOVIE_STATE_NONE && Movie.currentFrame >= Movie.header.length_frames); 2.2550 + return (Movie.state == MOVIE_STATE_END); 2.2551 + // return (Movie.state != MOVIE_STATE_NONE && Movie.currentFrame >= Movie.header.length_frames); 2.2552 } 2.2553 2.2554 bool VBAMovieAllowsRerecording() 2.2555 { 2.2556 - bool allows = (Movie.state != MOVIE_STATE_NONE) && (Movie.currentFrame <= Movie.header.length_frames); 2.2557 - return /*!VBAMovieReadOnly() &&*/ allows; 2.2558 + bool allows = (Movie.state != MOVIE_STATE_NONE) && (Movie.currentFrame <= Movie.header.length_frames); 2.2559 + return /*!VBAMovieReadOnly() &&*/ allows; 2.2560 } 2.2561 2.2562 bool VBAMovieSwitchToPlaying() 2.2563 { 2.2564 - if (!VBAMovieActive()) 2.2565 - return false; 2.2566 + if (!VBAMovieActive()) 2.2567 + return false; 2.2568 2.2569 - if (!Movie.readOnly) 2.2570 - { 2.2571 - VBAMovieToggleReadOnly(); 2.2572 - } 2.2573 + if (!Movie.readOnly) 2.2574 + { 2.2575 + VBAMovieToggleReadOnly(); 2.2576 + } 2.2577 2.2578 - change_state(MOVIE_STATE_PLAY); 2.2579 - if (Movie.state == MOVIE_STATE_PLAY) 2.2580 - systemScreenMessage("Movie replay (continue)"); 2.2581 - else 2.2582 - systemScreenMessage("Movie end"); 2.2583 + change_state(MOVIE_STATE_PLAY); 2.2584 + if (Movie.state == MOVIE_STATE_PLAY) 2.2585 + systemScreenMessage("Movie replay (continue)"); 2.2586 + else 2.2587 + systemScreenMessage("Movie end"); 2.2588 2.2589 - return true; 2.2590 + return true; 2.2591 } 2.2592 2.2593 bool VBAMovieSwitchToRecording() 2.2594 { 2.2595 - if (!VBAMovieAllowsRerecording()) 2.2596 - return false; 2.2597 + if (!VBAMovieAllowsRerecording()) 2.2598 + return false; 2.2599 2.2600 - if (Movie.readOnly) 2.2601 - { 2.2602 - VBAMovieToggleReadOnly(); 2.2603 - } 2.2604 + if (Movie.readOnly) 2.2605 + { 2.2606 + VBAMovieToggleReadOnly(); 2.2607 + } 2.2608 2.2609 - if (!VBALuaRerecordCountSkip()) 2.2610 - ++Movie.header.rerecord_count; 2.2611 + if (!VBALuaRerecordCountSkip()) 2.2612 + ++Movie.header.rerecord_count; 2.2613 2.2614 - change_state(MOVIE_STATE_RECORD); 2.2615 - systemScreenMessage("Movie re-record"); 2.2616 + change_state(MOVIE_STATE_RECORD); 2.2617 + systemScreenMessage("Movie re-record"); 2.2618 2.2619 - //truncate_movie(Movie.currentFrame); 2.2620 + //truncate_movie(Movie.currentFrame); 2.2621 2.2622 - return true; 2.2623 + return true; 2.2624 } 2.2625 2.2626 uint32 VBAMovieGetState() 2.2627 { 2.2628 - // ? 2.2629 - if (!VBAMovieActive()) 2.2630 - return MOVIE_STATE_NONE; 2.2631 + // ? 2.2632 + if (!VBAMovieActive()) 2.2633 + return MOVIE_STATE_NONE; 2.2634 2.2635 - return Movie.state; 2.2636 + return Movie.state; 2.2637 } 2.2638 2.2639 void VBAMovieSignalReset() 2.2640 { 2.2641 - if (VBAMovieActive()) 2.2642 - resetSignaled = true; 2.2643 + if (VBAMovieActive()) 2.2644 + resetSignaled = true; 2.2645 } 2.2646 2.2647 void VBAMovieResetIfRequested() 2.2648 { 2.2649 - if (resetSignaled) 2.2650 - { 2.2651 - theEmulator.emuReset(false); 2.2652 - resetSignaled = false; 2.2653 - resetSignaledLast = true; 2.2654 - } 2.2655 - else 2.2656 - { 2.2657 - resetSignaledLast = false; 2.2658 - } 2.2659 + if (resetSignaled) 2.2660 + { 2.2661 + theEmulator.emuReset(false); 2.2662 + resetSignaled = false; 2.2663 + resetSignaledLast = true; 2.2664 + } 2.2665 + else 2.2666 + { 2.2667 + resetSignaledLast = false; 2.2668 + } 2.2669 } 2.2670 2.2671 void VBAMovieSetMetadata(const char *info) 2.2672 { 2.2673 - if (!memcmp(Movie.authorInfo, info, MOVIE_METADATA_SIZE)) 2.2674 - return; 2.2675 + if (!memcmp(Movie.authorInfo, info, MOVIE_METADATA_SIZE)) 2.2676 + return; 2.2677 2.2678 - memcpy(Movie.authorInfo, info, MOVIE_METADATA_SIZE); // strncpy would omit post-0 bytes 2.2679 - Movie.authorInfo[MOVIE_METADATA_SIZE - 1] = '\0'; 2.2680 + memcpy(Movie.authorInfo, info, MOVIE_METADATA_SIZE); // strncpy would omit post-0 bytes 2.2681 + Movie.authorInfo[MOVIE_METADATA_SIZE - 1] = '\0'; 2.2682 2.2683 - if (Movie.file) 2.2684 - { 2.2685 - // (over-)write the header 2.2686 - fseek(Movie.file, 0, SEEK_SET); 2.2687 - write_movie_header(Movie.file, Movie); 2.2688 + if (Movie.file) 2.2689 + { 2.2690 + // (over-)write the header 2.2691 + fseek(Movie.file, 0, SEEK_SET); 2.2692 2.2693 - // write the metadata / author info to file 2.2694 - fwrite(Movie.authorInfo, 1, sizeof(char) * MOVIE_METADATA_SIZE, Movie.file); 2.2695 + write_movie_header(Movie.file, Movie); 2.2696 2.2697 - fflush(Movie.file); 2.2698 - } 2.2699 + // write the metadata / author info to file 2.2700 + fwrite(Movie.authorInfo, 1, sizeof(char) * MOVIE_METADATA_SIZE, Movie.file); 2.2701 + 2.2702 + fflush(Movie.file); 2.2703 + } 2.2704 + printf("RLM: setMetadata called\n"); 2.2705 + 2.2706 } 2.2707 2.2708 void VBAMovieRestart() 2.2709 { 2.2710 - if (VBAMovieActive()) 2.2711 - { 2.2712 - systemSoundClearBuffer(); 2.2713 + if (VBAMovieActive()) 2.2714 + { 2.2715 + systemSoundClearBuffer(); 2.2716 2.2717 - bool8 modified = Movie.RecordedThisSession; 2.2718 + bool8 modified = Movie.RecordedThisSession; 2.2719 2.2720 - VBAMovieStop(true); 2.2721 + VBAMovieStop(true); 2.2722 2.2723 - char movieName [_MAX_PATH]; 2.2724 - strncpy(movieName, Movie.filename, _MAX_PATH); 2.2725 - movieName[_MAX_PATH - 1] = '\0'; 2.2726 - VBAMovieOpen(movieName, Movie.readOnly); // can't just pass in Movie.filename, since VBAMovieOpen clears out Movie's 2.2727 - // variables 2.2728 + char movieName [_MAX_PATH]; 2.2729 + strncpy(movieName, Movie.filename, _MAX_PATH); 2.2730 + movieName[_MAX_PATH - 1] = '\0'; 2.2731 + VBAMovieOpen(movieName, Movie.readOnly); // can't just pass in Movie.filename, since VBAMovieOpen clears out Movie's 2.2732 + // variables 2.2733 2.2734 - Movie.RecordedThisSession = modified; 2.2735 + Movie.RecordedThisSession = modified; 2.2736 2.2737 - systemScreenMessage("Movie replay (restart)"); 2.2738 - } 2.2739 + systemScreenMessage("Movie replay (restart)"); 2.2740 + } 2.2741 } 2.2742 2.2743 int VBAMovieGetPauseAt() 2.2744 { 2.2745 - return Movie.pauseFrame; 2.2746 + return Movie.pauseFrame; 2.2747 } 2.2748 2.2749 void VBAMovieSetPauseAt(int at) 2.2750 { 2.2751 - Movie.pauseFrame = at; 2.2752 + Movie.pauseFrame = at; 2.2753 } 2.2754 2.2755 /////////////////////// 2.2756 @@ -1688,85 +1703,85 @@ 2.2757 // FIXME: is it safe to convert/flush a movie while recording it (considering fseek() problem)? 2.2758 int VBAMovieConvertCurrent() 2.2759 { 2.2760 - if (!VBAMovieActive()) 2.2761 + if (!VBAMovieActive()) 2.2762 + { 2.2763 + return MOVIE_NOTHING; 2.2764 + } 2.2765 + 2.2766 + if (Movie.header.minorVersion > VBM_REVISION) 2.2767 + { 2.2768 + return MOVIE_WRONG_VERSION; 2.2769 + } 2.2770 + 2.2771 + if (Movie.header.minorVersion == VBM_REVISION) 2.2772 + { 2.2773 + return MOVIE_NOTHING; 2.2774 + } 2.2775 + 2.2776 + Movie.header.minorVersion = VBM_REVISION; 2.2777 + 2.2778 + if (Movie.header.length_frames == 0) // this could happen 2.2779 + { 2.2780 + truncate_movie(0); 2.2781 + return MOVIE_SUCCESS; 2.2782 + } 2.2783 + 2.2784 + // fix movies recorded from snapshots 2.2785 + if (Movie.header.startFlags & MOVIE_START_FROM_SNAPSHOT) 2.2786 + { 2.2787 + uint8 *firstFramePtr = Movie.inputBuffer; 2.2788 + for (int i = 0; i < MOVIE_NUM_OF_POSSIBLE_CONTROLLERS; ++i) 2.2789 { 2.2790 - return MOVIE_NOTHING; 2.2791 + if (Movie.header.controllerFlags & MOVIE_CONTROLLER(i)) 2.2792 + { 2.2793 + Push16(initialInputs[i], firstFramePtr); 2.2794 + // note: this is correct since Push16 advances the dest pointer by sizeof u16 2.2795 + } 2.2796 } 2.2797 + } 2.2798 2.2799 - if (Movie.header.minorVersion > VBM_REVISION) 2.2800 + // convert old resets to new ones 2.2801 + const u8 OLD_RESET = u8(BUTTON_MASK_OLD_RESET >> 8); 2.2802 + const u8 NEW_RESET = u8(BUTTON_MASK_NEW_RESET >> 8); 2.2803 + for (int i = 0; i < MOVIE_NUM_OF_POSSIBLE_CONTROLLERS; ++i) 2.2804 + { 2.2805 + if (Movie.header.controllerFlags & MOVIE_CONTROLLER(i)) 2.2806 { 2.2807 - return MOVIE_WRONG_VERSION; 2.2808 + uint8 *startPtr = Movie.inputBuffer + sizeof(u16) * i + 1; 2.2809 + uint8 *endPtr = Movie.inputBuffer + Movie.bytesPerFrame * (Movie.header.length_frames - 1); 2.2810 + for (; startPtr < endPtr; startPtr += Movie.bytesPerFrame) 2.2811 + { 2.2812 + if (startPtr[Movie.bytesPerFrame] & OLD_RESET) 2.2813 + { 2.2814 + startPtr[0] |= NEW_RESET; 2.2815 + } 2.2816 + } 2.2817 } 2.2818 + } 2.2819 2.2820 - if (Movie.header.minorVersion == VBM_REVISION) 2.2821 - { 2.2822 - return MOVIE_NOTHING; 2.2823 - } 2.2824 - 2.2825 - Movie.header.minorVersion = VBM_REVISION; 2.2826 - 2.2827 - if (Movie.header.length_frames == 0) // this could happen 2.2828 - { 2.2829 - truncate_movie(0); 2.2830 - return MOVIE_SUCCESS; 2.2831 - } 2.2832 - 2.2833 - // fix movies recorded from snapshots 2.2834 - if (Movie.header.startFlags & MOVIE_START_FROM_SNAPSHOT) 2.2835 - { 2.2836 - uint8 *firstFramePtr = Movie.inputBuffer; 2.2837 - for (int i = 0; i < MOVIE_NUM_OF_POSSIBLE_CONTROLLERS; ++i) 2.2838 - { 2.2839 - if (Movie.header.controllerFlags & MOVIE_CONTROLLER(i)) 2.2840 - { 2.2841 - Push16(initialInputs[i], firstFramePtr); 2.2842 - // note: this is correct since Push16 advances the dest pointer by sizeof u16 2.2843 - } 2.2844 - } 2.2845 - } 2.2846 - 2.2847 - // convert old resets to new ones 2.2848 - const u8 OLD_RESET = u8(BUTTON_MASK_OLD_RESET >> 8); 2.2849 - const u8 NEW_RESET = u8(BUTTON_MASK_NEW_RESET >> 8); 2.2850 - for (int i = 0; i < MOVIE_NUM_OF_POSSIBLE_CONTROLLERS; ++i) 2.2851 - { 2.2852 - if (Movie.header.controllerFlags & MOVIE_CONTROLLER(i)) 2.2853 - { 2.2854 - uint8 *startPtr = Movie.inputBuffer + sizeof(u16) * i + 1; 2.2855 - uint8 *endPtr = Movie.inputBuffer + Movie.bytesPerFrame * (Movie.header.length_frames - 1); 2.2856 - for (; startPtr < endPtr; startPtr += Movie.bytesPerFrame) 2.2857 - { 2.2858 - if (startPtr[Movie.bytesPerFrame] & OLD_RESET) 2.2859 - { 2.2860 - startPtr[0] |= NEW_RESET; 2.2861 - } 2.2862 - } 2.2863 - } 2.2864 - } 2.2865 - 2.2866 - flush_movie_header(); 2.2867 - flush_movie_frames(); 2.2868 - return MOVIE_SUCCESS; 2.2869 + flush_movie_header(); 2.2870 + flush_movie_frames(); 2.2871 + return MOVIE_SUCCESS; 2.2872 } 2.2873 2.2874 bool VBAMovieTuncateAtCurrentFrame() 2.2875 { 2.2876 - if (!VBAMovieActive()) 2.2877 - return false; 2.2878 + if (!VBAMovieActive()) 2.2879 + return false; 2.2880 2.2881 - truncate_movie(Movie.currentFrame); 2.2882 - change_state(MOVIE_STATE_END); 2.2883 - systemScreenMessage("Movie truncated"); 2.2884 + truncate_movie(Movie.currentFrame); 2.2885 + change_state(MOVIE_STATE_END); 2.2886 + systemScreenMessage("Movie truncated"); 2.2887 2.2888 - return true; 2.2889 + return true; 2.2890 } 2.2891 2.2892 bool VBAMovieFixHeader() 2.2893 { 2.2894 - if (!VBAMovieActive()) 2.2895 - return false; 2.2896 + if (!VBAMovieActive()) 2.2897 + return false; 2.2898 2.2899 - flush_movie_header(); 2.2900 - systemScreenMessage("Movie header fixed"); 2.2901 - return true; 2.2902 + flush_movie_header(); 2.2903 + systemScreenMessage("Movie header fixed"); 2.2904 + return true; 2.2905 }
3.1 --- a/src/gb/GB.cpp Sun Mar 04 22:44:42 2012 -0600 3.2 +++ b/src/gb/GB.cpp Mon Mar 05 01:25:11 2012 -0600 3.3 @@ -163,361 +163,361 @@ 3.4 static bool pauseAfterFrameAdvance = false; 3.5 3.6 int32 gbRomSizes[] = { 0x00008000, // 32K 3.7 - 0x00010000, // 64K 3.8 - 0x00020000, // 128K 3.9 - 0x00040000, // 256K 3.10 - 0x00080000, // 512K 3.11 - 0x00100000, // 1024K 3.12 - 0x00200000, // 2048K 3.13 - 0x00400000, // 4096K 3.14 - 0x00800000 // 8192K 3.15 + 0x00010000, // 64K 3.16 + 0x00020000, // 128K 3.17 + 0x00040000, // 256K 3.18 + 0x00080000, // 512K 3.19 + 0x00100000, // 1024K 3.20 + 0x00200000, // 2048K 3.21 + 0x00400000, // 4096K 3.22 + 0x00800000 // 8192K 3.23 }; 3.24 int32 gbRomSizesMasks[] = { 0x00007fff, 3.25 - 0x0000ffff, 3.26 - 0x0001ffff, 3.27 - 0x0003ffff, 3.28 - 0x0007ffff, 3.29 - 0x000fffff, 3.30 - 0x001fffff, 3.31 - 0x003fffff, 3.32 - 0x007fffff }; 3.33 + 0x0000ffff, 3.34 + 0x0001ffff, 3.35 + 0x0003ffff, 3.36 + 0x0007ffff, 3.37 + 0x000fffff, 3.38 + 0x001fffff, 3.39 + 0x003fffff, 3.40 + 0x007fffff }; 3.41 3.42 int32 gbRamSizes[6] = { 0x00000000, // 0K 3.43 - 0x00000800, // 2K 3.44 - 0x00002000, // 8K 3.45 - 0x00008000, // 32K 3.46 - 0x00020000, // 128K 3.47 - 0x00010000 // 64K 3.48 + 0x00000800, // 2K 3.49 + 0x00002000, // 8K 3.50 + 0x00008000, // 32K 3.51 + 0x00020000, // 128K 3.52 + 0x00010000 // 64K 3.53 }; 3.54 3.55 int32 gbRamSizesMasks[6] = { 0x00000000, 3.56 - 0x000007ff, 3.57 - 0x00001fff, 3.58 - 0x00007fff, 3.59 - 0x0001ffff, 3.60 - 0x0000ffff }; 3.61 + 0x000007ff, 3.62 + 0x00001fff, 3.63 + 0x00007fff, 3.64 + 0x0001ffff, 3.65 + 0x0000ffff }; 3.66 3.67 int32 gbCycles[] = 3.68 -{ 3.69 -// 0 1 2 3 4 5 6 7 8 9 a b c d e f 3.70 - 1, 3, 2, 2, 1, 1, 2, 1, 5, 2, 2, 2, 1, 1, 2, 1, // 0 3.71 - 1, 3, 2, 2, 1, 1, 2, 1, 3, 2, 2, 2, 1, 1, 2, 1, // 1 3.72 - 2, 3, 2, 2, 1, 1, 2, 1, 2, 2, 2, 2, 1, 1, 2, 1, // 2 3.73 - 2, 3, 2, 2, 3, 3, 3, 1, 2, 2, 2, 2, 1, 1, 2, 1, // 3 3.74 - 1, 1, 1, 1, 1, 1, 2, 1, 1, 1, 1, 1, 1, 1, 2, 1, // 4 3.75 - 1, 1, 1, 1, 1, 1, 2, 1, 1, 1, 1, 1, 1, 1, 2, 1, // 5 3.76 - 1, 1, 1, 1, 1, 1, 2, 1, 1, 1, 1, 1, 1, 1, 2, 1, // 6 3.77 - 2, 2, 2, 2, 2, 2, 1, 2, 1, 1, 1, 1, 1, 1, 2, 1, // 7 3.78 - 1, 1, 1, 1, 1, 1, 2, 1, 1, 1, 1, 1, 1, 1, 2, 1, // 8 3.79 - 1, 1, 1, 1, 1, 1, 2, 1, 1, 1, 1, 1, 1, 1, 2, 1, // 9 3.80 - 1, 1, 1, 1, 1, 1, 2, 1, 1, 1, 1, 1, 1, 1, 2, 1, // a 3.81 - 1, 1, 1, 1, 1, 1, 2, 1, 1, 1, 1, 1, 1, 1, 2, 1, // b 3.82 - 2, 3, 3, 4, 3, 4, 2, 4, 2, 4, 3, 2, 3, 6, 2, 4, // c 3.83 - 2, 3, 3, 0, 3, 4, 2, 4, 2, 4, 3, 0, 3, 0, 2, 4, // d 3.84 - 3, 3, 2, 0, 0, 4, 2, 4, 4, 1, 4, 0, 0, 0, 2, 4, // e 3.85 - 3, 3, 2, 1, 0, 4, 2, 4, 3, 2, 4, 1, 0, 0, 2, 4 // f 3.86 -}; 3.87 + { 3.88 + // 0 1 2 3 4 5 6 7 8 9 a b c d e f 3.89 + 1, 3, 2, 2, 1, 1, 2, 1, 5, 2, 2, 2, 1, 1, 2, 1, // 0 3.90 + 1, 3, 2, 2, 1, 1, 2, 1, 3, 2, 2, 2, 1, 1, 2, 1, // 1 3.91 + 2, 3, 2, 2, 1, 1, 2, 1, 2, 2, 2, 2, 1, 1, 2, 1, // 2 3.92 + 2, 3, 2, 2, 3, 3, 3, 1, 2, 2, 2, 2, 1, 1, 2, 1, // 3 3.93 + 1, 1, 1, 1, 1, 1, 2, 1, 1, 1, 1, 1, 1, 1, 2, 1, // 4 3.94 + 1, 1, 1, 1, 1, 1, 2, 1, 1, 1, 1, 1, 1, 1, 2, 1, // 5 3.95 + 1, 1, 1, 1, 1, 1, 2, 1, 1, 1, 1, 1, 1, 1, 2, 1, // 6 3.96 + 2, 2, 2, 2, 2, 2, 1, 2, 1, 1, 1, 1, 1, 1, 2, 1, // 7 3.97 + 1, 1, 1, 1, 1, 1, 2, 1, 1, 1, 1, 1, 1, 1, 2, 1, // 8 3.98 + 1, 1, 1, 1, 1, 1, 2, 1, 1, 1, 1, 1, 1, 1, 2, 1, // 9 3.99 + 1, 1, 1, 1, 1, 1, 2, 1, 1, 1, 1, 1, 1, 1, 2, 1, // a 3.100 + 1, 1, 1, 1, 1, 1, 2, 1, 1, 1, 1, 1, 1, 1, 2, 1, // b 3.101 + 2, 3, 3, 4, 3, 4, 2, 4, 2, 4, 3, 2, 3, 6, 2, 4, // c 3.102 + 2, 3, 3, 0, 3, 4, 2, 4, 2, 4, 3, 0, 3, 0, 2, 4, // d 3.103 + 3, 3, 2, 0, 0, 4, 2, 4, 4, 1, 4, 0, 0, 0, 2, 4, // e 3.104 + 3, 3, 2, 1, 0, 4, 2, 4, 3, 2, 4, 1, 0, 0, 2, 4 // f 3.105 + }; 3.106 3.107 int32 gbCyclesCB[] = 3.108 -{ 3.109 -// 0 1 2 3 4 5 6 7 8 9 a b c d e f 3.110 - 2, 2, 2, 2, 2, 2, 4, 2, 2, 2, 2, 2, 2, 2, 4, 2, // 0 3.111 - 2, 2, 2, 2, 2, 2, 4, 2, 2, 2, 2, 2, 2, 2, 4, 2, // 1 3.112 - 2, 2, 2, 2, 2, 2, 4, 2, 2, 2, 2, 2, 2, 2, 4, 2, // 2 3.113 - 2, 2, 2, 2, 2, 2, 4, 2, 2, 2, 2, 2, 2, 2, 4, 2, // 3 3.114 - 2, 2, 2, 2, 2, 2, 3, 2, 2, 2, 2, 2, 2, 2, 3, 2, // 4 3.115 - 2, 2, 2, 2, 2, 2, 3, 2, 2, 2, 2, 2, 2, 2, 3, 2, // 5 3.116 - 2, 2, 2, 2, 2, 2, 3, 2, 2, 2, 2, 2, 2, 2, 3, 2, // 6 3.117 - 2, 2, 2, 2, 2, 2, 3, 2, 2, 2, 2, 2, 2, 2, 3, 2, // 7 3.118 - 2, 2, 2, 2, 2, 2, 3, 2, 2, 2, 2, 2, 2, 2, 3, 2, // 8 3.119 - 2, 2, 2, 2, 2, 2, 3, 2, 2, 2, 2, 2, 2, 2, 3, 2, // 9 3.120 - 2, 2, 2, 2, 2, 2, 3, 2, 2, 2, 2, 2, 2, 2, 3, 2, // a 3.121 - 2, 2, 2, 2, 2, 2, 3, 2, 2, 2, 2, 2, 2, 2, 3, 2, // b 3.122 - 2, 2, 2, 2, 2, 2, 3, 2, 2, 2, 2, 2, 2, 2, 3, 2, // c 3.123 - 2, 2, 2, 2, 2, 2, 3, 2, 2, 2, 2, 2, 2, 2, 3, 2, // d 3.124 - 2, 2, 2, 2, 2, 2, 3, 2, 2, 2, 2, 2, 2, 2, 3, 2, // e 3.125 - 2, 2, 2, 2, 2, 2, 3, 2, 2, 2, 2, 2, 2, 2, 3, 2 // f 3.126 -}; 3.127 + { 3.128 + // 0 1 2 3 4 5 6 7 8 9 a b c d e f 3.129 + 2, 2, 2, 2, 2, 2, 4, 2, 2, 2, 2, 2, 2, 2, 4, 2, // 0 3.130 + 2, 2, 2, 2, 2, 2, 4, 2, 2, 2, 2, 2, 2, 2, 4, 2, // 1 3.131 + 2, 2, 2, 2, 2, 2, 4, 2, 2, 2, 2, 2, 2, 2, 4, 2, // 2 3.132 + 2, 2, 2, 2, 2, 2, 4, 2, 2, 2, 2, 2, 2, 2, 4, 2, // 3 3.133 + 2, 2, 2, 2, 2, 2, 3, 2, 2, 2, 2, 2, 2, 2, 3, 2, // 4 3.134 + 2, 2, 2, 2, 2, 2, 3, 2, 2, 2, 2, 2, 2, 2, 3, 2, // 5 3.135 + 2, 2, 2, 2, 2, 2, 3, 2, 2, 2, 2, 2, 2, 2, 3, 2, // 6 3.136 + 2, 2, 2, 2, 2, 2, 3, 2, 2, 2, 2, 2, 2, 2, 3, 2, // 7 3.137 + 2, 2, 2, 2, 2, 2, 3, 2, 2, 2, 2, 2, 2, 2, 3, 2, // 8 3.138 + 2, 2, 2, 2, 2, 2, 3, 2, 2, 2, 2, 2, 2, 2, 3, 2, // 9 3.139 + 2, 2, 2, 2, 2, 2, 3, 2, 2, 2, 2, 2, 2, 2, 3, 2, // a 3.140 + 2, 2, 2, 2, 2, 2, 3, 2, 2, 2, 2, 2, 2, 2, 3, 2, // b 3.141 + 2, 2, 2, 2, 2, 2, 3, 2, 2, 2, 2, 2, 2, 2, 3, 2, // c 3.142 + 2, 2, 2, 2, 2, 2, 3, 2, 2, 2, 2, 2, 2, 2, 3, 2, // d 3.143 + 2, 2, 2, 2, 2, 2, 3, 2, 2, 2, 2, 2, 2, 2, 3, 2, // e 3.144 + 2, 2, 2, 2, 2, 2, 3, 2, 2, 2, 2, 2, 2, 2, 3, 2 // f 3.145 + }; 3.146 3.147 u16 DAATable[] = 3.148 -{ 3.149 - 0x0080, 0x0100, 0x0200, 0x0300, 0x0400, 0x0500, 0x0600, 0x0700, 3.150 - 0x0800, 0x0900, 0x1020, 0x1120, 0x1220, 0x1320, 0x1420, 0x1520, 3.151 - 0x1000, 0x1100, 0x1200, 0x1300, 0x1400, 0x1500, 0x1600, 0x1700, 3.152 - 0x1800, 0x1900, 0x2020, 0x2120, 0x2220, 0x2320, 0x2420, 0x2520, 3.153 - 0x2000, 0x2100, 0x2200, 0x2300, 0x2400, 0x2500, 0x2600, 0x2700, 3.154 - 0x2800, 0x2900, 0x3020, 0x3120, 0x3220, 0x3320, 0x3420, 0x3520, 3.155 - 0x3000, 0x3100, 0x3200, 0x3300, 0x3400, 0x3500, 0x3600, 0x3700, 3.156 - 0x3800, 0x3900, 0x4020, 0x4120, 0x4220, 0x4320, 0x4420, 0x4520, 3.157 - 0x4000, 0x4100, 0x4200, 0x4300, 0x4400, 0x4500, 0x4600, 0x4700, 3.158 - 0x4800, 0x4900, 0x5020, 0x5120, 0x5220, 0x5320, 0x5420, 0x5520, 3.159 - 0x5000, 0x5100, 0x5200, 0x5300, 0x5400, 0x5500, 0x5600, 0x5700, 3.160 - 0x5800, 0x5900, 0x6020, 0x6120, 0x6220, 0x6320, 0x6420, 0x6520, 3.161 - 0x6000, 0x6100, 0x6200, 0x6300, 0x6400, 0x6500, 0x6600, 0x6700, 3.162 - 0x6800, 0x6900, 0x7020, 0x7120, 0x7220, 0x7320, 0x7420, 0x7520, 3.163 - 0x7000, 0x7100, 0x7200, 0x7300, 0x7400, 0x7500, 0x7600, 0x7700, 3.164 - 0x7800, 0x7900, 0x8020, 0x8120, 0x8220, 0x8320, 0x8420, 0x8520, 3.165 - 0x8000, 0x8100, 0x8200, 0x8300, 0x8400, 0x8500, 0x8600, 0x8700, 3.166 - 0x8800, 0x8900, 0x9020, 0x9120, 0x9220, 0x9320, 0x9420, 0x9520, 3.167 - 0x9000, 0x9100, 0x9200, 0x9300, 0x9400, 0x9500, 0x9600, 0x9700, 3.168 - 0x9800, 0x9900, 0x00B0, 0x0130, 0x0230, 0x0330, 0x0430, 0x0530, 3.169 - 0x0090, 0x0110, 0x0210, 0x0310, 0x0410, 0x0510, 0x0610, 0x0710, 3.170 - 0x0810, 0x0910, 0x1030, 0x1130, 0x1230, 0x1330, 0x1430, 0x1530, 3.171 - 0x1010, 0x1110, 0x1210, 0x1310, 0x1410, 0x1510, 0x1610, 0x1710, 3.172 - 0x1810, 0x1910, 0x2030, 0x2130, 0x2230, 0x2330, 0x2430, 0x2530, 3.173 - 0x2010, 0x2110, 0x2210, 0x2310, 0x2410, 0x2510, 0x2610, 0x2710, 3.174 - 0x2810, 0x2910, 0x3030, 0x3130, 0x3230, 0x3330, 0x3430, 0x3530, 3.175 - 0x3010, 0x3110, 0x3210, 0x3310, 0x3410, 0x3510, 0x3610, 0x3710, 3.176 - 0x3810, 0x3910, 0x4030, 0x4130, 0x4230, 0x4330, 0x4430, 0x4530, 3.177 - 0x4010, 0x4110, 0x4210, 0x4310, 0x4410, 0x4510, 0x4610, 0x4710, 3.178 - 0x4810, 0x4910, 0x5030, 0x5130, 0x5230, 0x5330, 0x5430, 0x5530, 3.179 - 0x5010, 0x5110, 0x5210, 0x5310, 0x5410, 0x5510, 0x5610, 0x5710, 3.180 - 0x5810, 0x5910, 0x6030, 0x6130, 0x6230, 0x6330, 0x6430, 0x6530, 3.181 - 0x6010, 0x6110, 0x6210, 0x6310, 0x6410, 0x6510, 0x6610, 0x6710, 3.182 - 0x6810, 0x6910, 0x7030, 0x7130, 0x7230, 0x7330, 0x7430, 0x7530, 3.183 - 0x7010, 0x7110, 0x7210, 0x7310, 0x7410, 0x7510, 0x7610, 0x7710, 3.184 - 0x7810, 0x7910, 0x8030, 0x8130, 0x8230, 0x8330, 0x8430, 0x8530, 3.185 - 0x8010, 0x8110, 0x8210, 0x8310, 0x8410, 0x8510, 0x8610, 0x8710, 3.186 - 0x8810, 0x8910, 0x9030, 0x9130, 0x9230, 0x9330, 0x9430, 0x9530, 3.187 - 0x9010, 0x9110, 0x9210, 0x9310, 0x9410, 0x9510, 0x9610, 0x9710, 3.188 - 0x9810, 0x9910, 0xA030, 0xA130, 0xA230, 0xA330, 0xA430, 0xA530, 3.189 - 0xA010, 0xA110, 0xA210, 0xA310, 0xA410, 0xA510, 0xA610, 0xA710, 3.190 - 0xA810, 0xA910, 0xB030, 0xB130, 0xB230, 0xB330, 0xB430, 0xB530, 3.191 - 0xB010, 0xB110, 0xB210, 0xB310, 0xB410, 0xB510, 0xB610, 0xB710, 3.192 - 0xB810, 0xB910, 0xC030, 0xC130, 0xC230, 0xC330, 0xC430, 0xC530, 3.193 - 0xC010, 0xC110, 0xC210, 0xC310, 0xC410, 0xC510, 0xC610, 0xC710, 3.194 - 0xC810, 0xC910, 0xD030, 0xD130, 0xD230, 0xD330, 0xD430, 0xD530, 3.195 - 0xD010, 0xD110, 0xD210, 0xD310, 0xD410, 0xD510, 0xD610, 0xD710, 3.196 - 0xD810, 0xD910, 0xE030, 0xE130, 0xE230, 0xE330, 0xE430, 0xE530, 3.197 - 0xE010, 0xE110, 0xE210, 0xE310, 0xE410, 0xE510, 0xE610, 0xE710, 3.198 - 0xE810, 0xE910, 0xF030, 0xF130, 0xF230, 0xF330, 0xF430, 0xF530, 3.199 - 0xF010, 0xF110, 0xF210, 0xF310, 0xF410, 0xF510, 0xF610, 0xF710, 3.200 - 0xF810, 0xF910, 0x00B0, 0x0130, 0x0230, 0x0330, 0x0430, 0x0530, 3.201 - 0x0090, 0x0110, 0x0210, 0x0310, 0x0410, 0x0510, 0x0610, 0x0710, 3.202 - 0x0810, 0x0910, 0x1030, 0x1130, 0x1230, 0x1330, 0x1430, 0x1530, 3.203 - 0x1010, 0x1110, 0x1210, 0x1310, 0x1410, 0x1510, 0x1610, 0x1710, 3.204 - 0x1810, 0x1910, 0x2030, 0x2130, 0x2230, 0x2330, 0x2430, 0x2530, 3.205 - 0x2010, 0x2110, 0x2210, 0x2310, 0x2410, 0x2510, 0x2610, 0x2710, 3.206 - 0x2810, 0x2910, 0x3030, 0x3130, 0x3230, 0x3330, 0x3430, 0x3530, 3.207 - 0x3010, 0x3110, 0x3210, 0x3310, 0x3410, 0x3510, 0x3610, 0x3710, 3.208 - 0x3810, 0x3910, 0x4030, 0x4130, 0x4230, 0x4330, 0x4430, 0x4530, 3.209 - 0x4010, 0x4110, 0x4210, 0x4310, 0x4410, 0x4510, 0x4610, 0x4710, 3.210 - 0x4810, 0x4910, 0x5030, 0x5130, 0x5230, 0x5330, 0x5430, 0x5530, 3.211 - 0x5010, 0x5110, 0x5210, 0x5310, 0x5410, 0x5510, 0x5610, 0x5710, 3.212 - 0x5810, 0x5910, 0x6030, 0x6130, 0x6230, 0x6330, 0x6430, 0x6530, 3.213 - 0x0600, 0x0700, 0x0800, 0x0900, 0x0A00, 0x0B00, 0x0C00, 0x0D00, 3.214 - 0x0E00, 0x0F00, 0x1020, 0x1120, 0x1220, 0x1320, 0x1420, 0x1520, 3.215 - 0x1600, 0x1700, 0x1800, 0x1900, 0x1A00, 0x1B00, 0x1C00, 0x1D00, 3.216 - 0x1E00, 0x1F00, 0x2020, 0x2120, 0x2220, 0x2320, 0x2420, 0x2520, 3.217 - 0x2600, 0x2700, 0x2800, 0x2900, 0x2A00, 0x2B00, 0x2C00, 0x2D00, 3.218 - 0x2E00, 0x2F00, 0x3020, 0x3120, 0x3220, 0x3320, 0x3420, 0x3520, 3.219 - 0x3600, 0x3700, 0x3800, 0x3900, 0x3A00, 0x3B00, 0x3C00, 0x3D00, 3.220 - 0x3E00, 0x3F00, 0x4020, 0x4120, 0x4220, 0x4320, 0x4420, 0x4520, 3.221 - 0x4600, 0x4700, 0x4800, 0x4900, 0x4A00, 0x4B00, 0x4C00, 0x4D00, 3.222 - 0x4E00, 0x4F00, 0x5020, 0x5120, 0x5220, 0x5320, 0x5420, 0x5520, 3.223 - 0x5600, 0x5700, 0x5800, 0x5900, 0x5A00, 0x5B00, 0x5C00, 0x5D00, 3.224 - 0x5E00, 0x5F00, 0x6020, 0x6120, 0x6220, 0x6320, 0x6420, 0x6520, 3.225 - 0x6600, 0x6700, 0x6800, 0x6900, 0x6A00, 0x6B00, 0x6C00, 0x6D00, 3.226 - 0x6E00, 0x6F00, 0x7020, 0x7120, 0x7220, 0x7320, 0x7420, 0x7520, 3.227 - 0x7600, 0x7700, 0x7800, 0x7900, 0x7A00, 0x7B00, 0x7C00, 0x7D00, 3.228 - 0x7E00, 0x7F00, 0x8020, 0x8120, 0x8220, 0x8320, 0x8420, 0x8520, 3.229 - 0x8600, 0x8700, 0x8800, 0x8900, 0x8A00, 0x8B00, 0x8C00, 0x8D00, 3.230 - 0x8E00, 0x8F00, 0x9020, 0x9120, 0x9220, 0x9320, 0x9420, 0x9520, 3.231 - 0x9600, 0x9700, 0x9800, 0x9900, 0x9A00, 0x9B00, 0x9C00, 0x9D00, 3.232 - 0x9E00, 0x9F00, 0x00B0, 0x0130, 0x0230, 0x0330, 0x0430, 0x0530, 3.233 - 0x0610, 0x0710, 0x0810, 0x0910, 0x0A10, 0x0B10, 0x0C10, 0x0D10, 3.234 - 0x0E10, 0x0F10, 0x1030, 0x1130, 0x1230, 0x1330, 0x1430, 0x1530, 3.235 - 0x1610, 0x1710, 0x1810, 0x1910, 0x1A10, 0x1B10, 0x1C10, 0x1D10, 3.236 - 0x1E10, 0x1F10, 0x2030, 0x2130, 0x2230, 0x2330, 0x2430, 0x2530, 3.237 - 0x2610, 0x2710, 0x2810, 0x2910, 0x2A10, 0x2B10, 0x2C10, 0x2D10, 3.238 - 0x2E10, 0x2F10, 0x3030, 0x3130, 0x3230, 0x3330, 0x3430, 0x3530, 3.239 - 0x3610, 0x3710, 0x3810, 0x3910, 0x3A10, 0x3B10, 0x3C10, 0x3D10, 3.240 - 0x3E10, 0x3F10, 0x4030, 0x4130, 0x4230, 0x4330, 0x4430, 0x4530, 3.241 - 0x4610, 0x4710, 0x4810, 0x4910, 0x4A10, 0x4B10, 0x4C10, 0x4D10, 3.242 - 0x4E10, 0x4F10, 0x5030, 0x5130, 0x5230, 0x5330, 0x5430, 0x5530, 3.243 - 0x5610, 0x5710, 0x5810, 0x5910, 0x5A10, 0x5B10, 0x5C10, 0x5D10, 3.244 - 0x5E10, 0x5F10, 0x6030, 0x6130, 0x6230, 0x6330, 0x6430, 0x6530, 3.245 - 0x6610, 0x6710, 0x6810, 0x6910, 0x6A10, 0x6B10, 0x6C10, 0x6D10, 3.246 - 0x6E10, 0x6F10, 0x7030, 0x7130, 0x7230, 0x7330, 0x7430, 0x7530, 3.247 - 0x7610, 0x7710, 0x7810, 0x7910, 0x7A10, 0x7B10, 0x7C10, 0x7D10, 3.248 - 0x7E10, 0x7F10, 0x8030, 0x8130, 0x8230, 0x8330, 0x8430, 0x8530, 3.249 - 0x8610, 0x8710, 0x8810, 0x8910, 0x8A10, 0x8B10, 0x8C10, 0x8D10, 3.250 - 0x8E10, 0x8F10, 0x9030, 0x9130, 0x9230, 0x9330, 0x9430, 0x9530, 3.251 - 0x9610, 0x9710, 0x9810, 0x9910, 0x9A10, 0x9B10, 0x9C10, 0x9D10, 3.252 - 0x9E10, 0x9F10, 0xA030, 0xA130, 0xA230, 0xA330, 0xA430, 0xA530, 3.253 - 0xA610, 0xA710, 0xA810, 0xA910, 0xAA10, 0xAB10, 0xAC10, 0xAD10, 3.254 - 0xAE10, 0xAF10, 0xB030, 0xB130, 0xB230, 0xB330, 0xB430, 0xB530, 3.255 - 0xB610, 0xB710, 0xB810, 0xB910, 0xBA10, 0xBB10, 0xBC10, 0xBD10, 3.256 - 0xBE10, 0xBF10, 0xC030, 0xC130, 0xC230, 0xC330, 0xC430, 0xC530, 3.257 - 0xC610, 0xC710, 0xC810, 0xC910, 0xCA10, 0xCB10, 0xCC10, 0xCD10, 3.258 - 0xCE10, 0xCF10, 0xD030, 0xD130, 0xD230, 0xD330, 0xD430, 0xD530, 3.259 - 0xD610, 0xD710, 0xD810, 0xD910, 0xDA10, 0xDB10, 0xDC10, 0xDD10, 3.260 - 0xDE10, 0xDF10, 0xE030, 0xE130, 0xE230, 0xE330, 0xE430, 0xE530, 3.261 - 0xE610, 0xE710, 0xE810, 0xE910, 0xEA10, 0xEB10, 0xEC10, 0xED10, 3.262 - 0xEE10, 0xEF10, 0xF030, 0xF130, 0xF230, 0xF330, 0xF430, 0xF530, 3.263 - 0xF610, 0xF710, 0xF810, 0xF910, 0xFA10, 0xFB10, 0xFC10, 0xFD10, 3.264 - 0xFE10, 0xFF10, 0x00B0, 0x0130, 0x0230, 0x0330, 0x0430, 0x0530, 3.265 - 0x0610, 0x0710, 0x0810, 0x0910, 0x0A10, 0x0B10, 0x0C10, 0x0D10, 3.266 - 0x0E10, 0x0F10, 0x1030, 0x1130, 0x1230, 0x1330, 0x1430, 0x1530, 3.267 - 0x1610, 0x1710, 0x1810, 0x1910, 0x1A10, 0x1B10, 0x1C10, 0x1D10, 3.268 - 0x1E10, 0x1F10, 0x2030, 0x2130, 0x2230, 0x2330, 0x2430, 0x2530, 3.269 - 0x2610, 0x2710, 0x2810, 0x2910, 0x2A10, 0x2B10, 0x2C10, 0x2D10, 3.270 - 0x2E10, 0x2F10, 0x3030, 0x3130, 0x3230, 0x3330, 0x3430, 0x3530, 3.271 - 0x3610, 0x3710, 0x3810, 0x3910, 0x3A10, 0x3B10, 0x3C10, 0x3D10, 3.272 - 0x3E10, 0x3F10, 0x4030, 0x4130, 0x4230, 0x4330, 0x4430, 0x4530, 3.273 - 0x4610, 0x4710, 0x4810, 0x4910, 0x4A10, 0x4B10, 0x4C10, 0x4D10, 3.274 - 0x4E10, 0x4F10, 0x5030, 0x5130, 0x5230, 0x5330, 0x5430, 0x5530, 3.275 - 0x5610, 0x5710, 0x5810, 0x5910, 0x5A10, 0x5B10, 0x5C10, 0x5D10, 3.276 - 0x5E10, 0x5F10, 0x6030, 0x6130, 0x6230, 0x6330, 0x6430, 0x6530, 3.277 - 0x00C0, 0x0140, 0x0240, 0x0340, 0x0440, 0x0540, 0x0640, 0x0740, 3.278 - 0x0840, 0x0940, 0x0440, 0x0540, 0x0640, 0x0740, 0x0840, 0x0940, 3.279 - 0x1040, 0x1140, 0x1240, 0x1340, 0x1440, 0x1540, 0x1640, 0x1740, 3.280 - 0x1840, 0x1940, 0x1440, 0x1540, 0x1640, 0x1740, 0x1840, 0x1940, 3.281 - 0x2040, 0x2140, 0x2240, 0x2340, 0x2440, 0x2540, 0x2640, 0x2740, 3.282 - 0x2840, 0x2940, 0x2440, 0x2540, 0x2640, 0x2740, 0x2840, 0x2940, 3.283 - 0x3040, 0x3140, 0x3240, 0x3340, 0x3440, 0x3540, 0x3640, 0x3740, 3.284 - 0x3840, 0x3940, 0x3440, 0x3540, 0x3640, 0x3740, 0x3840, 0x3940, 3.285 - 0x4040, 0x4140, 0x4240, 0x4340, 0x4440, 0x4540, 0x4640, 0x4740, 3.286 - 0x4840, 0x4940, 0x4440, 0x4540, 0x4640, 0x4740, 0x4840, 0x4940, 3.287 - 0x5040, 0x5140, 0x5240, 0x5340, 0x5440, 0x5540, 0x5640, 0x5740, 3.288 - 0x5840, 0x5940, 0x5440, 0x5540, 0x5640, 0x5740, 0x5840, 0x5940, 3.289 - 0x6040, 0x6140, 0x6240, 0x6340, 0x6440, 0x6540, 0x6640, 0x6740, 3.290 - 0x6840, 0x6940, 0x6440, 0x6540, 0x6640, 0x6740, 0x6840, 0x6940, 3.291 - 0x7040, 0x7140, 0x7240, 0x7340, 0x7440, 0x7540, 0x7640, 0x7740, 3.292 - 0x7840, 0x7940, 0x7440, 0x7540, 0x7640, 0x7740, 0x7840, 0x7940, 3.293 - 0x8040, 0x8140, 0x8240, 0x8340, 0x8440, 0x8540, 0x8640, 0x8740, 3.294 - 0x8840, 0x8940, 0x8440, 0x8540, 0x8640, 0x8740, 0x8840, 0x8940, 3.295 - 0x9040, 0x9140, 0x9240, 0x9340, 0x9440, 0x9540, 0x9640, 0x9740, 3.296 - 0x9840, 0x9940, 0x3450, 0x3550, 0x3650, 0x3750, 0x3850, 0x3950, 3.297 - 0x4050, 0x4150, 0x4250, 0x4350, 0x4450, 0x4550, 0x4650, 0x4750, 3.298 - 0x4850, 0x4950, 0x4450, 0x4550, 0x4650, 0x4750, 0x4850, 0x4950, 3.299 - 0x5050, 0x5150, 0x5250, 0x5350, 0x5450, 0x5550, 0x5650, 0x5750, 3.300 - 0x5850, 0x5950, 0x5450, 0x5550, 0x5650, 0x5750, 0x5850, 0x5950, 3.301 - 0x6050, 0x6150, 0x6250, 0x6350, 0x6450, 0x6550, 0x6650, 0x6750, 3.302 - 0x6850, 0x6950, 0x6450, 0x6550, 0x6650, 0x6750, 0x6850, 0x6950, 3.303 - 0x7050, 0x7150, 0x7250, 0x7350, 0x7450, 0x7550, 0x7650, 0x7750, 3.304 - 0x7850, 0x7950, 0x7450, 0x7550, 0x7650, 0x7750, 0x7850, 0x7950, 3.305 - 0x8050, 0x8150, 0x8250, 0x8350, 0x8450, 0x8550, 0x8650, 0x8750, 3.306 - 0x8850, 0x8950, 0x8450, 0x8550, 0x8650, 0x8750, 0x8850, 0x8950, 3.307 - 0x9050, 0x9150, 0x9250, 0x9350, 0x9450, 0x9550, 0x9650, 0x9750, 3.308 - 0x9850, 0x9950, 0x9450, 0x9550, 0x9650, 0x9750, 0x9850, 0x9950, 3.309 - 0xA050, 0xA150, 0xA250, 0xA350, 0xA450, 0xA550, 0xA650, 0xA750, 3.310 - 0xA850, 0xA950, 0xA450, 0xA550, 0xA650, 0xA750, 0xA850, 0xA950, 3.311 - 0xB050, 0xB150, 0xB250, 0xB350, 0xB450, 0xB550, 0xB650, 0xB750, 3.312 - 0xB850, 0xB950, 0xB450, 0xB550, 0xB650, 0xB750, 0xB850, 0xB950, 3.313 - 0xC050, 0xC150, 0xC250, 0xC350, 0xC450, 0xC550, 0xC650, 0xC750, 3.314 - 0xC850, 0xC950, 0xC450, 0xC550, 0xC650, 0xC750, 0xC850, 0xC950, 3.315 - 0xD050, 0xD150, 0xD250, 0xD350, 0xD450, 0xD550, 0xD650, 0xD750, 3.316 - 0xD850, 0xD950, 0xD450, 0xD550, 0xD650, 0xD750, 0xD850, 0xD950, 3.317 - 0xE050, 0xE150, 0xE250, 0xE350, 0xE450, 0xE550, 0xE650, 0xE750, 3.318 - 0xE850, 0xE950, 0xE450, 0xE550, 0xE650, 0xE750, 0xE850, 0xE950, 3.319 - 0xF050, 0xF150, 0xF250, 0xF350, 0xF450, 0xF550, 0xF650, 0xF750, 3.320 - 0xF850, 0xF950, 0xF450, 0xF550, 0xF650, 0xF750, 0xF850, 0xF950, 3.321 - 0x00D0, 0x0150, 0x0250, 0x0350, 0x0450, 0x0550, 0x0650, 0x0750, 3.322 - 0x0850, 0x0950, 0x0450, 0x0550, 0x0650, 0x0750, 0x0850, 0x0950, 3.323 - 0x1050, 0x1150, 0x1250, 0x1350, 0x1450, 0x1550, 0x1650, 0x1750, 3.324 - 0x1850, 0x1950, 0x1450, 0x1550, 0x1650, 0x1750, 0x1850, 0x1950, 3.325 - 0x2050, 0x2150, 0x2250, 0x2350, 0x2450, 0x2550, 0x2650, 0x2750, 3.326 - 0x2850, 0x2950, 0x2450, 0x2550, 0x2650, 0x2750, 0x2850, 0x2950, 3.327 - 0x3050, 0x3150, 0x3250, 0x3350, 0x3450, 0x3550, 0x3650, 0x3750, 3.328 - 0x3850, 0x3950, 0x3450, 0x3550, 0x3650, 0x3750, 0x3850, 0x3950, 3.329 - 0x4050, 0x4150, 0x4250, 0x4350, 0x4450, 0x4550, 0x4650, 0x4750, 3.330 - 0x4850, 0x4950, 0x4450, 0x4550, 0x4650, 0x4750, 0x4850, 0x4950, 3.331 - 0x5050, 0x5150, 0x5250, 0x5350, 0x5450, 0x5550, 0x5650, 0x5750, 3.332 - 0x5850, 0x5950, 0x5450, 0x5550, 0x5650, 0x5750, 0x5850, 0x5950, 3.333 - 0x6050, 0x6150, 0x6250, 0x6350, 0x6450, 0x6550, 0x6650, 0x6750, 3.334 - 0x6850, 0x6950, 0x6450, 0x6550, 0x6650, 0x6750, 0x6850, 0x6950, 3.335 - 0x7050, 0x7150, 0x7250, 0x7350, 0x7450, 0x7550, 0x7650, 0x7750, 3.336 - 0x7850, 0x7950, 0x7450, 0x7550, 0x7650, 0x7750, 0x7850, 0x7950, 3.337 - 0x8050, 0x8150, 0x8250, 0x8350, 0x8450, 0x8550, 0x8650, 0x8750, 3.338 - 0x8850, 0x8950, 0x8450, 0x8550, 0x8650, 0x8750, 0x8850, 0x8950, 3.339 - 0x9050, 0x9150, 0x9250, 0x9350, 0x9450, 0x9550, 0x9650, 0x9750, 3.340 - 0x9850, 0x9950, 0x9450, 0x9550, 0x9650, 0x9750, 0x9850, 0x9950, 3.341 - 0xFA60, 0xFB60, 0xFC60, 0xFD60, 0xFE60, 0xFF60, 0x00C0, 0x0140, 3.342 - 0x0240, 0x0340, 0x0440, 0x0540, 0x0640, 0x0740, 0x0840, 0x0940, 3.343 - 0x0A60, 0x0B60, 0x0C60, 0x0D60, 0x0E60, 0x0F60, 0x1040, 0x1140, 3.344 - 0x1240, 0x1340, 0x1440, 0x1540, 0x1640, 0x1740, 0x1840, 0x1940, 3.345 - 0x1A60, 0x1B60, 0x1C60, 0x1D60, 0x1E60, 0x1F60, 0x2040, 0x2140, 3.346 - 0x2240, 0x2340, 0x2440, 0x2540, 0x2640, 0x2740, 0x2840, 0x2940, 3.347 - 0x2A60, 0x2B60, 0x2C60, 0x2D60, 0x2E60, 0x2F60, 0x3040, 0x3140, 3.348 - 0x3240, 0x3340, 0x3440, 0x3540, 0x3640, 0x3740, 0x3840, 0x3940, 3.349 - 0x3A60, 0x3B60, 0x3C60, 0x3D60, 0x3E60, 0x3F60, 0x4040, 0x4140, 3.350 - 0x4240, 0x4340, 0x4440, 0x4540, 0x4640, 0x4740, 0x4840, 0x4940, 3.351 - 0x4A60, 0x4B60, 0x4C60, 0x4D60, 0x4E60, 0x4F60, 0x5040, 0x5140, 3.352 - 0x5240, 0x5340, 0x5440, 0x5540, 0x5640, 0x5740, 0x5840, 0x5940, 3.353 - 0x5A60, 0x5B60, 0x5C60, 0x5D60, 0x5E60, 0x5F60, 0x6040, 0x6140, 3.354 - 0x6240, 0x6340, 0x6440, 0x6540, 0x6640, 0x6740, 0x6840, 0x6940, 3.355 - 0x6A60, 0x6B60, 0x6C60, 0x6D60, 0x6E60, 0x6F60, 0x7040, 0x7140, 3.356 - 0x7240, 0x7340, 0x7440, 0x7540, 0x7640, 0x7740, 0x7840, 0x7940, 3.357 - 0x7A60, 0x7B60, 0x7C60, 0x7D60, 0x7E60, 0x7F60, 0x8040, 0x8140, 3.358 - 0x8240, 0x8340, 0x8440, 0x8540, 0x8640, 0x8740, 0x8840, 0x8940, 3.359 - 0x8A60, 0x8B60, 0x8C60, 0x8D60, 0x8E60, 0x8F60, 0x9040, 0x9140, 3.360 - 0x9240, 0x9340, 0x3450, 0x3550, 0x3650, 0x3750, 0x3850, 0x3950, 3.361 - 0x3A70, 0x3B70, 0x3C70, 0x3D70, 0x3E70, 0x3F70, 0x4050, 0x4150, 3.362 - 0x4250, 0x4350, 0x4450, 0x4550, 0x4650, 0x4750, 0x4850, 0x4950, 3.363 - 0x4A70, 0x4B70, 0x4C70, 0x4D70, 0x4E70, 0x4F70, 0x5050, 0x5150, 3.364 - 0x5250, 0x5350, 0x5450, 0x5550, 0x5650, 0x5750, 0x5850, 0x5950, 3.365 - 0x5A70, 0x5B70, 0x5C70, 0x5D70, 0x5E70, 0x5F70, 0x6050, 0x6150, 3.366 - 0x6250, 0x6350, 0x6450, 0x6550, 0x6650, 0x6750, 0x6850, 0x6950, 3.367 - 0x6A70, 0x6B70, 0x6C70, 0x6D70, 0x6E70, 0x6F70, 0x7050, 0x7150, 3.368 - 0x7250, 0x7350, 0x7450, 0x7550, 0x7650, 0x7750, 0x7850, 0x7950, 3.369 - 0x7A70, 0x7B70, 0x7C70, 0x7D70, 0x7E70, 0x7F70, 0x8050, 0x8150, 3.370 - 0x8250, 0x8350, 0x8450, 0x8550, 0x8650, 0x8750, 0x8850, 0x8950, 3.371 - 0x8A70, 0x8B70, 0x8C70, 0x8D70, 0x8E70, 0x8F70, 0x9050, 0x9150, 3.372 - 0x9250, 0x9350, 0x9450, 0x9550, 0x9650, 0x9750, 0x9850, 0x9950, 3.373 - 0x9A70, 0x9B70, 0x9C70, 0x9D70, 0x9E70, 0x9F70, 0xA050, 0xA150, 3.374 - 0xA250, 0xA350, 0xA450, 0xA550, 0xA650, 0xA750, 0xA850, 0xA950, 3.375 - 0xAA70, 0xAB70, 0xAC70, 0xAD70, 0xAE70, 0xAF70, 0xB050, 0xB150, 3.376 - 0xB250, 0xB350, 0xB450, 0xB550, 0xB650, 0xB750, 0xB850, 0xB950, 3.377 - 0xBA70, 0xBB70, 0xBC70, 0xBD70, 0xBE70, 0xBF70, 0xC050, 0xC150, 3.378 - 0xC250, 0xC350, 0xC450, 0xC550, 0xC650, 0xC750, 0xC850, 0xC950, 3.379 - 0xCA70, 0xCB70, 0xCC70, 0xCD70, 0xCE70, 0xCF70, 0xD050, 0xD150, 3.380 - 0xD250, 0xD350, 0xD450, 0xD550, 0xD650, 0xD750, 0xD850, 0xD950, 3.381 - 0xDA70, 0xDB70, 0xDC70, 0xDD70, 0xDE70, 0xDF70, 0xE050, 0xE150, 3.382 - 0xE250, 0xE350, 0xE450, 0xE550, 0xE650, 0xE750, 0xE850, 0xE950, 3.383 - 0xEA70, 0xEB70, 0xEC70, 0xED70, 0xEE70, 0xEF70, 0xF050, 0xF150, 3.384 - 0xF250, 0xF350, 0xF450, 0xF550, 0xF650, 0xF750, 0xF850, 0xF950, 3.385 - 0xFA70, 0xFB70, 0xFC70, 0xFD70, 0xFE70, 0xFF70, 0x00D0, 0x0150, 3.386 - 0x0250, 0x0350, 0x0450, 0x0550, 0x0650, 0x0750, 0x0850, 0x0950, 3.387 - 0x0A70, 0x0B70, 0x0C70, 0x0D70, 0x0E70, 0x0F70, 0x1050, 0x1150, 3.388 - 0x1250, 0x1350, 0x1450, 0x1550, 0x1650, 0x1750, 0x1850, 0x1950, 3.389 - 0x1A70, 0x1B70, 0x1C70, 0x1D70, 0x1E70, 0x1F70, 0x2050, 0x2150, 3.390 - 0x2250, 0x2350, 0x2450, 0x2550, 0x2650, 0x2750, 0x2850, 0x2950, 3.391 - 0x2A70, 0x2B70, 0x2C70, 0x2D70, 0x2E70, 0x2F70, 0x3050, 0x3150, 3.392 - 0x3250, 0x3350, 0x3450, 0x3550, 0x3650, 0x3750, 0x3850, 0x3950, 3.393 - 0x3A70, 0x3B70, 0x3C70, 0x3D70, 0x3E70, 0x3F70, 0x4050, 0x4150, 3.394 - 0x4250, 0x4350, 0x4450, 0x4550, 0x4650, 0x4750, 0x4850, 0x4950, 3.395 - 0x4A70, 0x4B70, 0x4C70, 0x4D70, 0x4E70, 0x4F70, 0x5050, 0x5150, 3.396 - 0x5250, 0x5350, 0x5450, 0x5550, 0x5650, 0x5750, 0x5850, 0x5950, 3.397 - 0x5A70, 0x5B70, 0x5C70, 0x5D70, 0x5E70, 0x5F70, 0x6050, 0x6150, 3.398 - 0x6250, 0x6350, 0x6450, 0x6550, 0x6650, 0x6750, 0x6850, 0x6950, 3.399 - 0x6A70, 0x6B70, 0x6C70, 0x6D70, 0x6E70, 0x6F70, 0x7050, 0x7150, 3.400 - 0x7250, 0x7350, 0x7450, 0x7550, 0x7650, 0x7750, 0x7850, 0x7950, 3.401 - 0x7A70, 0x7B70, 0x7C70, 0x7D70, 0x7E70, 0x7F70, 0x8050, 0x8150, 3.402 - 0x8250, 0x8350, 0x8450, 0x8550, 0x8650, 0x8750, 0x8850, 0x8950, 3.403 - 0x8A70, 0x8B70, 0x8C70, 0x8D70, 0x8E70, 0x8F70, 0x9050, 0x9150, 3.404 - 0x9250, 0x9350, 0x9450, 0x9550, 0x9650, 0x9750, 0x9850, 0x9950, 3.405 -}; 3.406 + { 3.407 + 0x0080, 0x0100, 0x0200, 0x0300, 0x0400, 0x0500, 0x0600, 0x0700, 3.408 + 0x0800, 0x0900, 0x1020, 0x1120, 0x1220, 0x1320, 0x1420, 0x1520, 3.409 + 0x1000, 0x1100, 0x1200, 0x1300, 0x1400, 0x1500, 0x1600, 0x1700, 3.410 + 0x1800, 0x1900, 0x2020, 0x2120, 0x2220, 0x2320, 0x2420, 0x2520, 3.411 + 0x2000, 0x2100, 0x2200, 0x2300, 0x2400, 0x2500, 0x2600, 0x2700, 3.412 + 0x2800, 0x2900, 0x3020, 0x3120, 0x3220, 0x3320, 0x3420, 0x3520, 3.413 + 0x3000, 0x3100, 0x3200, 0x3300, 0x3400, 0x3500, 0x3600, 0x3700, 3.414 + 0x3800, 0x3900, 0x4020, 0x4120, 0x4220, 0x4320, 0x4420, 0x4520, 3.415 + 0x4000, 0x4100, 0x4200, 0x4300, 0x4400, 0x4500, 0x4600, 0x4700, 3.416 + 0x4800, 0x4900, 0x5020, 0x5120, 0x5220, 0x5320, 0x5420, 0x5520, 3.417 + 0x5000, 0x5100, 0x5200, 0x5300, 0x5400, 0x5500, 0x5600, 0x5700, 3.418 + 0x5800, 0x5900, 0x6020, 0x6120, 0x6220, 0x6320, 0x6420, 0x6520, 3.419 + 0x6000, 0x6100, 0x6200, 0x6300, 0x6400, 0x6500, 0x6600, 0x6700, 3.420 + 0x6800, 0x6900, 0x7020, 0x7120, 0x7220, 0x7320, 0x7420, 0x7520, 3.421 + 0x7000, 0x7100, 0x7200, 0x7300, 0x7400, 0x7500, 0x7600, 0x7700, 3.422 + 0x7800, 0x7900, 0x8020, 0x8120, 0x8220, 0x8320, 0x8420, 0x8520, 3.423 + 0x8000, 0x8100, 0x8200, 0x8300, 0x8400, 0x8500, 0x8600, 0x8700, 3.424 + 0x8800, 0x8900, 0x9020, 0x9120, 0x9220, 0x9320, 0x9420, 0x9520, 3.425 + 0x9000, 0x9100, 0x9200, 0x9300, 0x9400, 0x9500, 0x9600, 0x9700, 3.426 + 0x9800, 0x9900, 0x00B0, 0x0130, 0x0230, 0x0330, 0x0430, 0x0530, 3.427 + 0x0090, 0x0110, 0x0210, 0x0310, 0x0410, 0x0510, 0x0610, 0x0710, 3.428 + 0x0810, 0x0910, 0x1030, 0x1130, 0x1230, 0x1330, 0x1430, 0x1530, 3.429 + 0x1010, 0x1110, 0x1210, 0x1310, 0x1410, 0x1510, 0x1610, 0x1710, 3.430 + 0x1810, 0x1910, 0x2030, 0x2130, 0x2230, 0x2330, 0x2430, 0x2530, 3.431 + 0x2010, 0x2110, 0x2210, 0x2310, 0x2410, 0x2510, 0x2610, 0x2710, 3.432 + 0x2810, 0x2910, 0x3030, 0x3130, 0x3230, 0x3330, 0x3430, 0x3530, 3.433 + 0x3010, 0x3110, 0x3210, 0x3310, 0x3410, 0x3510, 0x3610, 0x3710, 3.434 + 0x3810, 0x3910, 0x4030, 0x4130, 0x4230, 0x4330, 0x4430, 0x4530, 3.435 + 0x4010, 0x4110, 0x4210, 0x4310, 0x4410, 0x4510, 0x4610, 0x4710, 3.436 + 0x4810, 0x4910, 0x5030, 0x5130, 0x5230, 0x5330, 0x5430, 0x5530, 3.437 + 0x5010, 0x5110, 0x5210, 0x5310, 0x5410, 0x5510, 0x5610, 0x5710, 3.438 + 0x5810, 0x5910, 0x6030, 0x6130, 0x6230, 0x6330, 0x6430, 0x6530, 3.439 + 0x6010, 0x6110, 0x6210, 0x6310, 0x6410, 0x6510, 0x6610, 0x6710, 3.440 + 0x6810, 0x6910, 0x7030, 0x7130, 0x7230, 0x7330, 0x7430, 0x7530, 3.441 + 0x7010, 0x7110, 0x7210, 0x7310, 0x7410, 0x7510, 0x7610, 0x7710, 3.442 + 0x7810, 0x7910, 0x8030, 0x8130, 0x8230, 0x8330, 0x8430, 0x8530, 3.443 + 0x8010, 0x8110, 0x8210, 0x8310, 0x8410, 0x8510, 0x8610, 0x8710, 3.444 + 0x8810, 0x8910, 0x9030, 0x9130, 0x9230, 0x9330, 0x9430, 0x9530, 3.445 + 0x9010, 0x9110, 0x9210, 0x9310, 0x9410, 0x9510, 0x9610, 0x9710, 3.446 + 0x9810, 0x9910, 0xA030, 0xA130, 0xA230, 0xA330, 0xA430, 0xA530, 3.447 + 0xA010, 0xA110, 0xA210, 0xA310, 0xA410, 0xA510, 0xA610, 0xA710, 3.448 + 0xA810, 0xA910, 0xB030, 0xB130, 0xB230, 0xB330, 0xB430, 0xB530, 3.449 + 0xB010, 0xB110, 0xB210, 0xB310, 0xB410, 0xB510, 0xB610, 0xB710, 3.450 + 0xB810, 0xB910, 0xC030, 0xC130, 0xC230, 0xC330, 0xC430, 0xC530, 3.451 + 0xC010, 0xC110, 0xC210, 0xC310, 0xC410, 0xC510, 0xC610, 0xC710, 3.452 + 0xC810, 0xC910, 0xD030, 0xD130, 0xD230, 0xD330, 0xD430, 0xD530, 3.453 + 0xD010, 0xD110, 0xD210, 0xD310, 0xD410, 0xD510, 0xD610, 0xD710, 3.454 + 0xD810, 0xD910, 0xE030, 0xE130, 0xE230, 0xE330, 0xE430, 0xE530, 3.455 + 0xE010, 0xE110, 0xE210, 0xE310, 0xE410, 0xE510, 0xE610, 0xE710, 3.456 + 0xE810, 0xE910, 0xF030, 0xF130, 0xF230, 0xF330, 0xF430, 0xF530, 3.457 + 0xF010, 0xF110, 0xF210, 0xF310, 0xF410, 0xF510, 0xF610, 0xF710, 3.458 + 0xF810, 0xF910, 0x00B0, 0x0130, 0x0230, 0x0330, 0x0430, 0x0530, 3.459 + 0x0090, 0x0110, 0x0210, 0x0310, 0x0410, 0x0510, 0x0610, 0x0710, 3.460 + 0x0810, 0x0910, 0x1030, 0x1130, 0x1230, 0x1330, 0x1430, 0x1530, 3.461 + 0x1010, 0x1110, 0x1210, 0x1310, 0x1410, 0x1510, 0x1610, 0x1710, 3.462 + 0x1810, 0x1910, 0x2030, 0x2130, 0x2230, 0x2330, 0x2430, 0x2530, 3.463 + 0x2010, 0x2110, 0x2210, 0x2310, 0x2410, 0x2510, 0x2610, 0x2710, 3.464 + 0x2810, 0x2910, 0x3030, 0x3130, 0x3230, 0x3330, 0x3430, 0x3530, 3.465 + 0x3010, 0x3110, 0x3210, 0x3310, 0x3410, 0x3510, 0x3610, 0x3710, 3.466 + 0x3810, 0x3910, 0x4030, 0x4130, 0x4230, 0x4330, 0x4430, 0x4530, 3.467 + 0x4010, 0x4110, 0x4210, 0x4310, 0x4410, 0x4510, 0x4610, 0x4710, 3.468 + 0x4810, 0x4910, 0x5030, 0x5130, 0x5230, 0x5330, 0x5430, 0x5530, 3.469 + 0x5010, 0x5110, 0x5210, 0x5310, 0x5410, 0x5510, 0x5610, 0x5710, 3.470 + 0x5810, 0x5910, 0x6030, 0x6130, 0x6230, 0x6330, 0x6430, 0x6530, 3.471 + 0x0600, 0x0700, 0x0800, 0x0900, 0x0A00, 0x0B00, 0x0C00, 0x0D00, 3.472 + 0x0E00, 0x0F00, 0x1020, 0x1120, 0x1220, 0x1320, 0x1420, 0x1520, 3.473 + 0x1600, 0x1700, 0x1800, 0x1900, 0x1A00, 0x1B00, 0x1C00, 0x1D00, 3.474 + 0x1E00, 0x1F00, 0x2020, 0x2120, 0x2220, 0x2320, 0x2420, 0x2520, 3.475 + 0x2600, 0x2700, 0x2800, 0x2900, 0x2A00, 0x2B00, 0x2C00, 0x2D00, 3.476 + 0x2E00, 0x2F00, 0x3020, 0x3120, 0x3220, 0x3320, 0x3420, 0x3520, 3.477 + 0x3600, 0x3700, 0x3800, 0x3900, 0x3A00, 0x3B00, 0x3C00, 0x3D00, 3.478 + 0x3E00, 0x3F00, 0x4020, 0x4120, 0x4220, 0x4320, 0x4420, 0x4520, 3.479 + 0x4600, 0x4700, 0x4800, 0x4900, 0x4A00, 0x4B00, 0x4C00, 0x4D00, 3.480 + 0x4E00, 0x4F00, 0x5020, 0x5120, 0x5220, 0x5320, 0x5420, 0x5520, 3.481 + 0x5600, 0x5700, 0x5800, 0x5900, 0x5A00, 0x5B00, 0x5C00, 0x5D00, 3.482 + 0x5E00, 0x5F00, 0x6020, 0x6120, 0x6220, 0x6320, 0x6420, 0x6520, 3.483 + 0x6600, 0x6700, 0x6800, 0x6900, 0x6A00, 0x6B00, 0x6C00, 0x6D00, 3.484 + 0x6E00, 0x6F00, 0x7020, 0x7120, 0x7220, 0x7320, 0x7420, 0x7520, 3.485 + 0x7600, 0x7700, 0x7800, 0x7900, 0x7A00, 0x7B00, 0x7C00, 0x7D00, 3.486 + 0x7E00, 0x7F00, 0x8020, 0x8120, 0x8220, 0x8320, 0x8420, 0x8520, 3.487 + 0x8600, 0x8700, 0x8800, 0x8900, 0x8A00, 0x8B00, 0x8C00, 0x8D00, 3.488 + 0x8E00, 0x8F00, 0x9020, 0x9120, 0x9220, 0x9320, 0x9420, 0x9520, 3.489 + 0x9600, 0x9700, 0x9800, 0x9900, 0x9A00, 0x9B00, 0x9C00, 0x9D00, 3.490 + 0x9E00, 0x9F00, 0x00B0, 0x0130, 0x0230, 0x0330, 0x0430, 0x0530, 3.491 + 0x0610, 0x0710, 0x0810, 0x0910, 0x0A10, 0x0B10, 0x0C10, 0x0D10, 3.492 + 0x0E10, 0x0F10, 0x1030, 0x1130, 0x1230, 0x1330, 0x1430, 0x1530, 3.493 + 0x1610, 0x1710, 0x1810, 0x1910, 0x1A10, 0x1B10, 0x1C10, 0x1D10, 3.494 + 0x1E10, 0x1F10, 0x2030, 0x2130, 0x2230, 0x2330, 0x2430, 0x2530, 3.495 + 0x2610, 0x2710, 0x2810, 0x2910, 0x2A10, 0x2B10, 0x2C10, 0x2D10, 3.496 + 0x2E10, 0x2F10, 0x3030, 0x3130, 0x3230, 0x3330, 0x3430, 0x3530, 3.497 + 0x3610, 0x3710, 0x3810, 0x3910, 0x3A10, 0x3B10, 0x3C10, 0x3D10, 3.498 + 0x3E10, 0x3F10, 0x4030, 0x4130, 0x4230, 0x4330, 0x4430, 0x4530, 3.499 + 0x4610, 0x4710, 0x4810, 0x4910, 0x4A10, 0x4B10, 0x4C10, 0x4D10, 3.500 + 0x4E10, 0x4F10, 0x5030, 0x5130, 0x5230, 0x5330, 0x5430, 0x5530, 3.501 + 0x5610, 0x5710, 0x5810, 0x5910, 0x5A10, 0x5B10, 0x5C10, 0x5D10, 3.502 + 0x5E10, 0x5F10, 0x6030, 0x6130, 0x6230, 0x6330, 0x6430, 0x6530, 3.503 + 0x6610, 0x6710, 0x6810, 0x6910, 0x6A10, 0x6B10, 0x6C10, 0x6D10, 3.504 + 0x6E10, 0x6F10, 0x7030, 0x7130, 0x7230, 0x7330, 0x7430, 0x7530, 3.505 + 0x7610, 0x7710, 0x7810, 0x7910, 0x7A10, 0x7B10, 0x7C10, 0x7D10, 3.506 + 0x7E10, 0x7F10, 0x8030, 0x8130, 0x8230, 0x8330, 0x8430, 0x8530, 3.507 + 0x8610, 0x8710, 0x8810, 0x8910, 0x8A10, 0x8B10, 0x8C10, 0x8D10, 3.508 + 0x8E10, 0x8F10, 0x9030, 0x9130, 0x9230, 0x9330, 0x9430, 0x9530, 3.509 + 0x9610, 0x9710, 0x9810, 0x9910, 0x9A10, 0x9B10, 0x9C10, 0x9D10, 3.510 + 0x9E10, 0x9F10, 0xA030, 0xA130, 0xA230, 0xA330, 0xA430, 0xA530, 3.511 + 0xA610, 0xA710, 0xA810, 0xA910, 0xAA10, 0xAB10, 0xAC10, 0xAD10, 3.512 + 0xAE10, 0xAF10, 0xB030, 0xB130, 0xB230, 0xB330, 0xB430, 0xB530, 3.513 + 0xB610, 0xB710, 0xB810, 0xB910, 0xBA10, 0xBB10, 0xBC10, 0xBD10, 3.514 + 0xBE10, 0xBF10, 0xC030, 0xC130, 0xC230, 0xC330, 0xC430, 0xC530, 3.515 + 0xC610, 0xC710, 0xC810, 0xC910, 0xCA10, 0xCB10, 0xCC10, 0xCD10, 3.516 + 0xCE10, 0xCF10, 0xD030, 0xD130, 0xD230, 0xD330, 0xD430, 0xD530, 3.517 + 0xD610, 0xD710, 0xD810, 0xD910, 0xDA10, 0xDB10, 0xDC10, 0xDD10, 3.518 + 0xDE10, 0xDF10, 0xE030, 0xE130, 0xE230, 0xE330, 0xE430, 0xE530, 3.519 + 0xE610, 0xE710, 0xE810, 0xE910, 0xEA10, 0xEB10, 0xEC10, 0xED10, 3.520 + 0xEE10, 0xEF10, 0xF030, 0xF130, 0xF230, 0xF330, 0xF430, 0xF530, 3.521 + 0xF610, 0xF710, 0xF810, 0xF910, 0xFA10, 0xFB10, 0xFC10, 0xFD10, 3.522 + 0xFE10, 0xFF10, 0x00B0, 0x0130, 0x0230, 0x0330, 0x0430, 0x0530, 3.523 + 0x0610, 0x0710, 0x0810, 0x0910, 0x0A10, 0x0B10, 0x0C10, 0x0D10, 3.524 + 0x0E10, 0x0F10, 0x1030, 0x1130, 0x1230, 0x1330, 0x1430, 0x1530, 3.525 + 0x1610, 0x1710, 0x1810, 0x1910, 0x1A10, 0x1B10, 0x1C10, 0x1D10, 3.526 + 0x1E10, 0x1F10, 0x2030, 0x2130, 0x2230, 0x2330, 0x2430, 0x2530, 3.527 + 0x2610, 0x2710, 0x2810, 0x2910, 0x2A10, 0x2B10, 0x2C10, 0x2D10, 3.528 + 0x2E10, 0x2F10, 0x3030, 0x3130, 0x3230, 0x3330, 0x3430, 0x3530, 3.529 + 0x3610, 0x3710, 0x3810, 0x3910, 0x3A10, 0x3B10, 0x3C10, 0x3D10, 3.530 + 0x3E10, 0x3F10, 0x4030, 0x4130, 0x4230, 0x4330, 0x4430, 0x4530, 3.531 + 0x4610, 0x4710, 0x4810, 0x4910, 0x4A10, 0x4B10, 0x4C10, 0x4D10, 3.532 + 0x4E10, 0x4F10, 0x5030, 0x5130, 0x5230, 0x5330, 0x5430, 0x5530, 3.533 + 0x5610, 0x5710, 0x5810, 0x5910, 0x5A10, 0x5B10, 0x5C10, 0x5D10, 3.534 + 0x5E10, 0x5F10, 0x6030, 0x6130, 0x6230, 0x6330, 0x6430, 0x6530, 3.535 + 0x00C0, 0x0140, 0x0240, 0x0340, 0x0440, 0x0540, 0x0640, 0x0740, 3.536 + 0x0840, 0x0940, 0x0440, 0x0540, 0x0640, 0x0740, 0x0840, 0x0940, 3.537 + 0x1040, 0x1140, 0x1240, 0x1340, 0x1440, 0x1540, 0x1640, 0x1740, 3.538 + 0x1840, 0x1940, 0x1440, 0x1540, 0x1640, 0x1740, 0x1840, 0x1940, 3.539 + 0x2040, 0x2140, 0x2240, 0x2340, 0x2440, 0x2540, 0x2640, 0x2740, 3.540 + 0x2840, 0x2940, 0x2440, 0x2540, 0x2640, 0x2740, 0x2840, 0x2940, 3.541 + 0x3040, 0x3140, 0x3240, 0x3340, 0x3440, 0x3540, 0x3640, 0x3740, 3.542 + 0x3840, 0x3940, 0x3440, 0x3540, 0x3640, 0x3740, 0x3840, 0x3940, 3.543 + 0x4040, 0x4140, 0x4240, 0x4340, 0x4440, 0x4540, 0x4640, 0x4740, 3.544 + 0x4840, 0x4940, 0x4440, 0x4540, 0x4640, 0x4740, 0x4840, 0x4940, 3.545 + 0x5040, 0x5140, 0x5240, 0x5340, 0x5440, 0x5540, 0x5640, 0x5740, 3.546 + 0x5840, 0x5940, 0x5440, 0x5540, 0x5640, 0x5740, 0x5840, 0x5940, 3.547 + 0x6040, 0x6140, 0x6240, 0x6340, 0x6440, 0x6540, 0x6640, 0x6740, 3.548 + 0x6840, 0x6940, 0x6440, 0x6540, 0x6640, 0x6740, 0x6840, 0x6940, 3.549 + 0x7040, 0x7140, 0x7240, 0x7340, 0x7440, 0x7540, 0x7640, 0x7740, 3.550 + 0x7840, 0x7940, 0x7440, 0x7540, 0x7640, 0x7740, 0x7840, 0x7940, 3.551 + 0x8040, 0x8140, 0x8240, 0x8340, 0x8440, 0x8540, 0x8640, 0x8740, 3.552 + 0x8840, 0x8940, 0x8440, 0x8540, 0x8640, 0x8740, 0x8840, 0x8940, 3.553 + 0x9040, 0x9140, 0x9240, 0x9340, 0x9440, 0x9540, 0x9640, 0x9740, 3.554 + 0x9840, 0x9940, 0x3450, 0x3550, 0x3650, 0x3750, 0x3850, 0x3950, 3.555 + 0x4050, 0x4150, 0x4250, 0x4350, 0x4450, 0x4550, 0x4650, 0x4750, 3.556 + 0x4850, 0x4950, 0x4450, 0x4550, 0x4650, 0x4750, 0x4850, 0x4950, 3.557 + 0x5050, 0x5150, 0x5250, 0x5350, 0x5450, 0x5550, 0x5650, 0x5750, 3.558 + 0x5850, 0x5950, 0x5450, 0x5550, 0x5650, 0x5750, 0x5850, 0x5950, 3.559 + 0x6050, 0x6150, 0x6250, 0x6350, 0x6450, 0x6550, 0x6650, 0x6750, 3.560 + 0x6850, 0x6950, 0x6450, 0x6550, 0x6650, 0x6750, 0x6850, 0x6950, 3.561 + 0x7050, 0x7150, 0x7250, 0x7350, 0x7450, 0x7550, 0x7650, 0x7750, 3.562 + 0x7850, 0x7950, 0x7450, 0x7550, 0x7650, 0x7750, 0x7850, 0x7950, 3.563 + 0x8050, 0x8150, 0x8250, 0x8350, 0x8450, 0x8550, 0x8650, 0x8750, 3.564 + 0x8850, 0x8950, 0x8450, 0x8550, 0x8650, 0x8750, 0x8850, 0x8950, 3.565 + 0x9050, 0x9150, 0x9250, 0x9350, 0x9450, 0x9550, 0x9650, 0x9750, 3.566 + 0x9850, 0x9950, 0x9450, 0x9550, 0x9650, 0x9750, 0x9850, 0x9950, 3.567 + 0xA050, 0xA150, 0xA250, 0xA350, 0xA450, 0xA550, 0xA650, 0xA750, 3.568 + 0xA850, 0xA950, 0xA450, 0xA550, 0xA650, 0xA750, 0xA850, 0xA950, 3.569 + 0xB050, 0xB150, 0xB250, 0xB350, 0xB450, 0xB550, 0xB650, 0xB750, 3.570 + 0xB850, 0xB950, 0xB450, 0xB550, 0xB650, 0xB750, 0xB850, 0xB950, 3.571 + 0xC050, 0xC150, 0xC250, 0xC350, 0xC450, 0xC550, 0xC650, 0xC750, 3.572 + 0xC850, 0xC950, 0xC450, 0xC550, 0xC650, 0xC750, 0xC850, 0xC950, 3.573 + 0xD050, 0xD150, 0xD250, 0xD350, 0xD450, 0xD550, 0xD650, 0xD750, 3.574 + 0xD850, 0xD950, 0xD450, 0xD550, 0xD650, 0xD750, 0xD850, 0xD950, 3.575 + 0xE050, 0xE150, 0xE250, 0xE350, 0xE450, 0xE550, 0xE650, 0xE750, 3.576 + 0xE850, 0xE950, 0xE450, 0xE550, 0xE650, 0xE750, 0xE850, 0xE950, 3.577 + 0xF050, 0xF150, 0xF250, 0xF350, 0xF450, 0xF550, 0xF650, 0xF750, 3.578 + 0xF850, 0xF950, 0xF450, 0xF550, 0xF650, 0xF750, 0xF850, 0xF950, 3.579 + 0x00D0, 0x0150, 0x0250, 0x0350, 0x0450, 0x0550, 0x0650, 0x0750, 3.580 + 0x0850, 0x0950, 0x0450, 0x0550, 0x0650, 0x0750, 0x0850, 0x0950, 3.581 + 0x1050, 0x1150, 0x1250, 0x1350, 0x1450, 0x1550, 0x1650, 0x1750, 3.582 + 0x1850, 0x1950, 0x1450, 0x1550, 0x1650, 0x1750, 0x1850, 0x1950, 3.583 + 0x2050, 0x2150, 0x2250, 0x2350, 0x2450, 0x2550, 0x2650, 0x2750, 3.584 + 0x2850, 0x2950, 0x2450, 0x2550, 0x2650, 0x2750, 0x2850, 0x2950, 3.585 + 0x3050, 0x3150, 0x3250, 0x3350, 0x3450, 0x3550, 0x3650, 0x3750, 3.586 + 0x3850, 0x3950, 0x3450, 0x3550, 0x3650, 0x3750, 0x3850, 0x3950, 3.587 + 0x4050, 0x4150, 0x4250, 0x4350, 0x4450, 0x4550, 0x4650, 0x4750, 3.588 + 0x4850, 0x4950, 0x4450, 0x4550, 0x4650, 0x4750, 0x4850, 0x4950, 3.589 + 0x5050, 0x5150, 0x5250, 0x5350, 0x5450, 0x5550, 0x5650, 0x5750, 3.590 + 0x5850, 0x5950, 0x5450, 0x5550, 0x5650, 0x5750, 0x5850, 0x5950, 3.591 + 0x6050, 0x6150, 0x6250, 0x6350, 0x6450, 0x6550, 0x6650, 0x6750, 3.592 + 0x6850, 0x6950, 0x6450, 0x6550, 0x6650, 0x6750, 0x6850, 0x6950, 3.593 + 0x7050, 0x7150, 0x7250, 0x7350, 0x7450, 0x7550, 0x7650, 0x7750, 3.594 + 0x7850, 0x7950, 0x7450, 0x7550, 0x7650, 0x7750, 0x7850, 0x7950, 3.595 + 0x8050, 0x8150, 0x8250, 0x8350, 0x8450, 0x8550, 0x8650, 0x8750, 3.596 + 0x8850, 0x8950, 0x8450, 0x8550, 0x8650, 0x8750, 0x8850, 0x8950, 3.597 + 0x9050, 0x9150, 0x9250, 0x9350, 0x9450, 0x9550, 0x9650, 0x9750, 3.598 + 0x9850, 0x9950, 0x9450, 0x9550, 0x9650, 0x9750, 0x9850, 0x9950, 3.599 + 0xFA60, 0xFB60, 0xFC60, 0xFD60, 0xFE60, 0xFF60, 0x00C0, 0x0140, 3.600 + 0x0240, 0x0340, 0x0440, 0x0540, 0x0640, 0x0740, 0x0840, 0x0940, 3.601 + 0x0A60, 0x0B60, 0x0C60, 0x0D60, 0x0E60, 0x0F60, 0x1040, 0x1140, 3.602 + 0x1240, 0x1340, 0x1440, 0x1540, 0x1640, 0x1740, 0x1840, 0x1940, 3.603 + 0x1A60, 0x1B60, 0x1C60, 0x1D60, 0x1E60, 0x1F60, 0x2040, 0x2140, 3.604 + 0x2240, 0x2340, 0x2440, 0x2540, 0x2640, 0x2740, 0x2840, 0x2940, 3.605 + 0x2A60, 0x2B60, 0x2C60, 0x2D60, 0x2E60, 0x2F60, 0x3040, 0x3140, 3.606 + 0x3240, 0x3340, 0x3440, 0x3540, 0x3640, 0x3740, 0x3840, 0x3940, 3.607 + 0x3A60, 0x3B60, 0x3C60, 0x3D60, 0x3E60, 0x3F60, 0x4040, 0x4140, 3.608 + 0x4240, 0x4340, 0x4440, 0x4540, 0x4640, 0x4740, 0x4840, 0x4940, 3.609 + 0x4A60, 0x4B60, 0x4C60, 0x4D60, 0x4E60, 0x4F60, 0x5040, 0x5140, 3.610 + 0x5240, 0x5340, 0x5440, 0x5540, 0x5640, 0x5740, 0x5840, 0x5940, 3.611 + 0x5A60, 0x5B60, 0x5C60, 0x5D60, 0x5E60, 0x5F60, 0x6040, 0x6140, 3.612 + 0x6240, 0x6340, 0x6440, 0x6540, 0x6640, 0x6740, 0x6840, 0x6940, 3.613 + 0x6A60, 0x6B60, 0x6C60, 0x6D60, 0x6E60, 0x6F60, 0x7040, 0x7140, 3.614 + 0x7240, 0x7340, 0x7440, 0x7540, 0x7640, 0x7740, 0x7840, 0x7940, 3.615 + 0x7A60, 0x7B60, 0x7C60, 0x7D60, 0x7E60, 0x7F60, 0x8040, 0x8140, 3.616 + 0x8240, 0x8340, 0x8440, 0x8540, 0x8640, 0x8740, 0x8840, 0x8940, 3.617 + 0x8A60, 0x8B60, 0x8C60, 0x8D60, 0x8E60, 0x8F60, 0x9040, 0x9140, 3.618 + 0x9240, 0x9340, 0x3450, 0x3550, 0x3650, 0x3750, 0x3850, 0x3950, 3.619 + 0x3A70, 0x3B70, 0x3C70, 0x3D70, 0x3E70, 0x3F70, 0x4050, 0x4150, 3.620 + 0x4250, 0x4350, 0x4450, 0x4550, 0x4650, 0x4750, 0x4850, 0x4950, 3.621 + 0x4A70, 0x4B70, 0x4C70, 0x4D70, 0x4E70, 0x4F70, 0x5050, 0x5150, 3.622 + 0x5250, 0x5350, 0x5450, 0x5550, 0x5650, 0x5750, 0x5850, 0x5950, 3.623 + 0x5A70, 0x5B70, 0x5C70, 0x5D70, 0x5E70, 0x5F70, 0x6050, 0x6150, 3.624 + 0x6250, 0x6350, 0x6450, 0x6550, 0x6650, 0x6750, 0x6850, 0x6950, 3.625 + 0x6A70, 0x6B70, 0x6C70, 0x6D70, 0x6E70, 0x6F70, 0x7050, 0x7150, 3.626 + 0x7250, 0x7350, 0x7450, 0x7550, 0x7650, 0x7750, 0x7850, 0x7950, 3.627 + 0x7A70, 0x7B70, 0x7C70, 0x7D70, 0x7E70, 0x7F70, 0x8050, 0x8150, 3.628 + 0x8250, 0x8350, 0x8450, 0x8550, 0x8650, 0x8750, 0x8850, 0x8950, 3.629 + 0x8A70, 0x8B70, 0x8C70, 0x8D70, 0x8E70, 0x8F70, 0x9050, 0x9150, 3.630 + 0x9250, 0x9350, 0x9450, 0x9550, 0x9650, 0x9750, 0x9850, 0x9950, 3.631 + 0x9A70, 0x9B70, 0x9C70, 0x9D70, 0x9E70, 0x9F70, 0xA050, 0xA150, 3.632 + 0xA250, 0xA350, 0xA450, 0xA550, 0xA650, 0xA750, 0xA850, 0xA950, 3.633 + 0xAA70, 0xAB70, 0xAC70, 0xAD70, 0xAE70, 0xAF70, 0xB050, 0xB150, 3.634 + 0xB250, 0xB350, 0xB450, 0xB550, 0xB650, 0xB750, 0xB850, 0xB950, 3.635 + 0xBA70, 0xBB70, 0xBC70, 0xBD70, 0xBE70, 0xBF70, 0xC050, 0xC150, 3.636 + 0xC250, 0xC350, 0xC450, 0xC550, 0xC650, 0xC750, 0xC850, 0xC950, 3.637 + 0xCA70, 0xCB70, 0xCC70, 0xCD70, 0xCE70, 0xCF70, 0xD050, 0xD150, 3.638 + 0xD250, 0xD350, 0xD450, 0xD550, 0xD650, 0xD750, 0xD850, 0xD950, 3.639 + 0xDA70, 0xDB70, 0xDC70, 0xDD70, 0xDE70, 0xDF70, 0xE050, 0xE150, 3.640 + 0xE250, 0xE350, 0xE450, 0xE550, 0xE650, 0xE750, 0xE850, 0xE950, 3.641 + 0xEA70, 0xEB70, 0xEC70, 0xED70, 0xEE70, 0xEF70, 0xF050, 0xF150, 3.642 + 0xF250, 0xF350, 0xF450, 0xF550, 0xF650, 0xF750, 0xF850, 0xF950, 3.643 + 0xFA70, 0xFB70, 0xFC70, 0xFD70, 0xFE70, 0xFF70, 0x00D0, 0x0150, 3.644 + 0x0250, 0x0350, 0x0450, 0x0550, 0x0650, 0x0750, 0x0850, 0x0950, 3.645 + 0x0A70, 0x0B70, 0x0C70, 0x0D70, 0x0E70, 0x0F70, 0x1050, 0x1150, 3.646 + 0x1250, 0x1350, 0x1450, 0x1550, 0x1650, 0x1750, 0x1850, 0x1950, 3.647 + 0x1A70, 0x1B70, 0x1C70, 0x1D70, 0x1E70, 0x1F70, 0x2050, 0x2150, 3.648 + 0x2250, 0x2350, 0x2450, 0x2550, 0x2650, 0x2750, 0x2850, 0x2950, 3.649 + 0x2A70, 0x2B70, 0x2C70, 0x2D70, 0x2E70, 0x2F70, 0x3050, 0x3150, 3.650 + 0x3250, 0x3350, 0x3450, 0x3550, 0x3650, 0x3750, 0x3850, 0x3950, 3.651 + 0x3A70, 0x3B70, 0x3C70, 0x3D70, 0x3E70, 0x3F70, 0x4050, 0x4150, 3.652 + 0x4250, 0x4350, 0x4450, 0x4550, 0x4650, 0x4750, 0x4850, 0x4950, 3.653 + 0x4A70, 0x4B70, 0x4C70, 0x4D70, 0x4E70, 0x4F70, 0x5050, 0x5150, 3.654 + 0x5250, 0x5350, 0x5450, 0x5550, 0x5650, 0x5750, 0x5850, 0x5950, 3.655 + 0x5A70, 0x5B70, 0x5C70, 0x5D70, 0x5E70, 0x5F70, 0x6050, 0x6150, 3.656 + 0x6250, 0x6350, 0x6450, 0x6550, 0x6650, 0x6750, 0x6850, 0x6950, 3.657 + 0x6A70, 0x6B70, 0x6C70, 0x6D70, 0x6E70, 0x6F70, 0x7050, 0x7150, 3.658 + 0x7250, 0x7350, 0x7450, 0x7550, 0x7650, 0x7750, 0x7850, 0x7950, 3.659 + 0x7A70, 0x7B70, 0x7C70, 0x7D70, 0x7E70, 0x7F70, 0x8050, 0x8150, 3.660 + 0x8250, 0x8350, 0x8450, 0x8550, 0x8650, 0x8750, 0x8850, 0x8950, 3.661 + 0x8A70, 0x8B70, 0x8C70, 0x8D70, 0x8E70, 0x8F70, 0x9050, 0x9150, 3.662 + 0x9250, 0x9350, 0x9450, 0x9550, 0x9650, 0x9750, 0x9850, 0x9950, 3.663 + }; 3.664 3.665 u8 ZeroTable[] = 3.666 -{ 3.667 - 0x80, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 3.668 - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 3.669 - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 3.670 - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 3.671 - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 3.672 - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 3.673 - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 3.674 - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 3.675 - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 3.676 - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 3.677 - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 3.678 - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 3.679 - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 3.680 - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 3.681 - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 3.682 - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0 3.683 -}; 3.684 + { 3.685 + 0x80, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 3.686 + 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 3.687 + 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 3.688 + 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 3.689 + 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 3.690 + 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 3.691 + 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 3.692 + 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 3.693 + 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 3.694 + 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 3.695 + 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 3.696 + 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 3.697 + 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 3.698 + 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 3.699 + 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 3.700 + 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0 3.701 + }; 3.702 3.703 #define GBSAVE_GAME_VERSION_1 1 3.704 #define GBSAVE_GAME_VERSION_2 2 3.705 @@ -536,81 +536,81 @@ 3.706 3.707 int inline gbGetValue(int min, int max, int v) 3.708 { 3.709 - return (int)(min + (float)(max - min) * (2.0 * (v / 31.0) - (v / 31.0) * (v / 31.0))); 3.710 + return (int)(min + (float)(max - min) * (2.0 * (v / 31.0) - (v / 31.0) * (v / 31.0))); 3.711 } 3.712 3.713 void gbGenFilter() 3.714 { 3.715 - for (int r = 0; r < 32; r++) 3.716 + for (int r = 0; r < 32; r++) 3.717 + { 3.718 + for (int g = 0; g < 32; g++) 3.719 { 3.720 - for (int g = 0; g < 32; g++) 3.721 - { 3.722 - for (int b = 0; b < 32; b++) 3.723 - { 3.724 - int nr = gbGetValue(gbGetValue(4, 14, g), 3.725 - gbGetValue(24, 29, g), r) - 4; 3.726 - int ng = gbGetValue(gbGetValue(4 + gbGetValue(0, 5, r), 3.727 - 14 + gbGetValue(0, 3, r), b), 3.728 - gbGetValue(24 + gbGetValue(0, 3, r), 3.729 - 29 + gbGetValue(0, 1, r), b), g) - 4; 3.730 - int nb = gbGetValue(gbGetValue(4 + gbGetValue(0, 5, r), 3.731 - 14 + gbGetValue(0, 3, r), g), 3.732 - gbGetValue(24 + gbGetValue(0, 3, r), 3.733 - 29 + gbGetValue(0, 1, r), g), b) - 4; 3.734 - gbColorFilter[(b << 10) | (g << 5) | r] = (nb << 10) | (ng << 5) | nr; 3.735 - } 3.736 - } 3.737 + for (int b = 0; b < 32; b++) 3.738 + { 3.739 + int nr = gbGetValue(gbGetValue(4, 14, g), 3.740 + gbGetValue(24, 29, g), r) - 4; 3.741 + int ng = gbGetValue(gbGetValue(4 + gbGetValue(0, 5, r), 3.742 + 14 + gbGetValue(0, 3, r), b), 3.743 + gbGetValue(24 + gbGetValue(0, 3, r), 3.744 + 29 + gbGetValue(0, 1, r), b), g) - 4; 3.745 + int nb = gbGetValue(gbGetValue(4 + gbGetValue(0, 5, r), 3.746 + 14 + gbGetValue(0, 3, r), g), 3.747 + gbGetValue(24 + gbGetValue(0, 3, r), 3.748 + 29 + gbGetValue(0, 1, r), g), b) - 4; 3.749 + gbColorFilter[(b << 10) | (g << 5) | r] = (nb << 10) | (ng << 5) | nr; 3.750 + } 3.751 } 3.752 + } 3.753 } 3.754 3.755 void gbCopyMemory(u16 d, u16 s, int count) 3.756 { 3.757 - while (count) 3.758 - { 3.759 - gbWriteMemoryQuick(d, gbReadMemoryQuick(s)); 3.760 - s++; 3.761 - d++; 3.762 - count--; 3.763 - } 3.764 + while (count) 3.765 + { 3.766 + gbWriteMemoryQuick(d, gbReadMemoryQuick(s)); 3.767 + s++; 3.768 + d++; 3.769 + count--; 3.770 + } 3.771 } 3.772 3.773 void gbDoHdma() 3.774 { 3.775 - gbCopyMemory(gbHdmaDestination, gbHdmaSource, 0x10); 3.776 - 3.777 - gbHdmaDestination += 0x10; 3.778 - gbHdmaSource += 0x10; 3.779 - 3.780 - register_HDMA2 = (register_HDMA2 + 0x10) & 0xFF; 3.781 - if (register_HDMA2 == 0x00) 3.782 - register_HDMA1++; 3.783 - 3.784 - register_HDMA4 = (register_HDMA4 + 0x10) & 0xFF; 3.785 - if (register_HDMA4 == 0x00) 3.786 - register_HDMA3++; 3.787 - 3.788 - if (gbHdmaDestination == 0x96b0) 3.789 - gbHdmaBytes = gbHdmaBytes; 3.790 - gbHdmaBytes -= 0x10; 3.791 - register_HDMA5--; 3.792 - if (register_HDMA5 == 0xff) 3.793 - gbHdmaOn = 0; 3.794 + gbCopyMemory(gbHdmaDestination, gbHdmaSource, 0x10); 3.795 + 3.796 + gbHdmaDestination += 0x10; 3.797 + gbHdmaSource += 0x10; 3.798 + 3.799 + register_HDMA2 = (register_HDMA2 + 0x10) & 0xFF; 3.800 + if (register_HDMA2 == 0x00) 3.801 + register_HDMA1++; 3.802 + 3.803 + register_HDMA4 = (register_HDMA4 + 0x10) & 0xFF; 3.804 + if (register_HDMA4 == 0x00) 3.805 + register_HDMA3++; 3.806 + 3.807 + if (gbHdmaDestination == 0x96b0) 3.808 + gbHdmaBytes = gbHdmaBytes; 3.809 + gbHdmaBytes -= 0x10; 3.810 + register_HDMA5--; 3.811 + if (register_HDMA5 == 0xff) 3.812 + gbHdmaOn = 0; 3.813 } 3.814 3.815 // fix for Harley and Lego Racers 3.816 void gbCompareLYToLYC() 3.817 { 3.818 - if (register_LY == register_LYC) 3.819 - { 3.820 - // mark that we have a match 3.821 - register_STAT |= 4; 3.822 - 3.823 - // check if we need an interrupt 3.824 - if ((register_STAT & 0x40) && (register_IE & 2)) 3.825 - gbInterrupt |= 2; 3.826 - } 3.827 - else // no match 3.828 - register_STAT &= 0xfb; 3.829 + if (register_LY == register_LYC) 3.830 + { 3.831 + // mark that we have a match 3.832 + register_STAT |= 4; 3.833 + 3.834 + // check if we need an interrupt 3.835 + if ((register_STAT & 0x40) && (register_IE & 2)) 3.836 + gbInterrupt |= 2; 3.837 + } 3.838 + else // no match 3.839 + register_STAT &= 0xfb; 3.840 } 3.841 3.842 // FIXME: horrible kludge to workaround the frame timing bug 3.843 @@ -618,1971 +618,1971 @@ 3.844 3.845 void gbWriteMemoryWrapped(register u16 address, register u8 value) 3.846 { 3.847 - if (address < 0x8000) 3.848 + if (address < 0x8000) 3.849 + { 3.850 +#ifndef FINAL_VERSION 3.851 + if (memorydebug && (address > 0x3fff || address < 0x2000)) 3.852 { 3.853 + log("Memory register write %04x=%02x PC=%04x\n", 3.854 + address, 3.855 + value, 3.856 + PC.W); 3.857 + } 3.858 +#endif 3.859 + if (mapper) 3.860 + (*mapper)(address, value); 3.861 + return; 3.862 + } 3.863 + 3.864 + if (address < 0xa000) 3.865 + { 3.866 + gbWriteMemoryQuick(address, value); 3.867 + return; 3.868 + } 3.869 + 3.870 + if (address < 0xc000) 3.871 + { 3.872 #ifndef FINAL_VERSION 3.873 - if (memorydebug && (address > 0x3fff || address < 0x2000)) 3.874 - { 3.875 - log("Memory register write %04x=%02x PC=%04x\n", 3.876 - address, 3.877 - value, 3.878 - PC.W); 3.879 - } 3.880 + if (memorydebug) 3.881 + { 3.882 + log("Memory register write %04x=%02x PC=%04x\n", 3.883 + address, 3.884 + value, 3.885 + PC.W); 3.886 + } 3.887 #endif 3.888 - if (mapper) 3.889 - (*mapper)(address, value); 3.890 - return; 3.891 + 3.892 + if (mapper) 3.893 + (*mapperRAM)(address, value); 3.894 + return; 3.895 + } 3.896 + 3.897 + if (address < 0xfe00) 3.898 + { 3.899 + gbWriteMemoryQuick(address, value); 3.900 + return; 3.901 + } 3.902 + 3.903 + if (address < 0xff00) 3.904 + { 3.905 + gbMemory[address] = value; 3.906 + return; 3.907 + } 3.908 + 3.909 + switch (address & 0x00ff) 3.910 + { 3.911 + case 0x00: 3.912 + { 3.913 + gbMemory[0xff00] = ((gbMemory[0xff00] & 0xcf) | 3.914 + (value & 0x30)); 3.915 + if (gbSgbMode) 3.916 + { 3.917 + gbSgbDoBitTransfer(value); 3.918 + } 3.919 + 3.920 + return; 3.921 + } 3.922 + 3.923 + case 0x01: 3.924 + { 3.925 + gbMemory[0xff01] = value; 3.926 + return; 3.927 + } 3.928 + 3.929 + // serial control 3.930 + case 0x02: 3.931 + { 3.932 + gbSerialOn = (value & 0x80); 3.933 + gbMemory[0xff02] = value; 3.934 + if (gbSerialOn) 3.935 + { 3.936 + gbSerialTicks = GBSERIAL_CLOCK_TICKS; 3.937 +#ifdef LINK_EMULATION 3.938 + if (linkConnected) 3.939 + { 3.940 + if (value & 1) 3.941 + { 3.942 + linkSendByte(0x100 | gbMemory[0xFF01]); 3.943 + Sleep(5); 3.944 + } 3.945 + } 3.946 +#endif 3.947 + } 3.948 + 3.949 + gbSerialBits = 0; 3.950 + return; 3.951 + } 3.952 + 3.953 + // DIV register resets on any write 3.954 + case 0x04: 3.955 + { 3.956 + register_DIV = 0; 3.957 + return; 3.958 + } 3.959 + case 0x05: 3.960 + register_TIMA = value; 3.961 + return; 3.962 + 3.963 + case 0x06: 3.964 + register_TMA = value; 3.965 + return; 3.966 + 3.967 + // TIMER control 3.968 + case 0x07: 3.969 + { 3.970 + register_TAC = value; 3.971 + 3.972 + gbTimerOn = (value & 4); 3.973 + gbTimerMode = value & 3; 3.974 + // register_TIMA = register_TMA; 3.975 + switch (gbTimerMode) 3.976 + { 3.977 + case 0: 3.978 + gbTimerClockTicks = gbTimerTicks = GBTIMER_MODE_0_CLOCK_TICKS; 3.979 + break; 3.980 + case 1: 3.981 + gbTimerClockTicks = gbTimerTicks = GBTIMER_MODE_1_CLOCK_TICKS; 3.982 + break; 3.983 + case 2: 3.984 + gbTimerClockTicks = gbTimerTicks = GBTIMER_MODE_2_CLOCK_TICKS; 3.985 + break; 3.986 + case 3: 3.987 + gbTimerClockTicks = gbTimerTicks = GBTIMER_MODE_3_CLOCK_TICKS; 3.988 + break; 3.989 + } 3.990 + return; 3.991 + } 3.992 + 3.993 + case 0x0f: 3.994 + { 3.995 + register_IF = value; 3.996 + gbInterrupt = value; 3.997 + return; 3.998 + } 3.999 + 3.1000 + case 0x10: 3.1001 + case 0x11: 3.1002 + case 0x12: 3.1003 + case 0x13: 3.1004 + case 0x14: 3.1005 + case 0x15: 3.1006 + case 0x16: 3.1007 + case 0x17: 3.1008 + case 0x18: 3.1009 + case 0x19: 3.1010 + case 0x1a: 3.1011 + case 0x1b: 3.1012 + case 0x1c: 3.1013 + case 0x1d: 3.1014 + case 0x1e: 3.1015 + case 0x1f: 3.1016 + case 0x20: 3.1017 + case 0x21: 3.1018 + case 0x22: 3.1019 + case 0x23: 3.1020 + case 0x24: 3.1021 + case 0x25: 3.1022 + case 0x26: 3.1023 + { 3.1024 + SOUND_EVENT(address, value); 3.1025 + return; 3.1026 + } 3.1027 + case 0x40: 3.1028 + { 3.1029 + int lcdChange = (register_LCDC & 0x80) ^ (value & 0x80); 3.1030 + 3.1031 + if (lcdChange) 3.1032 + { 3.1033 + if (value & 0x80) 3.1034 + { 3.1035 + gbLcdTicks = GBLCD_MODE_1_CLOCK_TICKS; 3.1036 + gbLcdMode = 0; 3.1037 + register_STAT &= 0xfc; 3.1038 + register_LY = 0x00; 3.1039 + // FIXME: horrible workaround 3.1040 + if (gbNullInputHackTempEnabled && !useOldFrameTiming) 3.1041 + memcpy(gbJoymask, s_gbJoymask, sizeof(gbJoymask)); 3.1042 + } 3.1043 + else 3.1044 + { 3.1045 + gbLcdTicks = 0; 3.1046 + gbLcdMode = 0; 3.1047 + register_STAT &= 0xfc; 3.1048 + register_LY = 0x00; 3.1049 + // FIXME: horrible workaround 3.1050 + memcpy(s_gbJoymask, gbJoymask, sizeof(gbJoymask)); 3.1051 + if (gbNullInputHackTempEnabled && !useOldFrameTiming) 3.1052 + memset(gbJoymask, 0, sizeof(gbJoymask)); 3.1053 + } 3.1054 + // compareLYToLYC(); 3.1055 + } 3.1056 + // don't draw the window if it was not enabled and not being drawn before 3.1057 + if (!(register_LCDC & 0x20) && (value & 0x20) && gbWindowLine == -1 && 3.1058 + register_LY > register_WY) 3.1059 + gbWindowLine = 144; 3.1060 + 3.1061 + register_LCDC = value; 3.1062 + 3.1063 + return; 3.1064 + } 3.1065 + 3.1066 + // STAT 3.1067 + case 0x41: 3.1068 + { 3.1069 + //register_STAT = (register_STAT & 0x87) | 3.1070 + // (value & 0x7c); 3.1071 + register_STAT = (value & 0xf8) | (register_STAT & 0x07); // fix ? 3.1072 + // GB bug from Devrs FAQ 3.1073 + if (!gbCgbMode && (register_LCDC & 0x80) && gbLcdMode < 2) 3.1074 + gbInterrupt |= 2; 3.1075 + return; 3.1076 + } 3.1077 + 3.1078 + // SCY 3.1079 + case 0x42: 3.1080 + { 3.1081 + register_SCY = value; 3.1082 + return; 3.1083 + } 3.1084 + 3.1085 + // SCX 3.1086 + case 0x43: 3.1087 + { 3.1088 + register_SCX = value; 3.1089 + return; 3.1090 + } 3.1091 + 3.1092 + // LY 3.1093 + case 0x44: 3.1094 + { 3.1095 + // read only 3.1096 + return; 3.1097 + } 3.1098 + 3.1099 + // LYC 3.1100 + case 0x45: 3.1101 + { 3.1102 + register_LYC = value; 3.1103 + if ((register_LCDC & 0x80)) 3.1104 + { 3.1105 + gbCompareLYToLYC(); 3.1106 + } 3.1107 + return; 3.1108 + } 3.1109 + 3.1110 + // DMA! 3.1111 + case 0x46: 3.1112 + { 3.1113 + int source = value * 0x0100; 3.1114 + 3.1115 + gbCopyMemory(0xfe00, 3.1116 + source, 3.1117 + 0xa0); 3.1118 + register_DMA = value; 3.1119 + return; 3.1120 + } 3.1121 + 3.1122 + // BGP 3.1123 + case 0x47: 3.1124 + { 3.1125 + gbBgp[0] = value & 0x03; 3.1126 + gbBgp[1] = (value & 0x0c) >> 2; 3.1127 + gbBgp[2] = (value & 0x30) >> 4; 3.1128 + gbBgp[3] = (value & 0xc0) >> 6; 3.1129 + break; 3.1130 + } 3.1131 + 3.1132 + // OBP0 3.1133 + case 0x48: 3.1134 + { 3.1135 + gbObp0[0] = value & 0x03; 3.1136 + gbObp0[1] = (value & 0x0c) >> 2; 3.1137 + gbObp0[2] = (value & 0x30) >> 4; 3.1138 + gbObp0[3] = (value & 0xc0) >> 6; 3.1139 + break; 3.1140 + } 3.1141 + 3.1142 + // OBP1 3.1143 + case 0x49: 3.1144 + { 3.1145 + gbObp1[0] = value & 0x03; 3.1146 + gbObp1[1] = (value & 0x0c) >> 2; 3.1147 + gbObp1[2] = (value & 0x30) >> 4; 3.1148 + gbObp1[3] = (value & 0xc0) >> 6; 3.1149 + break; 3.1150 + } 3.1151 + 3.1152 + case 0x4a: 3.1153 + register_WY = value; 3.1154 + return; 3.1155 + 3.1156 + case 0x4b: 3.1157 + register_WX = value; 3.1158 + return; 3.1159 + 3.1160 + // KEY1 3.1161 + case 0x4d: 3.1162 + { 3.1163 + if (gbCgbMode) 3.1164 + { 3.1165 + gbMemory[0xff4d] = (gbMemory[0xff4d] & 0x80) | (value & 1); 3.1166 + return; 3.1167 + } 3.1168 + break; 3.1169 + } 3.1170 + 3.1171 + // VBK 3.1172 + case 0x4f: 3.1173 + { 3.1174 + if (gbCgbMode) 3.1175 + { 3.1176 + value = value & 1; 3.1177 + if (value == gbVramBank) 3.1178 + return; 3.1179 + 3.1180 + int vramAddress = value * 0x2000; 3.1181 + gbMemoryMap[0x08] = &gbVram[vramAddress]; 3.1182 + gbMemoryMap[0x09] = &gbVram[vramAddress + 0x1000]; 3.1183 + 3.1184 + gbVramBank = value; 3.1185 + register_VBK = value; 3.1186 + } 3.1187 + return; 3.1188 + break; 3.1189 + } 3.1190 + 3.1191 + // HDMA1 3.1192 + case 0x51: 3.1193 + { 3.1194 + if (gbCgbMode) 3.1195 + { 3.1196 + if (value > 0x7f && value < 0xa0) 3.1197 + value = 0; 3.1198 + 3.1199 + gbHdmaSource = (value << 8) | (register_HDMA2 & 0xf0); 3.1200 + 3.1201 + register_HDMA1 = value; 3.1202 + return; 3.1203 + } 3.1204 + break; 3.1205 + } 3.1206 + 3.1207 + // HDMA2 3.1208 + case 0x52: 3.1209 + { 3.1210 + if (gbCgbMode) 3.1211 + { 3.1212 + value = value & 0xf0; 3.1213 + 3.1214 + gbHdmaSource = (register_HDMA1 << 8) | (value); 3.1215 + 3.1216 + register_HDMA2 = value; 3.1217 + return; 3.1218 + } 3.1219 + break; 3.1220 + } 3.1221 + 3.1222 + // HDMA3 3.1223 + case 0x53: 3.1224 + { 3.1225 + if (gbCgbMode) 3.1226 + { 3.1227 + value = value & 0x1f; 3.1228 + gbHdmaDestination = (value << 8) | (register_HDMA4 & 0xf0); 3.1229 + gbHdmaDestination += 0x8000; 3.1230 + register_HDMA3 = value; 3.1231 + return; 3.1232 + } 3.1233 + break; 3.1234 + } 3.1235 + 3.1236 + // HDMA4 3.1237 + case 0x54: 3.1238 + { 3.1239 + if (gbCgbMode) 3.1240 + { 3.1241 + value = value & 0xf0; 3.1242 + gbHdmaDestination = ((register_HDMA3 & 0x1f) << 8) | value; 3.1243 + gbHdmaDestination += 0x8000; 3.1244 + register_HDMA4 = value; 3.1245 + return; 3.1246 + } 3.1247 + break; 3.1248 + } 3.1249 + 3.1250 + // HDMA5 3.1251 + case 0x55: 3.1252 + { 3.1253 + if (gbCgbMode) 3.1254 + { 3.1255 + gbHdmaBytes = 16 + (value & 0x7f) * 16; 3.1256 + if (gbHdmaOn) 3.1257 + { 3.1258 + if (value & 0x80) 3.1259 + { 3.1260 + register_HDMA5 = (value & 0x7f); 3.1261 + } 3.1262 + else 3.1263 + { 3.1264 + register_HDMA5 = 0xff; 3.1265 + gbHdmaOn = 0; 3.1266 + } 3.1267 + } 3.1268 + else 3.1269 + { 3.1270 + if (value & 0x80) 3.1271 + { 3.1272 + gbHdmaOn = 1; 3.1273 + register_HDMA5 = value & 0x7f; 3.1274 + if (gbLcdMode == 0) 3.1275 + gbDoHdma(); 3.1276 + } 3.1277 + else 3.1278 + { 3.1279 + // we need to take the time it takes to complete the transfer into 3.1280 + // account... according to GB DEV FAQs, the setup time is the same 3.1281 + // for single and double speed, but the actual transfer takes the 3.1282 + // same time // (is that a typo?) 3.1283 + switch (gbDMASpeedVersion) 3.1284 + { 3.1285 + case 1: // I believe this is more correct 3.1286 + // the lower 7 bits of FF55 specify the Transfer Length (divided by 16, minus 1) 3.1287 + // and we make gbDmaTicks twice as many cycles at double speed to make the transfer take the same time 3.1288 + if (gbSpeed) 3.1289 + gbDmaTicks = 16 * ((value & 0x7f) + 1); 3.1290 + else 3.1291 + gbDmaTicks = 8 * ((value & 0x7f) + 1); 3.1292 + break; 3.1293 + case 0: // here for backward compatibility 3.1294 + // I think this was a guess that approximates the above in most but not all games 3.1295 + if (gbSpeed) 3.1296 + gbDmaTicks = 231 + 16 * (value & 0x7f); 3.1297 + else 3.1298 + gbDmaTicks = 231 + 8 * (value & 0x7f); 3.1299 + break; 3.1300 + default: // shouldn't happen 3.1301 + //assert(0); 3.1302 + break; 3.1303 + } 3.1304 + gbCopyMemory(gbHdmaDestination, gbHdmaSource, gbHdmaBytes); 3.1305 + gbHdmaDestination += gbHdmaBytes; 3.1306 + gbHdmaSource += gbHdmaBytes; 3.1307 + 3.1308 + register_HDMA3 = ((gbHdmaDestination - 0x8000) >> 8) & 0x1f; 3.1309 + register_HDMA4 = gbHdmaDestination & 0xf0; 3.1310 + register_HDMA1 = (gbHdmaSource >> 8) & 0xff; 3.1311 + register_HDMA2 = gbHdmaSource & 0xf0; 3.1312 + } 3.1313 + } 3.1314 + return; 3.1315 + } 3.1316 + break; 3.1317 + } 3.1318 + 3.1319 + // BCPS 3.1320 + case 0x68: 3.1321 + { 3.1322 + if (gbCgbMode) 3.1323 + { 3.1324 + int paletteIndex = (value & 0x3f) >> 1; 3.1325 + int paletteHiLo = (value & 0x01); 3.1326 + 3.1327 + gbMemory[0xff68] = value; 3.1328 + gbMemory[0xff69] = (paletteHiLo ? 3.1329 + (gbPalette[paletteIndex] >> 8) : 3.1330 + (gbPalette[paletteIndex] & 0x00ff)); 3.1331 + return; 3.1332 + } 3.1333 + break; 3.1334 + } 3.1335 + 3.1336 + // BCPD 3.1337 + case 0x69: 3.1338 + { 3.1339 + if (gbCgbMode) 3.1340 + { 3.1341 + int v = gbMemory[0xff68]; 3.1342 + int paletteIndex = (v & 0x3f) >> 1; 3.1343 + int paletteHiLo = (v & 0x01); 3.1344 + gbMemory[0xff69] = value; 3.1345 + gbPalette[paletteIndex] = (paletteHiLo ? 3.1346 + ((value << 8) | (gbPalette[paletteIndex] & 0xff)) : 3.1347 + ((gbPalette[paletteIndex] & 0xff00) | (value))) & 0x7fff; 3.1348 + 3.1349 + if (gbMemory[0xff68] & 0x80) 3.1350 + { 3.1351 + int index = ((gbMemory[0xff68] & 0x3f) + 1) & 0x3f; 3.1352 + 3.1353 + gbMemory[0xff68] = (gbMemory[0xff68] & 0x80) | index; 3.1354 + 3.1355 + gbMemory[0xff69] = (index & 1 ? 3.1356 + (gbPalette[index >> 1] >> 8) : 3.1357 + (gbPalette[index >> 1] & 0x00ff)); 3.1358 + } 3.1359 + return; 3.1360 + } 3.1361 + break; 3.1362 + } 3.1363 + 3.1364 + // OCPS 3.1365 + case 0x6a: 3.1366 + { 3.1367 + if (gbCgbMode) 3.1368 + { 3.1369 + int paletteIndex = (value & 0x3f) >> 1; 3.1370 + int paletteHiLo = (value & 0x01); 3.1371 + 3.1372 + paletteIndex += 32; 3.1373 + 3.1374 + gbMemory[0xff6a] = value; 3.1375 + gbMemory[0xff6b] = (paletteHiLo ? 3.1376 + (gbPalette[paletteIndex] >> 8) : 3.1377 + (gbPalette[paletteIndex] & 0x00ff)); 3.1378 + return; 3.1379 + } 3.1380 + break; 3.1381 + } 3.1382 + 3.1383 + // OCPD 3.1384 + case 0x6b: 3.1385 + { 3.1386 + if (gbCgbMode) 3.1387 + { 3.1388 + int v = gbMemory[0xff6a]; 3.1389 + int paletteIndex = (v & 0x3f) >> 1; 3.1390 + int paletteHiLo = (v & 0x01); 3.1391 + 3.1392 + paletteIndex += 32; 3.1393 + 3.1394 + gbMemory[0xff6b] = value; 3.1395 + gbPalette[paletteIndex] = (paletteHiLo ? 3.1396 + ((value << 8) | (gbPalette[paletteIndex] & 0xff)) : 3.1397 + ((gbPalette[paletteIndex] & 0xff00) | (value))) & 0x7fff; 3.1398 + if (gbMemory[0xff6a] & 0x80) 3.1399 + { 3.1400 + int index = ((gbMemory[0xff6a] & 0x3f) + 1) & 0x3f; 3.1401 + 3.1402 + gbMemory[0xff6a] = (gbMemory[0xff6a] & 0x80) | index; 3.1403 + 3.1404 + gbMemory[0xff6b] = (index & 1 ? 3.1405 + (gbPalette[(index >> 1) + 32] >> 8) : 3.1406 + (gbPalette[(index >> 1) + 32] & 0x00ff)); 3.1407 + } 3.1408 + return; 3.1409 + } 3.1410 + break; 3.1411 + } 3.1412 + 3.1413 + // SVBK 3.1414 + case 0x70: 3.1415 + { 3.1416 + if (gbCgbMode) 3.1417 + { 3.1418 + value = value & 7; 3.1419 + 3.1420 + int bank = value; 3.1421 + if (value == 0) 3.1422 + bank = 1; 3.1423 + 3.1424 + if (bank == gbWramBank) 3.1425 + return; 3.1426 + 3.1427 + int wramAddress = bank * 0x1000; 3.1428 + gbMemoryMap[0x0d] = &gbWram[wramAddress]; 3.1429 + 3.1430 + gbWramBank = bank; 3.1431 + register_SVBK = value; 3.1432 + return; 3.1433 + } 3.1434 + break; 3.1435 + } 3.1436 + 3.1437 + case 0xff: 3.1438 + { 3.1439 + register_IE = value; 3.1440 + register_IF &= value; 3.1441 + return; 3.1442 + } 3.1443 + } 3.1444 + 3.1445 + gbWriteMemoryQuick(address, value); 3.1446 +} 3.1447 + 3.1448 +u8 gbReadOpcode(register u16 address) 3.1449 +{ 3.1450 + if (gbCheatMap[address]) 3.1451 + return gbCheatRead(address); 3.1452 + 3.1453 + // the following fix does more than Echo RAM fix, anyway... 3.1454 + switch (gbEchoRAMFixOn ? (address >> 12) & 0x000f : address & 0xf000) 3.1455 + { 3.1456 + case 0x0a: 3.1457 + case 0x0b: 3.1458 + if (mapperReadRAM) 3.1459 + return mapperReadRAM(address); 3.1460 + break; 3.1461 + case 0x0f: 3.1462 + if (address > 0xff00) 3.1463 + { 3.1464 + switch (address & 0x00ff) 3.1465 + { 3.1466 + case 0x04: 3.1467 + return register_DIV; 3.1468 + case 0x05: 3.1469 + return register_TIMA; 3.1470 + case 0x06: 3.1471 + return register_TMA; 3.1472 + case 0x07: 3.1473 + return (0xf8 | register_TAC); 3.1474 + case 0x0f: 3.1475 + return (0xe0 | register_IF); 3.1476 + case 0x40: 3.1477 + return register_LCDC; 3.1478 + case 0x41: 3.1479 + return (0x80 | register_STAT); 3.1480 + case 0x42: 3.1481 + return register_SCY; 3.1482 + case 0x43: 3.1483 + return register_SCX; 3.1484 + case 0x44: 3.1485 + return register_LY; 3.1486 + case 0x45: 3.1487 + return register_LYC; 3.1488 + case 0x46: 3.1489 + return register_DMA; 3.1490 + case 0x4a: 3.1491 + return register_WY; 3.1492 + case 0x4b: 3.1493 + return register_WX; 3.1494 + case 0x4f: 3.1495 + return (0xfe | register_VBK); 3.1496 + case 0x51: 3.1497 + return register_HDMA1; 3.1498 + case 0x52: 3.1499 + return register_HDMA2; 3.1500 + case 0x53: 3.1501 + return register_HDMA3; 3.1502 + case 0x54: 3.1503 + return register_HDMA4; 3.1504 + case 0x55: 3.1505 + return register_HDMA5; 3.1506 + case 0x70: 3.1507 + return (0xf8 | register_SVBK); 3.1508 + case 0xff: 3.1509 + return register_IE; 3.1510 + } 3.1511 } 3.1512 - 3.1513 - if (address < 0xa000) 3.1514 + break; 3.1515 + } 3.1516 + return gbReadMemoryQuick(address); 3.1517 +} 3.1518 + 3.1519 +void gbWriteMemory(register u16 address, register u8 value) 3.1520 +{ 3.1521 + gbWriteMemoryWrapped(address, value); 3.1522 + CallRegisteredLuaMemHook(address, 1, value, LUAMEMHOOK_WRITE); 3.1523 +} 3.1524 + 3.1525 +u8 gbReadMemory(register u16 address) 3.1526 +{ 3.1527 + if (gbCheatMap[address]) 3.1528 + return gbCheatRead(address); 3.1529 + 3.1530 + if (address < 0xa000) 3.1531 + return gbReadMemoryQuick(address); 3.1532 + 3.1533 + if (address < 0xc000) 3.1534 + { 3.1535 +#ifndef FINAL_VERSION 3.1536 + if (memorydebug) 3.1537 { 3.1538 - gbWriteMemoryQuick(address, value); 3.1539 - return; 3.1540 + log("Memory register read %04x PC=%04x\n", 3.1541 + address, 3.1542 + PC.W); 3.1543 } 3.1544 - 3.1545 - if (address < 0xc000) 3.1546 - { 3.1547 -#ifndef FINAL_VERSION 3.1548 - if (memorydebug) 3.1549 - { 3.1550 - log("Memory register write %04x=%02x PC=%04x\n", 3.1551 - address, 3.1552 - value, 3.1553 - PC.W); 3.1554 - } 3.1555 #endif 3.1556 3.1557 - if (mapper) 3.1558 - (*mapperRAM)(address, value); 3.1559 - return; 3.1560 - } 3.1561 - 3.1562 - if (address < 0xfe00) 3.1563 - { 3.1564 - gbWriteMemoryQuick(address, value); 3.1565 - return; 3.1566 - } 3.1567 - 3.1568 - if (address < 0xff00) 3.1569 - { 3.1570 - gbMemory[address] = value; 3.1571 - return; 3.1572 - } 3.1573 - 3.1574 - switch (address & 0x00ff) 3.1575 + if (mapperReadRAM) 3.1576 + return mapperReadRAM(address); 3.1577 + return gbReadMemoryQuick(address); 3.1578 + } 3.1579 + 3.1580 + if (address >= 0xff00) 3.1581 + { 3.1582 + switch (address & 0x00ff) 3.1583 { 3.1584 case 0x00: 3.1585 - { 3.1586 - gbMemory[0xff00] = ((gbMemory[0xff00] & 0xcf) | 3.1587 - (value & 0x30)); 3.1588 - if (gbSgbMode) 3.1589 - { 3.1590 - gbSgbDoBitTransfer(value); 3.1591 - } 3.1592 - 3.1593 - return; 3.1594 + { 3.1595 + if (gbSgbMode) 3.1596 + { 3.1597 + gbSgbReadingController |= 4; 3.1598 + gbSgbResetPacketState(); 3.1599 + } 3.1600 + 3.1601 + int b = gbMemory[0xff00]; 3.1602 + 3.1603 + if ((b & 0x30) == 0x20) 3.1604 + { 3.1605 + b &= 0xf0; 3.1606 + 3.1607 + int joy = 0; 3.1608 + if (gbSgbMode && gbSgbMultiplayer) 3.1609 + { 3.1610 + switch (gbSgbNextController) 3.1611 + { 3.1612 + case 0x0f: 3.1613 + joy = 0; 3.1614 + break; 3.1615 + case 0x0e: 3.1616 + joy = 1; 3.1617 + break; 3.1618 + case 0x0d: 3.1619 + joy = 2; 3.1620 + break; 3.1621 + case 0x0c: 3.1622 + joy = 3; 3.1623 + break; 3.1624 + default: 3.1625 + joy = 0; 3.1626 + } 3.1627 + } 3.1628 + int joystate = gbJoymask[joy]; 3.1629 + if (!(joystate & 128)) 3.1630 + b |= 0x08; 3.1631 + if (!(joystate & 64)) 3.1632 + b |= 0x04; 3.1633 + if (!(joystate & 32)) 3.1634 + b |= 0x02; 3.1635 + if (!(joystate & 16)) 3.1636 + b |= 0x01; 3.1637 + 3.1638 + gbMemory[0xff00] = b; 3.1639 + } 3.1640 + else if ((b & 0x30) == 0x10) 3.1641 + { 3.1642 + b &= 0xf0; 3.1643 + 3.1644 + int joy = 0; 3.1645 + if (gbSgbMode && gbSgbMultiplayer) 3.1646 + { 3.1647 + switch (gbSgbNextController) 3.1648 + { 3.1649 + case 0x0f: 3.1650 + joy = 0; 3.1651 + break; 3.1652 + case 0x0e: 3.1653 + joy = 1; 3.1654 + break; 3.1655 + case 0x0d: 3.1656 + joy = 2; 3.1657 + break; 3.1658 + case 0x0c: 3.1659 + joy = 3; 3.1660 + break; 3.1661 + default: 3.1662 + joy = 0; 3.1663 + } 3.1664 + } 3.1665 + int joystate = gbJoymask[joy]; 3.1666 + if (!(joystate & 8)) 3.1667 + b |= 0x08; 3.1668 + if (!(joystate & 4)) 3.1669 + b |= 0x04; 3.1670 + if (!(joystate & 2)) 3.1671 + b |= 0x02; 3.1672 + if (!(joystate & 1)) 3.1673 + b |= 0x01; 3.1674 + 3.1675 + gbMemory[0xff00] = b; 3.1676 + } 3.1677 + else 3.1678 + { 3.1679 + if (gbSgbMode && gbSgbMultiplayer) 3.1680 + { 3.1681 + gbMemory[0xff00] = 0xf0 | gbSgbNextController; 3.1682 + } 3.1683 + else 3.1684 + { 3.1685 + gbMemory[0xff00] = 0xff; 3.1686 + } 3.1687 + } 3.1688 + } 3.1689 + GBSystemCounters.lagged = false; 3.1690 + return gbMemory[0xff00]; 3.1691 + break; 3.1692 + case 0x01: 3.1693 + return gbMemory[0xff01]; 3.1694 + case 0x04: 3.1695 + return register_DIV; 3.1696 + case 0x05: 3.1697 + return register_TIMA; 3.1698 + case 0x06: 3.1699 + return register_TMA; 3.1700 + case 0x07: 3.1701 + return (0xf8 | register_TAC); 3.1702 + case 0x0f: 3.1703 + return (0xe0 | register_IF); 3.1704 + case 0x40: 3.1705 + return register_LCDC; 3.1706 + case 0x41: 3.1707 + return (0x80 | register_STAT); 3.1708 + case 0x42: 3.1709 + return register_SCY; 3.1710 + case 0x43: 3.1711 + return register_SCX; 3.1712 + case 0x44: 3.1713 + return register_LY; 3.1714 + case 0x45: 3.1715 + return register_LYC; 3.1716 + case 0x46: 3.1717 + return register_DMA; 3.1718 + case 0x4a: 3.1719 + return register_WY; 3.1720 + case 0x4b: 3.1721 + return register_WX; 3.1722 + case 0x4f: 3.1723 + return (0xfe | register_VBK); 3.1724 + case 0x51: 3.1725 + return register_HDMA1; 3.1726 + case 0x52: 3.1727 + return register_HDMA2; 3.1728 + case 0x53: 3.1729 + return register_HDMA3; 3.1730 + case 0x54: 3.1731 + return register_HDMA4; 3.1732 + case 0x55: 3.1733 + return register_HDMA5; 3.1734 + case 0x70: 3.1735 + return (0xf8 | register_SVBK); 3.1736 + case 0xff: 3.1737 + return register_IE; 3.1738 } 3.1739 - 3.1740 - case 0x01: 3.1741 - { 3.1742 - gbMemory[0xff01] = value; 3.1743 - return; 3.1744 - } 3.1745 - 3.1746 - // serial control 3.1747 - case 0x02: 3.1748 - { 3.1749 - gbSerialOn = (value & 0x80); 3.1750 - gbMemory[0xff02] = value; 3.1751 - if (gbSerialOn) 3.1752 - { 3.1753 - gbSerialTicks = GBSERIAL_CLOCK_TICKS; 3.1754 -#ifdef LINK_EMULATION 3.1755 - if (linkConnected) 3.1756 - { 3.1757 - if (value & 1) 3.1758 - { 3.1759 - linkSendByte(0x100 | gbMemory[0xFF01]); 3.1760 - Sleep(5); 3.1761 - } 3.1762 - } 3.1763 -#endif 3.1764 - } 3.1765 - 3.1766 - gbSerialBits = 0; 3.1767 - return; 3.1768 - } 3.1769 - 3.1770 - // DIV register resets on any write 3.1771 - case 0x04: 3.1772 - { 3.1773 - register_DIV = 0; 3.1774 - return; 3.1775 - } 3.1776 - case 0x05: 3.1777 - register_TIMA = value; 3.1778 - return; 3.1779 - 3.1780 - case 0x06: 3.1781 - register_TMA = value; 3.1782 - return; 3.1783 - 3.1784 - // TIMER control 3.1785 - case 0x07: 3.1786 - { 3.1787 - register_TAC = value; 3.1788 - 3.1789 - gbTimerOn = (value & 4); 3.1790 - gbTimerMode = value & 3; 3.1791 - // register_TIMA = register_TMA; 3.1792 - switch (gbTimerMode) 3.1793 - { 3.1794 - case 0: 3.1795 - gbTimerClockTicks = gbTimerTicks = GBTIMER_MODE_0_CLOCK_TICKS; 3.1796 - break; 3.1797 - case 1: 3.1798 - gbTimerClockTicks = gbTimerTicks = GBTIMER_MODE_1_CLOCK_TICKS; 3.1799 - break; 3.1800 - case 2: 3.1801 - gbTimerClockTicks = gbTimerTicks = GBTIMER_MODE_2_CLOCK_TICKS; 3.1802 - break; 3.1803 - case 3: 3.1804 - gbTimerClockTicks = gbTimerTicks = GBTIMER_MODE_3_CLOCK_TICKS; 3.1805 - break; 3.1806 - } 3.1807 - return; 3.1808 - } 3.1809 - 3.1810 - case 0x0f: 3.1811 - { 3.1812 - register_IF = value; 3.1813 - gbInterrupt = value; 3.1814 - return; 3.1815 - } 3.1816 - 3.1817 - case 0x10: 3.1818 - case 0x11: 3.1819 - case 0x12: 3.1820 - case 0x13: 3.1821 - case 0x14: 3.1822 - case 0x15: 3.1823 - case 0x16: 3.1824 - case 0x17: 3.1825 - case 0x18: 3.1826 - case 0x19: 3.1827 - case 0x1a: 3.1828 - case 0x1b: 3.1829 - case 0x1c: 3.1830 - case 0x1d: 3.1831 - case 0x1e: 3.1832 - case 0x1f: 3.1833 - case 0x20: 3.1834 - case 0x21: 3.1835 - case 0x22: 3.1836 - case 0x23: 3.1837 - case 0x24: 3.1838 - case 0x25: 3.1839 - case 0x26: 3.1840 - { 3.1841 - SOUND_EVENT(address, value); 3.1842 - return; 3.1843 - } 3.1844 - case 0x40: 3.1845 - { 3.1846 - int lcdChange = (register_LCDC & 0x80) ^ (value & 0x80); 3.1847 - 3.1848 - if (lcdChange) 3.1849 - { 3.1850 - if (value & 0x80) 3.1851 - { 3.1852 - gbLcdTicks = GBLCD_MODE_1_CLOCK_TICKS; 3.1853 - gbLcdMode = 0; 3.1854 - register_STAT &= 0xfc; 3.1855 - register_LY = 0x00; 3.1856 - // FIXME: horrible workaround 3.1857 - if (gbNullInputHackTempEnabled && !useOldFrameTiming) 3.1858 - memcpy(gbJoymask, s_gbJoymask, sizeof(gbJoymask)); 3.1859 - } 3.1860 - else 3.1861 - { 3.1862 - gbLcdTicks = 0; 3.1863 - gbLcdMode = 0; 3.1864 - register_STAT &= 0xfc; 3.1865 - register_LY = 0x00; 3.1866 - // FIXME: horrible workaround 3.1867 - memcpy(s_gbJoymask, gbJoymask, sizeof(gbJoymask)); 3.1868 - if (gbNullInputHackTempEnabled && !useOldFrameTiming) 3.1869 - memset(gbJoymask, 0, sizeof(gbJoymask)); 3.1870 - } 3.1871 - // compareLYToLYC(); 3.1872 - } 3.1873 - // don't draw the window if it was not enabled and not being drawn before 3.1874 - if (!(register_LCDC & 0x20) && (value & 0x20) && gbWindowLine == -1 && 3.1875 - register_LY > register_WY) 3.1876 - gbWindowLine = 144; 3.1877 - 3.1878 - register_LCDC = value; 3.1879 - 3.1880 - return; 3.1881 - } 3.1882 - 3.1883 - // STAT 3.1884 - case 0x41: 3.1885 - { 3.1886 - //register_STAT = (register_STAT & 0x87) | 3.1887 - // (value & 0x7c); 3.1888 - register_STAT = (value & 0xf8) | (register_STAT & 0x07); // fix ? 3.1889 - // GB bug from Devrs FAQ 3.1890 - if (!gbCgbMode && (register_LCDC & 0x80) && gbLcdMode < 2) 3.1891 - gbInterrupt |= 2; 3.1892 - return; 3.1893 - } 3.1894 - 3.1895 - // SCY 3.1896 - case 0x42: 3.1897 - { 3.1898 - register_SCY = value; 3.1899 - return; 3.1900 - } 3.1901 - 3.1902 - // SCX 3.1903 - case 0x43: 3.1904 - { 3.1905 - register_SCX = value; 3.1906 - return; 3.1907 - } 3.1908 - 3.1909 - // LY 3.1910 - case 0x44: 3.1911 - { 3.1912 - // read only 3.1913 - return; 3.1914 - } 3.1915 - 3.1916 - // LYC 3.1917 - case 0x45: 3.1918 - { 3.1919 - register_LYC = value; 3.1920 - if ((register_LCDC & 0x80)) 3.1921 - { 3.1922 - gbCompareLYToLYC(); 3.1923 - } 3.1924 - return; 3.1925 - } 3.1926 - 3.1927 - // DMA! 3.1928 - case 0x46: 3.1929 - { 3.1930 - int source = value * 0x0100; 3.1931 - 3.1932 - gbCopyMemory(0xfe00, 3.1933 - source, 3.1934 - 0xa0); 3.1935 - register_DMA = value; 3.1936 - return; 3.1937 - } 3.1938 - 3.1939 - // BGP 3.1940 - case 0x47: 3.1941 - { 3.1942 - gbBgp[0] = value & 0x03; 3.1943 - gbBgp[1] = (value & 0x0c) >> 2; 3.1944 - gbBgp[2] = (value & 0x30) >> 4; 3.1945 - gbBgp[3] = (value & 0xc0) >> 6; 3.1946 - break; 3.1947 - } 3.1948 - 3.1949 - // OBP0 3.1950 - case 0x48: 3.1951 - { 3.1952 - gbObp0[0] = value & 0x03; 3.1953 - gbObp0[1] = (value & 0x0c) >> 2; 3.1954 - gbObp0[2] = (value & 0x30) >> 4; 3.1955 - gbObp0[3] = (value & 0xc0) >> 6; 3.1956 - break; 3.1957 - } 3.1958 - 3.1959 - // OBP1 3.1960 - case 0x49: 3.1961 - { 3.1962 - gbObp1[0] = value & 0x03; 3.1963 - gbObp1[1] = (value & 0x0c) >> 2; 3.1964 - gbObp1[2] = (value & 0x30) >> 4; 3.1965 - gbObp1[3] = (value & 0xc0) >> 6; 3.1966 - break; 3.1967 - } 3.1968 - 3.1969 - case 0x4a: 3.1970 - register_WY = value; 3.1971 - return; 3.1972 - 3.1973 - case 0x4b: 3.1974 - register_WX = value; 3.1975 - return; 3.1976 - 3.1977 - // KEY1 3.1978 - case 0x4d: 3.1979 - { 3.1980 - if (gbCgbMode) 3.1981 - { 3.1982 - gbMemory[0xff4d] = (gbMemory[0xff4d] & 0x80) | (value & 1); 3.1983 - return; 3.1984 - } 3.1985 - break; 3.1986 - } 3.1987 - 3.1988 - // VBK 3.1989 - case 0x4f: 3.1990 - { 3.1991 - if (gbCgbMode) 3.1992 - { 3.1993 - value = value & 1; 3.1994 - if (value == gbVramBank) 3.1995 - return; 3.1996 - 3.1997 - int vramAddress = value * 0x2000; 3.1998 - gbMemoryMap[0x08] = &gbVram[vramAddress]; 3.1999 - gbMemoryMap[0x09] = &gbVram[vramAddress + 0x1000]; 3.2000 - 3.2001 - gbVramBank = value; 3.2002 - register_VBK = value; 3.2003 - } 3.2004 - return; 3.2005 - break; 3.2006 - } 3.2007 - 3.2008 - // HDMA1 3.2009 - case 0x51: 3.2010 - { 3.2011 - if (gbCgbMode) 3.2012 - { 3.2013 - if (value > 0x7f && value < 0xa0) 3.2014 - value = 0; 3.2015 - 3.2016 - gbHdmaSource = (value << 8) | (register_HDMA2 & 0xf0); 3.2017 - 3.2018 - register_HDMA1 = value; 3.2019 - return; 3.2020 - } 3.2021 - break; 3.2022 - } 3.2023 - 3.2024 - // HDMA2 3.2025 - case 0x52: 3.2026 - { 3.2027 - if (gbCgbMode) 3.2028 - { 3.2029 - value = value & 0xf0; 3.2030 - 3.2031 - gbHdmaSource = (register_HDMA1 << 8) | (value); 3.2032 - 3.2033 - register_HDMA2 = value; 3.2034 - return; 3.2035 - } 3.2036 - break; 3.2037 - } 3.2038 - 3.2039 - // HDMA3 3.2040 - case 0x53: 3.2041 - { 3.2042 - if (gbCgbMode) 3.2043 - { 3.2044 - value = value & 0x1f; 3.2045 - gbHdmaDestination = (value << 8) | (register_HDMA4 & 0xf0); 3.2046 - gbHdmaDestination += 0x8000; 3.2047 - register_HDMA3 = value; 3.2048 - return; 3.2049 - } 3.2050 - break; 3.2051 - } 3.2052 - 3.2053 - // HDMA4 3.2054 - case 0x54: 3.2055 - { 3.2056 - if (gbCgbMode) 3.2057 - { 3.2058 - value = value & 0xf0; 3.2059 - gbHdmaDestination = ((register_HDMA3 & 0x1f) << 8) | value; 3.2060 - gbHdmaDestination += 0x8000; 3.2061 - register_HDMA4 = value; 3.2062 - return; 3.2063 - } 3.2064 - break; 3.2065 - } 3.2066 - 3.2067 - // HDMA5 3.2068 - case 0x55: 3.2069 - { 3.2070 - if (gbCgbMode) 3.2071 - { 3.2072 - gbHdmaBytes = 16 + (value & 0x7f) * 16; 3.2073 - if (gbHdmaOn) 3.2074 - { 3.2075 - if (value & 0x80) 3.2076 - { 3.2077 - register_HDMA5 = (value & 0x7f); 3.2078 - } 3.2079 - else 3.2080 - { 3.2081 - register_HDMA5 = 0xff; 3.2082 - gbHdmaOn = 0; 3.2083 - } 3.2084 - } 3.2085 - else 3.2086 - { 3.2087 - if (value & 0x80) 3.2088 - { 3.2089 - gbHdmaOn = 1; 3.2090 - register_HDMA5 = value & 0x7f; 3.2091 - if (gbLcdMode == 0) 3.2092 - gbDoHdma(); 3.2093 - } 3.2094 - else 3.2095 - { 3.2096 - // we need to take the time it takes to complete the transfer into 3.2097 - // account... according to GB DEV FAQs, the setup time is the same 3.2098 - // for single and double speed, but the actual transfer takes the 3.2099 - // same time // (is that a typo?) 3.2100 - switch (gbDMASpeedVersion) 3.2101 - { 3.2102 - case 1: // I believe this is more correct 3.2103 - // the lower 7 bits of FF55 specify the Transfer Length (divided by 16, minus 1) 3.2104 - // and we make gbDmaTicks twice as many cycles at double speed to make the transfer take the same time 3.2105 - if (gbSpeed) 3.2106 - gbDmaTicks = 16 * ((value & 0x7f) + 1); 3.2107 - else 3.2108 - gbDmaTicks = 8 * ((value & 0x7f) + 1); 3.2109 - break; 3.2110 - case 0: // here for backward compatibility 3.2111 - // I think this was a guess that approximates the above in most but not all games 3.2112 - if (gbSpeed) 3.2113 - gbDmaTicks = 231 + 16 * (value & 0x7f); 3.2114 - else 3.2115 - gbDmaTicks = 231 + 8 * (value & 0x7f); 3.2116 - break; 3.2117 - default: // shouldn't happen 3.2118 - //assert(0); 3.2119 - break; 3.2120 - } 3.2121 - gbCopyMemory(gbHdmaDestination, gbHdmaSource, gbHdmaBytes); 3.2122 - gbHdmaDestination += gbHdmaBytes; 3.2123 - gbHdmaSource += gbHdmaBytes; 3.2124 - 3.2125 - register_HDMA3 = ((gbHdmaDestination - 0x8000) >> 8) & 0x1f; 3.2126 - register_HDMA4 = gbHdmaDestination & 0xf0; 3.2127 - register_HDMA1 = (gbHdmaSource >> 8) & 0xff; 3.2128 - register_HDMA2 = gbHdmaSource & 0xf0; 3.2129 - } 3.2130 - } 3.2131 - return; 3.2132 - } 3.2133 - break; 3.2134 - } 3.2135 - 3.2136 - // BCPS 3.2137 - case 0x68: 3.2138 - { 3.2139 - if (gbCgbMode) 3.2140 - { 3.2141 - int paletteIndex = (value & 0x3f) >> 1; 3.2142 - int paletteHiLo = (value & 0x01); 3.2143 - 3.2144 - gbMemory[0xff68] = value; 3.2145 - gbMemory[0xff69] = (paletteHiLo ? 3.2146 - (gbPalette[paletteIndex] >> 8) : 3.2147 - (gbPalette[paletteIndex] & 0x00ff)); 3.2148 - return; 3.2149 - } 3.2150 - break; 3.2151 - } 3.2152 - 3.2153 - // BCPD 3.2154 - case 0x69: 3.2155 - { 3.2156 - if (gbCgbMode) 3.2157 - { 3.2158 - int v = gbMemory[0xff68]; 3.2159 - int paletteIndex = (v & 0x3f) >> 1; 3.2160 - int paletteHiLo = (v & 0x01); 3.2161 - gbMemory[0xff69] = value; 3.2162 - gbPalette[paletteIndex] = (paletteHiLo ? 3.2163 - ((value << 8) | (gbPalette[paletteIndex] & 0xff)) : 3.2164 - ((gbPalette[paletteIndex] & 0xff00) | (value))) & 0x7fff; 3.2165 - 3.2166 - if (gbMemory[0xff68] & 0x80) 3.2167 - { 3.2168 - int index = ((gbMemory[0xff68] & 0x3f) + 1) & 0x3f; 3.2169 - 3.2170 - gbMemory[0xff68] = (gbMemory[0xff68] & 0x80) | index; 3.2171 - 3.2172 - gbMemory[0xff69] = (index & 1 ? 3.2173 - (gbPalette[index >> 1] >> 8) : 3.2174 - (gbPalette[index >> 1] & 0x00ff)); 3.2175 - } 3.2176 - return; 3.2177 - } 3.2178 - break; 3.2179 - } 3.2180 - 3.2181 - // OCPS 3.2182 - case 0x6a: 3.2183 - { 3.2184 - if (gbCgbMode) 3.2185 - { 3.2186 - int paletteIndex = (value & 0x3f) >> 1; 3.2187 - int paletteHiLo = (value & 0x01); 3.2188 - 3.2189 - paletteIndex += 32; 3.2190 - 3.2191 - gbMemory[0xff6a] = value; 3.2192 - gbMemory[0xff6b] = (paletteHiLo ? 3.2193 - (gbPalette[paletteIndex] >> 8) : 3.2194 - (gbPalette[paletteIndex] & 0x00ff)); 3.2195 - return; 3.2196 - } 3.2197 - break; 3.2198 - } 3.2199 - 3.2200 - // OCPD 3.2201 - case 0x6b: 3.2202 - { 3.2203 - if (gbCgbMode) 3.2204 - { 3.2205 - int v = gbMemory[0xff6a]; 3.2206 - int paletteIndex = (v & 0x3f) >> 1; 3.2207 - int paletteHiLo = (v & 0x01); 3.2208 - 3.2209 - paletteIndex += 32; 3.2210 - 3.2211 - gbMemory[0xff6b] = value; 3.2212 - gbPalette[paletteIndex] = (paletteHiLo ? 3.2213 - ((value << 8) | (gbPalette[paletteIndex] & 0xff)) : 3.2214 - ((gbPalette[paletteIndex] & 0xff00) | (value))) & 0x7fff; 3.2215 - if (gbMemory[0xff6a] & 0x80) 3.2216 - { 3.2217 - int index = ((gbMemory[0xff6a] & 0x3f) + 1) & 0x3f; 3.2218 - 3.2219 - gbMemory[0xff6a] = (gbMemory[0xff6a] & 0x80) | index; 3.2220 - 3.2221 - gbMemory[0xff6b] = (index & 1 ? 3.2222 - (gbPalette[(index >> 1) + 32] >> 8) : 3.2223 - (gbPalette[(index >> 1) + 32] & 0x00ff)); 3.2224 - } 3.2225 - return; 3.2226 - } 3.2227 - break; 3.2228 - } 3.2229 - 3.2230 - // SVBK 3.2231 - case 0x70: 3.2232 - { 3.2233 - if (gbCgbMode) 3.2234 - { 3.2235 - value = value & 7; 3.2236 - 3.2237 - int bank = value; 3.2238 - if (value == 0) 3.2239 - bank = 1; 3.2240 - 3.2241 - if (bank == gbWramBank) 3.2242 - return; 3.2243 - 3.2244 - int wramAddress = bank * 0x1000; 3.2245 - gbMemoryMap[0x0d] = &gbWram[wramAddress]; 3.2246 - 3.2247 - gbWramBank = bank; 3.2248 - register_SVBK = value; 3.2249 - return; 3.2250 - } 3.2251 - break; 3.2252 - } 3.2253 - 3.2254 - case 0xff: 3.2255 - { 3.2256 - register_IE = value; 3.2257 - register_IF &= value; 3.2258 - return; 3.2259 - } 3.2260 - } 3.2261 - 3.2262 - gbWriteMemoryQuick(address, value); 3.2263 -} 3.2264 - 3.2265 -u8 gbReadOpcode(register u16 address) 3.2266 -{ 3.2267 - if (gbCheatMap[address]) 3.2268 - return gbCheatRead(address); 3.2269 - 3.2270 - // the following fix does more than Echo RAM fix, anyway... 3.2271 - switch (gbEchoRAMFixOn ? (address >> 12) & 0x000f : address & 0xf000) 3.2272 - { 3.2273 - case 0x0a: 3.2274 - case 0x0b: 3.2275 - if (mapperReadRAM) 3.2276 - return mapperReadRAM(address); 3.2277 - break; 3.2278 - case 0x0f: 3.2279 - if (address > 0xff00) 3.2280 - { 3.2281 - switch (address & 0x00ff) 3.2282 - { 3.2283 - case 0x04: 3.2284 - return register_DIV; 3.2285 - case 0x05: 3.2286 - return register_TIMA; 3.2287 - case 0x06: 3.2288 - return register_TMA; 3.2289 - case 0x07: 3.2290 - return (0xf8 | register_TAC); 3.2291 - case 0x0f: 3.2292 - return (0xe0 | register_IF); 3.2293 - case 0x40: 3.2294 - return register_LCDC; 3.2295 - case 0x41: 3.2296 - return (0x80 | register_STAT); 3.2297 - case 0x42: 3.2298 - return register_SCY; 3.2299 - case 0x43: 3.2300 - return register_SCX; 3.2301 - case 0x44: 3.2302 - return register_LY; 3.2303 - case 0x45: 3.2304 - return register_LYC; 3.2305 - case 0x46: 3.2306 - return register_DMA; 3.2307 - case 0x4a: 3.2308 - return register_WY; 3.2309 - case 0x4b: 3.2310 - return register_WX; 3.2311 - case 0x4f: 3.2312 - return (0xfe | register_VBK); 3.2313 - case 0x51: 3.2314 - return register_HDMA1; 3.2315 - case 0x52: 3.2316 - return register_HDMA2; 3.2317 - case 0x53: 3.2318 - return register_HDMA3; 3.2319 - case 0x54: 3.2320 - return register_HDMA4; 3.2321 - case 0x55: 3.2322 - return register_HDMA5; 3.2323 - case 0x70: 3.2324 - return (0xf8 | register_SVBK); 3.2325 - case 0xff: 3.2326 - return register_IE; 3.2327 - } 3.2328 - } 3.2329 - break; 3.2330 - } 3.2331 - return gbReadMemoryQuick(address); 3.2332 -} 3.2333 - 3.2334 -void gbWriteMemory(register u16 address, register u8 value) 3.2335 -{ 3.2336 - gbWriteMemoryWrapped(address, value); 3.2337 - CallRegisteredLuaMemHook(address, 1, value, LUAMEMHOOK_WRITE); 3.2338 -} 3.2339 - 3.2340 -u8 gbReadMemory(register u16 address) 3.2341 -{ 3.2342 - if (gbCheatMap[address]) 3.2343 - return gbCheatRead(address); 3.2344 - 3.2345 - if (address < 0xa000) 3.2346 - return gbReadMemoryQuick(address); 3.2347 - 3.2348 - if (address < 0xc000) 3.2349 - { 3.2350 -#ifndef FINAL_VERSION 3.2351 - if (memorydebug) 3.2352 - { 3.2353 - log("Memory register read %04x PC=%04x\n", 3.2354 - address, 3.2355 - PC.W); 3.2356 - } 3.2357 -#endif 3.2358 - 3.2359 - if (mapperReadRAM) 3.2360 - return mapperReadRAM(address); 3.2361 - return gbReadMemoryQuick(address); 3.2362 - } 3.2363 - 3.2364 - if (address >= 0xff00) 3.2365 - { 3.2366 - switch (address & 0x00ff) 3.2367 - { 3.2368 - case 0x00: 3.2369 - { 3.2370 - if (gbSgbMode) 3.2371 - { 3.2372 - gbSgbReadingController |= 4; 3.2373 - gbSgbResetPacketState(); 3.2374 - } 3.2375 - 3.2376 - int b = gbMemory[0xff00]; 3.2377 - 3.2378 - if ((b & 0x30) == 0x20) 3.2379 - { 3.2380 - b &= 0xf0; 3.2381 - 3.2382 - int joy = 0; 3.2383 - if (gbSgbMode && gbSgbMultiplayer) 3.2384 - { 3.2385 - switch (gbSgbNextController) 3.2386 - { 3.2387 - case 0x0f: 3.2388 - joy = 0; 3.2389 - break; 3.2390 - case 0x0e: 3.2391 - joy = 1; 3.2392 - break; 3.2393 - case 0x0d: 3.2394 - joy = 2; 3.2395 - break; 3.2396 - case 0x0c: 3.2397 - joy = 3; 3.2398 - break; 3.2399 - default: 3.2400 - joy = 0; 3.2401 - } 3.2402 - } 3.2403 - int joystate = gbJoymask[joy]; 3.2404 - if (!(joystate & 128)) 3.2405 - b |= 0x08; 3.2406 - if (!(joystate & 64)) 3.2407 - b |= 0x04; 3.2408 - if (!(joystate & 32)) 3.2409 - b |= 0x02; 3.2410 - if (!(joystate & 16)) 3.2411 - b |= 0x01; 3.2412 - 3.2413 - gbMemory[0xff00] = b; 3.2414 - } 3.2415 - else if ((b & 0x30) == 0x10) 3.2416 - { 3.2417 - b &= 0xf0; 3.2418 - 3.2419 - int joy = 0; 3.2420 - if (gbSgbMode && gbSgbMultiplayer) 3.2421 - { 3.2422 - switch (gbSgbNextController) 3.2423 - { 3.2424 - case 0x0f: 3.2425 - joy = 0; 3.2426 - break; 3.2427 - case 0x0e: 3.2428 - joy = 1; 3.2429 - break; 3.2430 - case 0x0d: 3.2431 - joy = 2; 3.2432 - break; 3.2433 - case 0x0c: 3.2434 - joy = 3; 3.2435 - break; 3.2436 - default: 3.2437 - joy = 0; 3.2438 - } 3.2439 - } 3.2440 - int joystate = gbJoymask[joy]; 3.2441 - if (!(joystate & 8)) 3.2442 - b |= 0x08; 3.2443 - if (!(joystate & 4)) 3.2444 - b |= 0x04; 3.2445 - if (!(joystate & 2)) 3.2446 - b |= 0x02; 3.2447 - if (!(joystate & 1)) 3.2448 - b |= 0x01; 3.2449 - 3.2450 - gbMemory[0xff00] = b; 3.2451 - } 3.2452 - else 3.2453 - { 3.2454 - if (gbSgbMode && gbSgbMultiplayer) 3.2455 - { 3.2456 - gbMemory[0xff00] = 0xf0 | gbSgbNextController; 3.2457 - } 3.2458 - else 3.2459 - { 3.2460 - gbMemory[0xff00] = 0xff; 3.2461 - } 3.2462 - } 3.2463 - } 3.2464 - GBSystemCounters.lagged = false; 3.2465 - return gbMemory[0xff00]; 3.2466 - break; 3.2467 - case 0x01: 3.2468 - return gbMemory[0xff01]; 3.2469 - case 0x04: 3.2470 - return register_DIV; 3.2471 - case 0x05: 3.2472 - return register_TIMA; 3.2473 - case 0x06: 3.2474 - return register_TMA; 3.2475 - case 0x07: 3.2476 - return (0xf8 | register_TAC); 3.2477 - case 0x0f: 3.2478 - return (0xe0 | register_IF); 3.2479 - case 0x40: 3.2480 - return register_LCDC; 3.2481 - case 0x41: 3.2482 - return (0x80 | register_STAT); 3.2483 - case 0x42: 3.2484 - return register_SCY; 3.2485 - case 0x43: 3.2486 - return register_SCX; 3.2487 - case 0x44: 3.2488 - return register_LY; 3.2489 - case 0x45: 3.2490 - return register_LYC; 3.2491 - case 0x46: 3.2492 - return register_DMA; 3.2493 - case 0x4a: 3.2494 - return register_WY; 3.2495 - case 0x4b: 3.2496 - return register_WX; 3.2497 - case 0x4f: 3.2498 - return (0xfe | register_VBK); 3.2499 - case 0x51: 3.2500 - return register_HDMA1; 3.2501 - case 0x52: 3.2502 - return register_HDMA2; 3.2503 - case 0x53: 3.2504 - return register_HDMA3; 3.2505 - case 0x54: 3.2506 - return register_HDMA4; 3.2507 - case 0x55: 3.2508 - return register_HDMA5; 3.2509 - case 0x70: 3.2510 - return (0xf8 | register_SVBK); 3.2511 - case 0xff: 3.2512 - return register_IE; 3.2513 - } 3.2514 - } 3.2515 - 3.2516 - return gbReadMemoryQuick(address); 3.2517 + } 3.2518 + 3.2519 + return gbReadMemoryQuick(address); 3.2520 } 3.2521 3.2522 void gbVblank_interrupt() 3.2523 { 3.2524 - if (IFF & 0x80) 3.2525 - { 3.2526 - PC.W++; 3.2527 - IFF &= 0x7f; 3.2528 - } 3.2529 - gbInterrupt &= 0xfe; 3.2530 - 3.2531 - IFF &= 0x7e; 3.2532 - register_IF &= 0xfe; 3.2533 - 3.2534 - gbWriteMemory(--SP.W, PC.B.B1); 3.2535 - gbWriteMemory(--SP.W, PC.B.B0); 3.2536 - PC.W = 0x40; 3.2537 + if (IFF & 0x80) 3.2538 + { 3.2539 + PC.W++; 3.2540 + IFF &= 0x7f; 3.2541 + } 3.2542 + gbInterrupt &= 0xfe; 3.2543 + 3.2544 + IFF &= 0x7e; 3.2545 + register_IF &= 0xfe; 3.2546 + 3.2547 + gbWriteMemory(--SP.W, PC.B.B1); 3.2548 + gbWriteMemory(--SP.W, PC.B.B0); 3.2549 + PC.W = 0x40; 3.2550 } 3.2551 3.2552 void gbLcd_interrupt() 3.2553 { 3.2554 - if (IFF & 0x80) 3.2555 - { 3.2556 - PC.W++; 3.2557 - IFF &= 0x7f; 3.2558 - } 3.2559 - gbInterrupt &= 0xfd; 3.2560 - IFF &= 0x7e; 3.2561 - register_IF &= 0xfd; 3.2562 - 3.2563 - gbWriteMemory(--SP.W, PC.B.B1); 3.2564 - gbWriteMemory(--SP.W, PC.B.B0); 3.2565 - 3.2566 - PC.W = 0x48; 3.2567 + if (IFF & 0x80) 3.2568 + { 3.2569 + PC.W++; 3.2570 + IFF &= 0x7f; 3.2571 + } 3.2572 + gbInterrupt &= 0xfd; 3.2573 + IFF &= 0x7e; 3.2574 + register_IF &= 0xfd; 3.2575 + 3.2576 + gbWriteMemory(--SP.W, PC.B.B1); 3.2577 + gbWriteMemory(--SP.W, PC.B.B0); 3.2578 + 3.2579 + PC.W = 0x48; 3.2580 } 3.2581 3.2582 void gbTimer_interrupt() 3.2583 { 3.2584 - if (IFF & 0x80) 3.2585 - { 3.2586 - PC.W++; 3.2587 - IFF &= 0x7f; 3.2588 - } 3.2589 - IFF &= 0x7e; 3.2590 - gbInterrupt &= 0xfb; 3.2591 - register_IF &= 0xfb; 3.2592 - 3.2593 - gbWriteMemory(--SP.W, PC.B.B1); 3.2594 - gbWriteMemory(--SP.W, PC.B.B0); 3.2595 - 3.2596 - PC.W = 0x50; 3.2597 + if (IFF & 0x80) 3.2598 + { 3.2599 + PC.W++; 3.2600 + IFF &= 0x7f; 3.2601 + } 3.2602 + IFF &= 0x7e; 3.2603 + gbInterrupt &= 0xfb; 3.2604 + register_IF &= 0xfb; 3.2605 + 3.2606 + gbWriteMemory(--SP.W, PC.B.B1); 3.2607 + gbWriteMemory(--SP.W, PC.B.B0); 3.2608 + 3.2609 + PC.W = 0x50; 3.2610 } 3.2611 3.2612 void gbSerial_interrupt() 3.2613 { 3.2614 - if (IFF & 0x80) 3.2615 - { 3.2616 - PC.W++; 3.2617 - IFF &= 0x7f; 3.2618 - } 3.2619 - IFF &= 0x7e; 3.2620 - gbInterrupt &= 0xf7; 3.2621 - register_IF &= 0xf7; 3.2622 - 3.2623 - gbWriteMemory(--SP.W, PC.B.B1); 3.2624 - gbWriteMemory(--SP.W, PC.B.B0); 3.2625 - 3.2626 - PC.W = 0x58; 3.2627 + if (IFF & 0x80) 3.2628 + { 3.2629 + PC.W++; 3.2630 + IFF &= 0x7f; 3.2631 + } 3.2632 + IFF &= 0x7e; 3.2633 + gbInterrupt &= 0xf7; 3.2634 + register_IF &= 0xf7; 3.2635 + 3.2636 + gbWriteMemory(--SP.W, PC.B.B1); 3.2637 + gbWriteMemory(--SP.W, PC.B.B0); 3.2638 + 3.2639 + PC.W = 0x58; 3.2640 } 3.2641 3.2642 void gbJoypad_interrupt() 3.2643 { 3.2644 - if (IFF & 0x80) 3.2645 - { 3.2646 - PC.W++; 3.2647 - IFF &= 0x7f; 3.2648 - } 3.2649 - IFF &= 0x7e; 3.2650 - gbInterrupt &= 0xef; 3.2651 - register_IF &= 0xef; 3.2652 - 3.2653 - gbWriteMemory(--SP.W, PC.B.B1); 3.2654 - gbWriteMemory(--SP.W, PC.B.B0); 3.2655 - 3.2656 - PC.W = 0x60; 3.2657 + if (IFF & 0x80) 3.2658 + { 3.2659 + PC.W++; 3.2660 + IFF &= 0x7f; 3.2661 + } 3.2662 + IFF &= 0x7e; 3.2663 + gbInterrupt &= 0xef; 3.2664 + register_IF &= 0xef; 3.2665 + 3.2666 + gbWriteMemory(--SP.W, PC.B.B1); 3.2667 + gbWriteMemory(--SP.W, PC.B.B0); 3.2668 + 3.2669 + PC.W = 0x60; 3.2670 } 3.2671 3.2672 void gbSpeedSwitch() 3.2673 { 3.2674 - if (gbSpeed == 0) 3.2675 - { 3.2676 - gbSpeed = 1; 3.2677 - GBLCD_MODE_0_CLOCK_TICKS = 51 * 2; //127; //51 * 2; 3.2678 - GBLCD_MODE_1_CLOCK_TICKS = 1140 * 2; 3.2679 - GBLCD_MODE_2_CLOCK_TICKS = 20 * 2; //52; //20 * 2; 3.2680 - GBLCD_MODE_3_CLOCK_TICKS = 43 * 2; //99; //43 * 2; 3.2681 - GBDIV_CLOCK_TICKS = 64 * 2; 3.2682 - GBLY_INCREMENT_CLOCK_TICKS = 114 * 2; 3.2683 - GBTIMER_MODE_0_CLOCK_TICKS = 256; //256*2; 3.2684 - GBTIMER_MODE_1_CLOCK_TICKS = 4; //4*2; 3.2685 - GBTIMER_MODE_2_CLOCK_TICKS = 16; //16*2; 3.2686 - GBTIMER_MODE_3_CLOCK_TICKS = 64; //64*2; 3.2687 - GBSERIAL_CLOCK_TICKS = 128 * 2; 3.2688 - gbDivTicks *= 2; 3.2689 - gbLcdTicks *= 2; 3.2690 - gbLcdLYIncrementTicks *= 2; 3.2691 - // timerTicks *= 2; 3.2692 - // timerClockTicks *= 2; 3.2693 - gbSerialTicks *= 2; 3.2694 - SOUND_CLOCK_TICKS = soundQuality * GB_USE_TICKS_AS * 2; 3.2695 - soundTicks *= 2; 3.2696 - // synchronizeTicks *= 2; 3.2697 - // SYNCHRONIZE_CLOCK_TICKS *= 2; 3.2698 - } 3.2699 - else 3.2700 - { 3.2701 - gbSpeed = 0; 3.2702 - GBLCD_MODE_0_CLOCK_TICKS = 51; 3.2703 - GBLCD_MODE_1_CLOCK_TICKS = 1140; 3.2704 - GBLCD_MODE_2_CLOCK_TICKS = 20; 3.2705 - GBLCD_MODE_3_CLOCK_TICKS = 43; 3.2706 - GBDIV_CLOCK_TICKS = 64; 3.2707 - GBLY_INCREMENT_CLOCK_TICKS = 114; 3.2708 - GBTIMER_MODE_0_CLOCK_TICKS = 256; 3.2709 - GBTIMER_MODE_1_CLOCK_TICKS = 4; 3.2710 - GBTIMER_MODE_2_CLOCK_TICKS = 16; 3.2711 - GBTIMER_MODE_3_CLOCK_TICKS = 64; 3.2712 - GBSERIAL_CLOCK_TICKS = 128; 3.2713 - gbDivTicks /= 2; 3.2714 - gbLcdTicks /= 2; 3.2715 - gbLcdLYIncrementTicks /= 2; 3.2716 - // timerTicks /= 2; 3.2717 - // timerClockTicks /= 2; 3.2718 - gbSerialTicks /= 2; 3.2719 - SOUND_CLOCK_TICKS = soundQuality * GB_USE_TICKS_AS; 3.2720 - soundTicks /= 2; 3.2721 - // synchronizeTicks /= 2; 3.2722 - // SYNCHRONIZE_CLOCK_TICKS /= 2; 3.2723 - } 3.2724 + if (gbSpeed == 0) 3.2725 + { 3.2726 + gbSpeed = 1; 3.2727 + GBLCD_MODE_0_CLOCK_TICKS = 51 * 2; //127; //51 * 2; 3.2728 + GBLCD_MODE_1_CLOCK_TICKS = 1140 * 2; 3.2729 + GBLCD_MODE_2_CLOCK_TICKS = 20 * 2; //52; //20 * 2; 3.2730 + GBLCD_MODE_3_CLOCK_TICKS = 43 * 2; //99; //43 * 2; 3.2731 + GBDIV_CLOCK_TICKS = 64 * 2; 3.2732 + GBLY_INCREMENT_CLOCK_TICKS = 114 * 2; 3.2733 + GBTIMER_MODE_0_CLOCK_TICKS = 256; //256*2; 3.2734 + GBTIMER_MODE_1_CLOCK_TICKS = 4; //4*2; 3.2735 + GBTIMER_MODE_2_CLOCK_TICKS = 16; //16*2; 3.2736 + GBTIMER_MODE_3_CLOCK_TICKS = 64; //64*2; 3.2737 + GBSERIAL_CLOCK_TICKS = 128 * 2; 3.2738 + gbDivTicks *= 2; 3.2739 + gbLcdTicks *= 2; 3.2740 + gbLcdLYIncrementTicks *= 2; 3.2741 + // timerTicks *= 2; 3.2742 + // timerClockTicks *= 2; 3.2743 + gbSerialTicks *= 2; 3.2744 + SOUND_CLOCK_TICKS = soundQuality * GB_USE_TICKS_AS * 2; 3.2745 + soundTicks *= 2; 3.2746 + // synchronizeTicks *= 2; 3.2747 + // SYNCHRONIZE_CLOCK_TICKS *= 2; 3.2748 + } 3.2749 + else 3.2750 + { 3.2751 + gbSpeed = 0; 3.2752 + GBLCD_MODE_0_CLOCK_TICKS = 51; 3.2753 + GBLCD_MODE_1_CLOCK_TICKS = 1140; 3.2754 + GBLCD_MODE_2_CLOCK_TICKS = 20; 3.2755 + GBLCD_MODE_3_CLOCK_TICKS = 43; 3.2756 + GBDIV_CLOCK_TICKS = 64; 3.2757 + GBLY_INCREMENT_CLOCK_TICKS = 114; 3.2758 + GBTIMER_MODE_0_CLOCK_TICKS = 256; 3.2759 + GBTIMER_MODE_1_CLOCK_TICKS = 4; 3.2760 + GBTIMER_MODE_2_CLOCK_TICKS = 16; 3.2761 + GBTIMER_MODE_3_CLOCK_TICKS = 64; 3.2762 + GBSERIAL_CLOCK_TICKS = 128; 3.2763 + gbDivTicks /= 2; 3.2764 + gbLcdTicks /= 2; 3.2765 + gbLcdLYIncrementTicks /= 2; 3.2766 + // timerTicks /= 2; 3.2767 + // timerClockTicks /= 2; 3.2768 + gbSerialTicks /= 2; 3.2769 + SOUND_CLOCK_TICKS = soundQuality * GB_USE_TICKS_AS; 3.2770 + soundTicks /= 2; 3.2771 + // synchronizeTicks /= 2; 3.2772 + // SYNCHRONIZE_CLOCK_TICKS /= 2; 3.2773 + } 3.2774 } 3.2775 3.2776 void gbGetHardwareType() 3.2777 { 3.2778 - gbCgbMode = 0; 3.2779 - if (gbRom[0x143] & 0x80) 3.2780 + gbCgbMode = 0; 3.2781 + if (gbRom[0x143] & 0x80) 3.2782 + { 3.2783 + if (gbEmulatorType == 0 || 3.2784 + gbEmulatorType == 1 || 3.2785 + gbEmulatorType == 4 || 3.2786 + gbEmulatorType == 5 || 3.2787 + (gbRom[0x146] != 0x03 && (gbEmulatorType == 2))) 3.2788 { 3.2789 - if (gbEmulatorType == 0 || 3.2790 - gbEmulatorType == 1 || 3.2791 - gbEmulatorType == 4 || 3.2792 - gbEmulatorType == 5 || 3.2793 - (gbRom[0x146] != 0x03 && (gbEmulatorType == 2))) 3.2794 - { 3.2795 - gbCgbMode = 1; 3.2796 - } 3.2797 + gbCgbMode = 1; 3.2798 } 3.2799 - 3.2800 - if (gbSgbMode == 2) 3.2801 - { 3.2802 - gbSgbMode = 0; 3.2803 - return; 3.2804 - } 3.2805 - 3.2806 - gbSgbMode = 0; 3.2807 - if (gbRom[0x146] == 0x03) 3.2808 - { 3.2809 - if (gbEmulatorType == 0 || 3.2810 - gbEmulatorType == 2 || 3.2811 - gbEmulatorType == 5 || 3.2812 - (!(gbRom[0x143] & 0x80) && (gbEmulatorType == 1 || gbEmulatorType == 4))) 3.2813 - gbSgbMode = 1; 3.2814 - } 3.2815 + } 3.2816 + 3.2817 + if (gbSgbMode == 2) 3.2818 + { 3.2819 + gbSgbMode = 0; 3.2820 + return; 3.2821 + } 3.2822 + 3.2823 + gbSgbMode = 0; 3.2824 + if (gbRom[0x146] == 0x03) 3.2825 + { 3.2826 + if (gbEmulatorType == 0 || 3.2827 + gbEmulatorType == 2 || 3.2828 + gbEmulatorType == 5 || 3.2829 + (!(gbRom[0x143] & 0x80) && (gbEmulatorType == 1 || gbEmulatorType == 4))) 3.2830 + gbSgbMode = 1; 3.2831 + } 3.2832 } 3.2833 3.2834 void gbReset(bool userReset) 3.2835 { 3.2836 - // movie must be closed while opening/creating a movie 3.2837 - if (userReset && VBAMovieRecording()) 3.2838 + // movie must be closed while opening/creating a movie 3.2839 + if (userReset && VBAMovieRecording()) 3.2840 + { 3.2841 + VBAMovieSignalReset(); 3.2842 + return; 3.2843 + } 3.2844 + 3.2845 + if (!VBAMovieActive()) 3.2846 + { 3.2847 + GBSystemCounters.frameCount = 0; 3.2848 + GBSystemCounters.lagCount = 0; 3.2849 + GBSystemCounters.extraCount = 0; 3.2850 + GBSystemCounters.lagged = true; 3.2851 + GBSystemCounters.laggedLast = true; 3.2852 + } 3.2853 + 3.2854 + SP.W = 0xfffe; 3.2855 + AF.W = 0x01b0; 3.2856 + BC.W = 0x0013; 3.2857 + DE.W = 0x00d8; 3.2858 + HL.W = 0x014d; 3.2859 + PC.W = 0x0100; 3.2860 + IFF = 0; 3.2861 + gbInterrupt = 1; 3.2862 + gbInterruptWait = 0; 3.2863 + 3.2864 + register_DIV = 0; 3.2865 + register_TIMA = 0; 3.2866 + register_TMA = 0; 3.2867 + register_TAC = 0; 3.2868 + register_IF = 1; 3.2869 + register_LCDC = 0x91; 3.2870 + register_STAT = 0; 3.2871 + register_SCY = 0; 3.2872 + register_SCX = 0; 3.2873 + register_LY = 0; 3.2874 + register_LYC = 0; 3.2875 + register_DMA = 0; 3.2876 + register_WY = 0; 3.2877 + register_WX = 0; 3.2878 + register_VBK = 0; 3.2879 + register_HDMA1 = 0; 3.2880 + register_HDMA2 = 0; 3.2881 + register_HDMA3 = 0; 3.2882 + register_HDMA4 = 0; 3.2883 + register_HDMA5 = 0; 3.2884 + register_SVBK = 0; 3.2885 + register_IE = 0; 3.2886 + 3.2887 + gbGetHardwareType(); 3.2888 + if (gbCgbMode) 3.2889 + { 3.2890 + if (!gbVram) 3.2891 + gbVram = (u8 *)malloc(0x4000 + 4); 3.2892 + if (!gbWram) 3.2893 + gbWram = (u8 *)malloc(0x8000 + 4); 3.2894 + memset(gbVram, 0, 0x4000 + 4); 3.2895 + memset(gbWram, 0, 0x8000 + 4); 3.2896 + } 3.2897 + else 3.2898 + { 3.2899 + if (gbVram) 3.2900 { 3.2901 - VBAMovieSignalReset(); 3.2902 - return; 3.2903 + free(gbVram); 3.2904 + gbVram = NULL; 3.2905 } 3.2906 - 3.2907 - if (!VBAMovieActive()) 3.2908 + if (gbWram) 3.2909 { 3.2910 - GBSystemCounters.frameCount = 0; 3.2911 - GBSystemCounters.lagCount = 0; 3.2912 - GBSystemCounters.extraCount = 0; 3.2913 - GBSystemCounters.lagged = true; 3.2914 - GBSystemCounters.laggedLast = true; 3.2915 + free(gbWram); 3.2916 + gbWram = NULL; 3.2917 } 3.2918 - 3.2919 - SP.W = 0xfffe; 3.2920 - AF.W = 0x01b0; 3.2921 - BC.W = 0x0013; 3.2922 - DE.W = 0x00d8; 3.2923 - HL.W = 0x014d; 3.2924 - PC.W = 0x0100; 3.2925 - IFF = 0; 3.2926 - gbInterrupt = 1; 3.2927 - gbInterruptWait = 0; 3.2928 - 3.2929 - register_DIV = 0; 3.2930 - register_TIMA = 0; 3.2931 - register_TMA = 0; 3.2932 - register_TAC = 0; 3.2933 - register_IF = 1; 3.2934 - register_LCDC = 0x91; 3.2935 - register_STAT = 0; 3.2936 - register_SCY = 0; 3.2937 - register_SCX = 0; 3.2938 - register_LY = 0; 3.2939 - register_LYC = 0; 3.2940 - register_DMA = 0; 3.2941 - register_WY = 0; 3.2942 - register_WX = 0; 3.2943 - register_VBK = 0; 3.2944 - register_HDMA1 = 0; 3.2945 - register_HDMA2 = 0; 3.2946 - register_HDMA3 = 0; 3.2947 - register_HDMA4 = 0; 3.2948 - register_HDMA5 = 0; 3.2949 - register_SVBK = 0; 3.2950 - register_IE = 0; 3.2951 - 3.2952 - gbGetHardwareType(); 3.2953 - if (gbCgbMode) 3.2954 + } 3.2955 + 3.2956 + // clean LineBuffer 3.2957 + if (gbLineBuffer) 3.2958 + memset(gbLineBuffer, 0, 160 * sizeof(u16)); 3.2959 + // clean Pix 3.2960 + if (pix) 3.2961 + memset(pix, 0, 4 * 257 * 226); 3.2962 + 3.2963 + if (gbCgbMode) 3.2964 + { 3.2965 + if (gbSgbMode) 3.2966 { 3.2967 - if (!gbVram) 3.2968 - gbVram = (u8 *)malloc(0x4000 + 4); 3.2969 - if (!gbWram) 3.2970 - gbWram = (u8 *)malloc(0x8000 + 4); 3.2971 - memset(gbVram, 0, 0x4000 + 4); 3.2972 - memset(gbWram, 0, 0x8000 + 4); 3.2973 + if (gbEmulatorType == 5) 3.2974 + AF.W = 0xffb0; 3.2975 + else 3.2976 + AF.W = 0x01b0; 3.2977 + BC.W = 0x0013; 3.2978 + DE.W = 0x00d8; 3.2979 + HL.W = 0x014d; 3.2980 } 3.2981 - else 3.2982 + else 3.2983 { 3.2984 - if (gbVram) 3.2985 - { 3.2986 - free(gbVram); 3.2987 - gbVram = NULL; 3.2988 - } 3.2989 - if (gbWram) 3.2990 - { 3.2991 - free(gbWram); 3.2992 - gbWram = NULL; 3.2993 - } 3.2994 + AF.W = 0x11b0; 3.2995 + BC.W = 0x0000; 3.2996 + DE.W = 0xff56; 3.2997 + HL.W = 0x000d; 3.2998 } 3.2999 - 3.3000 - // clean LineBuffer 3.3001 - if (gbLineBuffer) 3.3002 - memset(gbLineBuffer, 0, 160 * sizeof(u16)); 3.3003 - // clean Pix 3.3004 - if (pix) 3.3005 - memset(pix, 0, 4 * 257 * 226); 3.3006 - 3.3007 - if (gbCgbMode) 3.3008 - { 3.3009 - if (gbSgbMode) 3.3010 - { 3.3011 - if (gbEmulatorType == 5) 3.3012 - AF.W = 0xffb0; 3.3013 - else 3.3014 - AF.W = 0x01b0; 3.3015 - BC.W = 0x0013; 3.3016 - DE.W = 0x00d8; 3.3017 - HL.W = 0x014d; 3.3018 - } 3.3019 - else 3.3020 - { 3.3021 - AF.W = 0x11b0; 3.3022 - BC.W = 0x0000; 3.3023 - DE.W = 0xff56; 3.3024 - HL.W = 0x000d; 3.3025 - } 3.3026 - if (gbEmulatorType == 4) 3.3027 - BC.B.B1 |= 0x01; 3.3028 - 3.3029 - register_HDMA5 = 0xff; 3.3030 - gbMemory[0xff68] = 0xc0; 3.3031 - gbMemory[0xff6a] = 0xc0; 3.3032 - 3.3033 - for (int i = 0; i < 64; i++) 3.3034 - gbPalette[i] = 0x7fff; 3.3035 - } 3.3036 - else 3.3037 - { 3.3038 - for (int i = 0; i < 8; i++) 3.3039 - gbPalette[i] = systemGbPalette[gbPaletteOption * 8 + i]; 3.3040 - } 3.3041 - 3.3042 - if (gbSpeed) 3.3043 - { 3.3044 - gbSpeedSwitch(); 3.3045 - gbMemory[0xff4d] = 0; 3.3046 - } 3.3047 - 3.3048 - gbDivTicks = GBDIV_CLOCK_TICKS; 3.3049 - gbLcdMode = 2; 3.3050 - gbLcdTicks = GBLCD_MODE_2_CLOCK_TICKS; 3.3051 - gbLcdLYIncrementTicks = 0; 3.3052 - gbTimerTicks = 0; 3.3053 - gbTimerClockTicks = 0; 3.3054 - gbSerialTicks = 0; 3.3055 - gbSerialBits = 0; 3.3056 - gbSerialOn = 0; 3.3057 - gbWindowLine = -1; 3.3058 - gbTimerOn = 0; 3.3059 - gbTimerMode = 0; 3.3060 - // gbSynchronizeTicks = GBSYNCHRONIZE_CLOCK_TICKS; 3.3061 - gbSpeed = 0; 3.3062 - gbJoymask[0] = gbJoymask[1] = gbJoymask[2] = gbJoymask[3] = 0; 3.3063 - 3.3064 - // FIXME: horrible kludge 3.3065 - memset(s_gbJoymask, 0, sizeof(s_gbJoymask)); 3.3066 - 3.3067 - if (gbCgbMode) 3.3068 - { 3.3069 - gbSpeed = 0; 3.3070 - gbHdmaOn = 0; 3.3071 - gbHdmaSource = 0x0000; 3.3072 - gbHdmaDestination = 0x8000; 3.3073 - gbVramBank = 0; 3.3074 - gbWramBank = 1; 3.3075 - register_LY = 0x90; 3.3076 - gbLcdMode = 1; 3.3077 - } 3.3078 - 3.3079 - if (gbSgbMode) 3.3080 - { 3.3081 - gbSgbReset(); 3.3082 - } 3.3083 - 3.3084 - for (int i = 0; i < 4; i++) 3.3085 - gbBgp[i] = gbObp0[i] = gbObp1[i] = i; 3.3086 - 3.3087 - memset(&gbDataMBC1, 0, sizeof(gbDataMBC1)); 3.3088 - gbDataMBC1.mapperROMBank = 1; 3.3089 - 3.3090 - gbDataMBC2.mapperRAMEnable = 0; 3.3091 - gbDataMBC2.mapperROMBank = 1; 3.3092 - 3.3093 - memset(&gbDataMBC3, 0, 6 * sizeof(int32)); 3.3094 - gbDataMBC3.mapperROMBank = 1; 3.3095 - 3.3096 - memset(&gbDataMBC5, 0, sizeof(gbDataMBC5)); 3.3097 - gbDataMBC5.mapperROMBank = 1; 3.3098 - switch (gbRom[0x147]) 3.3099 - { 3.3100 - case 0x1c: 3.3101 - case 0x1d: 3.3102 - case 0x1e: 3.3103 - gbDataMBC5.isRumbleCartridge = 1; 3.3104 - } 3.3105 - 3.3106 - memset(&gbDataHuC1, 0, sizeof(gbDataHuC1)); 3.3107 - gbDataHuC1.mapperROMBank = 1; 3.3108 - 3.3109 - memset(&gbDataHuC3, 0, sizeof(gbDataHuC3)); 3.3110 - gbDataHuC3.mapperROMBank = 1; 3.3111 - 3.3112 - gbMemoryMap[0x00] = &gbRom[0x0000]; 3.3113 - gbMemoryMap[0x01] = &gbRom[0x1000]; 3.3114 - gbMemoryMap[0x02] = &gbRom[0x2000]; 3.3115 - gbMemoryMap[0x03] = &gbRom[0x3000]; 3.3116 - gbMemoryMap[0x04] = &gbRom[0x4000]; 3.3117 - gbMemoryMap[0x05] = &gbRom[0x5000]; 3.3118 - gbMemoryMap[0x06] = &gbRom[0x6000]; 3.3119 - gbMemoryMap[0x07] = &gbRom[0x7000]; 3.3120 - if (gbCgbMode) 3.3121 - { 3.3122 - gbMemoryMap[0x08] = &gbVram[0x0000]; 3.3123 - gbMemoryMap[0x09] = &gbVram[0x1000]; 3.3124 - gbMemoryMap[0x0a] = &gbMemory[0xa000]; 3.3125 - gbMemoryMap[0x0b] = &gbMemory[0xb000]; 3.3126 - gbMemoryMap[0x0c] = &gbMemory[0xc000]; 3.3127 - gbMemoryMap[0x0d] = &gbWram[0x1000]; 3.3128 - gbMemoryMap[0x0e] = &gbMemory[0xe000]; 3.3129 - gbMemoryMap[0x0f] = &gbMemory[0xf000]; 3.3130 - } 3.3131 - else 3.3132 - { 3.3133 - gbMemoryMap[0x08] = &gbMemory[0x8000]; 3.3134 - gbMemoryMap[0x09] = &gbMemory[0x9000]; 3.3135 - gbMemoryMap[0x0a] = &gbMemory[0xa000]; 3.3136 - gbMemoryMap[0x0b] = &gbMemory[0xb000]; 3.3137 - gbMemoryMap[0x0c] = &gbMemory[0xc000]; 3.3138 - gbMemoryMap[0x0d] = &gbMemory[0xd000]; 3.3139 - gbMemoryMap[0x0e] = &gbMemory[0xe000]; 3.3140 - gbMemoryMap[0x0f] = &gbMemory[0xf000]; 3.3141 - } 3.3142 - 3.3143 - if (gbRam) 3.3144 - { 3.3145 - gbMemoryMap[0x0a] = &gbRam[0x0000]; 3.3146 - gbMemoryMap[0x0b] = &gbRam[0x1000]; 3.3147 - } 3.3148 - 3.3149 - gbSoundReset(); 3.3150 - 3.3151 - systemResetSensor(); 3.3152 - 3.3153 - systemSaveUpdateCounter = SYSTEM_SAVE_NOT_UPDATED; 3.3154 - 3.3155 - gbLastTime = systemGetClock(); 3.3156 - gbFrameCount = 0; 3.3157 - 3.3158 - systemRefreshScreen(); 3.3159 + if (gbEmulatorType == 4) 3.3160 + BC.B.B1 |= 0x01; 3.3161 + 3.3162 + register_HDMA5 = 0xff; 3.3163 + gbMemory[0xff68] = 0xc0; 3.3164 + gbMemory[0xff6a] = 0xc0; 3.3165 + 3.3166 + for (int i = 0; i < 64; i++) 3.3167 + gbPalette[i] = 0x7fff; 3.3168 + } 3.3169 + else 3.3170 + { 3.3171 + for (int i = 0; i < 8; i++) 3.3172 + gbPalette[i] = systemGbPalette[gbPaletteOption * 8 + i]; 3.3173 + } 3.3174 + 3.3175 + if (gbSpeed) 3.3176 + { 3.3177 + gbSpeedSwitch(); 3.3178 + gbMemory[0xff4d] = 0; 3.3179 + } 3.3180 + 3.3181 + gbDivTicks = GBDIV_CLOCK_TICKS; 3.3182 + gbLcdMode = 2; 3.3183 + gbLcdTicks = GBLCD_MODE_2_CLOCK_TICKS; 3.3184 + gbLcdLYIncrementTicks = 0; 3.3185 + gbTimerTicks = 0; 3.3186 + gbTimerClockTicks = 0; 3.3187 + gbSerialTicks = 0; 3.3188 + gbSerialBits = 0; 3.3189 + gbSerialOn = 0; 3.3190 + gbWindowLine = -1; 3.3191 + gbTimerOn = 0; 3.3192 + gbTimerMode = 0; 3.3193 + // gbSynchronizeTicks = GBSYNCHRONIZE_CLOCK_TICKS; 3.3194 + gbSpeed = 0; 3.3195 + gbJoymask[0] = gbJoymask[1] = gbJoymask[2] = gbJoymask[3] = 0; 3.3196 + 3.3197 + // FIXME: horrible kludge 3.3198 + memset(s_gbJoymask, 0, sizeof(s_gbJoymask)); 3.3199 + 3.3200 + if (gbCgbMode) 3.3201 + { 3.3202 + gbSpeed = 0; 3.3203 + gbHdmaOn = 0; 3.3204 + gbHdmaSource = 0x0000; 3.3205 + gbHdmaDestination = 0x8000; 3.3206 + gbVramBank = 0; 3.3207 + gbWramBank = 1; 3.3208 + register_LY = 0x90; 3.3209 + gbLcdMode = 1; 3.3210 + } 3.3211 + 3.3212 + if (gbSgbMode) 3.3213 + { 3.3214 + gbSgbReset(); 3.3215 + } 3.3216 + 3.3217 + for (int i = 0; i < 4; i++) 3.3218 + gbBgp[i] = gbObp0[i] = gbObp1[i] = i; 3.3219 + 3.3220 + memset(&gbDataMBC1, 0, sizeof(gbDataMBC1)); 3.3221 + gbDataMBC1.mapperROMBank = 1; 3.3222 + 3.3223 + gbDataMBC2.mapperRAMEnable = 0; 3.3224 + gbDataMBC2.mapperROMBank = 1; 3.3225 + 3.3226 + memset(&gbDataMBC3, 0, 6 * sizeof(int32)); 3.3227 + gbDataMBC3.mapperROMBank = 1; 3.3228 + 3.3229 + memset(&gbDataMBC5, 0, sizeof(gbDataMBC5)); 3.3230 + gbDataMBC5.mapperROMBank = 1; 3.3231 + switch (gbRom[0x147]) 3.3232 + { 3.3233 + case 0x1c: 3.3234 + case 0x1d: 3.3235 + case 0x1e: 3.3236 + gbDataMBC5.isRumbleCartridge = 1; 3.3237 + } 3.3238 + 3.3239 + memset(&gbDataHuC1, 0, sizeof(gbDataHuC1)); 3.3240 + gbDataHuC1.mapperROMBank = 1; 3.3241 + 3.3242 + memset(&gbDataHuC3, 0, sizeof(gbDataHuC3)); 3.3243 + gbDataHuC3.mapperROMBank = 1; 3.3244 + 3.3245 + gbMemoryMap[0x00] = &gbRom[0x0000]; 3.3246 + gbMemoryMap[0x01] = &gbRom[0x1000]; 3.3247 + gbMemoryMap[0x02] = &gbRom[0x2000]; 3.3248 + gbMemoryMap[0x03] = &gbRom[0x3000]; 3.3249 + gbMemoryMap[0x04] = &gbRom[0x4000]; 3.3250 + gbMemoryMap[0x05] = &gbRom[0x5000]; 3.3251 + gbMemoryMap[0x06] = &gbRom[0x6000]; 3.3252 + gbMemoryMap[0x07] = &gbRom[0x7000]; 3.3253 + if (gbCgbMode) 3.3254 + { 3.3255 + gbMemoryMap[0x08] = &gbVram[0x0000]; 3.3256 + gbMemoryMap[0x09] = &gbVram[0x1000]; 3.3257 + gbMemoryMap[0x0a] = &gbMemory[0xa000]; 3.3258 + gbMemoryMap[0x0b] = &gbMemory[0xb000]; 3.3259 + gbMemoryMap[0x0c] = &gbMemory[0xc000]; 3.3260 + gbMemoryMap[0x0d] = &gbWram[0x1000]; 3.3261 + gbMemoryMap[0x0e] = &gbMemory[0xe000]; 3.3262 + gbMemoryMap[0x0f] = &gbMemory[0xf000]; 3.3263 + } 3.3264 + else 3.3265 + { 3.3266 + gbMemoryMap[0x08] = &gbMemory[0x8000]; 3.3267 + gbMemoryMap[0x09] = &gbMemory[0x9000]; 3.3268 + gbMemoryMap[0x0a] = &gbMemory[0xa000]; 3.3269 + gbMemoryMap[0x0b] = &gbMemory[0xb000]; 3.3270 + gbMemoryMap[0x0c] = &gbMemory[0xc000]; 3.3271 + gbMemoryMap[0x0d] = &gbMemory[0xd000]; 3.3272 + gbMemoryMap[0x0e] = &gbMemory[0xe000]; 3.3273 + gbMemoryMap[0x0f] = &gbMemory[0xf000]; 3.3274 + } 3.3275 + 3.3276 + if (gbRam) 3.3277 + { 3.3278 + gbMemoryMap[0x0a] = &gbRam[0x0000]; 3.3279 + gbMemoryMap[0x0b] = &gbRam[0x1000]; 3.3280 + } 3.3281 + 3.3282 + gbSoundReset(); 3.3283 + 3.3284 + systemResetSensor(); 3.3285 + 3.3286 + systemSaveUpdateCounter = SYSTEM_SAVE_NOT_UPDATED; 3.3287 + 3.3288 + gbLastTime = systemGetClock(); 3.3289 + gbFrameCount = 0; 3.3290 + 3.3291 + systemRefreshScreen(); 3.3292 } 3.3293 3.3294 void gbWriteSaveMBC1(const char *name) 3.3295 { 3.3296 - FILE *gzFile = fopen(name, "wb"); 3.3297 - 3.3298 - if (gzFile == NULL) 3.3299 - { 3.3300 - systemMessage(MSG_ERROR_CREATING_FILE, N_("Error creating file %s"), name); 3.3301 - return; 3.3302 - } 3.3303 - 3.3304 - fwrite(gbRam, 3.3305 - 1, 3.3306 - gbRamSize, 3.3307 - gzFile); 3.3308 - 3.3309 - fclose(gzFile); 3.3310 + FILE *gzFile = fopen(name, "wb"); 3.3311 + 3.3312 + if (gzFile == NULL) 3.3313 + { 3.3314 + systemMessage(MSG_ERROR_CREATING_FILE, N_("Error creating file %s"), name); 3.3315 + return; 3.3316 + } 3.3317 + 3.3318 + fwrite(gbRam, 3.3319 + 1, 3.3320 + gbRamSize, 3.3321 + gzFile); 3.3322 + 3.3323 + fclose(gzFile); 3.3324 } 3.3325 3.3326 void gbWriteSaveMBC2(const char *name) 3.3327 { 3.3328 - FILE *file = fopen(name, "wb"); 3.3329 - 3.3330 - if (file == NULL) 3.3331 - { 3.3332 - systemMessage(MSG_ERROR_CREATING_FILE, N_("Error creating file %s"), name); 3.3333 - return; 3.3334 - } 3.3335 - 3.3336 - fwrite(&gbMemory[0xa000], 3.3337 - 1, 3.3338 - 256, 3.3339 - file); 3.3340 - 3.3341 - fclose(file); 3.3342 + FILE *file = fopen(name, "wb"); 3.3343 + 3.3344 + if (file == NULL) 3.3345 + { 3.3346 + systemMessage(MSG_ERROR_CREATING_FILE, N_("Error creating file %s"), name); 3.3347 + return; 3.3348 + } 3.3349 + 3.3350 + fwrite(&gbMemory[0xa000], 3.3351 + 1, 3.3352 + 256, 3.3353 + file); 3.3354 + 3.3355 + fclose(file); 3.3356 } 3.3357 3.3358 void gbWriteSaveMBC3(const char *name, bool extendedSave) 3.3359 { 3.3360 - FILE *gzFile = fopen(name, "wb"); 3.3361 - 3.3362 - if (gzFile == NULL) 3.3363 - { 3.3364 - systemMessage(MSG_ERROR_CREATING_FILE, N_("Error creating file %s"), name); 3.3365 - return; 3.3366 - } 3.3367 - 3.3368 - fwrite(gbRam, 3.3369 - 1, 3.3370 - gbRamSize, 3.3371 - gzFile); 3.3372 - 3.3373 - if (extendedSave) 3.3374 - { 3.3375 - //assert(sizeof(time_t) == 4); 3.3376 - fwrite(&gbDataMBC3.mapperSeconds, 3.3377 - 1, 3.3378 - 10 * sizeof(int32) + /*sizeof(time_t)*/4, 3.3379 - gzFile); 3.3380 - } 3.3381 - 3.3382 - fclose(gzFile); 3.3383 + FILE *gzFile = fopen(name, "wb"); 3.3384 + 3.3385 + if (gzFile == NULL) 3.3386 + { 3.3387 + systemMessage(MSG_ERROR_CREATING_FILE, N_("Error creating file %s"), name); 3.3388 + return; 3.3389 + } 3.3390 + 3.3391 + fwrite(gbRam, 3.3392 + 1, 3.3393 + gbRamSize, 3.3394 + gzFile); 3.3395 + 3.3396 + if (extendedSave) 3.3397 + { 3.3398 + //assert(sizeof(time_t) == 4); 3.3399 + fwrite(&gbDataMBC3.mapperSeconds, 3.3400 + 1, 3.3401 + 10 * sizeof(int32) + /*sizeof(time_t)*/4, 3.3402 + gzFile); 3.3403 + } 3.3404 + 3.3405 + fclose(gzFile); 3.3406 } 3.3407 3.3408 void gbWriteSaveMBC5(const char *name) 3.3409 { 3.3410 - FILE *gzFile = fopen(name, "wb"); 3.3411 - 3.3412 - if (gzFile == NULL) 3.3413 - { 3.3414 - systemMessage(MSG_ERROR_CREATING_FILE, N_("Error creating file %s"), name); 3.3415 - return; 3.3416 - } 3.3417 - 3.3418 - fwrite(gbRam, 3.3419 - 1, 3.3420 - gbRamSize, 3.3421 - gzFile); 3.3422 - 3.3423 - fclose(gzFile); 3.3424 + FILE *gzFile = fopen(name, "wb"); 3.3425 + 3.3426 + if (gzFile == NULL) 3.3427 + { 3.3428 + systemMessage(MSG_ERROR_CREATING_FILE, N_("Error creating file %s"), name); 3.3429 + return; 3.3430 + } 3.3431 + 3.3432 + fwrite(gbRam, 3.3433 + 1, 3.3434 + gbRamSize, 3.3435 + gzFile); 3.3436 + 3.3437 + fclose(gzFile); 3.3438 } 3.3439 3.3440 void gbWriteSaveMBC7(const char *name) 3.3441 { 3.3442 - FILE *file = fopen(name, "wb"); 3.3443 - 3.3444 - if (file == NULL) 3.3445 - { 3.3446 - systemMessage(MSG_ERROR_CREATING_FILE, N_("Error creating file %s"), name); 3.3447 - return; 3.3448 - } 3.3449 - 3.3450 - fwrite(&gbMemory[0xa000], 3.3451 - 1, 3.3452 - 256, 3.3453 - file); 3.3454 - 3.3455 - fclose(file); 3.3456 + FILE *file = fopen(name, "wb"); 3.3457 + 3.3458 + if (file == NULL) 3.3459 + { 3.3460 + systemMessage(MSG_ERROR_CREATING_FILE, N_("Error creating file %s"), name); 3.3461 + return; 3.3462 + } 3.3463 + 3.3464 + fwrite(&gbMemory[0xa000], 3.3465 + 1, 3.3466 + 256, 3.3467 + file); 3.3468 + 3.3469 + fclose(file); 3.3470 } 3.3471 3.3472 bool gbReadSaveMBC1(const char *name) 3.3473 { 3.3474 - gzFile gzFile = gzopen(name, "rb"); 3.3475 - 3.3476 - if (gzFile == NULL) 3.3477 - { 3.3478 - return false; 3.3479 - } 3.3480 - 3.3481 - int read = gzread(gzFile, 3.3482 - gbRam, 3.3483 - gbRamSize); 3.3484 - 3.3485 - if (read != gbRamSize) 3.3486 - { 3.3487 - systemMessage(MSG_FAILED_TO_READ_SGM, N_("Failed to read complete save game %s (%d)"), name, read); 3.3488 - gzclose(gzFile); 3.3489 - return false; 3.3490 - } 3.3491 - 3.3492 - gzclose(gzFile); 3.3493 - return true; 3.3494 + gzFile gzFile = gzopen(name, "rb"); 3.3495 + 3.3496 + if (gzFile == NULL) 3.3497 + { 3.3498 + return false; 3.3499 + } 3.3500 + 3.3501 + int read = gzread(gzFile, 3.3502 + gbRam, 3.3503 + gbRamSize); 3.3504 + 3.3505 + if (read != gbRamSize) 3.3506 + { 3.3507 + systemMessage(MSG_FAILED_TO_READ_SGM, N_("Failed to read complete save game %s (%d)"), name, read); 3.3508 + gzclose(gzFile); 3.3509 + return false; 3.3510 + } 3.3511 + 3.3512 + gzclose(gzFile); 3.3513 + return true; 3.3514 } 3.3515 3.3516 bool gbReadSaveMBC2(const char *name) 3.3517 { 3.3518 - FILE *file = fopen(name, "rb"); 3.3519 - 3.3520 - if (file == NULL) 3.3521 - { 3.3522 - return false; 3.3523 - } 3.3524 - 3.3525 - int read = fread(&gbMemory[0xa000], 3.3526 - 1, 3.3527 - 256, 3.3528 - file); 3.3529 - 3.3530 - if (read != 256) 3.3531 - { 3.3532 - systemMessage(MSG_FAILED_TO_READ_SGM, 3.3533 - N_("Failed to read complete save game %s (%d)"), name, read); 3.3534 - fclose(file); 3.3535 - return false; 3.3536 - } 3.3537 - 3.3538 - fclose(file); 3.3539 - return true; 3.3540 + FILE *file = fopen(name, "rb"); 3.3541 + 3.3542 + if (file == NULL) 3.3543 + { 3.3544 + return false; 3.3545 + } 3.3546 + 3.3547 + int read = fread(&gbMemory[0xa000], 3.3548 + 1, 3.3549 + 256, 3.3550 + file); 3.3551 + 3.3552 + if (read != 256) 3.3553 + { 3.3554 + systemMessage(MSG_FAILED_TO_READ_SGM, 3.3555 + N_("Failed to read complete save game %s (%d)"), name, read); 3.3556 + fclose(file); 3.3557 + return false; 3.3558 + } 3.3559 + 3.3560 + fclose(file); 3.3561 + return true; 3.3562 } 3.3563 3.3564 bool gbReadSaveMBC3(const char *name) 3.3565 { 3.3566 - gzFile gzFile = gzopen(name, "rb"); 3.3567 - 3.3568 - if (gzFile == NULL) 3.3569 + gzFile gzFile = gzopen(name, "rb"); 3.3570 + 3.3571 + if (gzFile == NULL) 3.3572 + { 3.3573 + return false; 3.3574 + } 3.3575 + 3.3576 + int read = gzread(gzFile, 3.3577 + gbRam, 3.3578 + gbRamSize); 3.3579 + 3.3580 + bool res = true; 3.3581 + 3.3582 + if (read != gbRamSize) 3.3583 + { 3.3584 + systemMessage(MSG_FAILED_TO_READ_SGM, 3.3585 + N_("Failed to read complete save game %s (%d)"), name, read); 3.3586 + } 3.3587 + else 3.3588 + { 3.3589 + //assert(sizeof(time_t) == 4); 3.3590 + read = gzread(gzFile, 3.3591 + &gbDataMBC3.mapperSeconds, 3.3592 + sizeof(int32) * 10 + /*sizeof(time_t)*/4); 3.3593 + 3.3594 + if (read != (sizeof(int32) * 10 + /*sizeof(time_t)*/4) && read != 0) 3.3595 { 3.3596 - return false; 3.3597 + systemMessage(MSG_FAILED_TO_READ_RTC, 3.3598 + N_("Failed to read RTC from save game %s (continuing)"), 3.3599 + name); 3.3600 + res = false; 3.3601 } 3.3602 - 3.3603 - int read = gzread(gzFile, 3.3604 - gbRam, 3.3605 - gbRamSize); 3.3606 - 3.3607 - bool res = true; 3.3608 - 3.3609 - if (read != gbRamSize) 3.3610 - { 3.3611 - systemMessage(MSG_FAILED_TO_READ_SGM, 3.3612 - N_("Failed to read complete save game %s (%d)"), name, read); 3.3613 - } 3.3614 - else 3.3615 - { 3.3616 - //assert(sizeof(time_t) == 4); 3.3617 - read = gzread(gzFile, 3.3618 - &gbDataMBC3.mapperSeconds, 3.3619 - sizeof(int32) * 10 + /*sizeof(time_t)*/4); 3.3620 - 3.3621 - if (read != (sizeof(int32) * 10 + /*sizeof(time_t)*/4) && read != 0) 3.3622 - { 3.3623 - systemMessage(MSG_FAILED_TO_READ_RTC, 3.3624 - N_("Failed to read RTC from save game %s (continuing)"), 3.3625 - name); 3.3626 - res = false; 3.3627 - } 3.3628 - } 3.3629 - 3.3630 - gzclose(gzFile); 3.3631 - return res; 3.3632 + } 3.3633 + 3.3634 + gzclose(gzFile); 3.3635 + return res; 3.3636 } 3.3637 3.3638 bool gbReadSaveMBC5(const char *name) 3.3639 { 3.3640 - gzFile gzFile = gzopen(name, "rb"); 3.3641 - 3.3642 - if (gzFile == NULL) 3.3643 - { 3.3644 - return false; 3.3645 - } 3.3646 - 3.3647 - int read = gzread(gzFile, 3.3648 - gbRam, 3.3649 - gbRamSize); 3.3650 - 3.3651 - if (read != gbRamSize) 3.3652 - { 3.3653 - systemMessage(MSG_FAILED_TO_READ_SGM, 3.3654 - N_("Failed to read complete save game %s (%d)"), name, read); 3.3655 - gzclose(gzFile); 3.3656 - return false; 3.3657 - } 3.3658 - 3.3659 - gzclose(gzFile); 3.3660 - return true; 3.3661 + gzFile gzFile = gzopen(name, "rb"); 3.3662 + 3.3663 + if (gzFile == NULL) 3.3664 + { 3.3665 + return false; 3.3666 + } 3.3667 + 3.3668 + int read = gzread(gzFile, 3.3669 + gbRam, 3.3670 + gbRamSize); 3.3671 + 3.3672 + if (read != gbRamSize) 3.3673 + { 3.3674 + systemMessage(MSG_FAILED_TO_READ_SGM, 3.3675 + N_("Failed to read complete save game %s (%d)"), name, read); 3.3676 + gzclose(gzFile); 3.3677 + return false; 3.3678 + } 3.3679 + 3.3680 + gzclose(gzFile); 3.3681 + return true; 3.3682 } 3.3683 3.3684 bool gbReadSaveMBC7(const char *name) 3.3685 { 3.3686 - FILE *file = fopen(name, "rb"); 3.3687 - 3.3688 - if (file == NULL) 3.3689 - { 3.3690 - return false; 3.3691 - } 3.3692 - 3.3693 - int read = fread(&gbMemory[0xa000], 3.3694 - 1, 3.3695 - 256, 3.3696 - file); 3.3697 - 3.3698 - if (read != 256) 3.3699 - { 3.3700 - systemMessage(MSG_FAILED_TO_READ_SGM, 3.3701 - N_("Failed to read complete save game %s (%d)"), name, read); 3.3702 - fclose(file); 3.3703 - return false; 3.3704 - } 3.3705 - 3.3706 - fclose(file); 3.3707 - return true; 3.3708 + FILE *file = fopen(name, "rb"); 3.3709 + 3.3710 + if (file == NULL) 3.3711 + { 3.3712 + return false; 3.3713 + } 3.3714 + 3.3715 + int read = fread(&gbMemory[0xa000], 3.3716 + 1, 3.3717 + 256, 3.3718 + file); 3.3719 + 3.3720 + if (read != 256) 3.3721 + { 3.3722 + systemMessage(MSG_FAILED_TO_READ_SGM, 3.3723 + N_("Failed to read complete save game %s (%d)"), name, read); 3.3724 + fclose(file); 3.3725 + return false; 3.3726 + } 3.3727 + 3.3728 + fclose(file); 3.3729 + return true; 3.3730 } 3.3731 3.3732 #if 0 3.3733 bool gbLoadBIOS(const char *biosFileName, bool useBiosFile) 3.3734 { 3.3735 - useBios = false; 3.3736 - if (useBiosFile) 3.3737 + useBios = false; 3.3738 + if (useBiosFile) 3.3739 + { 3.3740 + useBios = utilLoadBIOS(bios, biosFileName, gbEmulatorType); 3.3741 + if (!useBios) 3.3742 { 3.3743 - useBios = utilLoadBIOS(bios, biosFileName, gbEmulatorType); 3.3744 - if (!useBios) 3.3745 - { 3.3746 - systemMessage(MSG_INVALID_BIOS_FILE_SIZE, N_("Invalid BOOTROM file")); 3.3747 - } 3.3748 + systemMessage(MSG_INVALID_BIOS_FILE_SIZE, N_("Invalid BOOTROM file")); 3.3749 } 3.3750 - return useBios; 3.3751 + } 3.3752 + return useBios; 3.3753 } 3.3754 #endif 3.3755 3.3756 void gbInit() 3.3757 { 3.3758 - gbGenFilter(); 3.3759 - gbSgbInit(); // calls gbSgbReset()... whatever 3.3760 - 3.3761 - gbMemory = (u8 *)malloc(65536 + 4); 3.3762 - memset(gbMemory, 0, 65536 + 4); 3.3763 - memset(gbPalette, 0, 2 * 128); 3.3764 - 3.3765 - // HACK: +4 at start to accomodate the 2xSaI filter reading out of bounds of the leftmost pixel 3.3766 - origPix = (u8 *)calloc(1, 4 * 257 * 226 + 4); 3.3767 - pix = origPix + 4; 3.3768 - 3.3769 - gbLineBuffer = (u16 *)malloc(160 * sizeof(u16)); 3.3770 + gbGenFilter(); 3.3771 + gbSgbInit(); // calls gbSgbReset()... whatever 3.3772 + 3.3773 + gbMemory = (u8 *)malloc(65536 + 4); 3.3774 + memset(gbMemory, 0, 65536 + 4); 3.3775 + memset(gbPalette, 0, 2 * 128); 3.3776 + 3.3777 + // HACK: +4 at start to accomodate the 2xSaI filter reading out of bounds of the leftmost pixel 3.3778 + origPix = (u8 *)calloc(1, 4 * 257 * 226 + 4); 3.3779 + pix = origPix + 4; 3.3780 + 3.3781 + gbLineBuffer = (u16 *)malloc(160 * sizeof(u16)); 3.3782 } 3.3783 3.3784 bool gbWriteBatteryFile(const char *file, bool extendedSave) 3.3785 { 3.3786 - if (gbBattery) 3.3787 - { 3.3788 - int type = gbRom[0x147]; 3.3789 - 3.3790 - switch (type) 3.3791 - { 3.3792 - case 0x03: 3.3793 - gbWriteSaveMBC1(file); 3.3794 - break; 3.3795 - case 0x06: 3.3796 - gbWriteSaveMBC2(file); 3.3797 - break; 3.3798 - case 0x0f: 3.3799 - case 0x10: 3.3800 - case 0x13: 3.3801 - gbWriteSaveMBC3(file, extendedSave); 3.3802 - break; 3.3803 - case 0x1b: 3.3804 - case 0x1e: 3.3805 - gbWriteSaveMBC5(file); 3.3806 - break; 3.3807 - case 0x22: 3.3808 - gbWriteSaveMBC7(file); 3.3809 - break; 3.3810 - case 0xff: 3.3811 - gbWriteSaveMBC1(file); 3.3812 - break; 3.3813 - } 3.3814 - } 3.3815 - return true; 3.3816 -} 3.3817 - 3.3818 -bool gbWriteBatteryFile(const char *file) 3.3819 -{ 3.3820 - gbWriteBatteryFile(file, true); 3.3821 - return true; 3.3822 -} 3.3823 - 3.3824 -bool gbWriteBatteryToStream(gzFile gzfile) 3.3825 -{ 3.3826 - // the GB save code is ugly, so rather than convert it all to use gzFiles, just save it to a temp file... 3.3827 -#define TEMP_SAVE_FNAME ("tempvbawrite.sav") 3.3828 - bool retVal = gbWriteBatteryFile(TEMP_SAVE_FNAME, true); 3.3829 - 3.3830 - // ...open the temp file and figure out its size... 3.3831 - FILE *fileTemp = fopen(TEMP_SAVE_FNAME, "rb"); 3.3832 - if (fileTemp == NULL) 3.3833 - return false; 3.3834 - fseek(fileTemp, 0, SEEK_END); 3.3835 - int len = (int) ftell(fileTemp); 3.3836 - 3.3837 - // ...copy over the temp file... 3.3838 - char *temp = new char [len]; 3.3839 - fseek(fileTemp, 0, SEEK_SET); 3.3840 - if (fread(temp, len, 1, fileTemp) != 1) 3.3841 - { 3.3842 - delete [] temp; 3.3843 - fclose(fileTemp); 3.3844 - return false; 3.3845 - } 3.3846 - fclose(fileTemp); 3.3847 - utilGzWrite(gzfile, temp, len); 3.3848 - delete [] temp; 3.3849 - 3.3850 - // ... and delete the temp file 3.3851 - remove(TEMP_SAVE_FNAME); 3.3852 -#undef TEMP_SAVE_FNAME 3.3853 - 3.3854 - return retVal; 3.3855 -} 3.3856 - 3.3857 -bool gbReadBatteryFile(const char *file) 3.3858 -{ 3.3859 - bool res = false; 3.3860 - if (gbBattery) 3.3861 - { 3.3862 - int type = gbRom[0x147]; 3.3863 - 3.3864 - switch (type) 3.3865 - { 3.3866 - case 0x03: 3.3867 - res = gbReadSaveMBC1(file); 3.3868 - break; 3.3869 - case 0x06: 3.3870 - res = gbReadSaveMBC2(file); 3.3871 - break; 3.3872 - case 0x0f: 3.3873 - case 0x10: 3.3874 - case 0x13: 3.3875 - if (!gbReadSaveMBC3(file)) 3.3876 - { 3.3877 - struct tm *lt; 3.3878 - time_t tmp; //Small kludge to get it working on some systems where time_t has size 8. 3.3879 - 3.3880 - if (VBAMovieActive() || VBAMovieLoading()) 3.3881 - { 3.3882 - gbDataMBC3.mapperLastTime = VBAMovieGetId() + VBAMovieGetFrameCounter() / 60; 3.3883 - lt = gmtime(&tmp); 3.3884 - gbDataMBC3.mapperLastTime=(u32)tmp; 3.3885 - } 3.3886 - else 3.3887 - { 3.3888 - time(&tmp); 3.3889 - gbDataMBC3.mapperLastTime=(u32)tmp; 3.3890 - lt = localtime(&tmp); 3.3891 - } 3.3892 - systemScreenMessage(ctime(&tmp), 4); 3.3893 - gbDataMBC3.mapperLastTime=(u32)tmp; 3.3894 - 3.3895 - gbDataMBC3.mapperSeconds = lt->tm_sec; 3.3896 - gbDataMBC3.mapperMinutes = lt->tm_min; 3.3897 - gbDataMBC3.mapperHours = lt->tm_hour; 3.3898 - gbDataMBC3.mapperDays = lt->tm_yday & 255; 3.3899 - gbDataMBC3.mapperControl = (gbDataMBC3.mapperControl & 0xfe) | 3.3900 - (lt->tm_yday > 255 ? 1 : 0); 3.3901 - res = false; 3.3902 - break; 3.3903 - } 3.3904 - time_t tmp; 3.3905 - systemScreenMessage(ctime(&tmp), 4); 3.3906 - gbDataMBC3.mapperLastTime=(u32)tmp; 3.3907 - res = true; 3.3908 - break; 3.3909 - case 0x1b: 3.3910 - case 0x1e: 3.3911 - res = gbReadSaveMBC5(file); 3.3912 - break; 3.3913 - case 0x22: 3.3914 - res = gbReadSaveMBC7(file); 3.3915 - case 0xff: 3.3916 - res = gbReadSaveMBC1(file); 3.3917 - break; 3.3918 - } 3.3919 - } 3.3920 - systemSaveUpdateCounter = SYSTEM_SAVE_NOT_UPDATED; 3.3921 - return res; 3.3922 -} 3.3923 - 3.3924 -bool gbReadBatteryFromStream(gzFile gzfile) 3.3925 -{ 3.3926 - // the GB save code is ugly, so rather than convert it all to use gzFiles, just copy it to temp RAM... 3.3927 -#define TEMP_SAVE_FNAME ("tempvbaread.sav") 3.3928 - int pos = gztell(gzfile); 3.3929 - int buflen = 1024; 3.3930 - // ...make a temp file and write it there... 3.3931 - FILE *fileTemp = fopen(TEMP_SAVE_FNAME, "wb"); 3.3932 - if (fileTemp == NULL) 3.3933 - return false; 3.3934 - int gzDeflated; 3.3935 - char *temp = new char [buflen]; 3.3936 - while ((gzDeflated = utilGzRead(gzfile, temp, buflen)) != 0) 3.3937 - { 3.3938 - if (gzDeflated == -1 || fwrite(temp, gzDeflated, 1, fileTemp) != 1) 3.3939 - { 3.3940 - delete [] temp; 3.3941 - fclose(fileTemp); 3.3942 - gzseek(gzfile, pos, SEEK_SET); /// FIXME: leaves pos in gzfile before save instead of after it (everything that 3.3943 - // calls this right now does a seek afterwards so it doesn't matter for now, but it's 3.3944 - // still bad) 3.3945 - return false; 3.3946 - } 3.3947 - } 3.3948 - gzseek(gzfile, pos, SEEK_SET); /// FIXME: leaves pos in gzfile before save instead of after it (everything that calls this 3.3949 - // right now does a seek afterwards so it doesn't matter for now, but it's still bad) 3.3950 - fclose(fileTemp); 3.3951 - delete [] temp; 3.3952 - 3.3953 - // ... load from the temp file... 3.3954 - bool retVal = gbReadBatteryFile(TEMP_SAVE_FNAME); 3.3955 - 3.3956 - // ... and delete the temp file 3.3957 - remove(TEMP_SAVE_FNAME); 3.3958 -#undef TEMP_SAVE_FNAME 3.3959 - 3.3960 - return retVal; 3.3961 -} 3.3962 - 3.3963 -bool gbReadGSASnapshot(const char *fileName) 3.3964 -{ 3.3965 - FILE *file = fopen(fileName, "rb"); 3.3966 - 3.3967 - if (!file) 3.3968 - { 3.3969 - systemMessage(MSG_CANNOT_OPEN_FILE, N_("Cannot open file %s"), fileName); 3.3970 - return false; 3.3971 - } 3.3972 - 3.3973 - // long size = ftell(file); 3.3974 - fseek(file, 0x4, SEEK_SET); 3.3975 - char buffer[16]; 3.3976 - char buffer2[16]; 3.3977 - fread(buffer, 1, 15, file); 3.3978 - buffer[15] = 0; 3.3979 - memcpy(buffer2, &gbRom[0x134], 15); 3.3980 - buffer2[15] = 0; 3.3981 - if (memcmp(buffer, buffer2, 15)) 3.3982 - { 3.3983 - systemMessage(MSG_CANNOT_IMPORT_SNAPSHOT_FOR, 3.3984 - N_("Cannot import snapshot for %s. Current game is %s"), 3.3985 - buffer, 3.3986 - buffer2); 3.3987 - fclose(file); 3.3988 - return false; 3.3989 - } 3.3990 - fseek(file, 0x13, SEEK_SET); 3.3991 - int read = 0; 3.3992 - int toRead = 0; 3.3993 - switch (gbRom[0x147]) 3.3994 + if (gbBattery) 3.3995 + { 3.3996 + int type = gbRom[0x147]; 3.3997 + 3.3998 + switch (type) 3.3999 { 3.4000 case 0x03: 3.4001 + gbWriteSaveMBC1(file); 3.4002 + break; 3.4003 + case 0x06: 3.4004 + gbWriteSaveMBC2(file); 3.4005 + break; 3.4006 case 0x0f: 3.4007 case 0x10: 3.4008 case 0x13: 3.4009 + gbWriteSaveMBC3(file, extendedSave); 3.4010 + break; 3.4011 case 0x1b: 3.4012 case 0x1e: 3.4013 + gbWriteSaveMBC5(file); 3.4014 + break; 3.4015 + case 0x22: 3.4016 + gbWriteSaveMBC7(file); 3.4017 + break; 3.4018 case 0xff: 3.4019 - read = fread(gbRam, 1, gbRamSize, file); 3.4020 - toRead = gbRamSize; 3.4021 - break; 3.4022 + gbWriteSaveMBC1(file); 3.4023 + break; 3.4024 + } 3.4025 + } 3.4026 + return true; 3.4027 +} 3.4028 + 3.4029 +bool gbWriteBatteryFile(const char *file) 3.4030 +{ 3.4031 + gbWriteBatteryFile(file, true); 3.4032 + return true; 3.4033 +} 3.4034 + 3.4035 +bool gbWriteBatteryToStream(gzFile gzfile) 3.4036 +{ 3.4037 + // the GB save code is ugly, so rather than convert it all to use gzFiles, just save it to a temp file... 3.4038 +#define TEMP_SAVE_FNAME ("tempvbawrite.sav") 3.4039 + bool retVal = gbWriteBatteryFile(TEMP_SAVE_FNAME, true); 3.4040 + 3.4041 + // ...open the temp file and figure out its size... 3.4042 + FILE *fileTemp = fopen(TEMP_SAVE_FNAME, "rb"); 3.4043 + if (fileTemp == NULL) 3.4044 + return false; 3.4045 + fseek(fileTemp, 0, SEEK_END); 3.4046 + int len = (int) ftell(fileTemp); 3.4047 + 3.4048 + // ...copy over the temp file... 3.4049 + char *temp = new char [len]; 3.4050 + fseek(fileTemp, 0, SEEK_SET); 3.4051 + if (fread(temp, len, 1, fileTemp) != 1) 3.4052 + { 3.4053 + delete [] temp; 3.4054 + fclose(fileTemp); 3.4055 + return false; 3.4056 + } 3.4057 + fclose(fileTemp); 3.4058 + utilGzWrite(gzfile, temp, len); 3.4059 + delete [] temp; 3.4060 + 3.4061 + // ... and delete the temp file 3.4062 + remove(TEMP_SAVE_FNAME); 3.4063 +#undef TEMP_SAVE_FNAME 3.4064 + 3.4065 + return retVal; 3.4066 +} 3.4067 + 3.4068 +bool gbReadBatteryFile(const char *file) 3.4069 +{ 3.4070 + bool res = false; 3.4071 + if (gbBattery) 3.4072 + { 3.4073 + int type = gbRom[0x147]; 3.4074 + 3.4075 + switch (type) 3.4076 + { 3.4077 + case 0x03: 3.4078 + res = gbReadSaveMBC1(file); 3.4079 + break; 3.4080 case 0x06: 3.4081 + res = gbReadSaveMBC2(file); 3.4082 + break; 3.4083 + case 0x0f: 3.4084 + case 0x10: 3.4085 + case 0x13: 3.4086 + if (!gbReadSaveMBC3(file)) 3.4087 + { 3.4088 + struct tm *lt; 3.4089 + time_t tmp; //Small kludge to get it working on some systems where time_t has size 8. 3.4090 + 3.4091 + if (VBAMovieActive() || VBAMovieLoading()) 3.4092 + { 3.4093 + gbDataMBC3.mapperLastTime = VBAMovieGetId() + VBAMovieGetFrameCounter() / 60; 3.4094 + lt = gmtime(&tmp); 3.4095 + gbDataMBC3.mapperLastTime=(u32)tmp; 3.4096 + } 3.4097 + else 3.4098 + { 3.4099 + time(&tmp); 3.4100 + gbDataMBC3.mapperLastTime=(u32)tmp; 3.4101 + lt = localtime(&tmp); 3.4102 + } 3.4103 + systemScreenMessage(ctime(&tmp), 4); 3.4104 + gbDataMBC3.mapperLastTime=(u32)tmp; 3.4105 + 3.4106 + gbDataMBC3.mapperSeconds = lt->tm_sec; 3.4107 + gbDataMBC3.mapperMinutes = lt->tm_min; 3.4108 + gbDataMBC3.mapperHours = lt->tm_hour; 3.4109 + gbDataMBC3.mapperDays = lt->tm_yday & 255; 3.4110 + gbDataMBC3.mapperControl = (gbDataMBC3.mapperControl & 0xfe) | 3.4111 + (lt->tm_yday > 255 ? 1 : 0); 3.4112 + res = false; 3.4113 + break; 3.4114 + } 3.4115 + time_t tmp; 3.4116 + systemScreenMessage(ctime(&tmp), 4); 3.4117 + gbDataMBC3.mapperLastTime=(u32)tmp; 3.4118 + res = true; 3.4119 + break; 3.4120 + case 0x1b: 3.4121 + case 0x1e: 3.4122 + res = gbReadSaveMBC5(file); 3.4123 + break; 3.4124 case 0x22: 3.4125 - read = fread(&gbMemory[0xa000], 1, 256, file); 3.4126 - toRead = 256; 3.4127 - break; 3.4128 - default: 3.4129 - systemMessage(MSG_UNSUPPORTED_SNAPSHOT_FILE, 3.4130 - N_("Unsupported snapshot file %s"), 3.4131 - fileName); 3.4132 - fclose(file); 3.4133 - return false; 3.4134 + res = gbReadSaveMBC7(file); 3.4135 + case 0xff: 3.4136 + res = gbReadSaveMBC1(file); 3.4137 + break; 3.4138 } 3.4139 - fclose(file); 3.4140 - gbReset(); 3.4141 - return true; 3.4142 + } 3.4143 + systemSaveUpdateCounter = SYSTEM_SAVE_NOT_UPDATED; 3.4144 + return res; 3.4145 } 3.4146 3.4147 +bool gbReadBatteryFromStream(gzFile gzfile) 3.4148 +{ 3.4149 + // the GB save code is ugly, so rather than convert it all to use gzFiles, just copy it to temp RAM... 3.4150 +#define TEMP_SAVE_FNAME ("tempvbaread.sav") 3.4151 + int pos = gztell(gzfile); 3.4152 + int buflen = 1024; 3.4153 + // ...make a temp file and write it there... 3.4154 + FILE *fileTemp = fopen(TEMP_SAVE_FNAME, "wb"); 3.4155 + if (fileTemp == NULL) 3.4156 + return false; 3.4157 + int gzDeflated; 3.4158 + char *temp = new char [buflen]; 3.4159 + while ((gzDeflated = utilGzRead(gzfile, temp, buflen)) != 0) 3.4160 + { 3.4161 + if (gzDeflated == -1 || fwrite(temp, gzDeflated, 1, fileTemp) != 1) 3.4162 + { 3.4163 + delete [] temp; 3.4164 + fclose(fileTemp); 3.4165 + gzseek(gzfile, pos, SEEK_SET); /// FIXME: leaves pos in gzfile before save instead of after it (everything that 3.4166 + // calls this right now does a seek afterwards so it doesn't matter for now, but it's 3.4167 + // still bad) 3.4168 + return false; 3.4169 + } 3.4170 + } 3.4171 + gzseek(gzfile, pos, SEEK_SET); /// FIXME: leaves pos in gzfile before save instead of after it (everything that calls this 3.4172 + // right now does a seek afterwards so it doesn't matter for now, but it's still bad) 3.4173 + fclose(fileTemp); 3.4174 + delete [] temp; 3.4175 + 3.4176 + // ... load from the temp file... 3.4177 + bool retVal = gbReadBatteryFile(TEMP_SAVE_FNAME); 3.4178 + 3.4179 + // ... and delete the temp file 3.4180 + remove(TEMP_SAVE_FNAME); 3.4181 +#undef TEMP_SAVE_FNAME 3.4182 + 3.4183 + return retVal; 3.4184 +} 3.4185 + 3.4186 +bool gbReadGSASnapshot(const char *fileName) 3.4187 +{ 3.4188 + FILE *file = fopen(fileName, "rb"); 3.4189 + 3.4190 + if (!file) 3.4191 + { 3.4192 + systemMessage(MSG_CANNOT_OPEN_FILE, N_("Cannot open file %s"), fileName); 3.4193 + return false; 3.4194 + } 3.4195 + 3.4196 + // long size = ftell(file); 3.4197 + fseek(file, 0x4, SEEK_SET); 3.4198 + char buffer[16]; 3.4199 + char buffer2[16]; 3.4200 + fread(buffer, 1, 15, file); 3.4201 + buffer[15] = 0; 3.4202 + memcpy(buffer2, &gbRom[0x134], 15); 3.4203 + buffer2[15] = 0; 3.4204 + if (memcmp(buffer, buffer2, 15)) 3.4205 + { 3.4206 + systemMessage(MSG_CANNOT_IMPORT_SNAPSHOT_FOR, 3.4207 + N_("Cannot import snapshot for %s. Current game is %s"), 3.4208 + buffer, 3.4209 + buffer2); 3.4210 + fclose(file); 3.4211 + return false; 3.4212 + } 3.4213 + fseek(file, 0x13, SEEK_SET); 3.4214 + int read = 0; 3.4215 + int toRead = 0; 3.4216 + switch (gbRom[0x147]) 3.4217 + { 3.4218 + case 0x03: 3.4219 + case 0x0f: 3.4220 + case 0x10: 3.4221 + case 0x13: 3.4222 + case 0x1b: 3.4223 + case 0x1e: 3.4224 + case 0xff: 3.4225 + read = fread(gbRam, 1, gbRamSize, file); 3.4226 + toRead = gbRamSize; 3.4227 + break; 3.4228 + case 0x06: 3.4229 + case 0x22: 3.4230 + read = fread(&gbMemory[0xa000], 1, 256, file); 3.4231 + toRead = 256; 3.4232 + break; 3.4233 + default: 3.4234 + systemMessage(MSG_UNSUPPORTED_SNAPSHOT_FILE, 3.4235 + N_("Unsupported snapshot file %s"), 3.4236 + fileName); 3.4237 + fclose(file); 3.4238 + return false; 3.4239 + } 3.4240 + fclose(file); 3.4241 + gbReset(); 3.4242 + return true; 3.4243 +} 3.4244 + 3.4245 variable_desc gbSaveGameStruct[] = 3.4246 -{ 3.4247 - { &PC.W, sizeof(u16) }, 3.4248 - { &SP.W, sizeof(u16) }, 3.4249 - { &AF.W, sizeof(u16) }, 3.4250 - { &BC.W, sizeof(u16) }, 3.4251 - { &DE.W, sizeof(u16) }, 3.4252 - { &HL.W, sizeof(u16) }, 3.4253 - { &IFF, sizeof(u8) }, 3.4254 - { &GBLCD_MODE_0_CLOCK_TICKS, sizeof(int32) }, 3.4255 - { &GBLCD_MODE_1_CLOCK_TICKS, sizeof(int32) }, 3.4256 - { &GBLCD_MODE_2_CLOCK_TICKS, sizeof(int32) }, 3.4257 - { &GBLCD_MODE_3_CLOCK_TICKS, sizeof(int32) }, 3.4258 - { &GBDIV_CLOCK_TICKS, sizeof(int32) }, 3.4259 - { &GBLY_INCREMENT_CLOCK_TICKS, sizeof(int32) }, 3.4260 - { &GBTIMER_MODE_0_CLOCK_TICKS, sizeof(int32) }, 3.4261 - { &GBTIMER_MODE_1_CLOCK_TICKS, sizeof(int32) }, 3.4262 - { &GBTIMER_MODE_2_CLOCK_TICKS, sizeof(int32) }, 3.4263 - { &GBTIMER_MODE_3_CLOCK_TICKS, sizeof(int32) }, 3.4264 - { &GBSERIAL_CLOCK_TICKS, sizeof(int32) }, 3.4265 - { &GBSYNCHRONIZE_CLOCK_TICKS, sizeof(int32) }, 3.4266 - { &gbDivTicks, sizeof(int32) }, 3.4267 - { &gbLcdMode, sizeof(int32) }, 3.4268 - { &gbLcdTicks, sizeof(int32) }, 3.4269 - { &gbLcdLYIncrementTicks, sizeof(int32) }, 3.4270 - { &gbTimerTicks, sizeof(int32) }, 3.4271 - { &gbTimerClockTicks, sizeof(int32) }, 3.4272 - { &gbSerialTicks, sizeof(int32) }, 3.4273 - { &gbSerialBits, sizeof(int32) }, 3.4274 - { &gbInterrupt, sizeof(int32) }, 3.4275 - { &gbInterruptWait, sizeof(int32) }, 3.4276 - { &gbSynchronizeTicks, sizeof(int32) }, 3.4277 - { &gbTimerOn, sizeof(int32) }, 3.4278 - { &gbTimerMode, sizeof(int32) }, 3.4279 - { &gbSerialOn, sizeof(int32) }, 3.4280 - { &gbWindowLine, sizeof(int32) }, 3.4281 - { &gbCgbMode, sizeof(int32) }, 3.4282 - { &gbVramBank, sizeof(int32) }, 3.4283 - { &gbWramBank, sizeof(int32) }, 3.4284 - { &gbHdmaSource, sizeof(int32) }, 3.4285 - { &gbHdmaDestination, sizeof(int32) }, 3.4286 - { &gbHdmaBytes, sizeof(int32) }, 3.4287 - { &gbHdmaOn, sizeof(int32) }, 3.4288 - { &gbSpeed, sizeof(int32) }, 3.4289 - { &gbSgbMode, sizeof(int32) }, 3.4290 - { ®ister_DIV, sizeof(u8) }, 3.4291 - { ®ister_TIMA, sizeof(u8) }, 3.4292 - { ®ister_TMA, sizeof(u8) }, 3.4293 - { ®ister_TAC, sizeof(u8) }, 3.4294 - { ®ister_IF, sizeof(u8) }, 3.4295 - { ®ister_LCDC, sizeof(u8) }, 3.4296 - { ®ister_STAT, sizeof(u8) }, 3.4297 - { ®ister_SCY, sizeof(u8) }, 3.4298 - { ®ister_SCX, sizeof(u8) }, 3.4299 - { ®ister_LY, sizeof(u8) }, 3.4300 - { ®ister_LYC, sizeof(u8) }, 3.4301 - { ®ister_DMA, sizeof(u8) }, 3.4302 - { ®ister_WY, sizeof(u8) }, 3.4303 - { ®ister_WX, sizeof(u8) }, 3.4304 - { ®ister_VBK, sizeof(u8) }, 3.4305 - { ®ister_HDMA1, sizeof(u8) }, 3.4306 - { ®ister_HDMA2, sizeof(u8) }, 3.4307 - { ®ister_HDMA3, sizeof(u8) }, 3.4308 - { ®ister_HDMA4, sizeof(u8) }, 3.4309 - { ®ister_HDMA5, sizeof(u8) }, 3.4310 - { ®ister_SVBK, sizeof(u8) }, 3.4311 - { ®ister_IE, sizeof(u8) }, 3.4312 - { &gbBgp[0], sizeof(u8) }, 3.4313 - { &gbBgp[1], sizeof(u8) }, 3.4314 - { &gbBgp[2], sizeof(u8) }, 3.4315 - { &gbBgp[3], sizeof(u8) }, 3.4316 - { &gbObp0[0], sizeof(u8) }, 3.4317 - { &gbObp0[1], sizeof(u8) }, 3.4318 - { &gbObp0[2], sizeof(u8) }, 3.4319 - { &gbObp0[3], sizeof(u8) }, 3.4320 - { &gbObp1[0], sizeof(u8) }, 3.4321 - { &gbObp1[1], sizeof(u8) }, 3.4322 - { &gbObp1[2], sizeof(u8) }, 3.4323 - { &gbObp1[3], sizeof(u8) }, 3.4324 - { NULL, 0 } 3.4325 -}; 3.4326 + { 3.4327 + { &PC.W, sizeof(u16) }, 3.4328 + { &SP.W, sizeof(u16) }, 3.4329 + { &AF.W, sizeof(u16) }, 3.4330 + { &BC.W, sizeof(u16) }, 3.4331 + { &DE.W, sizeof(u16) }, 3.4332 + { &HL.W, sizeof(u16) }, 3.4333 + { &IFF, sizeof(u8) }, 3.4334 + { &GBLCD_MODE_0_CLOCK_TICKS, sizeof(int32) }, 3.4335 + { &GBLCD_MODE_1_CLOCK_TICKS, sizeof(int32) }, 3.4336 + { &GBLCD_MODE_2_CLOCK_TICKS, sizeof(int32) }, 3.4337 + { &GBLCD_MODE_3_CLOCK_TICKS, sizeof(int32) }, 3.4338 + { &GBDIV_CLOCK_TICKS, sizeof(int32) }, 3.4339 + { &GBLY_INCREMENT_CLOCK_TICKS, sizeof(int32) }, 3.4340 + { &GBTIMER_MODE_0_CLOCK_TICKS, sizeof(int32) }, 3.4341 + { &GBTIMER_MODE_1_CLOCK_TICKS, sizeof(int32) }, 3.4342 + { &GBTIMER_MODE_2_CLOCK_TICKS, sizeof(int32) }, 3.4343 + { &GBTIMER_MODE_3_CLOCK_TICKS, sizeof(int32) }, 3.4344 + { &GBSERIAL_CLOCK_TICKS, sizeof(int32) }, 3.4345 + { &GBSYNCHRONIZE_CLOCK_TICKS, sizeof(int32) }, 3.4346 + { &gbDivTicks, sizeof(int32) }, 3.4347 + { &gbLcdMode, sizeof(int32) }, 3.4348 + { &gbLcdTicks, sizeof(int32) }, 3.4349 + { &gbLcdLYIncrementTicks, sizeof(int32) }, 3.4350 + { &gbTimerTicks, sizeof(int32) }, 3.4351 + { &gbTimerClockTicks, sizeof(int32) }, 3.4352 + { &gbSerialTicks, sizeof(int32) }, 3.4353 + { &gbSerialBits, sizeof(int32) }, 3.4354 + { &gbInterrupt, sizeof(int32) }, 3.4355 + { &gbInterruptWait, sizeof(int32) }, 3.4356 + { &gbSynchronizeTicks, sizeof(int32) }, 3.4357 + { &gbTimerOn, sizeof(int32) }, 3.4358 + { &gbTimerMode, sizeof(int32) }, 3.4359 + { &gbSerialOn, sizeof(int32) }, 3.4360 + { &gbWindowLine, sizeof(int32) }, 3.4361 + { &gbCgbMode, sizeof(int32) }, 3.4362 + { &gbVramBank, sizeof(int32) }, 3.4363 + { &gbWramBank, sizeof(int32) }, 3.4364 + { &gbHdmaSource, sizeof(int32) }, 3.4365 + { &gbHdmaDestination, sizeof(int32) }, 3.4366 + { &gbHdmaBytes, sizeof(int32) }, 3.4367 + { &gbHdmaOn, sizeof(int32) }, 3.4368 + { &gbSpeed, sizeof(int32) }, 3.4369 + { &gbSgbMode, sizeof(int32) }, 3.4370 + { ®ister_DIV, sizeof(u8) }, 3.4371 + { ®ister_TIMA, sizeof(u8) }, 3.4372 + { ®ister_TMA, sizeof(u8) }, 3.4373 + { ®ister_TAC, sizeof(u8) }, 3.4374 + { ®ister_IF, sizeof(u8) }, 3.4375 + { ®ister_LCDC, sizeof(u8) }, 3.4376 + { ®ister_STAT, sizeof(u8) }, 3.4377 + { ®ister_SCY, sizeof(u8) }, 3.4378 + { ®ister_SCX, sizeof(u8) }, 3.4379 + { ®ister_LY, sizeof(u8) }, 3.4380 + { ®ister_LYC, sizeof(u8) }, 3.4381 + { ®ister_DMA, sizeof(u8) }, 3.4382 + { ®ister_WY, sizeof(u8) }, 3.4383 + { ®ister_WX, sizeof(u8) }, 3.4384 + { ®ister_VBK, sizeof(u8) }, 3.4385 + { ®ister_HDMA1, sizeof(u8) }, 3.4386 + { ®ister_HDMA2, sizeof(u8) }, 3.4387 + { ®ister_HDMA3, sizeof(u8) }, 3.4388 + { ®ister_HDMA4, sizeof(u8) }, 3.4389 + { ®ister_HDMA5, sizeof(u8) }, 3.4390 + { ®ister_SVBK, sizeof(u8) }, 3.4391 + { ®ister_IE, sizeof(u8) }, 3.4392 + { &gbBgp[0], sizeof(u8) }, 3.4393 + { &gbBgp[1], sizeof(u8) }, 3.4394 + { &gbBgp[2], sizeof(u8) }, 3.4395 + { &gbBgp[3], sizeof(u8) }, 3.4396 + { &gbObp0[0], sizeof(u8) }, 3.4397 + { &gbObp0[1], sizeof(u8) }, 3.4398 + { &gbObp0[2], sizeof(u8) }, 3.4399 + { &gbObp0[3], sizeof(u8) }, 3.4400 + { &gbObp1[0], sizeof(u8) }, 3.4401 + { &gbObp1[1], sizeof(u8) }, 3.4402 + { &gbObp1[2], sizeof(u8) }, 3.4403 + { &gbObp1[3], sizeof(u8) }, 3.4404 + { NULL, 0 } 3.4405 + }; 3.4406 3.4407 bool gbWriteSaveStateToStream(gzFile gzFile) 3.4408 { 3.4409 - utilWriteInt(gzFile, GBSAVE_GAME_VERSION); 3.4410 - 3.4411 - utilGzWrite(gzFile, &gbRom[0x134], 15); 3.4412 - 3.4413 - utilWriteData(gzFile, gbSaveGameStruct); 3.4414 - 3.4415 - utilGzWrite(gzFile, &IFF, 2); 3.4416 - 3.4417 - if (gbSgbMode) 3.4418 - { 3.4419 - gbSgbSaveGame(gzFile); 3.4420 - } 3.4421 - 3.4422 - utilGzWrite(gzFile, &gbDataMBC1, sizeof(gbDataMBC1)); 3.4423 - utilGzWrite(gzFile, &gbDataMBC2, sizeof(gbDataMBC2)); 3.4424 - //assert(sizeof(time_t) == 4); 3.4425 - utilGzWrite(gzFile, &gbDataMBC3, sizeof(gbDataMBC3)); 3.4426 - utilGzWrite(gzFile, &gbDataMBC5, sizeof(gbDataMBC5)); 3.4427 - utilGzWrite(gzFile, &gbDataHuC1, sizeof(gbDataHuC1)); 3.4428 - utilGzWrite(gzFile, &gbDataHuC3, sizeof(gbDataHuC3)); 3.4429 - 3.4430 - // yes, this definitely needs to be saved, or loading paused games will show a black screen 3.4431 - // this is also necessary to be consistent with what the GBA saving does 3.4432 - utilGzWrite(gzFile, pix, 4 * 257 * 226); 3.4433 - 3.4434 - utilGzWrite(gzFile, gbPalette, 128 * sizeof(u16)); 3.4435 - // todo: remove 3.4436 - utilGzWrite(gzFile, gbPalette, 128 * sizeof(u16)); 3.4437 - 3.4438 - utilGzWrite(gzFile, &gbMemory[0x8000], 0x8000); 3.4439 - 3.4440 - if (gbRamSize && gbRam) 3.4441 - { 3.4442 - utilGzWrite(gzFile, gbRam, gbRamSize); 3.4443 - } 3.4444 - 3.4445 - if (gbCgbMode) 3.4446 - { 3.4447 - utilGzWrite(gzFile, gbVram, 0x4000); 3.4448 - utilGzWrite(gzFile, gbWram, 0x8000); 3.4449 - } 3.4450 - 3.4451 - gbSoundSaveGame(gzFile); 3.4452 - 3.4453 - gbCheatsSaveGame(gzFile); 3.4454 - 3.4455 - // new to re-recording version: 3.4456 - { 3.4457 - extern int32 sensorX, sensorY; 3.4458 - utilGzWrite(gzFile, &sensorX, sizeof(sensorX)); 3.4459 - utilGzWrite(gzFile, &sensorY, sizeof(sensorY)); 3.4460 - utilGzWrite(gzFile, gbJoymask, 4 * sizeof(*gbJoymask)); // this has to be saved or old input will incorrectly get 3.4461 - // carried 3.4462 - // back on loading a snapshot! 3.4463 - 3.4464 - bool8 movieActive = VBAMovieActive(); 3.4465 - utilGzWrite(gzFile, &movieActive, sizeof(movieActive)); 3.4466 - if (movieActive) 3.4467 - { 3.4468 - uint8 *movie_freeze_buf = NULL; 3.4469 - uint32 movie_freeze_size = 0; 3.4470 - 3.4471 - VBAMovieFreeze(&movie_freeze_buf, &movie_freeze_size); 3.4472 - if (movie_freeze_buf) 3.4473 - { 3.4474 - utilGzWrite(gzFile, &movie_freeze_size, sizeof(movie_freeze_size)); 3.4475 - utilGzWrite(gzFile, movie_freeze_buf, movie_freeze_size); 3.4476 - delete [] movie_freeze_buf; 3.4477 - } 3.4478 - else 3.4479 - { 3.4480 - systemMessage(0, N_("Failed to save movie snapshot.")); 3.4481 - return false; 3.4482 - } 3.4483 - } 3.4484 - utilGzWrite(gzFile, &GBSystemCounters.frameCount, sizeof(GBSystemCounters.frameCount)); 3.4485 - } 3.4486 - 3.4487 - // new to rerecording 19.4 wip (svn r22+): 3.4488 - { 3.4489 - utilGzWrite(gzFile, &GBSystemCounters.lagCount, sizeof(GBSystemCounters.lagCount)); 3.4490 - utilGzWrite(gzFile, &GBSystemCounters.lagged, sizeof(GBSystemCounters.lagged)); 3.4491 - utilGzWrite(gzFile, &GBSystemCounters.laggedLast, sizeof(GBSystemCounters.laggedLast)); 3.4492 - } 3.4493 - 3.4494 - return true; 3.4495 + utilWriteInt(gzFile, GBSAVE_GAME_VERSION); 3.4496 + 3.4497 + utilGzWrite(gzFile, &gbRom[0x134], 15); 3.4498 + 3.4499 + utilWriteData(gzFile, gbSaveGameStruct); 3.4500 + 3.4501 + utilGzWrite(gzFile, &IFF, 2); 3.4502 + 3.4503 + if (gbSgbMode) 3.4504 + { 3.4505 + gbSgbSaveGame(gzFile); 3.4506 + } 3.4507 + 3.4508 + utilGzWrite(gzFile, &gbDataMBC1, sizeof(gbDataMBC1)); 3.4509 + utilGzWrite(gzFile, &gbDataMBC2, sizeof(gbDataMBC2)); 3.4510 + //assert(sizeof(time_t) == 4); 3.4511 + utilGzWrite(gzFile, &gbDataMBC3, sizeof(gbDataMBC3)); 3.4512 + utilGzWrite(gzFile, &gbDataMBC5, sizeof(gbDataMBC5)); 3.4513 + utilGzWrite(gzFile, &gbDataHuC1, sizeof(gbDataHuC1)); 3.4514 + utilGzWrite(gzFile, &gbDataHuC3, sizeof(gbDataHuC3)); 3.4515 + 3.4516 + // yes, this definitely needs to be saved, or loading paused games will show a black screen 3.4517 + // this is also necessary to be consistent with what the GBA saving does 3.4518 + utilGzWrite(gzFile, pix, 4 * 257 * 226); 3.4519 + 3.4520 + utilGzWrite(gzFile, gbPalette, 128 * sizeof(u16)); 3.4521 + // todo: remove 3.4522 + utilGzWrite(gzFile, gbPalette, 128 * sizeof(u16)); 3.4523 + 3.4524 + utilGzWrite(gzFile, &gbMemory[0x8000], 0x8000); 3.4525 + 3.4526 + if (gbRamSize && gbRam) 3.4527 + { 3.4528 + utilGzWrite(gzFile, gbRam, gbRamSize); 3.4529 + } 3.4530 + 3.4531 + if (gbCgbMode) 3.4532 + { 3.4533 + utilGzWrite(gzFile, gbVram, 0x4000); 3.4534 + utilGzWrite(gzFile, gbWram, 0x8000); 3.4535 + } 3.4536 + 3.4537 + gbSoundSaveGame(gzFile); 3.4538 + 3.4539 + gbCheatsSaveGame(gzFile); 3.4540 + 3.4541 + // new to re-recording version: 3.4542 + { 3.4543 + extern int32 sensorX, sensorY; 3.4544 + utilGzWrite(gzFile, &sensorX, sizeof(sensorX)); 3.4545 + utilGzWrite(gzFile, &sensorY, sizeof(sensorY)); 3.4546 + utilGzWrite(gzFile, gbJoymask, 4 * sizeof(*gbJoymask)); // this has to be saved or old input will incorrectly get 3.4547 + // carried 3.4548 + // back on loading a snapshot! 3.4549 + 3.4550 + bool8 movieActive = VBAMovieActive(); 3.4551 + utilGzWrite(gzFile, &movieActive, sizeof(movieActive)); 3.4552 + if (movieActive) 3.4553 + { 3.4554 + uint8 *movie_freeze_buf = NULL; 3.4555 + uint32 movie_freeze_size = 0; 3.4556 + 3.4557 + VBAMovieFreeze(&movie_freeze_buf, &movie_freeze_size); 3.4558 + if (movie_freeze_buf) 3.4559 + { 3.4560 + utilGzWrite(gzFile, &movie_freeze_size, sizeof(movie_freeze_size)); 3.4561 + utilGzWrite(gzFile, movie_freeze_buf, movie_freeze_size); 3.4562 + delete [] movie_freeze_buf; 3.4563 + } 3.4564 + else 3.4565 + { 3.4566 + systemMessage(0, N_("Failed to save movie snapshot.")); 3.4567 + return false; 3.4568 + } 3.4569 + } 3.4570 + utilGzWrite(gzFile, &GBSystemCounters.frameCount, sizeof(GBSystemCounters.frameCount)); 3.4571 + } 3.4572 + 3.4573 + // new to rerecording 19.4 wip (svn r22+): 3.4574 + { 3.4575 + utilGzWrite(gzFile, &GBSystemCounters.lagCount, sizeof(GBSystemCounters.lagCount)); 3.4576 + utilGzWrite(gzFile, &GBSystemCounters.lagged, sizeof(GBSystemCounters.lagged)); 3.4577 + utilGzWrite(gzFile, &GBSystemCounters.laggedLast, sizeof(GBSystemCounters.laggedLast)); 3.4578 + } 3.4579 + 3.4580 + return true; 3.4581 } 3.4582 3.4583 bool gbWriteMemSaveState(char *memory, int available) 3.4584 { 3.4585 - gzFile gzFile = utilMemGzOpen(memory, available, "w"); 3.4586 - 3.4587 - if (gzFile == NULL) 3.4588 - { 3.4589 - return false; 3.4590 - } 3.4591 - 3.4592 - bool res = gbWriteSaveStateToStream(gzFile); 3.4593 - 3.4594 - long pos = utilGzTell(gzFile) + 8; 3.4595 - 3.4596 - if (pos >= (available)) 3.4597 - res = false; 3.4598 - 3.4599 - utilGzClose(gzFile); 3.4600 - 3.4601 - return res; 3.4602 + gzFile gzFile = utilMemGzOpen(memory, available, "w"); 3.4603 + 3.4604 + if (gzFile == NULL) 3.4605 + { 3.4606 + return false; 3.4607 + } 3.4608 + 3.4609 + bool res = gbWriteSaveStateToStream(gzFile); 3.4610 + 3.4611 + long pos = utilGzTell(gzFile) + 8; 3.4612 + 3.4613 + if (pos >= (available)) 3.4614 + res = false; 3.4615 + 3.4616 + utilGzClose(gzFile); 3.4617 + 3.4618 + return res; 3.4619 } 3.4620 3.4621 bool gbWriteSaveState(const char *name) 3.4622 { 3.4623 - gzFile gzFile = utilGzOpen(name, "wb"); 3.4624 - 3.4625 - if (gzFile == NULL) 3.4626 - return false; 3.4627 - 3.4628 - bool res = gbWriteSaveStateToStream(gzFile); 3.4629 - 3.4630 - utilGzClose(gzFile); 3.4631 - return res; 3.4632 + gzFile gzFile = utilGzOpen(name, "wb"); 3.4633 + 3.4634 + if (gzFile == NULL) 3.4635 + return false; 3.4636 + 3.4637 + bool res = gbWriteSaveStateToStream(gzFile); 3.4638 + 3.4639 + utilGzClose(gzFile); 3.4640 + return res; 3.4641 } 3.4642 3.4643 static int tempStateID = 0; 3.4644 @@ -2591,1328 +2591,1331 @@ 3.4645 3.4646 bool gbReadSaveStateFromStream(gzFile gzFile) 3.4647 { 3.4648 - int type; 3.4649 - char tempBackupName [128]; 3.4650 - if (backupSafe) 3.4651 + int type; 3.4652 + char tempBackupName [128]; 3.4653 + if (backupSafe) 3.4654 + { 3.4655 + sprintf(tempBackupName, "gbatempsave%d.sav", tempStateID++); 3.4656 + gbWriteSaveState(tempBackupName); 3.4657 + } 3.4658 + 3.4659 + int version = utilReadInt(gzFile); 3.4660 + 3.4661 + if (version > GBSAVE_GAME_VERSION || version < 0) 3.4662 + { 3.4663 + systemMessage(MSG_UNSUPPORTED_VB_SGM, 3.4664 + N_("Unsupported VisualBoy save game version %d"), version); 3.4665 + goto failedLoadGB; 3.4666 + } 3.4667 + 3.4668 + u8 romname[20]; 3.4669 + 3.4670 + utilGzRead(gzFile, romname, 15); 3.4671 + 3.4672 + if (memcmp(&gbRom[0x134], romname, 15) != 0) 3.4673 + { 3.4674 + systemMessage(MSG_CANNOT_LOAD_SGM_FOR, 3.4675 + N_("Cannot load save game for %s. Playing %s"), 3.4676 + romname, &gbRom[0x134]); 3.4677 + goto failedLoadGB; 3.4678 + } 3.4679 + 3.4680 + utilReadData(gzFile, gbSaveGameStruct); 3.4681 + 3.4682 + if (version >= GBSAVE_GAME_VERSION_7) 3.4683 + { 3.4684 + utilGzRead(gzFile, &IFF, 2); 3.4685 + } 3.4686 + 3.4687 + if (gbSgbMode) 3.4688 + { 3.4689 + gbSgbReadGame(gzFile, version); 3.4690 + } 3.4691 + else 3.4692 + { 3.4693 + gbSgbMask = 0; // loading a game at the wrong time causes no display 3.4694 + } 3.4695 + 3.4696 + utilGzRead(gzFile, &gbDataMBC1, sizeof(gbDataMBC1)); 3.4697 + utilGzRead(gzFile, &gbDataMBC2, sizeof(gbDataMBC2)); 3.4698 + if (version < GBSAVE_GAME_VERSION_4) 3.4699 + // prior to version 4, there was no adjustment for the time the game 3.4700 + // was last played, so we have less to read. This needs update if the 3.4701 + // structure changes again. 3.4702 + utilGzRead(gzFile, &gbDataMBC3, sizeof(int32) * 10); 3.4703 + else 3.4704 + { 3.4705 + //assert(sizeof(time_t) == 4); 3.4706 + utilGzRead(gzFile, &gbDataMBC3, sizeof(gbDataMBC3)); 3.4707 + } 3.4708 + utilGzRead(gzFile, &gbDataMBC5, sizeof(gbDataMBC5)); 3.4709 + utilGzRead(gzFile, &gbDataHuC1, sizeof(gbDataHuC1)); 3.4710 + utilGzRead(gzFile, &gbDataHuC3, sizeof(gbDataHuC3)); 3.4711 + 3.4712 + if (version >= GBSAVE_GAME_VERSION_12) 3.4713 + { 3.4714 + utilGzRead(gzFile, pix, 4 * 257 * 226); 3.4715 + } 3.4716 + else 3.4717 + { 3.4718 + memset(pix, 0, 257 * 226 * sizeof(u32)); 3.4719 + // if(version < GBSAVE_GAME_VERSION_5) 3.4720 + // utilGzRead(gzFile, pix, 256*224*sizeof(u16)); 3.4721 + } 3.4722 + 3.4723 + if (version < GBSAVE_GAME_VERSION_6) 3.4724 + { 3.4725 + utilGzRead(gzFile, gbPalette, 64 * sizeof(u16)); 3.4726 + } 3.4727 + else 3.4728 + utilGzRead(gzFile, gbPalette, 128 * sizeof(u16)); 3.4729 + 3.4730 + // todo: remove 3.4731 + utilGzRead(gzFile, gbPalette, 128 * sizeof(u16)); 3.4732 + 3.4733 + if (version < GBSAVE_GAME_VERSION_10) 3.4734 + { 3.4735 + if (!gbCgbMode && !gbSgbMode) 3.4736 { 3.4737 - sprintf(tempBackupName, "gbatempsave%d.sav", tempStateID++); 3.4738 - gbWriteSaveState(tempBackupName); 3.4739 + for (int i = 0; i < 8; i++) 3.4740 + gbPalette[i] = systemGbPalette[gbPaletteOption * 8 + i]; 3.4741 } 3.4742 - 3.4743 - int version = utilReadInt(gzFile); 3.4744 - 3.4745 - if (version > GBSAVE_GAME_VERSION || version < 0) 3.4746 + } 3.4747 + 3.4748 + utilGzRead(gzFile, &gbMemory[0x8000], 0x8000); 3.4749 + 3.4750 + if (gbRamSize && gbRam) 3.4751 + { 3.4752 + utilGzRead(gzFile, gbRam, gbRamSize); 3.4753 + } 3.4754 + 3.4755 + gbMemoryMap[0x00] = &gbRom[0x0000]; 3.4756 + gbMemoryMap[0x01] = &gbRom[0x1000]; 3.4757 + gbMemoryMap[0x02] = &gbRom[0x2000]; 3.4758 + gbMemoryMap[0x03] = &gbRom[0x3000]; 3.4759 + gbMemoryMap[0x04] = &gbRom[0x4000]; 3.4760 + gbMemoryMap[0x05] = &gbRom[0x5000]; 3.4761 + gbMemoryMap[0x06] = &gbRom[0x6000]; 3.4762 + gbMemoryMap[0x07] = &gbRom[0x7000]; 3.4763 + gbMemoryMap[0x08] = &gbMemory[0x8000]; 3.4764 + gbMemoryMap[0x09] = &gbMemory[0x9000]; 3.4765 + gbMemoryMap[0x0a] = &gbMemory[0xa000]; 3.4766 + gbMemoryMap[0x0b] = &gbMemory[0xb000]; 3.4767 + gbMemoryMap[0x0c] = &gbMemory[0xc000]; 3.4768 + gbMemoryMap[0x0d] = &gbMemory[0xd000]; 3.4769 + gbMemoryMap[0x0e] = &gbMemory[0xe000]; 3.4770 + gbMemoryMap[0x0f] = &gbMemory[0xf000]; 3.4771 + 3.4772 + type = gbRom[0x147]; 3.4773 + 3.4774 + switch (type) 3.4775 + { 3.4776 + case 0x00: 3.4777 + case 0x01: 3.4778 + case 0x02: 3.4779 + case 0x03: 3.4780 + // MBC 1 3.4781 + memoryUpdateMapMBC1(); 3.4782 + break; 3.4783 + case 0x05: 3.4784 + case 0x06: 3.4785 + // MBC2 3.4786 + memoryUpdateMapMBC2(); 3.4787 + break; 3.4788 + case 0x0f: 3.4789 + case 0x10: 3.4790 + case 0x11: 3.4791 + case 0x12: 3.4792 + case 0x13: 3.4793 + // MBC 3 3.4794 + memoryUpdateMapMBC3(); 3.4795 + break; 3.4796 + case 0x19: 3.4797 + case 0x1a: 3.4798 + case 0x1b: 3.4799 + // MBC5 3.4800 + memoryUpdateMapMBC5(); 3.4801 + break; 3.4802 + case 0x1c: 3.4803 + case 0x1d: 3.4804 + case 0x1e: 3.4805 + // MBC 5 Rumble 3.4806 + memoryUpdateMapMBC5(); 3.4807 + break; 3.4808 + case 0x22: 3.4809 + // MBC 7 3.4810 + memoryUpdateMapMBC7(); 3.4811 + break; 3.4812 + case 0xfe: 3.4813 + // HuC3 3.4814 + memoryUpdateMapHuC3(); 3.4815 + break; 3.4816 + case 0xff: 3.4817 + // HuC1 3.4818 + memoryUpdateMapHuC1(); 3.4819 + break; 3.4820 + } 3.4821 + 3.4822 + if (gbCgbMode) 3.4823 + { 3.4824 + if (!gbVram) 3.4825 + gbVram = (u8 *)malloc(0x4000 + 4); 3.4826 + if (!gbWram) 3.4827 + gbWram = (u8 *)malloc(0x8000 + 4); 3.4828 + utilGzRead(gzFile, gbVram, 0x4000); 3.4829 + utilGzRead(gzFile, gbWram, 0x8000); 3.4830 + 3.4831 + int value = register_SVBK; 3.4832 + if (value == 0) 3.4833 + value = 1; 3.4834 + 3.4835 + gbMemoryMap[0x08] = &gbVram[register_VBK * 0x2000]; 3.4836 + gbMemoryMap[0x09] = &gbVram[register_VBK * 0x2000 + 0x1000]; 3.4837 + gbMemoryMap[0x0d] = &gbWram[value * 0x1000]; 3.4838 + } 3.4839 + else 3.4840 + { 3.4841 + if (gbVram) 3.4842 { 3.4843 - systemMessage(MSG_UNSUPPORTED_VB_SGM, 3.4844 - N_("Unsupported VisualBoy save game version %d"), version); 3.4845 - goto failedLoadGB; 3.4846 + free(gbVram); 3.4847 + gbVram = NULL; 3.4848 } 3.4849 - 3.4850 - u8 romname[20]; 3.4851 - 3.4852 - utilGzRead(gzFile, romname, 15); 3.4853 - 3.4854 - if (memcmp(&gbRom[0x134], romname, 15) != 0) 3.4855 + if (gbWram) 3.4856 { 3.4857 - systemMessage(MSG_CANNOT_LOAD_SGM_FOR, 3.4858 - N_("Cannot load save game for %s. Playing %s"), 3.4859 - romname, &gbRom[0x134]); 3.4860 - goto failedLoadGB; 3.4861 + free(gbWram); 3.4862 + gbWram = NULL; 3.4863 } 3.4864 - 3.4865 - utilReadData(gzFile, gbSaveGameStruct); 3.4866 - 3.4867 - if (version >= GBSAVE_GAME_VERSION_7) 3.4868 + } 3.4869 + 3.4870 + gbSoundReadGame(version, gzFile); 3.4871 + 3.4872 +#if 0 3.4873 + if (gbBorderOn) 3.4874 + { 3.4875 + gbSgbRenderBorder(); 3.4876 + } 3.4877 + 3.4878 + systemRefreshScreen(); 3.4879 +#endif 3.4880 + 3.4881 + if (version > GBSAVE_GAME_VERSION_1) 3.4882 + gbCheatsReadGame(gzFile, version); 3.4883 + 3.4884 + systemSaveUpdateCounter = SYSTEM_SAVE_NOT_UPDATED; 3.4885 + 3.4886 + if (version >= GBSAVE_GAME_VERSION_11) // new to re-recording version: 3.4887 + { 3.4888 + extern int32 sensorX, sensorY; // from SDL.cpp 3.4889 + utilGzRead(gzFile, &sensorX, sizeof(sensorX)); 3.4890 + utilGzRead(gzFile, &sensorY, sizeof(sensorY)); 3.4891 + utilGzRead(gzFile, gbJoymask, 4 * sizeof(*gbJoymask)); // this has to be saved or old input will incorrectly get carried 3.4892 + // back on loading a snapshot! 3.4893 + 3.4894 + bool8 movieSnapshot; 3.4895 + utilGzRead(gzFile, &movieSnapshot, sizeof(movieSnapshot)); 3.4896 + if (VBAMovieActive() && !movieSnapshot) 3.4897 { 3.4898 - utilGzRead(gzFile, &IFF, 2); 3.4899 + systemMessage(0, N_("Can't load a non-movie snapshot while a movie is active.")); 3.4900 + goto failedLoadGB; 3.4901 } 3.4902 3.4903 - if (gbSgbMode) 3.4904 + if (movieSnapshot) // even if a movie isn't active we still want to parse through this in case other stuff is added 3.4905 + // later on in the save format 3.4906 { 3.4907 - gbSgbReadGame(gzFile, version); 3.4908 + uint32 movieInputDataSize = 0; 3.4909 + utilGzRead(gzFile, &movieInputDataSize, sizeof(movieInputDataSize)); 3.4910 + uint8 *local_movie_data = new uint8 [movieInputDataSize]; 3.4911 + int readBytes = utilGzRead(gzFile, local_movie_data, movieInputDataSize); 3.4912 + if (readBytes != movieInputDataSize) 3.4913 + { 3.4914 + systemMessage(0, N_("Corrupt movie snapshot.")); 3.4915 + if (local_movie_data) 3.4916 + delete [] local_movie_data; 3.4917 + goto failedLoadGB; 3.4918 + } 3.4919 + int code = VBAMovieUnfreeze(local_movie_data, movieInputDataSize); 3.4920 + if (local_movie_data) 3.4921 + delete [] local_movie_data; 3.4922 + if (code != MOVIE_SUCCESS && VBAMovieActive()) 3.4923 + { 3.4924 + char errStr [1024]; 3.4925 + strcpy(errStr, "Failed to load movie snapshot"); 3.4926 + switch (code) 3.4927 + { 3.4928 + case MOVIE_NOT_FROM_THIS_MOVIE: 3.4929 + strcat(errStr, ";\nSnapshot not from this movie"); break; 3.4930 + case MOVIE_NOT_FROM_A_MOVIE: 3.4931 + strcat(errStr, ";\nNot a movie snapshot"); break; // shouldn't get here... 3.4932 + case MOVIE_SNAPSHOT_INCONSISTENT: 3.4933 + strcat(errStr, ";\nSnapshot inconsistent with movie"); break; 3.4934 + case MOVIE_WRONG_FORMAT: 3.4935 + strcat(errStr, ";\nWrong format"); break; 3.4936 + } 3.4937 + strcat(errStr, "."); 3.4938 + systemMessage(0, N_(errStr)); 3.4939 + goto failedLoadGB; 3.4940 + } 3.4941 } 3.4942 - else 3.4943 - { 3.4944 - gbSgbMask = 0; // loading a game at the wrong time causes no display 3.4945 - } 3.4946 - 3.4947 - utilGzRead(gzFile, &gbDataMBC1, sizeof(gbDataMBC1)); 3.4948 - utilGzRead(gzFile, &gbDataMBC2, sizeof(gbDataMBC2)); 3.4949 - if (version < GBSAVE_GAME_VERSION_4) 3.4950 - // prior to version 4, there was no adjustment for the time the game 3.4951 - // was last played, so we have less to read. This needs update if the 3.4952 - // structure changes again. 3.4953 - utilGzRead(gzFile, &gbDataMBC3, sizeof(int32) * 10); 3.4954 - else 3.4955 - { 3.4956 - //assert(sizeof(time_t) == 4); 3.4957 - utilGzRead(gzFile, &gbDataMBC3, sizeof(gbDataMBC3)); 3.4958 - } 3.4959 - utilGzRead(gzFile, &gbDataMBC5, sizeof(gbDataMBC5)); 3.4960 - utilGzRead(gzFile, &gbDataHuC1, sizeof(gbDataHuC1)); 3.4961 - utilGzRead(gzFile, &gbDataHuC3, sizeof(gbDataHuC3)); 3.4962 - 3.4963 - if (version >= GBSAVE_GAME_VERSION_12) 3.4964 - { 3.4965 - utilGzRead(gzFile, pix, 4 * 257 * 226); 3.4966 - } 3.4967 - else 3.4968 - { 3.4969 - memset(pix, 0, 257 * 226 * sizeof(u32)); 3.4970 -// if(version < GBSAVE_GAME_VERSION_5) 3.4971 -// utilGzRead(gzFile, pix, 256*224*sizeof(u16)); 3.4972 - } 3.4973 - 3.4974 - if (version < GBSAVE_GAME_VERSION_6) 3.4975 - { 3.4976 - utilGzRead(gzFile, gbPalette, 64 * sizeof(u16)); 3.4977 - } 3.4978 - else 3.4979 - utilGzRead(gzFile, gbPalette, 128 * sizeof(u16)); 3.4980 - 3.4981 - // todo: remove 3.4982 - utilGzRead(gzFile, gbPalette, 128 * sizeof(u16)); 3.4983 - 3.4984 - if (version < GBSAVE_GAME_VERSION_10) 3.4985 - { 3.4986 - if (!gbCgbMode && !gbSgbMode) 3.4987 - { 3.4988 - for (int i = 0; i < 8; i++) 3.4989 - gbPalette[i] = systemGbPalette[gbPaletteOption * 8 + i]; 3.4990 - } 3.4991 - } 3.4992 - 3.4993 - utilGzRead(gzFile, &gbMemory[0x8000], 0x8000); 3.4994 - 3.4995 - if (gbRamSize && gbRam) 3.4996 - { 3.4997 - utilGzRead(gzFile, gbRam, gbRamSize); 3.4998 - } 3.4999 - 3.5000 - gbMemoryMap[0x00] = &gbRom[0x0000]; 3.5001 - gbMemoryMap[0x01] = &gbRom[0x1000]; 3.5002 - gbMemoryMap[0x02] = &gbRom[0x2000]; 3.5003 - gbMemoryMap[0x03] = &gbRom[0x3000]; 3.5004 - gbMemoryMap[0x04] = &gbRom[0x4000]; 3.5005 - gbMemoryMap[0x05] = &gbRom[0x5000]; 3.5006 - gbMemoryMap[0x06] = &gbRom[0x6000]; 3.5007 - gbMemoryMap[0x07] = &gbRom[0x7000]; 3.5008 - gbMemoryMap[0x08] = &gbMemory[0x8000]; 3.5009 - gbMemoryMap[0x09] = &gbMemory[0x9000]; 3.5010 - gbMemoryMap[0x0a] = &gbMemory[0xa000]; 3.5011 - gbMemoryMap[0x0b] = &gbMemory[0xb000]; 3.5012 - gbMemoryMap[0x0c] = &gbMemory[0xc000]; 3.5013 - gbMemoryMap[0x0d] = &gbMemory[0xd000]; 3.5014 - gbMemoryMap[0x0e] = &gbMemory[0xe000]; 3.5015 - gbMemoryMap[0x0f] = &gbMemory[0xf000]; 3.5016 - 3.5017 - type = gbRom[0x147]; 3.5018 - 3.5019 - switch (type) 3.5020 - { 3.5021 - case 0x00: 3.5022 - case 0x01: 3.5023 - case 0x02: 3.5024 - case 0x03: 3.5025 - // MBC 1 3.5026 - memoryUpdateMapMBC1(); 3.5027 - break; 3.5028 - case 0x05: 3.5029 - case 0x06: 3.5030 - // MBC2 3.5031 - memoryUpdateMapMBC2(); 3.5032 - break; 3.5033 - case 0x0f: 3.5034 - case 0x10: 3.5035 - case 0x11: 3.5036 - case 0x12: 3.5037 - case 0x13: 3.5038 - // MBC 3 3.5039 - memoryUpdateMapMBC3(); 3.5040 - break; 3.5041 - case 0x19: 3.5042 - case 0x1a: 3.5043 - case 0x1b: 3.5044 - // MBC5 3.5045 - memoryUpdateMapMBC5(); 3.5046 - break; 3.5047 - case 0x1c: 3.5048 - case 0x1d: 3.5049 - case 0x1e: 3.5050 - // MBC 5 Rumble 3.5051 - memoryUpdateMapMBC5(); 3.5052 - break; 3.5053 - case 0x22: 3.5054 - // MBC 7 3.5055 - memoryUpdateMapMBC7(); 3.5056 - break; 3.5057 - case 0xfe: 3.5058 - // HuC3 3.5059 - memoryUpdateMapHuC3(); 3.5060 - break; 3.5061 - case 0xff: 3.5062 - // HuC1 3.5063 - memoryUpdateMapHuC1(); 3.5064 - break; 3.5065 - } 3.5066 - 3.5067 - if (gbCgbMode) 3.5068 - { 3.5069 - if (!gbVram) 3.5070 - gbVram = (u8 *)malloc(0x4000 + 4); 3.5071 - if (!gbWram) 3.5072 - gbWram = (u8 *)malloc(0x8000 + 4); 3.5073 - utilGzRead(gzFile, gbVram, 0x4000); 3.5074 - utilGzRead(gzFile, gbWram, 0x8000); 3.5075 - 3.5076 - int value = register_SVBK; 3.5077 - if (value == 0) 3.5078 - value = 1; 3.5079 - 3.5080 - gbMemoryMap[0x08] = &gbVram[register_VBK * 0x2000]; 3.5081 - gbMemoryMap[0x09] = &gbVram[register_VBK * 0x2000 + 0x1000]; 3.5082 - gbMemoryMap[0x0d] = &gbWram[value * 0x1000]; 3.5083 - } 3.5084 - else 3.5085 - { 3.5086 - if (gbVram) 3.5087 - { 3.5088 - free(gbVram); 3.5089 - gbVram = NULL; 3.5090 - } 3.5091 - if (gbWram) 3.5092 - { 3.5093 - free(gbWram); 3.5094 - gbWram = NULL; 3.5095 - } 3.5096 - } 3.5097 - 3.5098 - gbSoundReadGame(version, gzFile); 3.5099 - 3.5100 -#if 0 3.5101 - if (gbBorderOn) 3.5102 - { 3.5103 - gbSgbRenderBorder(); 3.5104 - } 3.5105 - 3.5106 - systemRefreshScreen(); 3.5107 -#endif 3.5108 - 3.5109 - if (version > GBSAVE_GAME_VERSION_1) 3.5110 - gbCheatsReadGame(gzFile, version); 3.5111 - 3.5112 - systemSaveUpdateCounter = SYSTEM_SAVE_NOT_UPDATED; 3.5113 - 3.5114 - if (version >= GBSAVE_GAME_VERSION_11) // new to re-recording version: 3.5115 - { 3.5116 - extern int32 sensorX, sensorY; // from SDL.cpp 3.5117 - utilGzRead(gzFile, &sensorX, sizeof(sensorX)); 3.5118 - utilGzRead(gzFile, &sensorY, sizeof(sensorY)); 3.5119 - utilGzRead(gzFile, gbJoymask, 4 * sizeof(*gbJoymask)); // this has to be saved or old input will incorrectly get carried 3.5120 - // back on loading a snapshot! 3.5121 - 3.5122 - bool8 movieSnapshot; 3.5123 - utilGzRead(gzFile, &movieSnapshot, sizeof(movieSnapshot)); 3.5124 - if (VBAMovieActive() && !movieSnapshot) 3.5125 - { 3.5126 - systemMessage(0, N_("Can't load a non-movie snapshot while a movie is active.")); 3.5127 - goto failedLoadGB; 3.5128 - } 3.5129 - 3.5130 - if (movieSnapshot) // even if a movie isn't active we still want to parse through this in case other stuff is added 3.5131 - // later on in the save format 3.5132 - { 3.5133 - uint32 movieInputDataSize = 0; 3.5134 - utilGzRead(gzFile, &movieInputDataSize, sizeof(movieInputDataSize)); 3.5135 - uint8 *local_movie_data = new uint8 [movieInputDataSize]; 3.5136 - int readBytes = utilGzRead(gzFile, local_movie_data, movieInputDataSize); 3.5137 - if (readBytes != movieInputDataSize) 3.5138 - { 3.5139 - systemMessage(0, N_("Corrupt movie snapshot.")); 3.5140 - if (local_movie_data) 3.5141 - delete [] local_movie_data; 3.5142 - goto failedLoadGB; 3.5143 - } 3.5144 - int code = VBAMovieUnfreeze(local_movie_data, movieInputDataSize); 3.5145 - if (local_movie_data) 3.5146 - delete [] local_movie_data; 3.5147 - if (code != MOVIE_SUCCESS && VBAMovieActive()) 3.5148 - { 3.5149 - char errStr [1024]; 3.5150 - strcpy(errStr, "Failed to load movie snapshot"); 3.5151 - switch (code) 3.5152 - { 3.5153 - case MOVIE_NOT_FROM_THIS_MOVIE: 3.5154 - strcat(errStr, ";\nSnapshot not from this movie"); break; 3.5155 - case MOVIE_NOT_FROM_A_MOVIE: 3.5156 - strcat(errStr, ";\nNot a movie snapshot"); break; // shouldn't get here... 3.5157 - case MOVIE_SNAPSHOT_INCONSISTENT: 3.5158 - strcat(errStr, ";\nSnapshot inconsistent with movie"); break; 3.5159 - case MOVIE_WRONG_FORMAT: 3.5160 - strcat(errStr, ";\nWrong format"); break; 3.5161 - } 3.5162 - strcat(errStr, "."); 3.5163 - systemMessage(0, N_(errStr)); 3.5164 - goto failedLoadGB; 3.5165 - } 3.5166 - } 3.5167 - utilGzRead(gzFile, &GBSystemCounters.frameCount, sizeof(GBSystemCounters.frameCount)); 3.5168 - } 3.5169 - 3.5170 - if (version >= GBSAVE_GAME_VERSION_13) // new to rerecording 19.4 wip (svn r22+): 3.5171 - { 3.5172 - utilGzRead(gzFile, &GBSystemCounters.lagCount, sizeof(GBSystemCounters.lagCount)); 3.5173 - utilGzRead(gzFile, &GBSystemCounters.lagged, sizeof(GBSystemCounters.lagged)); 3.5174 - utilGzRead(gzFile, &GBSystemCounters.laggedLast, sizeof(GBSystemCounters.laggedLast)); 3.5175 - } 3.5176 - 3.5177 - if (backupSafe) 3.5178 - { 3.5179 - remove(tempBackupName); 3.5180 - tempFailCount = 0; 3.5181 - } 3.5182 - 3.5183 - for (int i = 0; i < 4; ++i) 3.5184 - systemSetJoypad(i, gbJoymask[i] & 0xFFFF); 3.5185 - 3.5186 - // FIXME: horrible kludge 3.5187 - memcpy(s_gbJoymask, gbJoymask, sizeof(gbJoymask)); 3.5188 - 3.5189 - VBAUpdateButtonPressDisplay(); 3.5190 - VBAUpdateFrameCountDisplay(); 3.5191 - systemRefreshScreen(); 3.5192 - return true; 3.5193 - 3.5194 -failedLoadGB: 3.5195 - if (backupSafe) 3.5196 - { 3.5197 - tempFailCount++; 3.5198 - if (tempFailCount < 3) // fail no more than 2 times in a row 3.5199 - gbReadSaveState(tempBackupName); 3.5200 - remove(tempBackupName); 3.5201 - } 3.5202 - return false; 3.5203 + utilGzRead(gzFile, &GBSystemCounters.frameCount, sizeof(GBSystemCounters.frameCount)); 3.5204 + } 3.5205 + 3.5206 + if (version >= GBSAVE_GAME_VERSION_13) // new to rerecording 19.4 wip (svn r22+): 3.5207 + { 3.5208 + utilGzRead(gzFile, &GBSystemCounters.lagCount, sizeof(GBSystemCounters.lagCount)); 3.5209 + utilGzRead(gzFile, &GBSystemCounters.lagged, sizeof(GBSystemCounters.lagged)); 3.5210 + utilGzRead(gzFile, &GBSystemCounters.laggedLast, sizeof(GBSystemCounters.laggedLast)); 3.5211 + } 3.5212 + 3.5213 + if (backupSafe) 3.5214 + { 3.5215 + remove(tempBackupName); 3.5216 + tempFailCount = 0; 3.5217 + } 3.5218 + 3.5219 + for (int i = 0; i < 4; ++i) 3.5220 + systemSetJoypad(i, gbJoymask[i] & 0xFFFF); 3.5221 + 3.5222 + // FIXME: horrible kludge 3.5223 + memcpy(s_gbJoymask, gbJoymask, sizeof(gbJoymask)); 3.5224 + 3.5225 + VBAUpdateButtonPressDisplay(); 3.5226 + VBAUpdateFrameCountDisplay(); 3.5227 + systemRefreshScreen(); 3.5228 + return true; 3.5229 + 3.5230 + failedLoadGB: 3.5231 + if (backupSafe) 3.5232 + { 3.5233 + tempFailCount++; 3.5234 + if (tempFailCount < 3) // fail no more than 2 times in a row 3.5235 + gbReadSaveState(tempBackupName); 3.5236 + remove(tempBackupName); 3.5237 + } 3.5238 + return false; 3.5239 } 3.5240 3.5241 bool gbReadMemSaveState(char *memory, int available) 3.5242 { 3.5243 - gzFile gzFile = utilMemGzOpen(memory, available, "r"); 3.5244 - 3.5245 - backupSafe = false; 3.5246 - bool res = gbReadSaveStateFromStream(gzFile); 3.5247 - backupSafe = true; 3.5248 - 3.5249 - utilGzClose(gzFile); 3.5250 - 3.5251 - return res; 3.5252 + gzFile gzFile = utilMemGzOpen(memory, available, "r"); 3.5253 + 3.5254 + backupSafe = false; 3.5255 + bool res = gbReadSaveStateFromStream(gzFile); 3.5256 + backupSafe = true; 3.5257 + 3.5258 + utilGzClose(gzFile); 3.5259 + 3.5260 + return res; 3.5261 } 3.5262 3.5263 bool gbReadSaveState(const char *name) 3.5264 { 3.5265 - gzFile gzFile = utilGzOpen(name, "rb"); 3.5266 - 3.5267 - if (gzFile == NULL) 3.5268 - { 3.5269 - return false; 3.5270 - } 3.5271 - 3.5272 - bool res = gbReadSaveStateFromStream(gzFile); 3.5273 - 3.5274 - utilGzClose(gzFile); 3.5275 - 3.5276 - return res; 3.5277 + gzFile gzFile = utilGzOpen(name, "rb"); 3.5278 + 3.5279 + if (gzFile == NULL) 3.5280 + { 3.5281 + return false; 3.5282 + } 3.5283 + 3.5284 + bool res = gbReadSaveStateFromStream(gzFile); 3.5285 + 3.5286 + utilGzClose(gzFile); 3.5287 + 3.5288 + return res; 3.5289 } 3.5290 3.5291 bool gbWritePNGFile(const char *fileName) 3.5292 { 3.5293 - if (gbBorderOn) 3.5294 - return utilWritePNGFile(fileName, 256, 224, pix); 3.5295 - return utilWritePNGFile(fileName, 160, 144, pix); 3.5296 + if (gbBorderOn) 3.5297 + return utilWritePNGFile(fileName, 256, 224, pix); 3.5298 + return utilWritePNGFile(fileName, 160, 144, pix); 3.5299 } 3.5300 3.5301 bool gbWriteBMPFile(const char *fileName) 3.5302 { 3.5303 - if (gbBorderOn) 3.5304 - return utilWriteBMPFile(fileName, 256, 224, pix); 3.5305 - return utilWriteBMPFile(fileName, 160, 144, pix); 3.5306 + if (gbBorderOn) 3.5307 + return utilWriteBMPFile(fileName, 256, 224, pix); 3.5308 + return utilWriteBMPFile(fileName, 160, 144, pix); 3.5309 } 3.5310 3.5311 void gbCleanUp() 3.5312 { 3.5313 - newFrame = true; 3.5314 - 3.5315 - GBSystemCounters.frameCount = 0; 3.5316 - GBSystemCounters.lagCount = 0; 3.5317 - GBSystemCounters.extraCount = 0; 3.5318 - GBSystemCounters.lagged = true; 3.5319 - GBSystemCounters.laggedLast = true; 3.5320 - 3.5321 - if (gbRam != NULL) 3.5322 - { 3.5323 - free(gbRam); 3.5324 - gbRam = NULL; 3.5325 - } 3.5326 - 3.5327 - if (gbRom != NULL) 3.5328 - { 3.5329 - free(gbRom); 3.5330 - gbRom = NULL; 3.5331 - } 3.5332 - 3.5333 - if (gbMemory != NULL) 3.5334 - { 3.5335 - free(gbMemory); 3.5336 - gbMemory = NULL; 3.5337 - } 3.5338 - 3.5339 - if (gbLineBuffer != NULL) 3.5340 - { 3.5341 - free(gbLineBuffer); 3.5342 - gbLineBuffer = NULL; 3.5343 - } 3.5344 - 3.5345 - if (origPix != NULL) 3.5346 - { 3.5347 - free(origPix); 3.5348 - origPix = NULL; 3.5349 - } 3.5350 - pix = NULL; 3.5351 - 3.5352 - gbSgbShutdown(); 3.5353 - 3.5354 - if (gbVram != NULL) 3.5355 - { 3.5356 - free(gbVram); 3.5357 - gbVram = NULL; 3.5358 - } 3.5359 - 3.5360 - if (gbWram != NULL) 3.5361 - { 3.5362 - free(gbWram); 3.5363 - gbWram = NULL; 3.5364 - } 3.5365 - 3.5366 - systemSaveUpdateCounter = SYSTEM_SAVE_NOT_UPDATED; 3.5367 - 3.5368 - memset(gbJoymask, 0, sizeof(gbJoymask)); 3.5369 - // FIXME: horrible kludge 3.5370 - memset(s_gbJoymask, 0, sizeof(s_gbJoymask)); 3.5371 - 3.5372 - systemClearJoypads(); 3.5373 - systemResetSensor(); 3.5374 - 3.5375 -// gbLastTime = gbFrameCount = 0; 3.5376 - systemRefreshScreen(); 3.5377 + newFrame = true; 3.5378 + 3.5379 + GBSystemCounters.frameCount = 0; 3.5380 + GBSystemCounters.lagCount = 0; 3.5381 + GBSystemCounters.extraCount = 0; 3.5382 + GBSystemCounters.lagged = true; 3.5383 + GBSystemCounters.laggedLast = true; 3.5384 + 3.5385 + if (gbRam != NULL) 3.5386 + { 3.5387 + free(gbRam); 3.5388 + gbRam = NULL; 3.5389 + } 3.5390 + 3.5391 + if (gbRom != NULL) 3.5392 + { 3.5393 + free(gbRom); 3.5394 + gbRom = NULL; 3.5395 + } 3.5396 + 3.5397 + if (gbMemory != NULL) 3.5398 + { 3.5399 + free(gbMemory); 3.5400 + gbMemory = NULL; 3.5401 + } 3.5402 + 3.5403 + if (gbLineBuffer != NULL) 3.5404 + { 3.5405 + free(gbLineBuffer); 3.5406 + gbLineBuffer = NULL; 3.5407 + } 3.5408 + 3.5409 + if (origPix != NULL) 3.5410 + { 3.5411 + free(origPix); 3.5412 + origPix = NULL; 3.5413 + } 3.5414 + pix = NULL; 3.5415 + 3.5416 + gbSgbShutdown(); 3.5417 + 3.5418 + if (gbVram != NULL) 3.5419 + { 3.5420 + free(gbVram); 3.5421 + gbVram = NULL; 3.5422 + } 3.5423 + 3.5424 + if (gbWram != NULL) 3.5425 + { 3.5426 + free(gbWram); 3.5427 + gbWram = NULL; 3.5428 + } 3.5429 + 3.5430 + systemSaveUpdateCounter = SYSTEM_SAVE_NOT_UPDATED; 3.5431 + 3.5432 + memset(gbJoymask, 0, sizeof(gbJoymask)); 3.5433 + // FIXME: horrible kludge 3.5434 + memset(s_gbJoymask, 0, sizeof(s_gbJoymask)); 3.5435 + 3.5436 + systemClearJoypads(); 3.5437 + systemResetSensor(); 3.5438 + 3.5439 + // gbLastTime = gbFrameCount = 0; 3.5440 + systemRefreshScreen(); 3.5441 } 3.5442 3.5443 bool gbLoadRom(const char *szFile) 3.5444 { 3.5445 - int size = 0; 3.5446 - 3.5447 - if (gbRom != NULL) 3.5448 - { 3.5449 - gbCleanUp(); 3.5450 - } 3.5451 - 3.5452 - systemSaveUpdateCounter = SYSTEM_SAVE_NOT_UPDATED; 3.5453 - 3.5454 - gbRom = utilLoad(szFile, 3.5455 - utilIsGBImage, 3.5456 - NULL, 3.5457 - size); 3.5458 - if (!gbRom) 3.5459 - return false; 3.5460 - 3.5461 - gbRomSize = size; 3.5462 - 3.5463 - return gbUpdateSizes(); 3.5464 + int size = 0; 3.5465 + 3.5466 + if (gbRom != NULL) 3.5467 + { 3.5468 + gbCleanUp(); 3.5469 + } 3.5470 + 3.5471 + systemSaveUpdateCounter = SYSTEM_SAVE_NOT_UPDATED; 3.5472 + 3.5473 + gbRom = utilLoad(szFile, 3.5474 + utilIsGBImage, 3.5475 + NULL, 3.5476 + size); 3.5477 + if (!gbRom) 3.5478 + return false; 3.5479 + 3.5480 + gbRomSize = size; 3.5481 + 3.5482 + return gbUpdateSizes(); 3.5483 } 3.5484 3.5485 bool gbUpdateSizes() 3.5486 { 3.5487 - if (gbRom[0x148] > 8) 3.5488 - { 3.5489 - systemMessage(MSG_UNSUPPORTED_ROM_SIZE, 3.5490 - N_("Unsupported rom size %02x"), gbRom[0x148]); 3.5491 - return false; 3.5492 - } 3.5493 - 3.5494 - if (gbRomSize < gbRomSizes[gbRom[0x148]]) 3.5495 - { 3.5496 - gbRom = (u8 *)realloc(gbRom, gbRomSizes[gbRom[0x148]]); 3.5497 - } 3.5498 - gbRomSize = gbRomSizes[gbRom[0x148]]; 3.5499 - gbRomSizeMask = gbRomSizesMasks[gbRom[0x148]]; 3.5500 - 3.5501 - if (gbRom[0x149] > 5) 3.5502 - { 3.5503 - systemMessage(MSG_UNSUPPORTED_RAM_SIZE, 3.5504 - N_("Unsupported ram size %02x"), gbRom[0x149]); 3.5505 - return false; 3.5506 - } 3.5507 - 3.5508 - gbRamSize = gbRamSizes[gbRom[0x149]]; 3.5509 - gbRamSizeMask = gbRamSizesMasks[gbRom[0x149]]; 3.5510 - 3.5511 - if (gbRamSize) 3.5512 - { 3.5513 - gbRam = (u8 *)malloc(gbRamSize + 4); 3.5514 - memset(gbRam, 0xFF, gbRamSize + 4); 3.5515 - } 3.5516 - 3.5517 - int type = gbRom[0x147]; 3.5518 - 3.5519 - mapperReadRAM = NULL; 3.5520 - 3.5521 - switch (type) 3.5522 - { 3.5523 - case 0x00: 3.5524 - case 0x01: 3.5525 - case 0x02: 3.5526 - case 0x03: 3.5527 - // MBC 1 3.5528 - mapper = mapperMBC1ROM; 3.5529 - mapperRAM = mapperMBC1RAM; 3.5530 - break; 3.5531 - case 0x05: 3.5532 - case 0x06: 3.5533 - // MBC2 3.5534 - mapper = mapperMBC2ROM; 3.5535 - mapperRAM = mapperMBC2RAM; 3.5536 - gbRamSize = 0x200; 3.5537 - gbRamSizeMask = 0x1ff; 3.5538 - break; 3.5539 - case 0x0f: 3.5540 - case 0x10: 3.5541 - case 0x11: 3.5542 - case 0x12: 3.5543 - case 0x13: 3.5544 - // MBC 3 3.5545 - mapper = mapperMBC3ROM; 3.5546 - mapperRAM = mapperMBC3RAM; 3.5547 - mapperReadRAM = mapperMBC3ReadRAM; 3.5548 - break; 3.5549 - case 0x19: 3.5550 - case 0x1a: 3.5551 - case 0x1b: 3.5552 - // MBC5 3.5553 - mapper = mapperMBC5ROM; 3.5554 - mapperRAM = mapperMBC5RAM; 3.5555 - break; 3.5556 - case 0x1c: 3.5557 - case 0x1d: 3.5558 - case 0x1e: 3.5559 - // MBC 5 Rumble 3.5560 - mapper = mapperMBC5ROM; 3.5561 - mapperRAM = mapperMBC5RAM; 3.5562 - break; 3.5563 - case 0x22: 3.5564 - // MBC 7 3.5565 - mapper = mapperMBC7ROM; 3.5566 - mapperRAM = mapperMBC7RAM; 3.5567 - mapperReadRAM = mapperMBC7ReadRAM; 3.5568 - break; 3.5569 - case 0xfe: 3.5570 - // HuC3 3.5571 - mapper = mapperHuC3ROM; 3.5572 - mapperRAM = mapperHuC3RAM; 3.5573 - mapperReadRAM = mapperHuC3ReadRAM; 3.5574 - break; 3.5575 - case 0xff: 3.5576 - // HuC1 3.5577 - mapper = mapperHuC1ROM; 3.5578 - mapperRAM = mapperHuC1RAM; 3.5579 - break; 3.5580 - default: 3.5581 - systemMessage(MSG_UNKNOWN_CARTRIDGE_TYPE, 3.5582 - N_("Unknown cartridge type %02x"), type); 3.5583 - return false; 3.5584 - } 3.5585 - 3.5586 - switch (type) 3.5587 - { 3.5588 - case 0x03: 3.5589 - case 0x06: 3.5590 - case 0x0f: 3.5591 - case 0x10: 3.5592 - case 0x13: 3.5593 - case 0x1b: 3.5594 - case 0x1d: 3.5595 - case 0x1e: 3.5596 - case 0x22: 3.5597 - case 0xff: 3.5598 - gbBattery = 1; 3.5599 - break; 3.5600 - } 3.5601 - 3.5602 - gbInit(); 3.5603 - gbReset(); 3.5604 - 3.5605 - return true; 3.5606 + if (gbRom[0x148] > 8) 3.5607 + { 3.5608 + systemMessage(MSG_UNSUPPORTED_ROM_SIZE, 3.5609 + N_("Unsupported rom size %02x"), gbRom[0x148]); 3.5610 + return false; 3.5611 + } 3.5612 + 3.5613 + if (gbRomSize < gbRomSizes[gbRom[0x148]]) 3.5614 + { 3.5615 + gbRom = (u8 *)realloc(gbRom, gbRomSizes[gbRom[0x148]]); 3.5616 + } 3.5617 + gbRomSize = gbRomSizes[gbRom[0x148]]; 3.5618 + gbRomSizeMask = gbRomSizesMasks[gbRom[0x148]]; 3.5619 + 3.5620 + if (gbRom[0x149] > 5) 3.5621 + { 3.5622 + systemMessage(MSG_UNSUPPORTED_RAM_SIZE, 3.5623 + N_("Unsupported ram size %02x"), gbRom[0x149]); 3.5624 + return false; 3.5625 + } 3.5626 + 3.5627 + gbRamSize = gbRamSizes[gbRom[0x149]]; 3.5628 + gbRamSizeMask = gbRamSizesMasks[gbRom[0x149]]; 3.5629 + 3.5630 + if (gbRamSize) 3.5631 + { 3.5632 + gbRam = (u8 *)malloc(gbRamSize + 4); 3.5633 + memset(gbRam, 0xFF, gbRamSize + 4); 3.5634 + } 3.5635 + 3.5636 + int type = gbRom[0x147]; 3.5637 + 3.5638 + mapperReadRAM = NULL; 3.5639 + 3.5640 + switch (type) 3.5641 + { 3.5642 + case 0x00: 3.5643 + case 0x01: 3.5644 + case 0x02: 3.5645 + case 0x03: 3.5646 + // MBC 1 3.5647 + mapper = mapperMBC1ROM; 3.5648 + mapperRAM = mapperMBC1RAM; 3.5649 + break; 3.5650 + case 0x05: 3.5651 + case 0x06: 3.5652 + // MBC2 3.5653 + mapper = mapperMBC2ROM; 3.5654 + mapperRAM = mapperMBC2RAM; 3.5655 + gbRamSize = 0x200; 3.5656 + gbRamSizeMask = 0x1ff; 3.5657 + break; 3.5658 + case 0x0f: 3.5659 + case 0x10: 3.5660 + case 0x11: 3.5661 + case 0x12: 3.5662 + case 0x13: 3.5663 + // MBC 3 3.5664 + mapper = mapperMBC3ROM; 3.5665 + mapperRAM = mapperMBC3RAM; 3.5666 + mapperReadRAM = mapperMBC3ReadRAM; 3.5667 + break; 3.5668 + case 0x19: 3.5669 + case 0x1a: 3.5670 + case 0x1b: 3.5671 + // MBC5 3.5672 + mapper = mapperMBC5ROM; 3.5673 + mapperRAM = mapperMBC5RAM; 3.5674 + break; 3.5675 + case 0x1c: 3.5676 + case 0x1d: 3.5677 + case 0x1e: 3.5678 + // MBC 5 Rumble 3.5679 + mapper = mapperMBC5ROM; 3.5680 + mapperRAM = mapperMBC5RAM; 3.5681 + break; 3.5682 + case 0x22: 3.5683 + // MBC 7 3.5684 + mapper = mapperMBC7ROM; 3.5685 + mapperRAM = mapperMBC7RAM; 3.5686 + mapperReadRAM = mapperMBC7ReadRAM; 3.5687 + break; 3.5688 + case 0xfe: 3.5689 + // HuC3 3.5690 + mapper = mapperHuC3ROM; 3.5691 + mapperRAM = mapperHuC3RAM; 3.5692 + mapperReadRAM = mapperHuC3ReadRAM; 3.5693 + break; 3.5694 + case 0xff: 3.5695 + // HuC1 3.5696 + mapper = mapperHuC1ROM; 3.5697 + mapperRAM = mapperHuC1RAM; 3.5698 + break; 3.5699 + default: 3.5700 + systemMessage(MSG_UNKNOWN_CARTRIDGE_TYPE, 3.5701 + N_("Unknown cartridge type %02x"), type); 3.5702 + return false; 3.5703 + } 3.5704 + 3.5705 + switch (type) 3.5706 + { 3.5707 + case 0x03: 3.5708 + case 0x06: 3.5709 + case 0x0f: 3.5710 + case 0x10: 3.5711 + case 0x13: 3.5712 + case 0x1b: 3.5713 + case 0x1d: 3.5714 + case 0x1e: 3.5715 + case 0x22: 3.5716 + case 0xff: 3.5717 + gbBattery = 1; 3.5718 + break; 3.5719 + } 3.5720 + 3.5721 + gbInit(); 3.5722 + gbReset(); 3.5723 + 3.5724 + return true; 3.5725 } 3.5726 3.5727 void gbEmulate(int ticksToStop) 3.5728 { 3.5729 - gbRegister tempRegister; 3.5730 - u8 tempValue; 3.5731 - s8 offset; 3.5732 - 3.5733 - int clockTicks = 0; 3.5734 - gbDmaTicks = 0; 3.5735 - 3.5736 - register int opcode = 0; 3.5737 - 3.5738 - u32 newmask = 0; 3.5739 - if (newFrame) 3.5740 + printf("RLM: Inside the GB!\n"); 3.5741 + gbRegister tempRegister; 3.5742 + u8 tempValue; 3.5743 + s8 offset; 3.5744 + 3.5745 + int clockTicks = 0; 3.5746 + gbDmaTicks = 0; 3.5747 + 3.5748 + register int opcode = 0; 3.5749 + 3.5750 + u32 newmask = 0; 3.5751 + printf("RLM: newframe = %d\n", newFrame); 3.5752 + if (newFrame) 3.5753 + { 3.5754 + extern void VBAOnExitingFrameBoundary(); 3.5755 + VBAOnExitingFrameBoundary(); 3.5756 + printf("RLM: exiting frame boundary?\n"); 3.5757 + // update joystick information 3.5758 + systemReadJoypads(); 3.5759 + 3.5760 + bool sensor = (gbRom[0x147] == 0x22); 3.5761 + 3.5762 + // read joystick 3.5763 + if (gbSgbMode && gbSgbMultiplayer) 3.5764 { 3.5765 - extern void VBAOnExitingFrameBoundary(); 3.5766 - VBAOnExitingFrameBoundary(); 3.5767 - 3.5768 - // update joystick information 3.5769 - systemReadJoypads(); 3.5770 - 3.5771 - bool sensor = (gbRom[0x147] == 0x22); 3.5772 - 3.5773 - // read joystick 3.5774 - if (gbSgbMode && gbSgbMultiplayer) 3.5775 + if (gbSgbFourPlayers) 3.5776 + { 3.5777 + gbJoymask[0] = systemGetJoypad(0, sensor); 3.5778 + gbJoymask[1] = systemGetJoypad(1, false); 3.5779 + gbJoymask[2] = systemGetJoypad(2, false); 3.5780 + gbJoymask[3] = systemGetJoypad(3, false); 3.5781 + } 3.5782 + else 3.5783 + { 3.5784 + gbJoymask[0] = systemGetJoypad(0, sensor); 3.5785 + gbJoymask[1] = systemGetJoypad(1, false); 3.5786 + } 3.5787 + } 3.5788 + else 3.5789 + { 3.5790 + gbJoymask[0] = systemGetJoypad(0, sensor); 3.5791 + } 3.5792 + 3.5793 + // FIXME: horrible kludge 3.5794 + memcpy(s_gbJoymask, gbJoymask, sizeof(gbJoymask)); 3.5795 + 3.5796 + // if (sensor) 3.5797 + // systemUpdateMotionSensor(0); 3.5798 + 3.5799 + newmask = gbJoymask[0]; 3.5800 + if (newmask & 0xFF) 3.5801 + { 3.5802 + gbInterrupt |= 16; 3.5803 + } 3.5804 + 3.5805 + extButtons = (newmask >> 18); 3.5806 + speedup = (extButtons & 1) != 0; 3.5807 + 3.5808 + VBAMovieResetIfRequested(); 3.5809 + printf("RLM: before Lua functions\n"); 3.5810 + //CallRegisteredLuaFunctions(LUACALL_BEFOREEMULATION); 3.5811 + printf("RLM: after Lua functions\n"); 3.5812 + newFrame = false; 3.5813 + } 3.5814 + 3.5815 + 3.5816 + for (;; ) 3.5817 + { 3.5818 +#ifndef FINAL_VERSION 3.5819 + if (systemDebug) 3.5820 + { 3.5821 + if (!(IFF & 0x80)) 3.5822 + { 3.5823 + if (systemDebug > 1) 3.5824 { 3.5825 - if (gbSgbFourPlayers) 3.5826 + sprintf(gbBuffer, "PC=%04x AF=%04x BC=%04x DE=%04x HL=%04x SP=%04x I=%04x\n", 3.5827 + PC.W, AF.W, BC.W, DE.W, HL.W, SP.W, IFF); 3.5828 + } 3.5829 + else 3.5830 + { 3.5831 + sprintf(gbBuffer, "PC=%04x I=%02x\n", PC.W, IFF); 3.5832 + } 3.5833 + log(gbBuffer); 3.5834 + } 3.5835 + } 3.5836 +#endif 3.5837 + if (IFF & 0x80) 3.5838 + { 3.5839 + if (register_LCDC & 0x80) 3.5840 + { 3.5841 + clockTicks = gbLcdTicks; 3.5842 + } 3.5843 + else 3.5844 + clockTicks = 100; 3.5845 + 3.5846 + if (gbLcdMode == 1 && (gbLcdLYIncrementTicks < clockTicks)) 3.5847 + clockTicks = gbLcdLYIncrementTicks; 3.5848 + 3.5849 + if (gbSerialOn && (gbSerialTicks < clockTicks)) 3.5850 + clockTicks = gbSerialTicks; 3.5851 + 3.5852 + if (gbTimerOn && (gbTimerTicks < clockTicks)) 3.5853 + clockTicks = gbTimerTicks; 3.5854 + 3.5855 + if (soundTicks && (soundTicks < clockTicks)) 3.5856 + clockTicks = soundTicks; 3.5857 + } 3.5858 + else 3.5859 + { 3.5860 + opcode = gbReadOpcode(PC.W); 3.5861 + CallRegisteredLuaMemHook(PC.W, 1, opcode, LUAMEMHOOK_EXEC); 3.5862 + PC.W++; 3.5863 + 3.5864 + if (IFF & 0x100) 3.5865 + { 3.5866 + IFF &= 0xff; 3.5867 + PC.W--; 3.5868 + } 3.5869 + 3.5870 + clockTicks = gbCycles[opcode]; 3.5871 + 3.5872 + switch (opcode) 3.5873 + { 3.5874 + case 0xCB: 3.5875 + // extended opcode 3.5876 + //CallRegisteredLuaMemHook(PC.W, 1, opcode, LUAMEMHOOK_EXEC); // is this desired? 3.5877 + opcode = gbReadOpcode(PC.W++); 3.5878 + clockTicks = gbCyclesCB[opcode]; 3.5879 + switch (opcode) 3.5880 + { 3.5881 +#include "gbCodesCB.h" 3.5882 + } 3.5883 + break; 3.5884 +#include "gbCodes.h" 3.5885 + } 3.5886 + } 3.5887 + 3.5888 + if (!emulating) 3.5889 + return; 3.5890 + 3.5891 + if (gbDmaTicks) 3.5892 + { 3.5893 + clockTicks += gbDmaTicks; 3.5894 + gbDmaTicks = 0; 3.5895 + } 3.5896 + 3.5897 + if (gbSgbMode) 3.5898 + { 3.5899 + if (gbSgbPacketTimeout) 3.5900 + { 3.5901 + gbSgbPacketTimeout -= clockTicks; 3.5902 + 3.5903 + if (gbSgbPacketTimeout <= 0) 3.5904 + gbSgbResetPacketState(); 3.5905 + } 3.5906 + } 3.5907 + 3.5908 + ticksToStop -= clockTicks; 3.5909 + 3.5910 + // DIV register emulation 3.5911 + gbDivTicks -= clockTicks; 3.5912 + while (gbDivTicks <= 0) 3.5913 + { 3.5914 + register_DIV++; 3.5915 + gbDivTicks += GBDIV_CLOCK_TICKS; 3.5916 + } 3.5917 + 3.5918 + if (register_LCDC & 0x80) 3.5919 + { 3.5920 + // LCD stuff 3.5921 + gbLcdTicks -= clockTicks; 3.5922 + if (gbLcdMode == 1) 3.5923 + { 3.5924 + // during V-BLANK,we need to increment LY at the same rate! 3.5925 + gbLcdLYIncrementTicks -= clockTicks; 3.5926 + while (gbLcdLYIncrementTicks <= 0) 3.5927 + { 3.5928 + gbLcdLYIncrementTicks += GBLY_INCREMENT_CLOCK_TICKS; 3.5929 + 3.5930 + if (register_LY < 153) 3.5931 + { 3.5932 + register_LY++; 3.5933 + 3.5934 + gbCompareLYToLYC(); 3.5935 + 3.5936 + if (register_LY >= 153) 3.5937 + gbLcdLYIncrementTicks = 6; 3.5938 + } 3.5939 + else 3.5940 + { 3.5941 + register_LY = 0x00; 3.5942 + // reset the window line 3.5943 + gbWindowLine = -1; 3.5944 + gbLcdLYIncrementTicks = GBLY_INCREMENT_CLOCK_TICKS * 2; 3.5945 + gbCompareLYToLYC(); 3.5946 + } 3.5947 + } 3.5948 + } 3.5949 + 3.5950 + // our counter is off, see what we need to do 3.5951 + while (gbLcdTicks <= 0) 3.5952 + { 3.5953 + int framesToSkip = systemFramesToSkip(); 3.5954 + 3.5955 + switch (gbLcdMode) 3.5956 + { 3.5957 + case 0: 3.5958 + // H-Blank 3.5959 + register_LY++; 3.5960 + 3.5961 + gbCompareLYToLYC(); 3.5962 + 3.5963 + // check if we reached the V-Blank period 3.5964 + if (register_LY == 144) 3.5965 + { 3.5966 + // Yes, V-Blank 3.5967 + // set the LY increment counter 3.5968 + gbLcdLYIncrementTicks = gbLcdTicks + GBLY_INCREMENT_CLOCK_TICKS; 3.5969 + gbLcdTicks += GBLCD_MODE_1_CLOCK_TICKS; 3.5970 + gbLcdMode = 1; 3.5971 + if (register_LCDC & 0x80) 3.5972 { 3.5973 - gbJoymask[0] = systemGetJoypad(0, sensor); 3.5974 - gbJoymask[1] = systemGetJoypad(1, false); 3.5975 - gbJoymask[2] = systemGetJoypad(2, false); 3.5976 - gbJoymask[3] = systemGetJoypad(3, false); 3.5977 + gbInterrupt |= 1; // V-Blank interrupt 3.5978 + gbInterruptWait = 6; 3.5979 + if (register_STAT & 0x10) 3.5980 + gbInterrupt |= 2; 3.5981 } 3.5982 - else 3.5983 + 3.5984 + systemFrame(); 3.5985 + 3.5986 + ++gbFrameCount; 3.5987 + u32 currentTime = systemGetClock(); 3.5988 + if (currentTime - gbLastTime >= 1000) 3.5989 { 3.5990 - gbJoymask[0] = systemGetJoypad(0, sensor); 3.5991 - gbJoymask[1] = systemGetJoypad(1, false); 3.5992 + systemShowSpeed(int(float(gbFrameCount) * 100000 / (float(currentTime - gbLastTime) * 60) + .5f)); 3.5993 + gbLastTime = currentTime; 3.5994 + gbFrameCount = 0; 3.5995 } 3.5996 + 3.5997 + ++GBSystemCounters.frameCount; 3.5998 + if (GBSystemCounters.lagged) 3.5999 + { 3.6000 + ++GBSystemCounters.lagCount; 3.6001 + } 3.6002 + GBSystemCounters.laggedLast = GBSystemCounters.lagged; 3.6003 + GBSystemCounters.lagged = true; 3.6004 + 3.6005 + extern void VBAOnEnteringFrameBoundary(); 3.6006 + VBAOnEnteringFrameBoundary(); 3.6007 + 3.6008 + newFrame = true; 3.6009 + 3.6010 + pauseAfterFrameAdvance = systemPauseOnFrame(); 3.6011 + 3.6012 + if (gbFrameSkipCount >= framesToSkip || pauseAfterFrameAdvance) 3.6013 + { 3.6014 + if (gbBorderOn) 3.6015 + gbSgbRenderBorder(); // clear unnecessary things on border (e.g. in-game text message) 3.6016 + 3.6017 + systemRenderFrame(); 3.6018 + gbFrameSkipCount = 0; 3.6019 + 3.6020 + bool capturePressed = (extButtons & 2) != 0; 3.6021 + if (capturePressed && !capturePrevious) 3.6022 + { 3.6023 + captureNumber = systemScreenCapture(captureNumber); 3.6024 + } 3.6025 + capturePrevious = capturePressed && !pauseAfterFrameAdvance; 3.6026 + } 3.6027 + else 3.6028 + { 3.6029 + ++gbFrameSkipCount; 3.6030 + } 3.6031 + 3.6032 + if (pauseAfterFrameAdvance) 3.6033 + { 3.6034 + systemSetPause(true); 3.6035 + } 3.6036 + } 3.6037 + else 3.6038 + { 3.6039 + // go the the OAM being accessed mode 3.6040 + gbLcdTicks += GBLCD_MODE_2_CLOCK_TICKS; 3.6041 + gbLcdMode = 2; 3.6042 + 3.6043 + // only one LCD interrupt per line. may need to generalize... 3.6044 + if (!(register_STAT & 0x40) || 3.6045 + (register_LY != register_LYC)) 3.6046 + { 3.6047 + if ((register_STAT & 0x28) == 0x20) 3.6048 + gbInterrupt |= 2; 3.6049 + } 3.6050 + } 3.6051 + 3.6052 + break; 3.6053 + case 1: 3.6054 + // V-Blank 3.6055 + // next mode is OAM being accessed mode 3.6056 + gbLcdTicks += GBLCD_MODE_2_CLOCK_TICKS; 3.6057 + gbLcdMode = 2; 3.6058 + if (!(register_STAT & 0x40) || 3.6059 + (register_LY != register_LYC)) 3.6060 + { 3.6061 + if ((register_STAT & 0x28) == 0x20) 3.6062 + gbInterrupt |= 2; 3.6063 + } 3.6064 + break; 3.6065 + case 2: 3.6066 + // OAM being accessed mode 3.6067 + 3.6068 + // next mode is OAM and VRAM in use 3.6069 + gbLcdTicks += GBLCD_MODE_3_CLOCK_TICKS; 3.6070 + gbLcdMode = 3; 3.6071 + break; 3.6072 + case 3: 3.6073 + // OAM and VRAM in use 3.6074 + // next mode is H-Blank 3.6075 + if (register_LY < 144) 3.6076 + { 3.6077 + if (!gbSgbMask) 3.6078 + { 3.6079 + if (gbFrameSkipCount >= framesToSkip || pauseAfterFrameAdvance) 3.6080 + { 3.6081 + gbRenderLine(); 3.6082 + gbDrawSprites(); 3.6083 + 3.6084 + switch (systemColorDepth) 3.6085 + { 3.6086 + case 16: 3.6087 + 3.6088 + { 3.6089 + u16 *dest = (u16 *)pix + 3.6090 + (gbBorderLineSkip + 2) * (register_LY + gbBorderRowSkip + 1) 3.6091 + + gbBorderColumnSkip; 3.6092 + for (int x = 0; x < 160; ) 3.6093 + { 3.6094 + *dest++ = systemColorMap16[gbLineMix[x++]]; 3.6095 + *dest++ = systemColorMap16[gbLineMix[x++]]; 3.6096 + *dest++ = systemColorMap16[gbLineMix[x++]]; 3.6097 + *dest++ = systemColorMap16[gbLineMix[x++]]; 3.6098 + 3.6099 + *dest++ = systemColorMap16[gbLineMix[x++]]; 3.6100 + *dest++ = systemColorMap16[gbLineMix[x++]]; 3.6101 + *dest++ = systemColorMap16[gbLineMix[x++]]; 3.6102 + *dest++ = systemColorMap16[gbLineMix[x++]]; 3.6103 + 3.6104 + *dest++ = systemColorMap16[gbLineMix[x++]]; 3.6105 + *dest++ = systemColorMap16[gbLineMix[x++]]; 3.6106 + *dest++ = systemColorMap16[gbLineMix[x++]]; 3.6107 + *dest++ = systemColorMap16[gbLineMix[x++]]; 3.6108 + 3.6109 + *dest++ = systemColorMap16[gbLineMix[x++]]; 3.6110 + *dest++ = systemColorMap16[gbLineMix[x++]]; 3.6111 + *dest++ = systemColorMap16[gbLineMix[x++]]; 3.6112 + *dest++ = systemColorMap16[gbLineMix[x++]]; 3.6113 + } 3.6114 + if (gbBorderOn) 3.6115 + dest += gbBorderColumnSkip; 3.6116 + *dest++ = 0; // for filters that read one pixel more 3.6117 + break; 3.6118 + } 3.6119 + case 24: 3.6120 + 3.6121 + { 3.6122 + u8 *dest = (u8 *)pix + 3.6123 + 3 * (gbBorderLineSkip * (register_LY + gbBorderRowSkip) + 3.6124 + gbBorderColumnSkip); 3.6125 + for (int x = 0; x < 160; ) 3.6126 + { 3.6127 + *((u32 *)dest) = systemColorMap32[gbLineMix[x++]]; 3.6128 + dest += 3; 3.6129 + *((u32 *)dest) = systemColorMap32[gbLineMix[x++]]; 3.6130 + dest += 3; 3.6131 + *((u32 *)dest) = systemColorMap32[gbLineMix[x++]]; 3.6132 + dest += 3; 3.6133 + *((u32 *)dest) = systemColorMap32[gbLineMix[x++]]; 3.6134 + dest += 3; 3.6135 + 3.6136 + *((u32 *)dest) = systemColorMap32[gbLineMix[x++]]; 3.6137 + dest += 3; 3.6138 + *((u32 *)dest) = systemColorMap32[gbLineMix[x++]]; 3.6139 + dest += 3; 3.6140 + *((u32 *)dest) = systemColorMap32[gbLineMix[x++]]; 3.6141 + dest += 3; 3.6142 + *((u32 *)dest) = systemColorMap32[gbLineMix[x++]]; 3.6143 + dest += 3; 3.6144 + 3.6145 + *((u32 *)dest) = systemColorMap32[gbLineMix[x++]]; 3.6146 + dest += 3; 3.6147 + *((u32 *)dest) = systemColorMap32[gbLineMix[x++]]; 3.6148 + dest += 3; 3.6149 + *((u32 *)dest) = systemColorMap32[gbLineMix[x++]]; 3.6150 + dest += 3; 3.6151 + *((u32 *)dest) = systemColorMap32[gbLineMix[x++]]; 3.6152 + dest += 3; 3.6153 + 3.6154 + *((u32 *)dest) = systemColorMap32[gbLineMix[x++]]; 3.6155 + dest += 3; 3.6156 + *((u32 *)dest) = systemColorMap32[gbLineMix[x++]]; 3.6157 + dest += 3; 3.6158 + *((u32 *)dest) = systemColorMap32[gbLineMix[x++]]; 3.6159 + dest += 3; 3.6160 + *((u32 *)dest) = systemColorMap32[gbLineMix[x++]]; 3.6161 + dest += 3; 3.6162 + } 3.6163 + break; 3.6164 + } 3.6165 + case 32: 3.6166 + 3.6167 + { 3.6168 + u32 *dest = (u32 *)pix + 3.6169 + (gbBorderLineSkip + 1) * (register_LY + gbBorderRowSkip + 1) 3.6170 + + gbBorderColumnSkip; 3.6171 + for (int x = 0; x < 160; ) 3.6172 + { 3.6173 + *dest++ = systemColorMap32[gbLineMix[x++]]; 3.6174 + *dest++ = systemColorMap32[gbLineMix[x++]]; 3.6175 + *dest++ = systemColorMap32[gbLineMix[x++]]; 3.6176 + *dest++ = systemColorMap32[gbLineMix[x++]]; 3.6177 + 3.6178 + *dest++ = systemColorMap32[gbLineMix[x++]]; 3.6179 + *dest++ = systemColorMap32[gbLineMix[x++]]; 3.6180 + *dest++ = systemColorMap32[gbLineMix[x++]]; 3.6181 + *dest++ = systemColorMap32[gbLineMix[x++]]; 3.6182 + 3.6183 + *dest++ = systemColorMap32[gbLineMix[x++]]; 3.6184 + *dest++ = systemColorMap32[gbLineMix[x++]]; 3.6185 + *dest++ = systemColorMap32[gbLineMix[x++]]; 3.6186 + *dest++ = systemColorMap32[gbLineMix[x++]]; 3.6187 + 3.6188 + *dest++ = systemColorMap32[gbLineMix[x++]]; 3.6189 + *dest++ = systemColorMap32[gbLineMix[x++]]; 3.6190 + *dest++ = systemColorMap32[gbLineMix[x++]]; 3.6191 + *dest++ = systemColorMap32[gbLineMix[x++]]; 3.6192 + } 3.6193 + break; 3.6194 + } 3.6195 + } 3.6196 + } 3.6197 + } 3.6198 + } 3.6199 + gbLcdTicks += GBLCD_MODE_0_CLOCK_TICKS; 3.6200 + gbLcdMode = 0; 3.6201 + // only one LCD interrupt per line. may need to generalize... 3.6202 + if (!(register_STAT & 0x40) || 3.6203 + (register_LY != register_LYC)) 3.6204 + { 3.6205 + if (register_STAT & 0x08) 3.6206 + gbInterrupt |= 2; 3.6207 + } 3.6208 + if (gbHdmaOn) 3.6209 + { 3.6210 + gbDoHdma(); 3.6211 + } 3.6212 + break; 3.6213 } 3.6214 - else 3.6215 + // mark the correct lcd mode on STAT register 3.6216 + register_STAT = (register_STAT & 0xfc) | gbLcdMode; 3.6217 + } 3.6218 + } 3.6219 + 3.6220 + // serial emulation 3.6221 + if (gbSerialOn) 3.6222 + { 3.6223 +#ifdef LINK_EMULATION 3.6224 + if (linkConnected) 3.6225 + { 3.6226 + gbSerialTicks -= clockTicks; 3.6227 + 3.6228 + while (gbSerialTicks <= 0) 3.6229 { 3.6230 - gbJoymask[0] = systemGetJoypad(0, sensor); 3.6231 + // increment number of shifted bits 3.6232 + gbSerialBits++; 3.6233 + linkProc(); 3.6234 + if (gbSerialOn && (gbMemory[0xff02] & 1)) 3.6235 + { 3.6236 + if (gbSerialBits == 8) 3.6237 + { 3.6238 + gbSerialBits = 0; 3.6239 + gbMemory[0xff01] = 0xff; 3.6240 + gbMemory[0xff02] &= 0x7f; 3.6241 + gbSerialOn = 0; 3.6242 + gbInterrupt |= 8; 3.6243 + gbSerialTicks = 0; 3.6244 + } 3.6245 + } 3.6246 + gbSerialTicks += GBSERIAL_CLOCK_TICKS; 3.6247 } 3.6248 - 3.6249 - // FIXME: horrible kludge 3.6250 - memcpy(s_gbJoymask, gbJoymask, sizeof(gbJoymask)); 3.6251 - 3.6252 -// if (sensor) 3.6253 -// systemUpdateMotionSensor(0); 3.6254 - 3.6255 - newmask = gbJoymask[0]; 3.6256 - if (newmask & 0xFF) 3.6257 + } 3.6258 + else 3.6259 + { 3.6260 +#endif 3.6261 + if (gbMemory[0xff02] & 1) 3.6262 { 3.6263 - gbInterrupt |= 16; 3.6264 + gbSerialTicks -= clockTicks; 3.6265 + 3.6266 + // overflow 3.6267 + while (gbSerialTicks <= 0) 3.6268 + { 3.6269 + // shift serial byte to right and put a 1 bit in its place 3.6270 + // gbMemory[0xff01] = 0x80 | (gbMemory[0xff01]>>1); 3.6271 + // increment number of shifted bits 3.6272 + gbSerialBits++; 3.6273 + if (gbSerialBits == 8) 3.6274 + { 3.6275 + // end of transmission 3.6276 + if (gbSerialFunction) // external device 3.6277 + gbMemory[0xff01] = gbSerialFunction(gbMemory[0xff01]); 3.6278 + else 3.6279 + gbMemory[0xff01] = 0xff; 3.6280 + gbSerialTicks = 0; 3.6281 + gbMemory[0xff02] &= 0x7f; 3.6282 + gbSerialOn = 0; 3.6283 + gbInterrupt |= 8; 3.6284 + gbSerialBits = 0; 3.6285 + } 3.6286 + else 3.6287 + gbSerialTicks += GBSERIAL_CLOCK_TICKS; 3.6288 + } 3.6289 } 3.6290 - 3.6291 - extButtons = (newmask >> 18); 3.6292 - speedup = (extButtons & 1) != 0; 3.6293 - 3.6294 - VBAMovieResetIfRequested(); 3.6295 - 3.6296 - CallRegisteredLuaFunctions(LUACALL_BEFOREEMULATION); 3.6297 - 3.6298 - newFrame = false; 3.6299 +#ifdef LINK_EMULATION 3.6300 + } 3.6301 +#endif 3.6302 } 3.6303 3.6304 - for (;; ) 3.6305 + // timer emulation 3.6306 + if (gbTimerOn) 3.6307 { 3.6308 -#ifndef FINAL_VERSION 3.6309 - if (systemDebug) 3.6310 + gbTimerTicks -= clockTicks; 3.6311 + 3.6312 + while (gbTimerTicks <= 0) 3.6313 + { 3.6314 + register_TIMA++; 3.6315 + 3.6316 + if (register_TIMA == 0) 3.6317 { 3.6318 - if (!(IFF & 0x80)) 3.6319 + // timer overflow! 3.6320 + 3.6321 + // reload timer modulo 3.6322 + register_TIMA = register_TMA; 3.6323 + 3.6324 + // flag interrupt 3.6325 + gbInterrupt |= 4; 3.6326 + } 3.6327 + 3.6328 + gbTimerTicks += gbTimerClockTicks; 3.6329 + } 3.6330 + } 3.6331 + 3.6332 + /* 3.6333 + if(soundOffFlag) 3.6334 + { 3.6335 + if(synchronize && !speedup) 3.6336 + { 3.6337 + synchronizeTicks -= clockTicks; 3.6338 + 3.6339 + while(synchronizeTicks < 0) 3.6340 + { 3.6341 + synchronizeTicks += SYNCHRONIZE_CLOCK_TICKS; 3.6342 + 3.6343 + DWORD now = timeGetTime(); 3.6344 + gbElapsedTime += (now - timeNow); 3.6345 + 3.6346 + if(gbElapsedTime < 50) 3.6347 + { 3.6348 + DWORD diff = 50 - gbElapsedTime; 3.6349 + Sleep(diff); 3.6350 + timeNow = timeGetTime(); 3.6351 + elapsedTime = timeNow - now - diff; 3.6352 + if((int)elapsedTime < 0) 3.6353 + elapsedTime = 0; 3.6354 + } else 3.6355 + { 3.6356 + timeNow = timeGetTime(); 3.6357 + elapsedTime = 0; 3.6358 + } 3.6359 + } 3.6360 + } 3.6361 + } 3.6362 + */ 3.6363 + 3.6364 + soundTicks -= clockTicks; 3.6365 + while (soundTicks < 0) // must be < 1 when soundtick_t is real data type 3.6366 + { 3.6367 + soundTicks += SOUND_CLOCK_TICKS; 3.6368 + 3.6369 + gbSoundTick(); 3.6370 + } 3.6371 + 3.6372 + register_IF = gbInterrupt; 3.6373 + 3.6374 + if (IFF & 0x20) 3.6375 + { 3.6376 + IFF &= 0xdf; 3.6377 + IFF |= 0x01; 3.6378 + gbInterruptWait = 0; 3.6379 + } 3.6380 + else if (gbInterrupt) 3.6381 + { 3.6382 + if (gbInterruptWait == 0) 3.6383 + { 3.6384 + // gbInterruptWait = 0; 3.6385 + 3.6386 + if (IFF & 0x01) 3.6387 + { 3.6388 + if ((gbInterrupt & 1) && (register_IE & 1)) 3.6389 + { 3.6390 + gbVblank_interrupt(); 3.6391 + continue; 3.6392 + } 3.6393 + 3.6394 + if ((gbInterrupt & 2) && (register_IE & 2)) 3.6395 + { 3.6396 + gbLcd_interrupt(); 3.6397 + continue; 3.6398 + } 3.6399 + 3.6400 + if ((gbInterrupt & 4) && (register_IE & 4)) 3.6401 + { 3.6402 + gbTimer_interrupt(); 3.6403 + continue; 3.6404 + } 3.6405 + 3.6406 + if ((gbInterrupt & 8) && (register_IE & 8)) 3.6407 + { 3.6408 + gbSerial_interrupt(); 3.6409 + continue; 3.6410 + } 3.6411 + 3.6412 + if ((gbInterrupt & 16) && (register_IE & 16)) 3.6413 + { 3.6414 + gbJoypad_interrupt(); 3.6415 + continue; 3.6416 + } 3.6417 + } 3.6418 + } 3.6419 + else 3.6420 + { 3.6421 + gbInterruptWait -= clockTicks; 3.6422 + if (gbInterruptWait < 0) 3.6423 + gbInterruptWait = 0; 3.6424 + } 3.6425 + } 3.6426 + 3.6427 + if (useOldFrameTiming) 3.6428 + { 3.6429 + // old timing code 3.6430 + if (ticksToStop > 0) 3.6431 + continue; 3.6432 + } 3.6433 + else 3.6434 + { 3.6435 + if (!newFrame && (register_LCDC & 0x80) != 0) 3.6436 + continue; 3.6437 + } 3.6438 + 3.6439 + if (!(register_LCDC & 0x80)) 3.6440 + { 3.6441 + if (!useOldFrameTiming) 3.6442 + { 3.6443 + // FIXME: since register_LY can be reset to 0 by some games, frame length is variable 3.6444 + // and infinite loops can occurr 3.6445 + // for now, it IS necessary to do something on this condition or games like 3.6446 + // Megaman would freeze upon low-level restart interrupt sequence (Start+Select+A+B). 3.6447 + // the only sensible way to fix this issue is to implement the RIGHT frame timing 3.6448 +#ifdef WANTS_INCOMPLETE_WORKAROUND 3.6449 + if (systemReadJoypads()) 3.6450 + { 3.6451 + if (gbSgbMode && gbSgbMultiplayer) 3.6452 + { 3.6453 + if (gbSgbFourPlayers) 3.6454 { 3.6455 - if (systemDebug > 1) 3.6456 - { 3.6457 - sprintf(gbBuffer, "PC=%04x AF=%04x BC=%04x DE=%04x HL=%04x SP=%04x I=%04x\n", 3.6458 - PC.W, AF.W, BC.W, DE.W, HL.W, SP.W, IFF); 3.6459 - } 3.6460 - else 3.6461 - { 3.6462 - sprintf(gbBuffer, "PC=%04x I=%02x\n", PC.W, IFF); 3.6463 - } 3.6464 - log(gbBuffer); 3.6465 + gbJoymask[0] = systemGetJoypad(0, false); 3.6466 + gbJoymask[1] = systemGetJoypad(1, false); 3.6467 + gbJoymask[2] = systemGetJoypad(2, false); 3.6468 + gbJoymask[3] = systemGetJoypad(3, false); 3.6469 } 3.6470 + else 3.6471 + { 3.6472 + gbJoymask[0] = systemGetJoypad(0, false); 3.6473 + gbJoymask[1] = systemGetJoypad(1, false); 3.6474 + } 3.6475 + } 3.6476 + else 3.6477 + { 3.6478 + gbJoymask[0] = systemGetJoypad(0, false); 3.6479 + } 3.6480 } 3.6481 -#endif 3.6482 - if (IFF & 0x80) 3.6483 - { 3.6484 - if (register_LCDC & 0x80) 3.6485 - { 3.6486 - clockTicks = gbLcdTicks; 3.6487 - } 3.6488 - else 3.6489 - clockTicks = 100; 3.6490 - 3.6491 - if (gbLcdMode == 1 && (gbLcdLYIncrementTicks < clockTicks)) 3.6492 - clockTicks = gbLcdLYIncrementTicks; 3.6493 - 3.6494 - if (gbSerialOn && (gbSerialTicks < clockTicks)) 3.6495 - clockTicks = gbSerialTicks; 3.6496 - 3.6497 - if (gbTimerOn && (gbTimerTicks < clockTicks)) 3.6498 - clockTicks = gbTimerTicks; 3.6499 - 3.6500 - if (soundTicks && (soundTicks < clockTicks)) 3.6501 - clockTicks = soundTicks; 3.6502 - } 3.6503 - else 3.6504 - { 3.6505 - opcode = gbReadOpcode(PC.W); 3.6506 - CallRegisteredLuaMemHook(PC.W, 1, opcode, LUAMEMHOOK_EXEC); 3.6507 - PC.W++; 3.6508 - 3.6509 - if (IFF & 0x100) 3.6510 - { 3.6511 - IFF &= 0xff; 3.6512 - PC.W--; 3.6513 - } 3.6514 - 3.6515 - clockTicks = gbCycles[opcode]; 3.6516 - 3.6517 - switch (opcode) 3.6518 - { 3.6519 - case 0xCB: 3.6520 - // extended opcode 3.6521 - //CallRegisteredLuaMemHook(PC.W, 1, opcode, LUAMEMHOOK_EXEC); // is this desired? 3.6522 - opcode = gbReadOpcode(PC.W++); 3.6523 - clockTicks = gbCyclesCB[opcode]; 3.6524 - switch (opcode) 3.6525 - { 3.6526 -#include "gbCodesCB.h" 3.6527 - } 3.6528 - break; 3.6529 -#include "gbCodes.h" 3.6530 - } 3.6531 - } 3.6532 - 3.6533 - if (!emulating) 3.6534 - return; 3.6535 - 3.6536 - if (gbDmaTicks) 3.6537 - { 3.6538 - clockTicks += gbDmaTicks; 3.6539 - gbDmaTicks = 0; 3.6540 - } 3.6541 - 3.6542 - if (gbSgbMode) 3.6543 - { 3.6544 - if (gbSgbPacketTimeout) 3.6545 - { 3.6546 - gbSgbPacketTimeout -= clockTicks; 3.6547 - 3.6548 - if (gbSgbPacketTimeout <= 0) 3.6549 - gbSgbResetPacketState(); 3.6550 - } 3.6551 - } 3.6552 - 3.6553 - ticksToStop -= clockTicks; 3.6554 - 3.6555 - // DIV register emulation 3.6556 - gbDivTicks -= clockTicks; 3.6557 - while (gbDivTicks <= 0) 3.6558 - { 3.6559 - register_DIV++; 3.6560 - gbDivTicks += GBDIV_CLOCK_TICKS; 3.6561 - } 3.6562 - 3.6563 - if (register_LCDC & 0x80) 3.6564 - { 3.6565 - // LCD stuff 3.6566 - gbLcdTicks -= clockTicks; 3.6567 - if (gbLcdMode == 1) 3.6568 - { 3.6569 - // during V-BLANK,we need to increment LY at the same rate! 3.6570 - gbLcdLYIncrementTicks -= clockTicks; 3.6571 - while (gbLcdLYIncrementTicks <= 0) 3.6572 - { 3.6573 - gbLcdLYIncrementTicks += GBLY_INCREMENT_CLOCK_TICKS; 3.6574 - 3.6575 - if (register_LY < 153) 3.6576 - { 3.6577 - register_LY++; 3.6578 - 3.6579 - gbCompareLYToLYC(); 3.6580 - 3.6581 - if (register_LY >= 153) 3.6582 - gbLcdLYIncrementTicks = 6; 3.6583 - } 3.6584 - else 3.6585 - { 3.6586 - register_LY = 0x00; 3.6587 - // reset the window line 3.6588 - gbWindowLine = -1; 3.6589 - gbLcdLYIncrementTicks = GBLY_INCREMENT_CLOCK_TICKS * 2; 3.6590 - gbCompareLYToLYC(); 3.6591 - } 3.6592 - } 3.6593 - } 3.6594 - 3.6595 - // our counter is off, see what we need to do 3.6596 - while (gbLcdTicks <= 0) 3.6597 - { 3.6598 - int framesToSkip = systemFramesToSkip(); 3.6599 - 3.6600 - switch (gbLcdMode) 3.6601 - { 3.6602 - case 0: 3.6603 - // H-Blank 3.6604 - register_LY++; 3.6605 - 3.6606 - gbCompareLYToLYC(); 3.6607 - 3.6608 - // check if we reached the V-Blank period 3.6609 - if (register_LY == 144) 3.6610 - { 3.6611 - // Yes, V-Blank 3.6612 - // set the LY increment counter 3.6613 - gbLcdLYIncrementTicks = gbLcdTicks + GBLY_INCREMENT_CLOCK_TICKS; 3.6614 - gbLcdTicks += GBLCD_MODE_1_CLOCK_TICKS; 3.6615 - gbLcdMode = 1; 3.6616 - if (register_LCDC & 0x80) 3.6617 - { 3.6618 - gbInterrupt |= 1; // V-Blank interrupt 3.6619 - gbInterruptWait = 6; 3.6620 - if (register_STAT & 0x10) 3.6621 - gbInterrupt |= 2; 3.6622 - } 3.6623 - 3.6624 - systemFrame(); 3.6625 - 3.6626 - ++gbFrameCount; 3.6627 - u32 currentTime = systemGetClock(); 3.6628 - if (currentTime - gbLastTime >= 1000) 3.6629 - { 3.6630 - systemShowSpeed(int(float(gbFrameCount) * 100000 / (float(currentTime - gbLastTime) * 60) + .5f)); 3.6631 - gbLastTime = currentTime; 3.6632 - gbFrameCount = 0; 3.6633 - } 3.6634 - 3.6635 - ++GBSystemCounters.frameCount; 3.6636 - if (GBSystemCounters.lagged) 3.6637 - { 3.6638 - ++GBSystemCounters.lagCount; 3.6639 - } 3.6640 - GBSystemCounters.laggedLast = GBSystemCounters.lagged; 3.6641 - GBSystemCounters.lagged = true; 3.6642 - 3.6643 - extern void VBAOnEnteringFrameBoundary(); 3.6644 - VBAOnEnteringFrameBoundary(); 3.6645 - 3.6646 - newFrame = true; 3.6647 - 3.6648 - pauseAfterFrameAdvance = systemPauseOnFrame(); 3.6649 - 3.6650 - if (gbFrameSkipCount >= framesToSkip || pauseAfterFrameAdvance) 3.6651 - { 3.6652 - if (gbBorderOn) 3.6653 - gbSgbRenderBorder(); // clear unnecessary things on border (e.g. in-game text message) 3.6654 - 3.6655 - systemRenderFrame(); 3.6656 - gbFrameSkipCount = 0; 3.6657 - 3.6658 - bool capturePressed = (extButtons & 2) != 0; 3.6659 - if (capturePressed && !capturePrevious) 3.6660 - { 3.6661 - captureNumber = systemScreenCapture(captureNumber); 3.6662 - } 3.6663 - capturePrevious = capturePressed && !pauseAfterFrameAdvance; 3.6664 - } 3.6665 - else 3.6666 - { 3.6667 - ++gbFrameSkipCount; 3.6668 - } 3.6669 - 3.6670 - if (pauseAfterFrameAdvance) 3.6671 - { 3.6672 - systemSetPause(true); 3.6673 - } 3.6674 - } 3.6675 - else 3.6676 - { 3.6677 - // go the the OAM being accessed mode 3.6678 - gbLcdTicks += GBLCD_MODE_2_CLOCK_TICKS; 3.6679 - gbLcdMode = 2; 3.6680 - 3.6681 - // only one LCD interrupt per line. may need to generalize... 3.6682 - if (!(register_STAT & 0x40) || 3.6683 - (register_LY != register_LYC)) 3.6684 - { 3.6685 - if ((register_STAT & 0x28) == 0x20) 3.6686 - gbInterrupt |= 2; 3.6687 - } 3.6688 - } 3.6689 - 3.6690 - break; 3.6691 - case 1: 3.6692 - // V-Blank 3.6693 - // next mode is OAM being accessed mode 3.6694 - gbLcdTicks += GBLCD_MODE_2_CLOCK_TICKS; 3.6695 - gbLcdMode = 2; 3.6696 - if (!(register_STAT & 0x40) || 3.6697 - (register_LY != register_LYC)) 3.6698 - { 3.6699 - if ((register_STAT & 0x28) == 0x20) 3.6700 - gbInterrupt |= 2; 3.6701 - } 3.6702 - break; 3.6703 - case 2: 3.6704 - // OAM being accessed mode 3.6705 - 3.6706 - // next mode is OAM and VRAM in use 3.6707 - gbLcdTicks += GBLCD_MODE_3_CLOCK_TICKS; 3.6708 - gbLcdMode = 3; 3.6709 - break; 3.6710 - case 3: 3.6711 - // OAM and VRAM in use 3.6712 - // next mode is H-Blank 3.6713 - if (register_LY < 144) 3.6714 - { 3.6715 - if (!gbSgbMask) 3.6716 - { 3.6717 - if (gbFrameSkipCount >= framesToSkip || pauseAfterFrameAdvance) 3.6718 - { 3.6719 - gbRenderLine(); 3.6720 - gbDrawSprites(); 3.6721 - 3.6722 - switch (systemColorDepth) 3.6723 - { 3.6724 - case 16: 3.6725 - 3.6726 - { 3.6727 - u16 *dest = (u16 *)pix + 3.6728 - (gbBorderLineSkip + 2) * (register_LY + gbBorderRowSkip + 1) 3.6729 - + gbBorderColumnSkip; 3.6730 - for (int x = 0; x < 160; ) 3.6731 - { 3.6732 - *dest++ = systemColorMap16[gbLineMix[x++]]; 3.6733 - *dest++ = systemColorMap16[gbLineMix[x++]]; 3.6734 - *dest++ = systemColorMap16[gbLineMix[x++]]; 3.6735 - *dest++ = systemColorMap16[gbLineMix[x++]]; 3.6736 - 3.6737 - *dest++ = systemColorMap16[gbLineMix[x++]]; 3.6738 - *dest++ = systemColorMap16[gbLineMix[x++]]; 3.6739 - *dest++ = systemColorMap16[gbLineMix[x++]]; 3.6740 - *dest++ = systemColorMap16[gbLineMix[x++]]; 3.6741 - 3.6742 - *dest++ = systemColorMap16[gbLineMix[x++]]; 3.6743 - *dest++ = systemColorMap16[gbLineMix[x++]]; 3.6744 - *dest++ = systemColorMap16[gbLineMix[x++]]; 3.6745 - *dest++ = systemColorMap16[gbLineMix[x++]]; 3.6746 - 3.6747 - *dest++ = systemColorMap16[gbLineMix[x++]]; 3.6748 - *dest++ = systemColorMap16[gbLineMix[x++]]; 3.6749 - *dest++ = systemColorMap16[gbLineMix[x++]]; 3.6750 - *dest++ = systemColorMap16[gbLineMix[x++]]; 3.6751 - } 3.6752 - if (gbBorderOn) 3.6753 - dest += gbBorderColumnSkip; 3.6754 - *dest++ = 0; // for filters that read one pixel more 3.6755 - break; 3.6756 - } 3.6757 - case 24: 3.6758 - 3.6759 - { 3.6760 - u8 *dest = (u8 *)pix + 3.6761 - 3 * (gbBorderLineSkip * (register_LY + gbBorderRowSkip) + 3.6762 - gbBorderColumnSkip); 3.6763 - for (int x = 0; x < 160; ) 3.6764 - { 3.6765 - *((u32 *)dest) = systemColorMap32[gbLineMix[x++]]; 3.6766 - dest += 3; 3.6767 - *((u32 *)dest) = systemColorMap32[gbLineMix[x++]]; 3.6768 - dest += 3; 3.6769 - *((u32 *)dest) = systemColorMap32[gbLineMix[x++]]; 3.6770 - dest += 3; 3.6771 - *((u32 *)dest) = systemColorMap32[gbLineMix[x++]]; 3.6772 - dest += 3; 3.6773 - 3.6774 - *((u32 *)dest) = systemColorMap32[gbLineMix[x++]]; 3.6775 - dest += 3; 3.6776 - *((u32 *)dest) = systemColorMap32[gbLineMix[x++]]; 3.6777 - dest += 3; 3.6778 - *((u32 *)dest) = systemColorMap32[gbLineMix[x++]]; 3.6779 - dest += 3; 3.6780 - *((u32 *)dest) = systemColorMap32[gbLineMix[x++]]; 3.6781 - dest += 3; 3.6782 - 3.6783 - *((u32 *)dest) = systemColorMap32[gbLineMix[x++]]; 3.6784 - dest += 3; 3.6785 - *((u32 *)dest) = systemColorMap32[gbLineMix[x++]]; 3.6786 - dest += 3; 3.6787 - *((u32 *)dest) = systemColorMap32[gbLineMix[x++]]; 3.6788 - dest += 3; 3.6789 - *((u32 *)dest) = systemColorMap32[gbLineMix[x++]]; 3.6790 - dest += 3; 3.6791 - 3.6792 - *((u32 *)dest) = systemColorMap32[gbLineMix[x++]]; 3.6793 - dest += 3; 3.6794 - *((u32 *)dest) = systemColorMap32[gbLineMix[x++]]; 3.6795 - dest += 3; 3.6796 - *((u32 *)dest) = systemColorMap32[gbLineMix[x++]]; 3.6797 - dest += 3; 3.6798 - *((u32 *)dest) = systemColorMap32[gbLineMix[x++]]; 3.6799 - dest += 3; 3.6800 - } 3.6801 - break; 3.6802 - } 3.6803 - case 32: 3.6804 - 3.6805 - { 3.6806 - u32 *dest = (u32 *)pix + 3.6807 - (gbBorderLineSkip + 1) * (register_LY + gbBorderRowSkip + 1) 3.6808 - + gbBorderColumnSkip; 3.6809 - for (int x = 0; x < 160; ) 3.6810 - { 3.6811 - *dest++ = systemColorMap32[gbLineMix[x++]]; 3.6812 - *dest++ = systemColorMap32[gbLineMix[x++]]; 3.6813 - *dest++ = systemColorMap32[gbLineMix[x++]]; 3.6814 - *dest++ = systemColorMap32[gbLineMix[x++]]; 3.6815 - 3.6816 - *dest++ = systemColorMap32[gbLineMix[x++]]; 3.6817 - *dest++ = systemColorMap32[gbLineMix[x++]]; 3.6818 - *dest++ = systemColorMap32[gbLineMix[x++]]; 3.6819 - *dest++ = systemColorMap32[gbLineMix[x++]]; 3.6820 - 3.6821 - *dest++ = systemColorMap32[gbLineMix[x++]]; 3.6822 - *dest++ = systemColorMap32[gbLineMix[x++]]; 3.6823 - *dest++ = systemColorMap32[gbLineMix[x++]]; 3.6824 - *dest++ = systemColorMap32[gbLineMix[x++]]; 3.6825 - 3.6826 - *dest++ = systemColorMap32[gbLineMix[x++]]; 3.6827 - *dest++ = systemColorMap32[gbLineMix[x++]]; 3.6828 - *dest++ = systemColorMap32[gbLineMix[x++]]; 3.6829 - *dest++ = systemColorMap32[gbLineMix[x++]]; 3.6830 - } 3.6831 - break; 3.6832 - } 3.6833 - } 3.6834 - } 3.6835 - } 3.6836 - } 3.6837 - gbLcdTicks += GBLCD_MODE_0_CLOCK_TICKS; 3.6838 - gbLcdMode = 0; 3.6839 - // only one LCD interrupt per line. may need to generalize... 3.6840 - if (!(register_STAT & 0x40) || 3.6841 - (register_LY != register_LYC)) 3.6842 - { 3.6843 - if (register_STAT & 0x08) 3.6844 - gbInterrupt |= 2; 3.6845 - } 3.6846 - if (gbHdmaOn) 3.6847 - { 3.6848 - gbDoHdma(); 3.6849 - } 3.6850 - break; 3.6851 - } 3.6852 - // mark the correct lcd mode on STAT register 3.6853 - register_STAT = (register_STAT & 0xfc) | gbLcdMode; 3.6854 - } 3.6855 - } 3.6856 - 3.6857 - // serial emulation 3.6858 - if (gbSerialOn) 3.6859 - { 3.6860 -#ifdef LINK_EMULATION 3.6861 - if (linkConnected) 3.6862 - { 3.6863 - gbSerialTicks -= clockTicks; 3.6864 - 3.6865 - while (gbSerialTicks <= 0) 3.6866 - { 3.6867 - // increment number of shifted bits 3.6868 - gbSerialBits++; 3.6869 - linkProc(); 3.6870 - if (gbSerialOn && (gbMemory[0xff02] & 1)) 3.6871 - { 3.6872 - if (gbSerialBits == 8) 3.6873 - { 3.6874 - gbSerialBits = 0; 3.6875 - gbMemory[0xff01] = 0xff; 3.6876 - gbMemory[0xff02] &= 0x7f; 3.6877 - gbSerialOn = 0; 3.6878 - gbInterrupt |= 8; 3.6879 - gbSerialTicks = 0; 3.6880 - } 3.6881 - } 3.6882 - gbSerialTicks += GBSERIAL_CLOCK_TICKS; 3.6883 - } 3.6884 - } 3.6885 - else 3.6886 - { 3.6887 -#endif 3.6888 - if (gbMemory[0xff02] & 1) 3.6889 - { 3.6890 - gbSerialTicks -= clockTicks; 3.6891 - 3.6892 - // overflow 3.6893 - while (gbSerialTicks <= 0) 3.6894 - { 3.6895 - // shift serial byte to right and put a 1 bit in its place 3.6896 - // gbMemory[0xff01] = 0x80 | (gbMemory[0xff01]>>1); 3.6897 - // increment number of shifted bits 3.6898 - gbSerialBits++; 3.6899 - if (gbSerialBits == 8) 3.6900 - { 3.6901 - // end of transmission 3.6902 - if (gbSerialFunction) // external device 3.6903 - gbMemory[0xff01] = gbSerialFunction(gbMemory[0xff01]); 3.6904 - else 3.6905 - gbMemory[0xff01] = 0xff; 3.6906 - gbSerialTicks = 0; 3.6907 - gbMemory[0xff02] &= 0x7f; 3.6908 - gbSerialOn = 0; 3.6909 - gbInterrupt |= 8; 3.6910 - gbSerialBits = 0; 3.6911 - } 3.6912 - else 3.6913 - gbSerialTicks += GBSERIAL_CLOCK_TICKS; 3.6914 - } 3.6915 - } 3.6916 -#ifdef LINK_EMULATION 3.6917 - } 3.6918 -#endif 3.6919 - } 3.6920 - 3.6921 - // timer emulation 3.6922 - if (gbTimerOn) 3.6923 - { 3.6924 - gbTimerTicks -= clockTicks; 3.6925 - 3.6926 - while (gbTimerTicks <= 0) 3.6927 - { 3.6928 - register_TIMA++; 3.6929 - 3.6930 - if (register_TIMA == 0) 3.6931 - { 3.6932 - // timer overflow! 3.6933 - 3.6934 - // reload timer modulo 3.6935 - register_TIMA = register_TMA; 3.6936 - 3.6937 - // flag interrupt 3.6938 - gbInterrupt |= 4; 3.6939 - } 3.6940 - 3.6941 - gbTimerTicks += gbTimerClockTicks; 3.6942 - } 3.6943 - } 3.6944 - 3.6945 - /* 3.6946 - if(soundOffFlag) 3.6947 - { 3.6948 - if(synchronize && !speedup) 3.6949 - { 3.6950 - synchronizeTicks -= clockTicks; 3.6951 - 3.6952 - while(synchronizeTicks < 0) 3.6953 - { 3.6954 - synchronizeTicks += SYNCHRONIZE_CLOCK_TICKS; 3.6955 - 3.6956 - DWORD now = timeGetTime(); 3.6957 - gbElapsedTime += (now - timeNow); 3.6958 - 3.6959 - if(gbElapsedTime < 50) 3.6960 - { 3.6961 - DWORD diff = 50 - gbElapsedTime; 3.6962 - Sleep(diff); 3.6963 - timeNow = timeGetTime(); 3.6964 - elapsedTime = timeNow - now - diff; 3.6965 - if((int)elapsedTime < 0) 3.6966 - elapsedTime = 0; 3.6967 - } else 3.6968 - { 3.6969 - timeNow = timeGetTime(); 3.6970 - elapsedTime = 0; 3.6971 - } 3.6972 - } 3.6973 - } 3.6974 - } 3.6975 - */ 3.6976 - 3.6977 - soundTicks -= clockTicks; 3.6978 - while (soundTicks < 0) // must be < 1 when soundtick_t is real data type 3.6979 - { 3.6980 - soundTicks += SOUND_CLOCK_TICKS; 3.6981 - 3.6982 - gbSoundTick(); 3.6983 - } 3.6984 - 3.6985 - register_IF = gbInterrupt; 3.6986 - 3.6987 - if (IFF & 0x20) 3.6988 - { 3.6989 - IFF &= 0xdf; 3.6990 - IFF |= 0x01; 3.6991 - gbInterruptWait = 0; 3.6992 - } 3.6993 - else if (gbInterrupt) 3.6994 - { 3.6995 - if (gbInterruptWait == 0) 3.6996 - { 3.6997 - // gbInterruptWait = 0; 3.6998 - 3.6999 - if (IFF & 0x01) 3.7000 - { 3.7001 - if ((gbInterrupt & 1) && (register_IE & 1)) 3.7002 - { 3.7003 - gbVblank_interrupt(); 3.7004 - continue; 3.7005 - } 3.7006 - 3.7007 - if ((gbInterrupt & 2) && (register_IE & 2)) 3.7008 - { 3.7009 - gbLcd_interrupt(); 3.7010 - continue; 3.7011 - } 3.7012 - 3.7013 - if ((gbInterrupt & 4) && (register_IE & 4)) 3.7014 - { 3.7015 - gbTimer_interrupt(); 3.7016 - continue; 3.7017 - } 3.7018 - 3.7019 - if ((gbInterrupt & 8) && (register_IE & 8)) 3.7020 - { 3.7021 - gbSerial_interrupt(); 3.7022 - continue; 3.7023 - } 3.7024 - 3.7025 - if ((gbInterrupt & 16) && (register_IE & 16)) 3.7026 - { 3.7027 - gbJoypad_interrupt(); 3.7028 - continue; 3.7029 - } 3.7030 - } 3.7031 - } 3.7032 - else 3.7033 - { 3.7034 - gbInterruptWait -= clockTicks; 3.7035 - if (gbInterruptWait < 0) 3.7036 - gbInterruptWait = 0; 3.7037 - } 3.7038 - } 3.7039 - 3.7040 - if (useOldFrameTiming) 3.7041 - { 3.7042 - // old timing code 3.7043 - if (ticksToStop > 0) 3.7044 - continue; 3.7045 - } 3.7046 - else 3.7047 - { 3.7048 - if (!newFrame && (register_LCDC & 0x80) != 0) 3.7049 - continue; 3.7050 - } 3.7051 - 3.7052 - if (!(register_LCDC & 0x80)) 3.7053 - { 3.7054 - if (!useOldFrameTiming) 3.7055 - { 3.7056 - // FIXME: since register_LY can be reset to 0 by some games, frame length is variable 3.7057 - // and infinite loops can occurr 3.7058 - // for now, it IS necessary to do something on this condition or games like 3.7059 - // Megaman would freeze upon low-level restart interrupt sequence (Start+Select+A+B). 3.7060 - // the only sensible way to fix this issue is to implement the RIGHT frame timing 3.7061 -#ifdef WANTS_INCOMPLETE_WORKAROUND 3.7062 - if (systemReadJoypads()) 3.7063 - { 3.7064 - if (gbSgbMode && gbSgbMultiplayer) 3.7065 - { 3.7066 - if (gbSgbFourPlayers) 3.7067 - { 3.7068 - gbJoymask[0] = systemGetJoypad(0, false); 3.7069 - gbJoymask[1] = systemGetJoypad(1, false); 3.7070 - gbJoymask[2] = systemGetJoypad(2, false); 3.7071 - gbJoymask[3] = systemGetJoypad(3, false); 3.7072 - } 3.7073 - else 3.7074 - { 3.7075 - gbJoymask[0] = systemGetJoypad(0, false); 3.7076 - gbJoymask[1] = systemGetJoypad(1, false); 3.7077 - } 3.7078 - } 3.7079 - else 3.7080 - { 3.7081 - gbJoymask[0] = systemGetJoypad(0, false); 3.7082 - } 3.7083 - } 3.7084 - else 3.7085 - gbJoymask[0] = gbJoymask[1] = gbJoymask[2] = gbJoymask[3] = 0; 3.7086 + else 3.7087 + gbJoymask[0] = gbJoymask[1] = gbJoymask[2] = gbJoymask[3] = 0; 3.7088 #else 3.7089 #endif 3.7090 - } 3.7091 - } 3.7092 - 3.7093 - // makes sure frames are really divided across input sampling boundaries which occur at a constant rate 3.7094 - if (newFrame || useOldFrameTiming) 3.7095 - { 3.7096 -/// extern void VBAOnEnteringFrameBoundary(); 3.7097 -/// VBAOnEnteringFrameBoundary(); 3.7098 - 3.7099 - break; 3.7100 - } 3.7101 + } 3.7102 } 3.7103 + 3.7104 + // makes sure frames are really divided across input sampling boundaries which occur at a constant rate 3.7105 + if (newFrame || useOldFrameTiming) 3.7106 + { 3.7107 + /// extern void VBAOnEnteringFrameBoundary(); 3.7108 + /// VBAOnEnteringFrameBoundary(); 3.7109 + 3.7110 + break; 3.7111 + } 3.7112 + } 3.7113 } 3.7114 3.7115 struct EmulatedSystem GBSystem = 3.7116 -{ 3.7117 - // emuMain 3.7118 - gbEmulate, 3.7119 - // emuReset 3.7120 - gbReset, 3.7121 - // emuCleanUp 3.7122 - gbCleanUp, 3.7123 - // emuReadBattery 3.7124 - gbReadBatteryFile, 3.7125 - // emuWriteBattery 3.7126 - gbWriteBatteryFile, 3.7127 - // emuReadBatteryFromStream 3.7128 - gbReadBatteryFromStream, 3.7129 - // emuWriteBatteryToStream 3.7130 - gbWriteBatteryToStream, 3.7131 - // emuReadState 3.7132 - gbReadSaveState, 3.7133 - // emuWriteState 3.7134 - gbWriteSaveState, 3.7135 - // emuReadStateFromStream 3.7136 - gbReadSaveStateFromStream, 3.7137 - // emuWriteStateToStream 3.7138 - gbWriteSaveStateToStream, 3.7139 - // emuReadMemState 3.7140 - gbReadMemSaveState, 3.7141 - // emuWriteMemState 3.7142 - gbWriteMemSaveState, 3.7143 - // emuWritePNG 3.7144 - gbWritePNGFile, 3.7145 - // emuWriteBMP 3.7146 - gbWriteBMPFile, 3.7147 - // emuUpdateCPSR 3.7148 - NULL, 3.7149 - // emuHasDebugger 3.7150 - false, 3.7151 - // emuCount 3.7152 + { 3.7153 + // emuMain 3.7154 + gbEmulate, 3.7155 + // emuReset 3.7156 + gbReset, 3.7157 + // emuCleanUp 3.7158 + gbCleanUp, 3.7159 + // emuReadBattery 3.7160 + gbReadBatteryFile, 3.7161 + // emuWriteBattery 3.7162 + gbWriteBatteryFile, 3.7163 + // emuReadBatteryFromStream 3.7164 + gbReadBatteryFromStream, 3.7165 + // emuWriteBatteryToStream 3.7166 + gbWriteBatteryToStream, 3.7167 + // emuReadState 3.7168 + gbReadSaveState, 3.7169 + // emuWriteState 3.7170 + gbWriteSaveState, 3.7171 + // emuReadStateFromStream 3.7172 + gbReadSaveStateFromStream, 3.7173 + // emuWriteStateToStream 3.7174 + gbWriteSaveStateToStream, 3.7175 + // emuReadMemState 3.7176 + gbReadMemSaveState, 3.7177 + // emuWriteMemState 3.7178 + gbWriteMemSaveState, 3.7179 + // emuWritePNG 3.7180 + gbWritePNGFile, 3.7181 + // emuWriteBMP 3.7182 + gbWriteBMPFile, 3.7183 + // emuUpdateCPSR 3.7184 + NULL, 3.7185 + // emuHasDebugger 3.7186 + false, 3.7187 + // emuCount 3.7188 #ifdef FINAL_VERSION 3.7189 - 70000 / 4, 3.7190 + 70000 / 4, 3.7191 #else 3.7192 - 1000, 3.7193 + 1000, 3.7194 #endif 3.7195 -}; 3.7196 + }; 3.7197 3.7198 // is there a reason to use more than one set of counters? 3.7199 EmulatedSystemCounters &GBSystemCounters = systemCounters; 3.7200 3.7201 /* 3.7202 - EmulatedSystemCounters GBSystemCounters = 3.7203 - { 3.7204 - // frameCount 3.7205 - 0, 3.7206 - // lagCount 3.7207 - 0, 3.7208 - // lagged 3.7209 - true, 3.7210 - // laggedLast 3.7211 - true, 3.7212 - }; 3.7213 - */ 3.7214 + EmulatedSystemCounters GBSystemCounters = 3.7215 + { 3.7216 + // frameCount 3.7217 + 0, 3.7218 + // lagCount 3.7219 + 0, 3.7220 + // lagged 3.7221 + true, 3.7222 + // laggedLast 3.7223 + true, 3.7224 + }; 3.7225 +*/
4.1 --- a/src/lua/loadlib.c Sun Mar 04 22:44:42 2012 -0600 4.2 +++ b/src/lua/loadlib.c Mon Mar 05 01:25:11 2012 -0600 4.3 @@ -424,7 +424,7 @@ 4.4 funcname = mkfuncname(L, name); 4.5 if ((stat = ll_loadfunc(L, filename, funcname)) != 0) { 4.6 if (stat != ERRFUNC) loaderror(L, filename); /* real error */ 4.7 - lua_pushfstring(L, "\n\tno module " LUA_QS " in file " LUA_QS, 4.8 + lua_pushfstring(L, "\n\tno module " LUA_QS " in derp " LUA_QS, 4.9 name, filename); 4.10 return 1; /* function not found */ 4.11 }
5.1 --- a/src/sdl/SDL.cpp Sun Mar 04 22:44:42 2012 -0600 5.2 +++ b/src/sdl/SDL.cpp Mon Mar 05 01:25:11 2012 -0600 5.3 @@ -1966,6 +1966,7 @@ 5.4 5.5 void file_run() 5.6 { 5.7 + printf("RLM: file_run\n"); 5.8 utilGetBaseName(szFile, filename); 5.9 char *p = strrchr(filename, '.'); 5.10 5.11 @@ -1989,7 +1990,8 @@ 5.12 failed = !gbLoadRom(szFile); 5.13 if(!failed) { 5.14 systemCartridgeType = 1; 5.15 - theEmulator = GBSystem; 5.16 + printf("RLM: choosing GBSystem\n"); 5.17 + theEmulator = GBSystem; 5.18 if(sdlAutoIPS) { 5.19 int size = gbRomSize; 5.20 utilApplyIPS(ipsname, &gbRom, &size); 5.21 @@ -2007,7 +2009,7 @@ 5.22 // if(cpuEnhancedDetection && cpuSaveType == 0) { 5.23 // utilGBAFindSave(rom, size); 5.24 // } 5.25 - 5.26 + 5.27 sdlApplyPerImagePreferences(); 5.28 5.29 systemCartridgeType = 0; 5.30 @@ -2183,7 +2185,7 @@ 5.31 5.32 case 'r': 5.33 if(optarg == NULL) { 5.34 - fprintf(stderr, "ERROR: --recordMovie ('r') needs movie filename as option\n"); 5.35 + fprintf(stderr, "ERROR: --recordmovie ('r') needs movie filename as option\n"); 5.36 exit(-1); 5.37 } 5.38 strcpy(movieFileName, optarg); 5.39 @@ -2192,7 +2194,7 @@ 5.40 case 'p': // play without read-only (editable) 5.41 fprintf (stderr, "-p got called!\n"); 5.42 if(optarg == NULL) { 5.43 - fprintf(stderr, "ERROR: --playMovie ('p') needs movie filename as option\n"); 5.44 + fprintf(stderr, "ERROR: --playmovie ('p') needs movie filename as option\n"); 5.45 exit(-1); 5.46 } 5.47 strcpy(movieFileName, optarg); 5.48 @@ -2201,7 +2203,7 @@ 5.49 case 'w': // play with read-only 5.50 fprintf (stderr, "-w got called!\n"); 5.51 if(optarg == NULL) { 5.52 - fprintf(stderr, "ERROR: --watchMovie ('w') needs movie filename as option\n"); 5.53 + fprintf(stderr, "ERROR: --watchmovie ('w') needs movie filename as option\n"); 5.54 exit(-1); 5.55 } 5.56 strcpy(movieFileName, optarg); 5.57 @@ -2271,6 +2273,8 @@ 5.58 } 5.59 } 5.60 5.61 + printf("RLM: derpy loves you!\n"); 5.62 + printf("RLM: useMovie: %d (1 is record)\n", useMovie); 5.63 if(sdlPrintUsage) { 5.64 usage(argv[0]); 5.65 exit(-1); 5.66 @@ -2317,6 +2321,7 @@ 5.67 { 5.68 szFile = argv[optind]; 5.69 file_run(); 5.70 + printf("RLM: file_run() done\n"); 5.71 } 5.72 else 5.73 { 5.74 @@ -2336,7 +2341,7 @@ 5.75 5.76 //CPUInit(biosFileName, useBios); 5.77 CPUInit(); 5.78 - CPUReset(); 5.79 + CPUReset(); 5.80 } 5.81 5.82 if(debuggerStub) 5.83 @@ -2615,6 +2620,7 @@ 5.84 soundInit(); 5.85 5.86 autoFrameSkipLastTime = throttleLastTime = systemGetClock(); 5.87 + printf("RLM: and now for the movie part!\n"); 5.88 5.89 switch(useMovie) 5.90 { 5.91 @@ -2635,13 +2641,14 @@ 5.92 sdlReadBattery(); 5.93 break; 5.94 } 5.95 + printf("RLM: still alive after movie switch\n"); 5.96 SDL_WM_SetCaption("VisualBoyAdvance", NULL); 5.97 5.98 char *moviefile = getenv("AUTODEMO"); 5.99 -// fprintf (stderr, "Checking for AUTODEMO...\n"); 5.100 + fprintf (stderr, "Checking for AUTODEMO...\n"); 5.101 if (moviefile) 5.102 { 5.103 -// fprintf (stderr, "I got a filename OMG!\nCalling VBAMovieOpen...\n"); 5.104 + fprintf (stderr, "I got a filename OMG!\nCalling VBAMovieOpen...\n"); 5.105 VBAMovieOpen(moviefile, true); 5.106 } 5.107 5.108 @@ -2650,7 +2657,9 @@ 5.109 if(debugger && theEmulator.emuHasDebugger) 5.110 dbgMain(); 5.111 else { 5.112 - theEmulator.emuMain(theEmulator.emuCount); 5.113 + printf("RLM: emulator main\n"); 5.114 + theEmulator.emuMain(theEmulator.emuCount); 5.115 + printf("RLM: emulator main called\n"); 5.116 if(rewindSaveNeeded && rewindMemory && theEmulator.emuWriteMemState) { 5.117 rewindCount++; 5.118 if(rewindCount > 8) 5.119 @@ -3605,14 +3614,17 @@ 5.120 5.121 void VBAOnEnteringFrameBoundary() 5.122 { 5.123 - CallRegisteredLuaFunctions(LUACALL_AFTEREMULATION); 5.124 - 5.125 - if (VBALuaRunning()) 5.126 - { 5.127 - VBALuaFrameBoundary(); 5.128 - } 5.129 - 5.130 - VBAMovieUpdateState(); 5.131 + printf("RLM: Entering Frame Boundary\n"); 5.132 + CallRegisteredLuaFunctions(LUACALL_AFTEREMULATION); 5.133 + 5.134 + if (VBALuaRunning()) 5.135 + { 5.136 + VBALuaFrameBoundary(); 5.137 + } 5.138 + 5.139 + printf("RLM: Movie state update pending\n"); 5.140 + VBAMovieUpdateState(); 5.141 + printf("RLM: Movie state updated\n"); 5.142 } 5.143 5.144 void VBAOnExitingFrameBoundary()