blob: 420574c7f5cc0ef1f87bada3033729f29e51a1ad [file] [log] [blame]
// Copyright 2022 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_OPENH264_OPENH264_VIDEO_DECODER_H_
#define STARBOARD_SHARED_OPENH264_OPENH264_VIDEO_DECODER_H_
#include <queue>
#include <string>
#include <vector>
#include "starboard/common/optional.h"
#include "starboard/common/ref_counted.h"
#include "starboard/decode_target.h"
#include "starboard/shared/internal_only.h"
#include "starboard/shared/starboard/media/codec_util.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"
#include "starboard/shared/starboard/player/job_thread.h"
#include "third_party/openh264/include/codec_api.h"
#include "third_party/openh264/include/codec_app_def.h"
#include "third_party/openh264/include/codec_def.h"
namespace starboard {
namespace shared {
namespace openh264 {
class VideoDecoder : public starboard::player::filter::VideoDecoder,
private starboard::player::JobQueue::JobOwner {
public:
VideoDecoder(SbMediaVideoCodec video_codec,
SbPlayerOutputMode output_mode,
SbDecodeTargetGraphicsContextProvider*
decode_target_graphics_context_provider);
~VideoDecoder() override;
void Initialize(const DecoderStatusCB& decoder_status_cb,
const ErrorCB& error_cb) override;
// TODO: Verify if these values are correct.
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;
SbDecodeTarget GetCurrentDecodeTarget() override;
private:
static const int kDefaultOpenH264BitsDepth = 8;
typedef ::starboard::shared::starboard::player::filter::CpuVideoFrame
CpuVideoFrame;
// Operator to compare CpuVideoFrame by timestamp.
struct VideoFrameTimeStampGreater {
bool operator()(const scoped_refptr<CpuVideoFrame>& left,
const scoped_refptr<CpuVideoFrame>& right) const {
// In chronological order.
return left->timestamp() > right->timestamp();
}
};
typedef std::priority_queue<scoped_refptr<CpuVideoFrame>,
std::vector<scoped_refptr<CpuVideoFrame>>,
VideoFrameTimeStampGreater>
TimeSequentialQueue;
void UpdateDecodeTarget_Locked(const scoped_refptr<CpuVideoFrame>& frame);
// The following functions are only called on the decoder thread.
void InitializeCodec();
void TeardownCodec();
void DecodeOneBuffer(const scoped_refptr<InputBuffer>& input_buffer);
void DecodeEndOfStream();
void ProcessDecodedImage(unsigned char* decoded_frame[],
const SBufferInfo& buffer_info,
bool flushing);
void FlushFrames();
void ReportError(const std::string& error_message);
SbDecodeTargetGraphicsContextProvider*
decode_target_graphics_context_provider_;
SbPlayerOutputMode output_mode_;
// 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_;
// Openh264 does NOT always output video frames in chronological order.
// |time_sequential_queue_| is used to reorder |CpuVideoFrame|
// chronologically.
TimeSequentialQueue time_sequential_queue_;
std::queue<scoped_refptr<CpuVideoFrame>> frames_;
bool stream_ended_ = false;
// If decode-to-texture is enabled, then we store the decode target texture
// inside of this |decode_target_| member.
SbDecodeTarget decode_target_ = kSbDecodeTargetInvalid;
// 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 we do so through this mutex.
Mutex decode_target_mutex_;
// Working thread to avoid lengthy decoding work block the player thread.
scoped_ptr<starboard::player::JobThread> decoder_thread_;
// Openh264 decode handler.
ISVCDecoder* decoder_ = nullptr;
// The number of frames which have been sent to decoder but not received yet.
int frames_being_decoded_ = 0;
// Store current avc level profile and resolution.
optional<shared::starboard::media::VideoConfig> video_config_;
};
} // namespace openh264
} // namespace shared
} // namespace starboard
#endif // STARBOARD_SHARED_OPENH264_OPENH264_VIDEO_DECODER_H_