+ if (global.autoplay_mode == AUTOPLAY_MODE_FIX)
+ {
+ if (tape.auto_play_level_solved)
+ {
+ if (patch_nr > 0)
+ {
+ // level solved by patched tape -- save fixed tape
+ char *filename = getTapeFilename(level_nr);
+ char *filename_orig = getStringCat2(filename, ".orig");
+
+ // 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;
+ autoplay.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
+ // * level set identifier is missing (old-style tape without INFO chunk)
+ // * solution tape may have native format (like Supaplex solution files)
+
+ SaveScoreTape(level_nr);
+ SaveServerScore(level_nr, TRUE);
+
+ AutoPlayTapes_WaitForUpload();
+ }
+
+ if (patch_nr == 0)
+ autoplay.num_levels_played++;
+
+ if (tape.auto_play_level_solved)
+ autoplay.num_levels_solved++;
+
+ if (level_nr >= 0 && level_nr < MAX_TAPES_PER_SET)
+ autoplay.level_failed[level_nr] = !tape.auto_play_level_solved;
+ }
+ else
+ {
+ if (strEqual(global.autoplay_leveldir, "ALL"))
+ {
+ autoplay.all_levelsets = TRUE;
+
+ // tape mass-uploading only allowed for private tapes
+ if (global.autoplay_mode == AUTOPLAY_MODE_UPLOAD)
+ options.mytapes = TRUE;
+ }
+
+ if ((global.autoplay_mode == AUTOPLAY_MODE_SAVE ||
+ global.autoplay_mode == AUTOPLAY_MODE_UPLOAD) &&
+ !options.mytapes &&
+ options.player_name == NULL)
+ {
+ Fail("specify player name when uploading solution tapes");
+ }
+
+ if (global.autoplay_mode != AUTOPLAY_MODE_UPLOAD)
+ DrawCompleteVideoDisplay();
+
+ if (program.headless)
+ {
+ audio.sound_enabled = FALSE;
+ setup.engine_snapshot_mode = getStringCopy(STR_SNAPSHOT_MODE_OFF);
+ }
+
+ if (strSuffix(global.autoplay_leveldir, ".tape"))
+ {
+ autoplay.tape_filename = global.autoplay_leveldir;
+
+ if (!fileExists(autoplay.tape_filename))
+ Fail("tape file '%s' does not exist", autoplay.tape_filename);
+
+ LoadTapeFromFilename(autoplay.tape_filename);
+
+ if (tape.no_valid_file)
+ Fail("cannot load tape file '%s'", autoplay.tape_filename);
+
+ if (tape.no_info_chunk && !options.identifier)
+ Fail("cannot get levelset from tape file '%s'", autoplay.tape_filename);
+
+ if (tape.no_info_chunk && !options.level_nr)
+ Fail("cannot get level nr from tape file '%s'", autoplay.tape_filename);
+
+ global.autoplay_leveldir = tape.level_identifier;
+
+ if (options.identifier != NULL)
+ global.autoplay_leveldir = options.identifier;
+
+ if (options.level_nr != NULL)
+ tape.level_nr = atoi(options.level_nr);
+
+ if (tape.level_nr >= 0 && tape.level_nr < MAX_TAPES_PER_SET)
+ global.autoplay_level[tape.level_nr] = TRUE;
+
+ global.autoplay_all = FALSE;
+ options.mytapes = FALSE;
+ }
+
+ if (autoplay.all_levelsets)
+ {
+ // start auto-playing first level set
+ autoplay.leveldir = getFirstValidAutoPlayEntry(leveldir_first);
+ }
+ else
+ {
+ // auto-play selected level set
+ autoplay.leveldir = getTreeInfoFromIdentifier(leveldir_first,
+ global.autoplay_leveldir);
+ }
+
+ if (autoplay.leveldir == NULL)
+ Fail("no such level identifier: '%s'", global.autoplay_leveldir);
+
+ // only private tapes may be modified
+ if (global.autoplay_mode == AUTOPLAY_MODE_FIX)
+ options.mytapes = TRUE;
+
+ // set timestamp for batch tape upload
+ global.autoplay_time = time(NULL);
+
+ num_tapes = 0;
+
+ init_level_set = TRUE;
+ }
+
+ while (1)
+ {
+ if (init_level_set)
+ {
+ 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;
+
+ autoplay.num_levels_played = 0;
+ autoplay.num_levels_solved = 0;
+ autoplay.num_tapes_patched = 0;
+ autoplay.num_tape_missing = 0;
+
+ for (i = 0; i < MAX_TAPES_PER_SET; i++)
+ autoplay.level_failed[i] = FALSE;
+
+ PrintTapeReplayHeader(&autoplay);
+
+ init_level_set = FALSE;
+ }
+
+ if (autoplay.all_levelsets && global.autoplay_mode == AUTOPLAY_MODE_UPLOAD)
+ {
+ boolean skip_levelset = FALSE;
+
+ if (!directoryExists(getTapeDir(autoplay.leveldir->subdir)))
+ {
+ Print("No tape directory for this level set found -- skipping.\n");
+
+ skip_levelset = TRUE;
+ }
+
+ if (CheckTapeDirectoryUploadsComplete(autoplay.leveldir->subdir))
+ {
+ Print("All tapes for this level set already uploaded -- skipping.\n");
+
+ skip_levelset = TRUE;
+ }
+
+ if (skip_levelset)
+ {
+ PrintTapeReplaySummary(&autoplay);
+
+ // continue with next level set
+ autoplay.leveldir = getNextValidAutoPlayEntry(autoplay.leveldir);
+
+ // all level sets processed
+ if (autoplay.leveldir == NULL)
+ break;
+
+ init_level_set = TRUE;
+
+ continue;
+ }
+ }
+
+ if (global.autoplay_mode != AUTOPLAY_MODE_FIX || patch_nr == 0)
+ level_nr = autoplay.level_nr++;
+
+ UPDATE_BUSY_STATE();
+
+ // check if all tapes for this level set have been processed
+ if (level_nr > autoplay.leveldir->last_level)
+ {
+ PrintTapeReplaySummary(&autoplay);
+
+ if (!autoplay.all_levelsets)
+ break;
+
+ if (global.autoplay_mode == AUTOPLAY_MODE_UPLOAD)
+ MarkTapeDirectoryUploadsAsComplete(autoplay.leveldir->subdir);
+
+ // continue with next level set
+ autoplay.leveldir = getNextValidAutoPlayEntry(autoplay.leveldir);
+
+ // all level sets processed
+ if (autoplay.leveldir == NULL)
+ break;
+
+ init_level_set = TRUE;
+
+ continue;
+ }
+
+ // 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]);