blob: f1df13e554e3ac1d4f8acbcaa268bb3baf60a96c [file] [log] [blame]
Anders Carlsson7bca8ca2018-08-30 09:30:29 +02001/*
2 * Copyright 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
11#import "RTCEncodedImage+Private.h"
12
Kári Tristan Helgason03e85d22019-04-25 08:57:41 +020013#import <objc/runtime.h>
14
Anders Carlsson7bca8ca2018-08-30 09:30:29 +020015#include "rtc_base/numerics/safe_conversions.h"
16
Niels Möller4d504c72019-06-18 15:56:56 +020017// A simple wrapper around webrtc::EncodedImageBufferInterface to make it usable with associated
18// objects.
19@interface RTCWrappedEncodedImageBuffer : NSObject
20@property(nonatomic) rtc::scoped_refptr<webrtc::EncodedImageBufferInterface> buffer;
21- (instancetype)initWithEncodedImageBuffer:
22 (rtc::scoped_refptr<webrtc::EncodedImageBufferInterface>)buffer;
Kári Tristan Helgason03e85d22019-04-25 08:57:41 +020023@end
Niels Möller4d504c72019-06-18 15:56:56 +020024@implementation RTCWrappedEncodedImageBuffer
Kári Tristan Helgason03e85d22019-04-25 08:57:41 +020025@synthesize buffer = _buffer;
Niels Möller4d504c72019-06-18 15:56:56 +020026- (instancetype)initWithEncodedImageBuffer:
27 (rtc::scoped_refptr<webrtc::EncodedImageBufferInterface>)buffer {
Kári Tristan Helgason03e85d22019-04-25 08:57:41 +020028 self = [super init];
29 if (self) {
30 _buffer = buffer;
31 }
32 return self;
33}
34@end
35
Anders Carlsson7bca8ca2018-08-30 09:30:29 +020036@implementation RTCEncodedImage (Private)
37
Niels Möller4d504c72019-06-18 15:56:56 +020038- (rtc::scoped_refptr<webrtc::EncodedImageBufferInterface>)encodedData {
39 RTCWrappedEncodedImageBuffer *wrappedBuffer =
40 objc_getAssociatedObject(self, @selector(encodedData));
Kári Tristan Helgason03e85d22019-04-25 08:57:41 +020041 return wrappedBuffer.buffer;
42}
43
Niels Möller4d504c72019-06-18 15:56:56 +020044- (void)setEncodedData:(rtc::scoped_refptr<webrtc::EncodedImageBufferInterface>)buffer {
45 return objc_setAssociatedObject(
46 self,
47 @selector(encodedData),
48 [[RTCWrappedEncodedImageBuffer alloc] initWithEncodedImageBuffer:buffer],
49 OBJC_ASSOCIATION_RETAIN_NONATOMIC);
Kári Tristan Helgason03e85d22019-04-25 08:57:41 +020050}
51
Dillon Cower5963fdd2019-02-07 16:17:43 -080052- (instancetype)initWithNativeEncodedImage:(const webrtc::EncodedImage &)encodedImage {
Anders Carlsson7bca8ca2018-08-30 09:30:29 +020053 if (self = [super init]) {
54 // Wrap the buffer in NSData without copying, do not take ownership.
Dillon Cower5963fdd2019-02-07 16:17:43 -080055 self.buffer = [NSData dataWithBytesNoCopy:encodedImage.mutable_data()
Niels Möller77536a22019-01-15 08:50:01 +010056 length:encodedImage.size()
57 freeWhenDone:NO];
Anders Carlsson7bca8ca2018-08-30 09:30:29 +020058 self.encodedWidth = rtc::dchecked_cast<int32_t>(encodedImage._encodedWidth);
59 self.encodedHeight = rtc::dchecked_cast<int32_t>(encodedImage._encodedHeight);
60 self.timeStamp = encodedImage.Timestamp();
61 self.captureTimeMs = encodedImage.capture_time_ms_;
62 self.ntpTimeMs = encodedImage.ntp_time_ms_;
63 self.flags = encodedImage.timing_.flags;
64 self.encodeStartMs = encodedImage.timing_.encode_start_ms;
65 self.encodeFinishMs = encodedImage.timing_.encode_finish_ms;
66 self.frameType = static_cast<RTCFrameType>(encodedImage._frameType);
67 self.rotation = static_cast<RTCVideoRotation>(encodedImage.rotation_);
68 self.completeFrame = encodedImage._completeFrame;
69 self.qp = @(encodedImage.qp_);
70 self.contentType = (encodedImage.content_type_ == webrtc::VideoContentType::SCREENSHARE) ?
71 RTCVideoContentTypeScreenshare :
72 RTCVideoContentTypeUnspecified;
73 }
74
75 return self;
76}
77
78- (webrtc::EncodedImage)nativeEncodedImage {
79 // Return the pointer without copying.
80 webrtc::EncodedImage encodedImage(
81 (uint8_t *)self.buffer.bytes, (size_t)self.buffer.length, (size_t)self.buffer.length);
82 encodedImage._encodedWidth = rtc::dchecked_cast<uint32_t>(self.encodedWidth);
83 encodedImage._encodedHeight = rtc::dchecked_cast<uint32_t>(self.encodedHeight);
84 encodedImage.SetTimestamp(self.timeStamp);
85 encodedImage.capture_time_ms_ = self.captureTimeMs;
86 encodedImage.ntp_time_ms_ = self.ntpTimeMs;
87 encodedImage.timing_.flags = self.flags;
88 encodedImage.timing_.encode_start_ms = self.encodeStartMs;
89 encodedImage.timing_.encode_finish_ms = self.encodeFinishMs;
Niels Möller87e2d782019-03-07 10:18:23 +010090 encodedImage._frameType = webrtc::VideoFrameType(self.frameType);
Anders Carlsson7bca8ca2018-08-30 09:30:29 +020091 encodedImage.rotation_ = webrtc::VideoRotation(self.rotation);
92 encodedImage._completeFrame = self.completeFrame;
93 encodedImage.qp_ = self.qp ? self.qp.intValue : -1;
94 encodedImage.content_type_ = (self.contentType == RTCVideoContentTypeScreenshare) ?
95 webrtc::VideoContentType::SCREENSHARE :
96 webrtc::VideoContentType::UNSPECIFIED;
97
98 return encodedImage;
99}
100
101@end