+ // create backup from old tape, if not yet existing
+ if (!fileExists(filename_orig))
+ rename(filename, filename_orig);
+
+ SaveTapeToFilename(filename);
+
+ tape.auto_play_level_fixed = TRUE;
+ num_tapes_patched++;
+ }
+
+ // continue with next tape
+ patch_nr = 0;
+ }
+ else if (patch_name[patch_nr + 1] != NULL)
+ {
+ // level not solved by patched tape -- continue with next patch
+ patch_nr++;
+ }
+ else
+ {
+ // level not solved by any patched tape -- continue with next tape
+ tape.auto_play_level_not_fixable = TRUE;
+ patch_nr = 0;
+ }
+ }
+
+ // just finished auto-playing tape
+ PrintTapeReplayProgress(TRUE);
+
+ if (options.tape_log_filename != NULL)
+ CloseTapeLogfile();
+
+ if (global.autoplay_mode == AUTOPLAY_MODE_SAVE &&
+ tape.auto_play_level_solved)
+ {
+ AutoPlayTapes_SetScoreEntry(game.score_final, game.score_time_final);
+
+ if (leveldir_current)
+ {
+ // the tape's level set identifier may differ from current level set
+ strncpy(tape.level_identifier, leveldir_current->identifier,
+ MAX_FILENAME_LEN);
+ tape.level_identifier[MAX_FILENAME_LEN] = '\0';
+
+ // the tape's level number may differ from current level number
+ tape.level_nr = level_nr;
+ }
+
+ // save score tape to upload to server; may be required for some reasons:
+ // * level set identifier in solution tapes may differ from level set
+ // * solution tape may have native format (like Supaplex solution files)
+
+ SaveScoreTape(level_nr);
+ SaveServerScore(level_nr);
+
+ AutoPlayTapes_WaitForUpload();
+ }
+
+ if (patch_nr == 0)
+ num_levels_played++;
+
+ if (tape.auto_play_level_solved)
+ num_levels_solved++;
+
+ if (level_nr >= 0 && level_nr < MAX_TAPES_PER_SET)
+ level_failed[level_nr] = !tape.auto_play_level_solved;
+ }
+ else
+ {
+ DrawCompleteVideoDisplay();
+
+ audio.sound_enabled = FALSE;
+ setup.engine_snapshot_mode = getStringCopy(STR_SNAPSHOT_MODE_OFF);
+
+ if (strSuffix(global.autoplay_leveldir, ".tape"))
+ {
+ tape_filename = global.autoplay_leveldir;
+
+ LoadTapeFromFilename(tape_filename);
+
+ if (tape.no_valid_file)
+ {
+ if (!fileExists(tape_filename))
+ Fail("tape file '%s' does not exist", tape_filename);
+ else
+ Fail("cannot load tape file '%s'", tape_filename);
+ }
+
+ global.autoplay_leveldir = tape.level_identifier;
+
+ if (tape.level_nr >= 0 && tape.level_nr < MAX_TAPES_PER_SET)
+ global.autoplay_level[tape.level_nr] = TRUE;
+
+ global.autoplay_all = FALSE;
+ }
+
+ autoplay_leveldir = getTreeInfoFromIdentifier(leveldir_first,
+ global.autoplay_leveldir);
+
+ if (autoplay_leveldir == NULL)
+ Fail("no such level identifier: '%s'", global.autoplay_leveldir);
+
+ leveldir_current = autoplay_leveldir;
+
+ if (autoplay_leveldir->first_level < 0)
+ autoplay_leveldir->first_level = 0;
+ if (autoplay_leveldir->last_level >= MAX_TAPES_PER_SET)
+ autoplay_leveldir->last_level = MAX_TAPES_PER_SET - 1;
+
+ autoplay_level_nr = autoplay_leveldir->first_level;
+
+ PrintLine("=", 79);
+ if (global.autoplay_mode == AUTOPLAY_MODE_FIX)
+ Print("Automatically fixing level tapes\n");
+ else if (global.autoplay_mode == AUTOPLAY_MODE_UPLOAD)
+ Print("Automatically uploading level tapes\n");
+ else
+ Print("Automatically playing level tapes\n");
+ PrintLine("-", 79);
+ Print("Level series identifier: '%s'\n", autoplay_leveldir->identifier);
+ Print("Level series name: '%s'\n", autoplay_leveldir->name);
+ Print("Level series author: '%s'\n", autoplay_leveldir->author);
+ Print("Number of levels: %d\n", autoplay_leveldir->levels);
+ PrintLine("=", 79);
+ Print("\n");
+
+ for (i = 0; i < MAX_TAPES_PER_SET; i++)
+ level_failed[i] = FALSE;
+
+ // only private tapes may be modified
+ if (global.autoplay_mode == AUTOPLAY_MODE_FIX)
+ options.mytapes = TRUE;
+
+ autoplay_initialized = TRUE;
+ }
+
+ while (1)
+ {
+ if (global.autoplay_mode != AUTOPLAY_MODE_FIX || patch_nr == 0)
+ level_nr = autoplay_level_nr++;
+
+ if (level_nr > autoplay_leveldir->last_level)
+ break;
+
+ // set patch info (required for progress output)
+ strcpy(tape_patch_info, "");
+ if (global.autoplay_mode == AUTOPLAY_MODE_FIX)
+ snprintf(tape_patch_info, MAX_OUTPUT_LINESIZE, "[%-13s] ",
+ patch_name[patch_nr]);
+
+ if (!global.autoplay_all && !global.autoplay_level[level_nr])
+ continue;
+
+ TapeErase();