| OLD | NEW |
| 1 /* | 1 /* |
| 2 * Copyright 2015 The WebRTC project authors. All Rights Reserved. | 2 * Copyright 2015 The WebRTC project authors. All Rights Reserved. |
| 3 * | 3 * |
| 4 * Use of this source code is governed by a BSD-style license | 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 | 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 | 6 * tree. An additional intellectual property rights grant can be found |
| 7 * in the file PATENTS. All contributing project authors may | 7 * in the file PATENTS. All contributing project authors may |
| 8 * be found in the AUTHORS file in the root of the source tree. | 8 * be found in the AUTHORS file in the root of the source tree. |
| 9 */ | 9 */ |
| 10 | 10 |
| 11 #include "webrtc/api/androidvideocapturer.h" | 11 #include "webrtc/api/androidvideocapturer.h" |
| 12 | 12 |
| 13 #include <memory> | 13 #include <memory> |
| 14 | 14 |
| 15 #include "webrtc/api/java/jni/native_handle_impl.h" | 15 #include "webrtc/api/java/jni/native_handle_impl.h" |
| 16 #include "webrtc/api/java/jni/surfacetexturehelper_jni.h" |
| 16 #include "webrtc/base/common.h" | 17 #include "webrtc/base/common.h" |
| 17 #include "webrtc/base/timeutils.h" | 18 #include "webrtc/base/timeutils.h" |
| 18 #include "webrtc/media/engine/webrtcvideoframe.h" | 19 #include "webrtc/media/engine/webrtcvideoframe.h" |
| 19 | 20 |
| 20 namespace webrtc { | 21 namespace webrtc { |
| 21 | 22 |
| 22 // A hack for avoiding deep frame copies in | |
| 23 // cricket::VideoCapturer.SignalFrameCaptured() using a custom FrameFactory. | |
| 24 // A frame is injected using UpdateCapturedFrame(), and converted into a | |
| 25 // cricket::VideoFrame with CreateAliasedFrame(). UpdateCapturedFrame() should | |
| 26 // be called before CreateAliasedFrame() for every frame. | |
| 27 // TODO(magjed): Add an interface cricket::VideoCapturer::OnFrameCaptured() | |
| 28 // for ref counted I420 frames instead of this hack. | |
| 29 class AndroidVideoCapturer::FrameFactory : public cricket::VideoFrameFactory { | |
| 30 public: | |
| 31 explicit FrameFactory( | |
| 32 const rtc::scoped_refptr<AndroidVideoCapturerDelegate>& delegate) | |
| 33 : delegate_(delegate) { | |
| 34 // Create a CapturedFrame that only contains header information, not the | |
| 35 // actual pixel data. | |
| 36 captured_frame_.pixel_height = 1; | |
| 37 captured_frame_.pixel_width = 1; | |
| 38 captured_frame_.data = nullptr; | |
| 39 captured_frame_.data_size = cricket::CapturedFrame::kUnknownDataSize; | |
| 40 captured_frame_.fourcc = static_cast<uint32_t>(cricket::FOURCC_ANY); | |
| 41 } | |
| 42 | |
| 43 void UpdateCapturedFrame( | |
| 44 const rtc::scoped_refptr<webrtc::VideoFrameBuffer>& buffer, | |
| 45 int rotation, | |
| 46 int64_t time_stamp_in_ns) { | |
| 47 RTC_DCHECK(rotation == 0 || rotation == 90 || rotation == 180 || | |
| 48 rotation == 270); | |
| 49 buffer_ = buffer; | |
| 50 captured_frame_.width = buffer->width(); | |
| 51 captured_frame_.height = buffer->height(); | |
| 52 captured_frame_.time_stamp = time_stamp_in_ns; | |
| 53 captured_frame_.rotation = static_cast<webrtc::VideoRotation>(rotation); | |
| 54 } | |
| 55 | |
| 56 void ClearCapturedFrame() { | |
| 57 buffer_ = nullptr; | |
| 58 captured_frame_.width = 0; | |
| 59 captured_frame_.height = 0; | |
| 60 captured_frame_.time_stamp = 0; | |
| 61 } | |
| 62 | |
| 63 const cricket::CapturedFrame* GetCapturedFrame() const { | |
| 64 return &captured_frame_; | |
| 65 } | |
| 66 | |
| 67 cricket::VideoFrame* CreateAliasedFrame( | |
| 68 const cricket::CapturedFrame* captured_frame, | |
| 69 int dst_width, | |
| 70 int dst_height) const override { | |
| 71 // Check that captured_frame is actually our frame. | |
| 72 RTC_CHECK(captured_frame == &captured_frame_); | |
| 73 RTC_CHECK(buffer_->native_handle() == nullptr); | |
| 74 | |
| 75 std::unique_ptr<cricket::VideoFrame> frame(new cricket::WebRtcVideoFrame( | |
| 76 ShallowCenterCrop(buffer_, dst_width, dst_height), | |
| 77 captured_frame->time_stamp, captured_frame->rotation)); | |
| 78 // Caller takes ownership. | |
| 79 // TODO(magjed): Change CreateAliasedFrame() to return a std::unique_ptr. | |
| 80 return apply_rotation_ ? frame->GetCopyWithRotationApplied()->Copy() | |
| 81 : frame.release(); | |
| 82 } | |
| 83 | |
| 84 cricket::VideoFrame* CreateAliasedFrame( | |
| 85 const cricket::CapturedFrame* input_frame, | |
| 86 int cropped_input_width, | |
| 87 int cropped_input_height, | |
| 88 int output_width, | |
| 89 int output_height) const override { | |
| 90 if (buffer_->native_handle() != nullptr) { | |
| 91 rtc::scoped_refptr<webrtc::VideoFrameBuffer> scaled_buffer( | |
| 92 static_cast<webrtc_jni::AndroidTextureBuffer*>(buffer_.get()) | |
| 93 ->CropScaleAndRotate(cropped_input_width, cropped_input_height, | |
| 94 output_width, output_height, | |
| 95 apply_rotation_ ? input_frame->rotation | |
| 96 : webrtc::kVideoRotation_0)); | |
| 97 return new cricket::WebRtcVideoFrame( | |
| 98 scaled_buffer, input_frame->time_stamp, | |
| 99 apply_rotation_ ? webrtc::kVideoRotation_0 : input_frame->rotation); | |
| 100 } | |
| 101 return VideoFrameFactory::CreateAliasedFrame(input_frame, | |
| 102 cropped_input_width, | |
| 103 cropped_input_height, | |
| 104 output_width, | |
| 105 output_height); | |
| 106 } | |
| 107 | |
| 108 private: | |
| 109 rtc::scoped_refptr<webrtc::VideoFrameBuffer> buffer_; | |
| 110 cricket::CapturedFrame captured_frame_; | |
| 111 rtc::scoped_refptr<AndroidVideoCapturerDelegate> delegate_; | |
| 112 }; | |
| 113 | |
| 114 AndroidVideoCapturer::AndroidVideoCapturer( | 23 AndroidVideoCapturer::AndroidVideoCapturer( |
| 115 const rtc::scoped_refptr<AndroidVideoCapturerDelegate>& delegate) | 24 const rtc::scoped_refptr<AndroidVideoCapturerDelegate>& delegate, |
| 25 rtc::scoped_refptr<webrtc_jni::SurfaceTextureHelper> helper) |
| 116 : running_(false), | 26 : running_(false), |
| 117 delegate_(delegate), | 27 delegate_(delegate), |
| 118 frame_factory_(NULL), | 28 surface_texture_helper_(helper), |
| 119 current_state_(cricket::CS_STOPPED) { | 29 current_state_(cricket::CS_STOPPED) { |
| 120 thread_checker_.DetachFromThread(); | 30 thread_checker_.DetachFromThread(); |
| 121 SetSupportedFormats(delegate_->GetSupportedFormats()); | 31 SetSupportedFormats(delegate_->GetSupportedFormats()); |
| 122 } | 32 } |
| 123 | 33 |
| 124 AndroidVideoCapturer::~AndroidVideoCapturer() { | 34 AndroidVideoCapturer::~AndroidVideoCapturer() { |
| 125 RTC_CHECK(!running_); | 35 RTC_CHECK(!running_); |
| 126 } | 36 } |
| 127 | 37 |
| 128 cricket::CaptureState AndroidVideoCapturer::Start( | 38 cricket::CaptureState AndroidVideoCapturer::Start( |
| 129 const cricket::VideoFormat& capture_format) { | 39 const cricket::VideoFormat& capture_format) { |
| 130 RTC_CHECK(thread_checker_.CalledOnValidThread()); | 40 RTC_CHECK(thread_checker_.CalledOnValidThread()); |
| 131 RTC_CHECK(!running_); | 41 RTC_CHECK(!running_); |
| 132 const int fps = cricket::VideoFormat::IntervalToFps(capture_format.interval); | 42 const int fps = cricket::VideoFormat::IntervalToFps(capture_format.interval); |
| 133 LOG(LS_INFO) << " AndroidVideoCapturer::Start " << capture_format.width << "x" | 43 LOG(LS_INFO) << " AndroidVideoCapturer::Start " << capture_format.width << "x" |
| 134 << capture_format.height << "@" << fps; | 44 << capture_format.height << "@" << fps; |
| 135 | 45 |
| 136 frame_factory_ = new AndroidVideoCapturer::FrameFactory(delegate_.get()); | |
| 137 set_frame_factory(frame_factory_); | |
| 138 | |
| 139 running_ = true; | 46 running_ = true; |
| 140 delegate_->Start(capture_format.width, capture_format.height, fps, this); | 47 delegate_->Start(capture_format.width, capture_format.height, fps, this); |
| 141 SetCaptureFormat(&capture_format); | 48 SetCaptureFormat(&capture_format); |
| 142 current_state_ = cricket::CS_STARTING; | 49 current_state_ = cricket::CS_STARTING; |
| 143 return current_state_; | 50 return current_state_; |
| 144 } | 51 } |
| 145 | 52 |
| 146 void AndroidVideoCapturer::Stop() { | 53 void AndroidVideoCapturer::Stop() { |
| 147 LOG(LS_INFO) << " AndroidVideoCapturer::Stop "; | 54 LOG(LS_INFO) << " AndroidVideoCapturer::Stop "; |
| 148 RTC_CHECK(thread_checker_.CalledOnValidThread()); | 55 RTC_CHECK(thread_checker_.CalledOnValidThread()); |
| (...skipping 20 matching lines...) Expand all Loading... |
| 169 void AndroidVideoCapturer::OnCapturerStarted(bool success) { | 76 void AndroidVideoCapturer::OnCapturerStarted(bool success) { |
| 170 RTC_CHECK(thread_checker_.CalledOnValidThread()); | 77 RTC_CHECK(thread_checker_.CalledOnValidThread()); |
| 171 cricket::CaptureState new_state = | 78 cricket::CaptureState new_state = |
| 172 success ? cricket::CS_RUNNING : cricket::CS_FAILED; | 79 success ? cricket::CS_RUNNING : cricket::CS_FAILED; |
| 173 if (new_state == current_state_) | 80 if (new_state == current_state_) |
| 174 return; | 81 return; |
| 175 current_state_ = new_state; | 82 current_state_ = new_state; |
| 176 SetCaptureState(new_state); | 83 SetCaptureState(new_state); |
| 177 } | 84 } |
| 178 | 85 |
| 179 void AndroidVideoCapturer::OnIncomingFrame( | 86 void AndroidVideoCapturer::OnNV21Frame(const uint8_t* video_frame, |
| 180 const rtc::scoped_refptr<webrtc::VideoFrameBuffer>& buffer, | 87 int length, |
| 181 int rotation, | 88 int width, |
| 182 int64_t time_stamp) { | 89 int height, |
| 183 RTC_CHECK(thread_checker_.CalledOnValidThread()); | 90 webrtc::VideoRotation rotation, |
| 184 frame_factory_->UpdateCapturedFrame(buffer, rotation, time_stamp); | 91 int64_t timestamp_ns) { |
| 185 SignalFrameCaptured(this, frame_factory_->GetCapturedFrame()); | 92 int adapted_width; |
| 186 frame_factory_->ClearCapturedFrame(); | 93 int adapted_height; |
| 94 int crop_width; |
| 95 int crop_height; |
| 96 int crop_x; |
| 97 int crop_y; |
| 98 |
| 99 if (!AdaptFrame(width, height, &adapted_width, &adapted_height, |
| 100 &crop_width, &crop_height, &crop_x, &crop_y)) { |
| 101 return; |
| 102 } |
| 103 |
| 104 int rotated_width = crop_width; |
| 105 int rotated_height = crop_height; |
| 106 |
| 107 if (apply_rotation() && (rotation == webrtc::kVideoRotation_90 || |
| 108 rotation == webrtc::kVideoRotation_270)) { |
| 109 std::swap(adapted_width, adapted_height); |
| 110 std::swap(rotated_width, rotated_height); |
| 111 } |
| 112 |
| 113 // TODO(nisse): Move buffer_pool_ here (from AndroidVideoCapturerJni) |
| 114 rtc::scoped_refptr<webrtc::I420Buffer> buffer( |
| 115 new rtc::RefCountedObject<I420Buffer>(rotated_width, rotated_height)); |
| 116 |
| 117 const uint8_t* y_plane = video_frame; |
| 118 const uint8_t* uv_plane = video_frame + width * height; |
| 119 |
| 120 // Can only crop at even pixels. |
| 121 crop_x &= ~1; |
| 122 |
| 123 libyuv::NV12ToI420Rotate( |
| 124 video_frame + width * crop_y + crop_x, width, // Y plane |
| 125 video_frame + width * (height + crop_y) + crop_x, width, // UV Plane |
| 126 buffer->MutableData(webrtc::kYPlane), buffer->stride(webrtc::kYPlane), |
| 127 // Swap U and V, since we have NV21, not NV12. |
| 128 buffer->MutableData(webrtc::kVPlane), buffer->stride(webrtc::kVPlane), |
| 129 buffer->MutableData(webrtc::kUPlane), buffer->stride(webrtc::kUPlane), |
| 130 crop_width, crop_height, |
| 131 static_cast<libyuv::RotationMode>( |
| 132 apply_rotation() ? rotation : webrtc::kVideoRotation_0)); |
| 133 |
| 134 if (adapted_width != rotated_width || adapted_height != rotated_height) { |
| 135 buffer = I420Buffer::CropAndScale( |
| 136 buffer, 0, 0, rotated_width, rotated_height, |
| 137 adapted_width, adapted_height); |
| 138 } |
| 139 // TODO(nisse): Use microsecond time instead. |
| 140 OnFrame(cricket::WebRtcVideoFrame( |
| 141 buffer, timestamp_ns, |
| 142 apply_rotation() ? webrtc::kVideoRotation_0 : rotation), |
| 143 width, height); |
| 144 } |
| 145 |
| 146 void AndroidVideoCapturer::OnTextureFrame( |
| 147 int width, |
| 148 int height, |
| 149 webrtc::VideoRotation rotation, |
| 150 int64_t timestamp_ns, |
| 151 const webrtc_jni::NativeHandleImpl& handle) { |
| 152 int adapted_width; |
| 153 int adapted_height; |
| 154 int crop_width; |
| 155 int crop_height; |
| 156 int crop_x; |
| 157 int crop_y; |
| 158 |
| 159 if (!AdaptFrame(width, height, &adapted_width, &adapted_height, |
| 160 &crop_width, &crop_height, &crop_x, &crop_y)) { |
| 161 return; |
| 162 } |
| 163 |
| 164 webrtc_jni::NativeHandleImpl::Matrix matrix = handle.sampling_matrix; |
| 165 |
| 166 matrix.Crop(crop_width / static_cast<float>(width), |
| 167 crop_height / static_cast<float>(height), |
| 168 crop_x / static_cast<float>(width), |
| 169 crop_y / static_cast<float>(height)); |
| 170 |
| 171 if (apply_rotation()) { |
| 172 if (rotation == webrtc::kVideoRotation_90 || |
| 173 rotation == webrtc::kVideoRotation_270) { |
| 174 std::swap(adapted_width, adapted_height); |
| 175 } |
| 176 matrix.Rotate(rotation); |
| 177 } |
| 178 |
| 179 // TODO(nisse): Use microsecond time instead. |
| 180 OnFrame(cricket::WebRtcVideoFrame( |
| 181 surface_texture_helper_->CreateTextureFrame( |
| 182 adapted_width, adapted_height, |
| 183 webrtc_jni::NativeHandleImpl(handle.oes_texture_id, matrix)), |
| 184 timestamp_ns, |
| 185 apply_rotation() ? webrtc::kVideoRotation_0 : rotation), |
| 186 width, height); |
| 187 } | 187 } |
| 188 | 188 |
| 189 void AndroidVideoCapturer::OnOutputFormatRequest( | 189 void AndroidVideoCapturer::OnOutputFormatRequest( |
| 190 int width, int height, int fps) { | 190 int width, int height, int fps) { |
| 191 RTC_CHECK(thread_checker_.CalledOnValidThread()); | 191 RTC_CHECK(thread_checker_.CalledOnValidThread()); |
| 192 cricket::VideoFormat format(width, height, | 192 cricket::VideoFormat format(width, height, |
| 193 cricket::VideoFormat::FpsToInterval(fps), 0); | 193 cricket::VideoFormat::FpsToInterval(fps), 0); |
| 194 video_adapter()->OnOutputFormatRequest(format); | 194 video_adapter()->OnOutputFormatRequest(format); |
| 195 } | 195 } |
| 196 | 196 |
| 197 bool AndroidVideoCapturer::GetBestCaptureFormat( | 197 bool AndroidVideoCapturer::GetBestCaptureFormat( |
| 198 const cricket::VideoFormat& desired, | 198 const cricket::VideoFormat& desired, |
| 199 cricket::VideoFormat* best_format) { | 199 cricket::VideoFormat* best_format) { |
| 200 // Delegate this choice to VideoCapturer.startCapture(). | 200 // Delegate this choice to VideoCapturer.startCapture(). |
| 201 *best_format = desired; | 201 *best_format = desired; |
| 202 return true; | 202 return true; |
| 203 } | 203 } |
| 204 | 204 |
| 205 } // namespace webrtc | 205 } // namespace webrtc |
| OLD | NEW |