blob: 239695ac706dd6817972d15c350e11fba551b763 [file] [log] [blame]
tkchin@webrtc.orgff273322014-04-30 18:32:33 +00001/*
2 * libjingle
jlmiller@webrtc.org5f93d0a2015-01-20 21:36:13 +00003 * Copyright 2014 Google Inc.
tkchin@webrtc.orgff273322014-04-30 18:32:33 +00004 *
5 * Redistribution and use in source and binary forms, with or without
6 * modification, are permitted provided that the following conditions are met:
7 *
8 * 1. Redistributions of source code must retain the above copyright notice,
9 * this list of conditions and the following disclaimer.
10 * 2. Redistributions in binary form must reproduce the above copyright notice,
11 * this list of conditions and the following disclaimer in the documentation
12 * and/or other materials provided with the distribution.
13 * 3. The name of the author may not be used to endorse or promote products
14 * derived from this software without specific prior written permission.
15 *
16 * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR IMPLIED
17 * WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
18 * MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO
19 * EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
20 * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
21 * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS;
22 * OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY,
23 * WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR
24 * OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF
25 * ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
26 */
27
28#if !defined(__has_feature) || !__has_feature(objc_arc)
29#error "This file requires ARC support."
30#endif
31
32#import "RTCDataChannel+Internal.h"
33
34#include "talk/app/webrtc/datachannelinterface.h"
35
36namespace webrtc {
37
38class RTCDataChannelObserver : public DataChannelObserver {
39 public:
40 RTCDataChannelObserver(RTCDataChannel* channel) { _channel = channel; }
41
kjellander@webrtc.org14665ff2015-03-04 12:58:35 +000042 void OnStateChange() override {
tkchin@webrtc.orgff273322014-04-30 18:32:33 +000043 [_channel.delegate channelDidChangeState:_channel];
44 }
45
kjellander@webrtc.org14665ff2015-03-04 12:58:35 +000046 void OnMessage(const DataBuffer& buffer) override {
tkchin@webrtc.orgff273322014-04-30 18:32:33 +000047 if (!_channel.delegate) {
48 return;
49 }
50 RTCDataBuffer* dataBuffer =
51 [[RTCDataBuffer alloc] initWithDataBuffer:buffer];
52 [_channel.delegate channel:_channel didReceiveMessageWithBuffer:dataBuffer];
53 }
54
55 private:
56 __weak RTCDataChannel* _channel;
57};
58}
59
60// TODO(tkchin): move to shared location
61NSString* NSStringFromStdString(const std::string& stdString) {
62 // std::string may contain null termination character so we construct
63 // using length.
64 return [[NSString alloc] initWithBytes:stdString.data()
65 length:stdString.length()
66 encoding:NSUTF8StringEncoding];
67}
68
69std::string StdStringFromNSString(NSString* nsString) {
70 NSData* charData = [nsString dataUsingEncoding:NSUTF8StringEncoding];
71 return std::string(reinterpret_cast<const char*>([charData bytes]),
72 [charData length]);
73}
74
75@implementation RTCDataChannelInit {
76 webrtc::DataChannelInit _dataChannelInit;
77}
78
79- (BOOL)isOrdered {
80 return _dataChannelInit.ordered;
81}
82
83- (void)setIsOrdered:(BOOL)isOrdered {
84 _dataChannelInit.ordered = isOrdered;
85}
86
87- (NSInteger)maxRetransmitTime {
88 return _dataChannelInit.maxRetransmitTime;
89}
90
91- (void)setMaxRetransmitTime:(NSInteger)maxRetransmitTime {
92 _dataChannelInit.maxRetransmitTime = maxRetransmitTime;
93}
94
95- (NSInteger)maxRetransmits {
96 return _dataChannelInit.maxRetransmits;
97}
98
99- (void)setMaxRetransmits:(NSInteger)maxRetransmits {
100 _dataChannelInit.maxRetransmits = maxRetransmits;
101}
102
103- (NSString*)protocol {
104 return NSStringFromStdString(_dataChannelInit.protocol);
105}
106
107- (void)setProtocol:(NSString*)protocol {
108 _dataChannelInit.protocol = StdStringFromNSString(protocol);
109}
110
111- (BOOL)isNegotiated {
112 return _dataChannelInit.negotiated;
113}
114
115- (void)setIsNegotiated:(BOOL)isNegotiated {
116 _dataChannelInit.negotiated = isNegotiated;
117}
118
119- (NSInteger)streamId {
120 return _dataChannelInit.id;
121}
122
123- (void)setStreamId:(NSInteger)streamId {
124 _dataChannelInit.id = streamId;
125}
126
127@end
128
129@implementation RTCDataChannelInit (Internal)
130
131- (const webrtc::DataChannelInit*)dataChannelInit {
132 return &_dataChannelInit;
133}
134
135@end
136
137@implementation RTCDataBuffer {
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +0000138 rtc::scoped_ptr<webrtc::DataBuffer> _dataBuffer;
tkchin@webrtc.orgff273322014-04-30 18:32:33 +0000139}
140
141- (instancetype)initWithData:(NSData*)data isBinary:(BOOL)isBinary {
142 NSAssert(data, @"data cannot be nil");
143 if (self = [super init]) {
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +0000144 rtc::Buffer buffer([data bytes], [data length]);
tkchin@webrtc.orgff273322014-04-30 18:32:33 +0000145 _dataBuffer.reset(new webrtc::DataBuffer(buffer, isBinary));
146 }
147 return self;
148}
149
150- (NSData*)data {
151 return [NSData dataWithBytes:_dataBuffer->data.data()
kwiberg@webrtc.orgeebcab52015-03-24 09:19:06 +0000152 length:_dataBuffer->data.size()];
tkchin@webrtc.orgff273322014-04-30 18:32:33 +0000153}
154
155- (BOOL)isBinary {
156 return _dataBuffer->binary;
157}
158
159@end
160
161@implementation RTCDataBuffer (Internal)
162
163- (instancetype)initWithDataBuffer:(const webrtc::DataBuffer&)buffer {
164 if (self = [super init]) {
165 _dataBuffer.reset(new webrtc::DataBuffer(buffer));
166 }
167 return self;
168}
169
170- (const webrtc::DataBuffer*)dataBuffer {
171 return _dataBuffer.get();
172}
173
174@end
175
176@implementation RTCDataChannel {
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +0000177 rtc::scoped_refptr<webrtc::DataChannelInterface> _dataChannel;
178 rtc::scoped_ptr<webrtc::RTCDataChannelObserver> _observer;
tkchin@webrtc.orgff273322014-04-30 18:32:33 +0000179 BOOL _isObserverRegistered;
180}
181
Zeke Chin48253562015-03-31 11:06:26 -0700182- (void)dealloc {
183 // Handles unregistering the observer properly. We need to do this because
184 // there may still be other references to the underlying data channel.
185 self.delegate = nil;
186}
187
tkchin@webrtc.orgff273322014-04-30 18:32:33 +0000188- (NSString*)label {
189 return NSStringFromStdString(_dataChannel->label());
190}
191
192- (BOOL)isReliable {
193 return _dataChannel->reliable();
194}
195
196- (BOOL)isOrdered {
197 return _dataChannel->ordered();
198}
199
200- (NSUInteger)maxRetransmitTimeMs {
201 return _dataChannel->maxRetransmitTime();
202}
203
204- (NSUInteger)maxRetransmits {
205 return _dataChannel->maxRetransmits();
206}
207
208- (NSString*)protocol {
209 return NSStringFromStdString(_dataChannel->protocol());
210}
211
212- (BOOL)isNegotiated {
213 return _dataChannel->negotiated();
214}
215
216- (NSInteger)streamId {
217 return _dataChannel->id();
218}
219
220- (RTCDataChannelState)state {
221 switch (_dataChannel->state()) {
222 case webrtc::DataChannelInterface::DataState::kConnecting:
223 return kRTCDataChannelStateConnecting;
224 case webrtc::DataChannelInterface::DataState::kOpen:
225 return kRTCDataChannelStateOpen;
226 case webrtc::DataChannelInterface::DataState::kClosing:
227 return kRTCDataChannelStateClosing;
228 case webrtc::DataChannelInterface::DataState::kClosed:
229 return kRTCDataChannelStateClosed;
230 }
231}
232
233- (NSUInteger)bufferedAmount {
234 return _dataChannel->buffered_amount();
235}
236
237- (void)setDelegate:(id<RTCDataChannelDelegate>)delegate {
238 if (_delegate == delegate) {
239 return;
240 }
241 if (_isObserverRegistered) {
242 _dataChannel->UnregisterObserver();
243 _isObserverRegistered = NO;
244 }
245 _delegate = delegate;
246 if (_delegate) {
247 _dataChannel->RegisterObserver(_observer.get());
248 _isObserverRegistered = YES;
249 }
250}
251
252- (void)close {
253 _dataChannel->Close();
254}
255
256- (BOOL)sendData:(RTCDataBuffer*)data {
257 return _dataChannel->Send(*data.dataBuffer);
258}
259
260@end
261
262@implementation RTCDataChannel (Internal)
263
264- (instancetype)initWithDataChannel:
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +0000265 (rtc::scoped_refptr<webrtc::DataChannelInterface>)
tkchin@webrtc.orgff273322014-04-30 18:32:33 +0000266 dataChannel {
267 NSAssert(dataChannel != NULL, @"dataChannel cannot be NULL");
268 if (self = [super init]) {
269 _dataChannel = dataChannel;
270 _observer.reset(new webrtc::RTCDataChannelObserver(self));
271 }
272 return self;
273}
274
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +0000275- (rtc::scoped_refptr<webrtc::DataChannelInterface>)dataChannel {
tkchin@webrtc.orgff273322014-04-30 18:32:33 +0000276 return _dataChannel;
277}
278
279@end