| Index: webrtc/common_video/corevideo_frame_buffer.cc
|
| diff --git a/webrtc/common_video/corevideo_frame_buffer.cc b/webrtc/common_video/corevideo_frame_buffer.cc
|
| index 3245bf5e498b17337fa6bdf5983bfecc5ebdd6c5..6455ed18f019b8ce3f5e535a97412f18db96eabe 100644
|
| --- a/webrtc/common_video/corevideo_frame_buffer.cc
|
| +++ b/webrtc/common_video/corevideo_frame_buffer.cc
|
| @@ -13,14 +13,35 @@
|
| #include "libyuv/convert.h"
|
| #include "webrtc/base/checks.h"
|
| #include "webrtc/base/logging.h"
|
| +#include "webrtc/common_video/libyuv/include/webrtc_libyuv.h"
|
|
|
| namespace webrtc {
|
|
|
| +CoreVideoFrameBuffer::CoreVideoFrameBuffer(CVPixelBufferRef pixel_buffer,
|
| + int adapted_width,
|
| + int adapted_height,
|
| + int crop_width,
|
| + int crop_height,
|
| + int crop_x,
|
| + int crop_y)
|
| + : NativeHandleBuffer(pixel_buffer, adapted_width, adapted_height),
|
| + pixel_buffer_(pixel_buffer),
|
| + buffer_width_(CVPixelBufferGetWidth(pixel_buffer)),
|
| + buffer_height_(CVPixelBufferGetHeight(pixel_buffer)),
|
| + crop_width_(crop_width), crop_height_(crop_height),
|
| + // Can only crop at even pixels.
|
| + crop_x_(crop_x & ~1), crop_y_(crop_y & ~1) {
|
| + CVBufferRetain(pixel_buffer_);
|
| +}
|
| +
|
| CoreVideoFrameBuffer::CoreVideoFrameBuffer(CVPixelBufferRef pixel_buffer)
|
| : NativeHandleBuffer(pixel_buffer,
|
| CVPixelBufferGetWidth(pixel_buffer),
|
| CVPixelBufferGetHeight(pixel_buffer)),
|
| - pixel_buffer_(pixel_buffer) {
|
| + pixel_buffer_(pixel_buffer),
|
| + buffer_width_(width_), buffer_height_(height_),
|
| + crop_width_(width_), crop_height_(height_),
|
| + crop_x_(0), crop_y_(0) {
|
| CVBufferRetain(pixel_buffer_);
|
| }
|
|
|
| @@ -30,32 +51,98 @@ CoreVideoFrameBuffer::~CoreVideoFrameBuffer() {
|
|
|
| rtc::scoped_refptr<VideoFrameBuffer>
|
| CoreVideoFrameBuffer::NativeToI420Buffer() {
|
| - RTC_DCHECK(CVPixelBufferGetPixelFormatType(pixel_buffer_) ==
|
| - kCVPixelFormatType_420YpCbCr8BiPlanarFullRange);
|
| - size_t width = CVPixelBufferGetWidthOfPlane(pixel_buffer_, 0);
|
| - size_t height = CVPixelBufferGetHeightOfPlane(pixel_buffer_, 0);
|
| - // TODO(tkchin): Use a frame buffer pool.
|
| - rtc::scoped_refptr<webrtc::I420Buffer> buffer =
|
| - new rtc::RefCountedObject<webrtc::I420Buffer>(width, height);
|
| + const OSType pixel_format = CVPixelBufferGetPixelFormatType(pixel_buffer_);
|
| + RTC_DCHECK(pixel_format == kCVPixelFormatType_420YpCbCr8BiPlanarFullRange ||
|
| + pixel_format == kCVPixelFormatType_420YpCbCr8BiPlanarVideoRange);
|
| +
|
| CVPixelBufferLockBaseAddress(pixel_buffer_, kCVPixelBufferLock_ReadOnly);
|
| const uint8_t* src_y = static_cast<const uint8_t*>(
|
| CVPixelBufferGetBaseAddressOfPlane(pixel_buffer_, 0));
|
| - int src_y_stride = CVPixelBufferGetBytesPerRowOfPlane(pixel_buffer_, 0);
|
| + const int src_y_stride = CVPixelBufferGetBytesPerRowOfPlane(pixel_buffer_, 0);
|
| const uint8_t* src_uv = static_cast<const uint8_t*>(
|
| CVPixelBufferGetBaseAddressOfPlane(pixel_buffer_, 1));
|
| - int src_uv_stride = CVPixelBufferGetBytesPerRowOfPlane(pixel_buffer_, 1);
|
| - int ret = libyuv::NV12ToI420(
|
| - src_y, src_y_stride, src_uv, src_uv_stride,
|
| + const int src_uv_stride =
|
| + CVPixelBufferGetBytesPerRowOfPlane(pixel_buffer_, 1);
|
| +
|
| + // Crop just by modifying pointers.
|
| + src_y += src_y_stride * crop_y_ + crop_x_;
|
| + src_uv += src_uv_stride * (crop_y_ / 2) + crop_x_;
|
| +
|
| + // TODO(magjed): Use a frame buffer pool.
|
| + NV12ToI420Scaler nv12_to_i420_scaler;
|
| + rtc::scoped_refptr<I420Buffer> buffer =
|
| + new rtc::RefCountedObject<I420Buffer>(width_, height_);
|
| + nv12_to_i420_scaler.NV12ToI420Scale(
|
| + src_y, src_y_stride,
|
| + src_uv, src_uv_stride,
|
| + crop_width_, crop_height_,
|
| buffer->MutableDataY(), buffer->StrideY(),
|
| buffer->MutableDataU(), buffer->StrideU(),
|
| buffer->MutableDataV(), buffer->StrideV(),
|
| - width, height);
|
| + buffer->width(), buffer->height());
|
| +
|
| CVPixelBufferUnlockBaseAddress(pixel_buffer_, kCVPixelBufferLock_ReadOnly);
|
| - if (ret) {
|
| - LOG(LS_ERROR) << "Error converting NV12 to I420: " << ret;
|
| - return nullptr;
|
| - }
|
| +
|
| return buffer;
|
| }
|
|
|
| +bool CoreVideoFrameBuffer::RequiresCropping() const {
|
| + return crop_width_ != buffer_width_ || crop_height_ != buffer_height_;
|
| +}
|
| +
|
| +bool CoreVideoFrameBuffer::CropAndScaleTo(
|
| + std::vector<uint8_t>* tmp_buffer,
|
| + CVPixelBufferRef output_pixel_buffer) const {
|
| + // Prepare output pointers.
|
| + RTC_DCHECK_EQ(CVPixelBufferGetPixelFormatType(output_pixel_buffer),
|
| + kCVPixelFormatType_420YpCbCr8BiPlanarFullRange);
|
| + CVReturn cv_ret = CVPixelBufferLockBaseAddress(output_pixel_buffer, 0);
|
| + if (cv_ret != kCVReturnSuccess) {
|
| + LOG(LS_ERROR) << "Failed to lock base address: " << cv_ret;
|
| + return false;
|
| + }
|
| + const int dst_width = CVPixelBufferGetWidth(output_pixel_buffer);
|
| + const int dst_height = CVPixelBufferGetHeight(output_pixel_buffer);
|
| + uint8_t* dst_y = reinterpret_cast<uint8_t*>(
|
| + CVPixelBufferGetBaseAddressOfPlane(output_pixel_buffer, 0));
|
| + const int dst_y_stride =
|
| + CVPixelBufferGetBytesPerRowOfPlane(output_pixel_buffer, 0);
|
| + uint8_t* dst_uv = reinterpret_cast<uint8_t*>(
|
| + CVPixelBufferGetBaseAddressOfPlane(output_pixel_buffer, 1));
|
| + const int dst_uv_stride =
|
| + CVPixelBufferGetBytesPerRowOfPlane(output_pixel_buffer, 1);
|
| +
|
| + // Prepare source pointers.
|
| + const OSType src_pixel_format =
|
| + CVPixelBufferGetPixelFormatType(pixel_buffer_);
|
| + RTC_DCHECK(
|
| + src_pixel_format == kCVPixelFormatType_420YpCbCr8BiPlanarFullRange ||
|
| + src_pixel_format == kCVPixelFormatType_420YpCbCr8BiPlanarVideoRange);
|
| + CVPixelBufferLockBaseAddress(pixel_buffer_, kCVPixelBufferLock_ReadOnly);
|
| + const uint8_t* src_y = static_cast<const uint8_t*>(
|
| + CVPixelBufferGetBaseAddressOfPlane(pixel_buffer_, 0));
|
| + const int src_y_stride = CVPixelBufferGetBytesPerRowOfPlane(pixel_buffer_, 0);
|
| + const uint8_t* src_uv = static_cast<const uint8_t*>(
|
| + CVPixelBufferGetBaseAddressOfPlane(pixel_buffer_, 1));
|
| + const int src_uv_stride =
|
| + CVPixelBufferGetBytesPerRowOfPlane(pixel_buffer_, 1);
|
| +
|
| + // Crop just by modifying pointers.
|
| + src_y += src_y_stride * crop_y_ + crop_x_;
|
| + src_uv += src_uv_stride * (crop_y_ / 2) + crop_x_;
|
| +
|
| + NV12Scale(tmp_buffer,
|
| + src_y, src_y_stride,
|
| + src_uv, src_uv_stride,
|
| + crop_width_, crop_height_,
|
| + dst_y, dst_y_stride,
|
| + dst_uv, dst_uv_stride,
|
| + dst_width, dst_height);
|
| +
|
| + CVPixelBufferUnlockBaseAddress(pixel_buffer_, kCVPixelBufferLock_ReadOnly);
|
| + CVPixelBufferUnlockBaseAddress(output_pixel_buffer, 0);
|
| +
|
| + return true;
|
| +}
|
| +
|
| } // namespace webrtc
|
|
|