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