PlaybackManager.cpp 8.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249
  1. /*
  2. * Copyright (c) 2022, Gregory Bertilson <zaggy1024@gmail.com>
  3. *
  4. * SPDX-License-Identifier: BSD-2-Clause
  5. */
  6. #include <AK/Format.h>
  7. #include <LibCore/Timer.h>
  8. #include <LibVideo/MatroskaReader.h>
  9. #include <LibVideo/VP9/Decoder.h>
  10. #include "MatroskaDemuxer.h"
  11. #include "PlaybackManager.h"
  12. namespace Video {
  13. DecoderErrorOr<NonnullRefPtr<PlaybackManager>> PlaybackManager::from_file(Object* event_handler, StringView filename)
  14. {
  15. NonnullOwnPtr<Demuxer> demuxer = TRY(MatroskaDemuxer::from_file(filename));
  16. auto video_tracks = demuxer->get_tracks_for_type(TrackType::Video);
  17. if (video_tracks.is_empty())
  18. return DecoderError::with_description(DecoderErrorCategory::Invalid, "No video track is present"sv);
  19. auto track = video_tracks[0];
  20. dbgln_if(PLAYBACK_MANAGER_DEBUG, "Selecting video track number {}", track.identifier());
  21. NonnullOwnPtr<VideoDecoder> decoder = make<VP9::Decoder>();
  22. return PlaybackManager::construct(event_handler, demuxer, track, decoder);
  23. }
  24. PlaybackManager::PlaybackManager(Object* event_handler, NonnullOwnPtr<Demuxer>& demuxer, Track video_track, NonnullOwnPtr<VideoDecoder>& decoder)
  25. : Object(event_handler)
  26. , m_main_loop(Core::EventLoop::current())
  27. , m_demuxer(move(demuxer))
  28. , m_selected_video_track(video_track)
  29. , m_decoder(move(decoder))
  30. , m_frame_queue(make<VideoFrameQueue>())
  31. , m_present_timer(Core::Timer::construct())
  32. , m_decode_timer(Core::Timer::construct())
  33. {
  34. m_present_timer->set_single_shot(true);
  35. m_present_timer->set_interval(0);
  36. m_present_timer->on_timeout = [&] { update_presented_frame(); };
  37. m_decode_timer->set_single_shot(true);
  38. m_decode_timer->set_interval(0);
  39. m_decode_timer->on_timeout = [&] { on_decode_timer(); };
  40. }
  41. void PlaybackManager::set_playback_status(PlaybackStatus status)
  42. {
  43. if (status != m_status) {
  44. auto old_status = m_status;
  45. m_status = status;
  46. dbgln_if(PLAYBACK_MANAGER_DEBUG, "Set playback status from {} to {}", playback_status_to_string(old_status), playback_status_to_string(m_status));
  47. if (status == PlaybackStatus::Playing) {
  48. if (old_status == PlaybackStatus::Stopped || old_status == PlaybackStatus::Corrupted) {
  49. restart_playback();
  50. m_frame_queue->clear();
  51. m_skipped_frames = 0;
  52. }
  53. m_last_present_in_real_time = Time::now_monotonic();
  54. m_present_timer->start();
  55. } else {
  56. m_last_present_in_media_time = current_playback_time();
  57. m_last_present_in_real_time = Time::zero();
  58. m_present_timer->stop();
  59. }
  60. m_main_loop.post_event(*this, make<PlaybackStatusChangeEvent>(status, old_status));
  61. }
  62. }
  63. void PlaybackManager::event(Core::Event& event)
  64. {
  65. if (event.type() == DecoderErrorOccurred) {
  66. auto& error_event = static_cast<DecoderErrorEvent&>(event);
  67. VERIFY(error_event.error().category() != DecoderErrorCategory::EndOfStream);
  68. set_playback_status(PlaybackStatus::Corrupted);
  69. }
  70. // Allow events to bubble up in all cases.
  71. event.ignore();
  72. }
  73. void PlaybackManager::resume_playback()
  74. {
  75. set_playback_status(PlaybackStatus::Playing);
  76. }
  77. void PlaybackManager::pause_playback()
  78. {
  79. set_playback_status(PlaybackStatus::Paused);
  80. }
  81. bool PlaybackManager::prepare_next_frame()
  82. {
  83. if (m_next_frame.has_value())
  84. return true;
  85. if (m_frame_queue->is_empty())
  86. return false;
  87. auto frame_item = m_frame_queue->dequeue();
  88. m_next_frame.emplace(move(frame_item));
  89. m_decode_timer->start();
  90. return true;
  91. }
  92. Time PlaybackManager::current_playback_time()
  93. {
  94. if (is_playing())
  95. return m_last_present_in_media_time + (Time::now_monotonic() - m_last_present_in_real_time);
  96. return m_last_present_in_media_time;
  97. }
  98. Time PlaybackManager::duration()
  99. {
  100. return m_demuxer->duration();
  101. }
  102. void PlaybackManager::on_decoder_error(DecoderError error)
  103. {
  104. dbgln("Playback error encountered: {}", error.string_literal());
  105. switch (error.category()) {
  106. case DecoderErrorCategory::EndOfStream:
  107. set_playback_status(PlaybackStatus::Stopped);
  108. break;
  109. default:
  110. set_playback_status(PlaybackStatus::Corrupted);
  111. m_main_loop.post_event(*this, make<DecoderErrorEvent>(move(error)));
  112. break;
  113. }
  114. }
  115. void PlaybackManager::update_presented_frame()
  116. {
  117. bool out_of_queued_frames = false;
  118. Optional<FrameQueueItem> frame_item_to_display;
  119. while (true) {
  120. out_of_queued_frames = out_of_queued_frames || !prepare_next_frame();
  121. if (out_of_queued_frames)
  122. break;
  123. VERIFY(m_next_frame.has_value());
  124. if (m_next_frame->is_error() || m_next_frame->timestamp() > current_playback_time())
  125. break;
  126. if (frame_item_to_display.has_value()) {
  127. dbgln_if(PLAYBACK_MANAGER_DEBUG, "At {}ms: Dropped {} in favor of {}", current_playback_time().to_milliseconds(), frame_item_to_display->debug_string(), m_next_frame->debug_string());
  128. m_skipped_frames++;
  129. }
  130. frame_item_to_display = m_next_frame.release_value();
  131. }
  132. if (!out_of_queued_frames && frame_item_to_display.has_value()) {
  133. m_main_loop.post_event(*this, make<VideoFramePresentEvent>(frame_item_to_display->bitmap()));
  134. m_last_present_in_media_time = current_playback_time();
  135. m_last_present_in_real_time = Time::now_monotonic();
  136. frame_item_to_display.clear();
  137. }
  138. if (frame_item_to_display.has_value()) {
  139. VERIFY(!m_next_frame.has_value());
  140. m_next_frame = frame_item_to_display;
  141. dbgln_if(PLAYBACK_MANAGER_DEBUG, "Set next frame back to dequeued item {}", m_next_frame->debug_string());
  142. }
  143. if (!is_playing())
  144. return;
  145. if (!out_of_queued_frames) {
  146. if (m_next_frame->is_error()) {
  147. on_decoder_error(m_next_frame->release_error());
  148. m_next_frame.clear();
  149. return;
  150. }
  151. auto frame_time_ms = (m_next_frame->timestamp() - current_playback_time()).to_milliseconds();
  152. VERIFY(frame_time_ms <= NumericLimits<int>::max());
  153. dbgln_if(PLAYBACK_MANAGER_DEBUG, "Time until next frame is {}ms", frame_time_ms);
  154. m_present_timer->start(max(static_cast<int>(frame_time_ms), 0));
  155. return;
  156. }
  157. set_playback_status(PlaybackStatus::Buffering);
  158. m_decode_timer->start();
  159. }
  160. void PlaybackManager::restart_playback()
  161. {
  162. m_last_present_in_media_time = Time::zero();
  163. m_last_present_in_real_time = Time::zero();
  164. auto result = m_demuxer->seek_to_most_recent_keyframe(m_selected_video_track, 0);
  165. if (result.is_error())
  166. on_decoder_error(result.release_error());
  167. }
  168. bool PlaybackManager::decode_and_queue_one_sample()
  169. {
  170. if (m_frame_queue->size() >= FRAME_BUFFER_COUNT)
  171. return false;
  172. #if PLAYBACK_MANAGER_DEBUG
  173. auto start_time = Time::now_monotonic();
  174. #endif
  175. #define TRY_OR_ENQUEUE_ERROR(expression) \
  176. ({ \
  177. auto _temporary_result = ((expression)); \
  178. if (_temporary_result.is_error()) { \
  179. dbgln("Enqueued decoder error: {}", _temporary_result.error().string_literal()); \
  180. m_frame_queue->enqueue(FrameQueueItem::error_marker(_temporary_result.release_error())); \
  181. return false; \
  182. } \
  183. _temporary_result.release_value(); \
  184. })
  185. auto frame_sample = TRY_OR_ENQUEUE_ERROR(m_demuxer->get_next_video_sample_for_track(m_selected_video_track));
  186. TRY_OR_ENQUEUE_ERROR(m_decoder->receive_sample(frame_sample->data()));
  187. auto decoded_frame = TRY_OR_ENQUEUE_ERROR(m_decoder->get_decoded_frame());
  188. auto& cicp = decoded_frame->cicp();
  189. cicp.adopt_specified_values(frame_sample->container_cicp());
  190. cicp.default_code_points_if_unspecified({ Video::ColorPrimaries::BT709, Video::TransferCharacteristics::BT709, Video::MatrixCoefficients::BT709, Video::ColorRange::Studio });
  191. auto bitmap = TRY_OR_ENQUEUE_ERROR(decoded_frame->to_bitmap());
  192. m_frame_queue->enqueue(FrameQueueItem::frame(bitmap, frame_sample->timestamp()));
  193. #if PLAYBACK_MANAGER_DEBUG
  194. auto end_time = Time::now_monotonic();
  195. dbgln("Decoding took {}ms", (end_time - start_time).to_milliseconds());
  196. #endif
  197. return true;
  198. }
  199. void PlaybackManager::on_decode_timer()
  200. {
  201. if (!decode_and_queue_one_sample() && is_buffering()) {
  202. set_playback_status(PlaybackStatus::Playing);
  203. return;
  204. }
  205. // Continually decode until buffering is complete
  206. if (is_buffering())
  207. m_decode_timer->start();
  208. }
  209. }