blob: 3feb7fea4454b637de3203fa326286f4eb3a99d9 [file] [log] [blame]
// Copyright 2018 The Cobalt Authors. All Rights Reserved.
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
#ifndef STARBOARD_SHARED_FFMPEG_FFMPEG_VIDEO_DECODER_IMPL_H_
#define STARBOARD_SHARED_FFMPEG_FFMPEG_VIDEO_DECODER_IMPL_H_
#include <pthread.h>
#include <queue>
#include "starboard/common/log.h"
#include "starboard/common/queue.h"
#include "starboard/common/ref_counted.h"
#include "starboard/media.h"
#include "starboard/shared/ffmpeg/ffmpeg_common.h"
#include "starboard/shared/ffmpeg/ffmpeg_dispatch.h"
#include "starboard/shared/ffmpeg/ffmpeg_video_decoder.h"
#include "starboard/shared/ffmpeg/ffmpeg_video_decoder_impl_interface.h"
#include "starboard/shared/internal_only.h"
#include "starboard/shared/starboard/player/filter/cpu_video_frame.h"
#include "starboard/shared/starboard/player/filter/video_decoder_internal.h"
#include "starboard/shared/starboard/player/input_buffer_internal.h"
namespace starboard {
namespace shared {
namespace ffmpeg {
// Forward class declaration of the explicit specialization with value FFMPEG.
template <>
class VideoDecoderImpl<FFMPEG>;
// Declare the explicit specialization of the class with value FFMPEG.
template <>
class VideoDecoderImpl<FFMPEG> : public VideoDecoder {
public:
VideoDecoderImpl(SbMediaVideoCodec video_codec,
SbPlayerOutputMode output_mode,
SbDecodeTargetGraphicsContextProvider*
decode_target_graphics_context_provider);
~VideoDecoderImpl() override;
// From: VideoDecoder
static VideoDecoder* Create(SbMediaVideoCodec video_codec,
SbPlayerOutputMode output_mode,
SbDecodeTargetGraphicsContextProvider*
decode_target_graphics_context_provider);
bool is_valid() const override;
// From: starboard::player::filter::VideoDecoder
void Initialize(const DecoderStatusCB& decoder_status_cb,
const ErrorCB& error_cb) override;
size_t GetPrerollFrameCount() const override { return 8; }
int64_t GetPrerollTimeout() const override { return kSbInt64Max; }
size_t GetMaxNumberOfCachedFrames() const override { return 12; }
void WriteInputBuffers(const InputBuffers& input_buffers) override;
void WriteEndOfStream() override;
void Reset() override;
#if LIBAVUTIL_VERSION_INT >= AV_VERSION_INT(52, 8, 0)
int AllocateBuffer(AVCodecContext* codec_context, AVFrame* frame, int flags);
#else // LIBAVUTIL_VERSION_INT >= AV_VERSION_INT(52, 8, 0)
int AllocateBuffer(AVCodecContext* codec_context, AVFrame* frame);
#endif // LIBAVUTIL_VERSION_INT >= AV_VERSION_INT(52, 8, 0)
private:
typedef ::starboard::shared::starboard::player::filter::CpuVideoFrame
CpuVideoFrame;
enum EventType {
kInvalid,
kReset,
kWriteInputBuffer,
kWriteEndOfStream,
};
struct Event {
EventType type;
// |input_buffer| is only used when |type| is kWriteInputBuffer.
scoped_refptr<InputBuffer> input_buffer;
explicit Event(EventType type = kInvalid) : type(type) {
SB_DCHECK(type != kWriteInputBuffer);
}
explicit Event(const scoped_refptr<InputBuffer>& input_buffer)
: type(kWriteInputBuffer), input_buffer(input_buffer) {}
};
static void* ThreadEntryPoint(void* context);
void DecoderThreadFunc();
bool DecodePacket(AVPacket* packet);
void InitializeCodec();
void TeardownCodec();
SbDecodeTarget GetCurrentDecodeTarget() override;
void UpdateDecodeTarget_Locked(const scoped_refptr<CpuVideoFrame>& frame);
// Processes a decoded video frame received from FFmpeg. The frame is
// ultimately passed to decoder_status_cb_.
//
// Returns false if the frame contains invalid data.
bool ProcessDecodedFrame(const AVFrame& av_frame);
FFMPEGDispatch* ffmpeg_;
// |video_codec_| will be initialized inside ctor and won't be changed during
// the life time of this class.
const SbMediaVideoCodec video_codec_;
// The following callbacks will be initialized in Initialize() and won't be
// changed during the life time of this class.
DecoderStatusCB decoder_status_cb_;
ErrorCB error_cb_;
Queue<Event> queue_;
// The AV related classes will only be created and accessed on the decoder
// thread.
AVCodecContext* codec_context_;
AVFrame* av_frame_;
bool stream_ended_;
bool error_occurred_;
// Working thread to avoid lengthy decoding work block the player thread.
pthread_t decoder_thread_;
// Decode-to-texture related state.
SbPlayerOutputMode output_mode_;
SbDecodeTargetGraphicsContextProvider*
decode_target_graphics_context_provider_;
// If decode-to-texture is enabled, then we store the decode target texture
// inside of this |decode_target_| member.
SbDecodeTarget decode_target_;
// GetCurrentDecodeTarget() needs to be called from an arbitrary thread
// to obtain the current decode target (which ultimately ends up being a
// copy of |decode_target_|), we need to safe-guard access to |decode_target_|
// and |frames_|, we do so through this mutex.
Mutex decode_target_and_frames_mutex_;
// int frame_last_rendered_pts_;
// scoped_refptr<VideoFrame> frame_;
std::queue<scoped_refptr<CpuVideoFrame>> frames_;
};
} // namespace ffmpeg
} // namespace shared
} // namespace starboard
#endif // STARBOARD_SHARED_FFMPEG_FFMPEG_VIDEO_DECODER_IMPL_H_