sprang | c5d62e2 | 2017-04-03 06:53:04 | [diff] [blame] | 1 | /* |
| 2 | * Copyright (c) 2017 The WebRTC project authors. All Rights Reserved. |
| 3 | * |
| 4 | * Use of this source code is governed by a BSD-style license |
| 5 | * that can be found in the LICENSE file in the root of the source |
| 6 | * tree. An additional intellectual property rights grant can be found |
| 7 | * in the file PATENTS. All contributing project authors may |
| 8 | * be found in the AUTHORS file in the root of the source tree. |
| 9 | */ |
| 10 | |
Sebastian Jansson | f1f363f | 2018-08-13 12:24:58 | [diff] [blame] | 11 | #include "test/test_video_capturer.h" |
sprang | c5d62e2 | 2017-04-03 06:53:04 | [diff] [blame] | 12 | |
Niels Möller | 3793bb4 | 2018-12-20 12:46:06 | [diff] [blame] | 13 | #include <algorithm> |
| 14 | |
Mirko Bonadei | d970807 | 2019-01-25 19:26:48 | [diff] [blame] | 15 | #include "api/scoped_refptr.h" |
Yves Gerey | 3e70781 | 2018-11-28 15:47:49 | [diff] [blame] | 16 | #include "api/video/i420_buffer.h" |
| 17 | #include "api/video/video_frame_buffer.h" |
| 18 | #include "api/video/video_rotation.h" |
sprang | c5d62e2 | 2017-04-03 06:53:04 | [diff] [blame] | 19 | |
| 20 | namespace webrtc { |
| 21 | namespace test { |
Artem Titov | 6fd5f33d | 2023-03-25 20:15:09 | [diff] [blame] | 22 | |
Niels Möller | 3793bb4 | 2018-12-20 12:46:06 | [diff] [blame] | 23 | TestVideoCapturer::~TestVideoCapturer() = default; |
sprang | c5d62e2 | 2017-04-03 06:53:04 | [diff] [blame] | 24 | |
Asa Persson | 42eec3d | 2022-01-13 16:51:18 | [diff] [blame] | 25 | void TestVideoCapturer::OnOutputFormatRequest( |
| 26 | int width, |
| 27 | int height, |
| 28 | const absl::optional<int>& max_fps) { |
| 29 | absl::optional<std::pair<int, int>> target_aspect_ratio = |
| 30 | std::make_pair(width, height); |
| 31 | absl::optional<int> max_pixel_count = width * height; |
| 32 | video_adapter_.OnOutputFormatRequest(target_aspect_ratio, max_pixel_count, |
| 33 | max_fps); |
| 34 | } |
| 35 | |
Artem Titov | 9afdddf | 2019-10-10 11:29:03 | [diff] [blame] | 36 | void TestVideoCapturer::OnFrame(const VideoFrame& original_frame) { |
sprang | c5d62e2 | 2017-04-03 06:53:04 | [diff] [blame] | 37 | int cropped_width = 0; |
| 38 | int cropped_height = 0; |
| 39 | int out_width = 0; |
| 40 | int out_height = 0; |
| 41 | |
Artem Titov | 9afdddf | 2019-10-10 11:29:03 | [diff] [blame] | 42 | VideoFrame frame = MaybePreprocess(original_frame); |
| 43 | |
Artem Titov | 6fd5f33d | 2023-03-25 20:15:09 | [diff] [blame] | 44 | bool enable_adaptation; |
Artem Titov | 6a78e93 | 2023-03-21 12:46:47 | [diff] [blame] | 45 | { |
| 46 | MutexLock lock(&lock_); |
Artem Titov | 6fd5f33d | 2023-03-25 20:15:09 | [diff] [blame] | 47 | enable_adaptation = enable_adaptation_; |
Artem Titov | 6a78e93 | 2023-03-21 12:46:47 | [diff] [blame] | 48 | } |
Artem Titov | 5246ae2 | 2023-06-21 13:09:33 | [diff] [blame] | 49 | if (!enable_adaptation) { |
Artem Titov | 6a78e93 | 2023-03-21 12:46:47 | [diff] [blame] | 50 | broadcaster_.OnFrame(frame); |
Artem Titov | 5246ae2 | 2023-06-21 13:09:33 | [diff] [blame] | 51 | return; |
Artem Titov | 6a78e93 | 2023-03-21 12:46:47 | [diff] [blame] | 52 | } |
| 53 | |
Niels Möller | 3793bb4 | 2018-12-20 12:46:06 | [diff] [blame] | 54 | if (!video_adapter_.AdaptFrameResolution( |
sprang | c5d62e2 | 2017-04-03 06:53:04 | [diff] [blame] | 55 | frame.width(), frame.height(), frame.timestamp_us() * 1000, |
| 56 | &cropped_width, &cropped_height, &out_width, &out_height)) { |
| 57 | // Drop frame in order to respect frame rate constraint. |
Niels Möller | 3793bb4 | 2018-12-20 12:46:06 | [diff] [blame] | 58 | return; |
sprang | c5d62e2 | 2017-04-03 06:53:04 | [diff] [blame] | 59 | } |
| 60 | |
sprang | c5d62e2 | 2017-04-03 06:53:04 | [diff] [blame] | 61 | if (out_height != frame.height() || out_width != frame.width()) { |
| 62 | // Video adapter has requested a down-scale. Allocate a new buffer and |
| 63 | // return scaled version. |
Ilya Nikolaevskiy | 648b9d7 | 2019-12-03 15:54:17 | [diff] [blame] | 64 | // For simplicity, only scale here without cropping. |
sprang | c5d62e2 | 2017-04-03 06:53:04 | [diff] [blame] | 65 | rtc::scoped_refptr<I420Buffer> scaled_buffer = |
| 66 | I420Buffer::Create(out_width, out_height); |
magjed | 3f07549 | 2017-06-01 17:02:26 | [diff] [blame] | 67 | scaled_buffer->ScaleFrom(*frame.video_frame_buffer()->ToI420()); |
Ilya Nikolaevskiy | 648b9d7 | 2019-12-03 15:54:17 | [diff] [blame] | 68 | VideoFrame::Builder new_frame_builder = |
| 69 | VideoFrame::Builder() |
| 70 | .set_video_frame_buffer(scaled_buffer) |
| 71 | .set_rotation(kVideoRotation_0) |
| 72 | .set_timestamp_us(frame.timestamp_us()) |
| 73 | .set_id(frame.id()); |
| 74 | if (frame.has_update_rect()) { |
| 75 | VideoFrame::UpdateRect new_rect = frame.update_rect().ScaleWithFrame( |
| 76 | frame.width(), frame.height(), 0, 0, frame.width(), frame.height(), |
| 77 | out_width, out_height); |
| 78 | new_frame_builder.set_update_rect(new_rect); |
| 79 | } |
| 80 | broadcaster_.OnFrame(new_frame_builder.build()); |
| 81 | |
sprang | c5d62e2 | 2017-04-03 06:53:04 | [diff] [blame] | 82 | } else { |
| 83 | // No adaptations needed, just return the frame as is. |
Niels Möller | 3793bb4 | 2018-12-20 12:46:06 | [diff] [blame] | 84 | broadcaster_.OnFrame(frame); |
sprang | c5d62e2 | 2017-04-03 06:53:04 | [diff] [blame] | 85 | } |
Niels Möller | 3793bb4 | 2018-12-20 12:46:06 | [diff] [blame] | 86 | } |
sprang | c5d62e2 | 2017-04-03 06:53:04 | [diff] [blame] | 87 | |
Niels Möller | 3793bb4 | 2018-12-20 12:46:06 | [diff] [blame] | 88 | rtc::VideoSinkWants TestVideoCapturer::GetSinkWants() { |
| 89 | return broadcaster_.wants(); |
sprang | c5d62e2 | 2017-04-03 06:53:04 | [diff] [blame] | 90 | } |
| 91 | |
Sebastian Jansson | f1f363f | 2018-08-13 12:24:58 | [diff] [blame] | 92 | void TestVideoCapturer::AddOrUpdateSink( |
| 93 | rtc::VideoSinkInterface<VideoFrame>* sink, |
| 94 | const rtc::VideoSinkWants& wants) { |
Niels Möller | 3793bb4 | 2018-12-20 12:46:06 | [diff] [blame] | 95 | broadcaster_.AddOrUpdateSink(sink, wants); |
| 96 | UpdateVideoAdapter(); |
| 97 | } |
| 98 | |
| 99 | void TestVideoCapturer::RemoveSink(rtc::VideoSinkInterface<VideoFrame>* sink) { |
| 100 | broadcaster_.RemoveSink(sink); |
| 101 | UpdateVideoAdapter(); |
| 102 | } |
| 103 | |
| 104 | void TestVideoCapturer::UpdateVideoAdapter() { |
Rasmus Brandt | 287e464 | 2019-11-15 15:56:01 | [diff] [blame] | 105 | video_adapter_.OnSinkWants(broadcaster_.wants()); |
sprang | c5d62e2 | 2017-04-03 06:53:04 | [diff] [blame] | 106 | } |
| 107 | |
Artem Titov | 9afdddf | 2019-10-10 11:29:03 | [diff] [blame] | 108 | VideoFrame TestVideoCapturer::MaybePreprocess(const VideoFrame& frame) { |
Markus Handell | a5a4be1 | 2020-07-08 14:09:21 | [diff] [blame] | 109 | MutexLock lock(&lock_); |
Artem Titov | 9afdddf | 2019-10-10 11:29:03 | [diff] [blame] | 110 | if (preprocessor_ != nullptr) { |
| 111 | return preprocessor_->Preprocess(frame); |
| 112 | } else { |
| 113 | return frame; |
| 114 | } |
| 115 | } |
| 116 | |
sprang | c5d62e2 | 2017-04-03 06:53:04 | [diff] [blame] | 117 | } // namespace test |
| 118 | } // namespace webrtc |