2017-12-17 03:43:09 +00:00
|
|
|
// Copyright 2017 Citra Emulator Project
|
|
|
|
// Licensed under GPLv2 or any later version
|
|
|
|
// Refer to the license.txt file included.
|
|
|
|
|
|
|
|
#pragma once
|
|
|
|
|
2018-07-07 14:30:37 +00:00
|
|
|
#include <functional>
|
2023-07-06 22:52:40 +00:00
|
|
|
#include <span>
|
2020-04-06 20:23:39 +00:00
|
|
|
#include <boost/serialization/vector.hpp>
|
2017-12-17 03:43:09 +00:00
|
|
|
#include "common/common_types.h"
|
|
|
|
|
|
|
|
namespace Service {
|
|
|
|
namespace HID {
|
|
|
|
struct AccelerometerDataEntry;
|
|
|
|
struct GyroscopeDataEntry;
|
|
|
|
struct PadState;
|
|
|
|
struct TouchDataEntry;
|
2018-03-09 17:54:43 +00:00
|
|
|
} // namespace HID
|
2017-12-17 03:43:09 +00:00
|
|
|
namespace IR {
|
|
|
|
struct ExtraHIDResponse;
|
|
|
|
union PadState;
|
2018-03-09 17:54:43 +00:00
|
|
|
} // namespace IR
|
|
|
|
} // namespace Service
|
2017-12-17 03:43:09 +00:00
|
|
|
|
2017-12-17 04:55:56 +00:00
|
|
|
namespace Core {
|
|
|
|
struct CTMHeader;
|
|
|
|
struct ControllerState;
|
|
|
|
|
|
|
|
class Movie {
|
|
|
|
public:
|
2020-07-08 15:56:37 +00:00
|
|
|
enum class PlayMode { None, Recording, Playing, MovieFinished };
|
2018-07-07 14:30:37 +00:00
|
|
|
enum class ValidationResult {
|
|
|
|
OK,
|
|
|
|
RevisionDismatch,
|
2020-06-30 14:48:10 +00:00
|
|
|
InputCountDismatch,
|
2018-07-07 14:30:37 +00:00
|
|
|
Invalid,
|
|
|
|
};
|
2017-12-17 04:55:56 +00:00
|
|
|
/**
|
2018-03-09 17:54:43 +00:00
|
|
|
* Gets the instance of the Movie singleton class.
|
|
|
|
* @returns Reference to the instance of the Movie singleton class.
|
|
|
|
*/
|
2017-12-17 04:55:56 +00:00
|
|
|
static Movie& GetInstance() {
|
|
|
|
return s_instance;
|
|
|
|
}
|
|
|
|
|
2020-07-07 08:38:24 +00:00
|
|
|
void SetPlaybackCompletionCallback(std::function<void()> completion_callback);
|
|
|
|
void StartPlayback(const std::string& movie_file);
|
2020-06-30 14:48:10 +00:00
|
|
|
void StartRecording(const std::string& movie_file, const std::string& author);
|
2018-09-26 12:38:57 +00:00
|
|
|
|
2020-06-29 14:32:24 +00:00
|
|
|
/**
|
|
|
|
* Sets the read-only status.
|
|
|
|
* When true, movies will be opened in read-only mode. Loading a state will resume playback
|
|
|
|
* from that state.
|
|
|
|
* When false, movies will be opened in read/write mode. Loading a state will start recording
|
|
|
|
* from that state (rerecording). To start rerecording without loading a state, one can save
|
|
|
|
* and then immediately load while in R/W.
|
|
|
|
*
|
|
|
|
* The default is true.
|
|
|
|
*/
|
|
|
|
void SetReadOnly(bool read_only);
|
|
|
|
|
2018-09-26 12:38:57 +00:00
|
|
|
/// Prepare to override the clock before playing back movies
|
|
|
|
void PrepareForPlayback(const std::string& movie_file);
|
|
|
|
|
|
|
|
/// Prepare to override the clock before recording movies
|
|
|
|
void PrepareForRecording();
|
|
|
|
|
2020-07-03 14:14:26 +00:00
|
|
|
ValidationResult ValidateMovie(const std::string& movie_file) const;
|
2018-09-26 12:38:57 +00:00
|
|
|
|
|
|
|
/// Get the init time that would override the one in the settings
|
|
|
|
u64 GetOverrideInitTime() const;
|
2020-06-30 14:48:10 +00:00
|
|
|
|
|
|
|
struct MovieMetadata {
|
|
|
|
u64 program_id;
|
|
|
|
std::string author;
|
|
|
|
u32 rerecord_count;
|
|
|
|
u64 input_count;
|
|
|
|
};
|
|
|
|
MovieMetadata GetMovieMetadata(const std::string& movie_file) const;
|
2017-12-17 04:55:56 +00:00
|
|
|
|
2020-06-29 14:32:24 +00:00
|
|
|
/// Get the current movie's unique ID. Used to provide separate savestate slots for movies.
|
|
|
|
u64 GetCurrentMovieID() const {
|
|
|
|
return id;
|
|
|
|
}
|
|
|
|
|
2017-12-17 04:55:56 +00:00
|
|
|
void Shutdown();
|
|
|
|
|
|
|
|
/**
|
|
|
|
* When recording: Takes a copy of the given input states so they can be used for playback
|
|
|
|
* When playing: Replaces the given input states with the ones stored in the playback file
|
|
|
|
*/
|
|
|
|
void HandlePadAndCircleStatus(Service::HID::PadState& pad_state, s16& circle_pad_x,
|
|
|
|
s16& circle_pad_y);
|
|
|
|
|
|
|
|
/**
|
2018-03-09 17:54:43 +00:00
|
|
|
* When recording: Takes a copy of the given input states so they can be used for playback
|
|
|
|
* When playing: Replaces the given input states with the ones stored in the playback file
|
|
|
|
*/
|
2017-12-17 04:55:56 +00:00
|
|
|
void HandleTouchStatus(Service::HID::TouchDataEntry& touch_data);
|
|
|
|
|
|
|
|
/**
|
2018-03-09 17:54:43 +00:00
|
|
|
* When recording: Takes a copy of the given input states so they can be used for playback
|
|
|
|
* When playing: Replaces the given input states with the ones stored in the playback file
|
|
|
|
*/
|
2017-12-17 04:55:56 +00:00
|
|
|
void HandleAccelerometerStatus(Service::HID::AccelerometerDataEntry& accelerometer_data);
|
|
|
|
|
|
|
|
/**
|
2018-03-09 17:54:43 +00:00
|
|
|
* When recording: Takes a copy of the given input states so they can be used for playback
|
|
|
|
* When playing: Replaces the given input states with the ones stored in the playback file
|
|
|
|
*/
|
2017-12-17 04:55:56 +00:00
|
|
|
void HandleGyroscopeStatus(Service::HID::GyroscopeDataEntry& gyroscope_data);
|
|
|
|
|
|
|
|
/**
|
2018-03-09 17:54:43 +00:00
|
|
|
* When recording: Takes a copy of the given input states so they can be used for playback
|
|
|
|
* When playing: Replaces the given input states with the ones stored in the playback file
|
|
|
|
*/
|
2017-12-17 04:55:56 +00:00
|
|
|
void HandleIrRst(Service::IR::PadState& pad_state, s16& c_stick_x, s16& c_stick_y);
|
|
|
|
|
|
|
|
/**
|
2018-03-09 17:54:43 +00:00
|
|
|
* When recording: Takes a copy of the given input states so they can be used for playback
|
|
|
|
* When playing: Replaces the given input states with the ones stored in the playback file
|
|
|
|
*/
|
2017-12-17 04:55:56 +00:00
|
|
|
void HandleExtraHidResponse(Service::IR::ExtraHIDResponse& extra_hid_response);
|
2020-07-08 15:56:37 +00:00
|
|
|
PlayMode GetPlayMode() const;
|
2017-12-17 04:55:56 +00:00
|
|
|
|
2020-07-06 16:15:26 +00:00
|
|
|
u64 GetCurrentInputIndex() const;
|
|
|
|
u64 GetTotalInputCount() const;
|
|
|
|
|
2020-10-14 14:35:20 +00:00
|
|
|
/**
|
|
|
|
* Saves the movie immediately, in its current state.
|
|
|
|
* This is called in Shutdown.
|
|
|
|
*/
|
|
|
|
void SaveMovie();
|
|
|
|
|
2017-12-17 04:55:56 +00:00
|
|
|
private:
|
|
|
|
static Movie s_instance;
|
|
|
|
|
|
|
|
void CheckInputEnd();
|
|
|
|
|
|
|
|
template <typename... Targs>
|
|
|
|
void Handle(Targs&... Fargs);
|
|
|
|
|
|
|
|
void Play(Service::HID::PadState& pad_state, s16& circle_pad_x, s16& circle_pad_y);
|
|
|
|
void Play(Service::HID::TouchDataEntry& touch_data);
|
|
|
|
void Play(Service::HID::AccelerometerDataEntry& accelerometer_data);
|
|
|
|
void Play(Service::HID::GyroscopeDataEntry& gyroscope_data);
|
|
|
|
void Play(Service::IR::PadState& pad_state, s16& c_stick_x, s16& c_stick_y);
|
|
|
|
void Play(Service::IR::ExtraHIDResponse& extra_hid_response);
|
|
|
|
|
|
|
|
void Record(const ControllerState& controller_state);
|
|
|
|
void Record(const Service::HID::PadState& pad_state, const s16& circle_pad_x,
|
|
|
|
const s16& circle_pad_y);
|
|
|
|
void Record(const Service::HID::TouchDataEntry& touch_data);
|
|
|
|
void Record(const Service::HID::AccelerometerDataEntry& accelerometer_data);
|
|
|
|
void Record(const Service::HID::GyroscopeDataEntry& gyroscope_data);
|
|
|
|
void Record(const Service::IR::PadState& pad_state, const s16& c_stick_x, const s16& c_stick_y);
|
|
|
|
void Record(const Service::IR::ExtraHIDResponse& extra_hid_response);
|
|
|
|
|
2020-07-03 14:14:26 +00:00
|
|
|
ValidationResult ValidateHeader(const CTMHeader& header) const;
|
2023-07-06 22:52:40 +00:00
|
|
|
ValidationResult ValidateInput(std::span<const u8> input, u64 expected_count) const;
|
2017-12-17 04:55:56 +00:00
|
|
|
|
|
|
|
PlayMode play_mode;
|
2020-06-30 14:48:10 +00:00
|
|
|
|
2018-07-07 14:30:37 +00:00
|
|
|
std::string record_movie_file;
|
2020-06-30 14:48:10 +00:00
|
|
|
std::string record_movie_author;
|
|
|
|
|
2020-08-27 14:27:29 +00:00
|
|
|
u64 init_time; // Clock init time override for RNG consistency
|
|
|
|
|
2017-12-17 04:55:56 +00:00
|
|
|
std::vector<u8> recorded_input;
|
2018-09-06 20:03:28 +00:00
|
|
|
std::size_t current_byte = 0;
|
2020-07-06 16:15:26 +00:00
|
|
|
u64 current_input = 0;
|
|
|
|
// Total input count of the current movie being played. Not used for recording.
|
|
|
|
u64 total_input = 0;
|
2020-06-30 14:48:10 +00:00
|
|
|
|
2020-06-29 14:32:24 +00:00
|
|
|
u64 id = 0; // ID of the current movie loaded
|
2020-08-27 14:27:29 +00:00
|
|
|
u64 program_id = 0;
|
2020-06-30 14:48:10 +00:00
|
|
|
u32 rerecord_count = 1;
|
2020-06-29 14:32:24 +00:00
|
|
|
bool read_only = true;
|
2020-04-06 20:23:39 +00:00
|
|
|
|
2020-07-07 08:38:24 +00:00
|
|
|
std::function<void()> playback_completion_callback = [] {};
|
2020-06-30 14:48:10 +00:00
|
|
|
|
2020-04-06 20:23:39 +00:00
|
|
|
template <class Archive>
|
2020-06-29 14:32:24 +00:00
|
|
|
void serialize(Archive& ar, const unsigned int file_version);
|
2020-04-06 20:23:39 +00:00
|
|
|
friend class boost::serialization::access;
|
2017-12-17 04:55:56 +00:00
|
|
|
};
|
2020-06-29 14:32:24 +00:00
|
|
|
} // namespace Core
|
|
|
|
|
|
|
|
BOOST_CLASS_VERSION(Core::Movie, 1)
|