// 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_
