|  | /* | 
|  | *  Copyright (c) 2020 The WebRTC project authors. All Rights Reserved. | 
|  | * | 
|  | *  Use of this source code is governed by a BSD-style license | 
|  | *  that can be found in the LICENSE file in the root of the source | 
|  | *  tree. An additional intellectual property rights grant can be found | 
|  | *  in the file PATENTS.  All contributing project authors may | 
|  | *  be found in the AUTHORS file in the root of the source tree. | 
|  | */ | 
|  |  | 
|  | #include "test/mappable_native_buffer.h" | 
|  |  | 
|  | #include "absl/algorithm/container.h" | 
|  | #include "api/video/i420_buffer.h" | 
|  | #include "api/video/nv12_buffer.h" | 
|  | #include "api/video/video_frame.h" | 
|  | #include "api/video/video_rotation.h" | 
|  | #include "common_video/include/video_frame_buffer.h" | 
|  | #include "rtc_base/checks.h" | 
|  |  | 
|  | namespace webrtc { | 
|  | namespace test { | 
|  |  | 
|  | namespace { | 
|  |  | 
|  | class NV12BufferWithDidConvertToI420 : public NV12Buffer { | 
|  | public: | 
|  | NV12BufferWithDidConvertToI420(int width, int height) | 
|  | : NV12Buffer(width, height), did_convert_to_i420_(false) {} | 
|  |  | 
|  | bool did_convert_to_i420() const { return did_convert_to_i420_; } | 
|  |  | 
|  | rtc::scoped_refptr<I420BufferInterface> ToI420() override { | 
|  | did_convert_to_i420_ = true; | 
|  | return NV12Buffer::ToI420(); | 
|  | } | 
|  |  | 
|  | private: | 
|  | bool did_convert_to_i420_; | 
|  | }; | 
|  |  | 
|  | }  // namespace | 
|  |  | 
|  | VideoFrame CreateMappableNativeFrame(int64_t ntp_time_ms, | 
|  | VideoFrameBuffer::Type mappable_type, | 
|  | int width, | 
|  | int height) { | 
|  | VideoFrame frame = | 
|  | VideoFrame::Builder() | 
|  | .set_video_frame_buffer(rtc::make_ref_counted<MappableNativeBuffer>( | 
|  | mappable_type, width, height)) | 
|  | .set_rtp_timestamp(99) | 
|  | .set_timestamp_ms(99) | 
|  | .set_rotation(kVideoRotation_0) | 
|  | .build(); | 
|  | frame.set_ntp_time_ms(ntp_time_ms); | 
|  | return frame; | 
|  | } | 
|  |  | 
|  | rtc::scoped_refptr<MappableNativeBuffer> GetMappableNativeBufferFromVideoFrame( | 
|  | const VideoFrame& frame) { | 
|  | return rtc::scoped_refptr<MappableNativeBuffer>( | 
|  | static_cast<MappableNativeBuffer*>(frame.video_frame_buffer().get())); | 
|  | } | 
|  |  | 
|  | MappableNativeBuffer::ScaledBuffer::ScaledBuffer( | 
|  | rtc::scoped_refptr<MappableNativeBuffer> parent, | 
|  | int width, | 
|  | int height) | 
|  | : parent_(std::move(parent)), width_(width), height_(height) {} | 
|  |  | 
|  | MappableNativeBuffer::ScaledBuffer::~ScaledBuffer() {} | 
|  |  | 
|  | rtc::scoped_refptr<VideoFrameBuffer> | 
|  | MappableNativeBuffer::ScaledBuffer::CropAndScale(int offset_x, | 
|  | int offset_y, | 
|  | int crop_width, | 
|  | int crop_height, | 
|  | int scaled_width, | 
|  | int scaled_height) { | 
|  | return rtc::make_ref_counted<ScaledBuffer>(parent_, scaled_width, | 
|  | scaled_height); | 
|  | } | 
|  |  | 
|  | rtc::scoped_refptr<I420BufferInterface> | 
|  | MappableNativeBuffer::ScaledBuffer::ToI420() { | 
|  | return parent_->GetOrCreateMappedBuffer(width_, height_)->ToI420(); | 
|  | } | 
|  |  | 
|  | rtc::scoped_refptr<VideoFrameBuffer> | 
|  | MappableNativeBuffer::ScaledBuffer::GetMappedFrameBuffer( | 
|  | rtc::ArrayView<VideoFrameBuffer::Type> types) { | 
|  | if (absl::c_find(types, parent_->mappable_type_) == types.end()) | 
|  | return nullptr; | 
|  | return parent_->GetOrCreateMappedBuffer(width_, height_); | 
|  | } | 
|  |  | 
|  | MappableNativeBuffer::MappableNativeBuffer(VideoFrameBuffer::Type mappable_type, | 
|  | int width, | 
|  | int height) | 
|  | : mappable_type_(mappable_type), width_(width), height_(height) { | 
|  | RTC_DCHECK(mappable_type_ == VideoFrameBuffer::Type::kI420 || | 
|  | mappable_type_ == VideoFrameBuffer::Type::kNV12); | 
|  | } | 
|  |  | 
|  | MappableNativeBuffer::~MappableNativeBuffer() {} | 
|  |  | 
|  | rtc::scoped_refptr<VideoFrameBuffer> MappableNativeBuffer::CropAndScale( | 
|  | int offset_x, | 
|  | int offset_y, | 
|  | int crop_width, | 
|  | int crop_height, | 
|  | int scaled_width, | 
|  | int scaled_height) { | 
|  | return FullSizeBuffer()->CropAndScale( | 
|  | offset_x, offset_y, crop_width, crop_height, scaled_width, scaled_height); | 
|  | } | 
|  |  | 
|  | rtc::scoped_refptr<I420BufferInterface> MappableNativeBuffer::ToI420() { | 
|  | return FullSizeBuffer()->ToI420(); | 
|  | } | 
|  |  | 
|  | rtc::scoped_refptr<VideoFrameBuffer> MappableNativeBuffer::GetMappedFrameBuffer( | 
|  | rtc::ArrayView<VideoFrameBuffer::Type> types) { | 
|  | return FullSizeBuffer()->GetMappedFrameBuffer(types); | 
|  | } | 
|  |  | 
|  | std::vector<rtc::scoped_refptr<VideoFrameBuffer>> | 
|  | MappableNativeBuffer::GetMappedFramedBuffers() const { | 
|  | MutexLock lock(&lock_); | 
|  | return mapped_buffers_; | 
|  | } | 
|  |  | 
|  | bool MappableNativeBuffer::DidConvertToI420() const { | 
|  | if (mappable_type_ != VideoFrameBuffer::Type::kNV12) | 
|  | return false; | 
|  | MutexLock lock(&lock_); | 
|  | for (auto& mapped_buffer : mapped_buffers_) { | 
|  | if (static_cast<NV12BufferWithDidConvertToI420*>(mapped_buffer.get()) | 
|  | ->did_convert_to_i420()) { | 
|  | return true; | 
|  | } | 
|  | } | 
|  | return false; | 
|  | } | 
|  |  | 
|  | rtc::scoped_refptr<MappableNativeBuffer::ScaledBuffer> | 
|  | MappableNativeBuffer::FullSizeBuffer() { | 
|  | return rtc::make_ref_counted<ScaledBuffer>( | 
|  | rtc::scoped_refptr<MappableNativeBuffer>(this), width_, height_); | 
|  | } | 
|  |  | 
|  | rtc::scoped_refptr<VideoFrameBuffer> | 
|  | MappableNativeBuffer::GetOrCreateMappedBuffer(int width, int height) { | 
|  | MutexLock lock(&lock_); | 
|  | for (auto& mapped_buffer : mapped_buffers_) { | 
|  | if (mapped_buffer->width() == width && mapped_buffer->height() == height) { | 
|  | return mapped_buffer; | 
|  | } | 
|  | } | 
|  | rtc::scoped_refptr<VideoFrameBuffer> mapped_buffer; | 
|  | switch (mappable_type_) { | 
|  | case VideoFrameBuffer::Type::kI420: { | 
|  | rtc::scoped_refptr<I420Buffer> i420_buffer = | 
|  | I420Buffer::Create(width, height); | 
|  | I420Buffer::SetBlack(i420_buffer.get()); | 
|  | mapped_buffer = i420_buffer; | 
|  | break; | 
|  | } | 
|  | case VideoFrameBuffer::Type::kNV12: { | 
|  | auto nv12_buffer = | 
|  | rtc::make_ref_counted<NV12BufferWithDidConvertToI420>(width, height); | 
|  | nv12_buffer->InitializeData(); | 
|  | mapped_buffer = std::move(nv12_buffer); | 
|  | break; | 
|  | } | 
|  | default: | 
|  | RTC_DCHECK_NOTREACHED(); | 
|  | } | 
|  | mapped_buffers_.push_back(mapped_buffer); | 
|  | return mapped_buffer; | 
|  | } | 
|  |  | 
|  | }  // namespace test | 
|  | }  // namespace webrtc |