• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /*
2  * libjingle
3  * Copyright 2013 Google Inc.
4  *
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 // Hints for future visitors:
29 // This entire file is an implementation detail of the org.webrtc Java package,
30 // the most interesting bits of which are org.webrtc.PeerConnection{,Factory}.
31 // The layout of this file is roughly:
32 // - various helper C++ functions & classes that wrap Java counterparts and
33 //   expose a C++ interface that can be passed to the C++ PeerConnection APIs
34 // - implementations of methods declared "static" in the Java package (named
35 //   things like Java_org_webrtc_OMG_Can_This_Name_Be_Any_Longer, prescribed by
36 //   the JNI spec).
37 //
38 // Lifecycle notes: objects are owned where they will be called; in other words
39 // FooObservers are owned by C++-land, and user-callable objects (e.g.
40 // PeerConnection and VideoTrack) are owned by Java-land.
41 // When this file allocates C++ RefCountInterfaces it AddRef()s an artificial
42 // ref simulating the jlong held in Java-land, and then Release()s the ref in
43 // the respective free call.  Sometimes this AddRef is implicit in the
44 // construction of a scoped_refptr<> which is then .release()d.
45 // Any persistent (non-local) references from C++ to Java must be global or weak
46 // (in which case they must be checked before use)!
47 //
48 // Exception notes: pretty much all JNI calls can throw Java exceptions, so each
49 // call through a JNIEnv* pointer needs to be followed by an ExceptionCheck()
50 // call.  In this file this is done in CHECK_EXCEPTION, making for much easier
51 // debugging in case of failure (the alternative is to wait for control to
52 // return to the Java frame that called code in this file, at which point it's
53 // impossible to tell which JNI call broke).
54 
55 #include <jni.h>
56 #undef JNIEXPORT
57 #define JNIEXPORT __attribute__((visibility("default")))
58 
59 #include <limits>
60 #include <utility>
61 
62 #include "talk/app/webrtc/java/jni/classreferenceholder.h"
63 #include "talk/app/webrtc/java/jni/jni_helpers.h"
64 #include "talk/app/webrtc/java/jni/native_handle_impl.h"
65 #include "talk/app/webrtc/dtlsidentitystore.h"
66 #include "talk/app/webrtc/mediaconstraintsinterface.h"
67 #include "talk/app/webrtc/peerconnectioninterface.h"
68 #include "talk/app/webrtc/rtpreceiverinterface.h"
69 #include "talk/app/webrtc/rtpsenderinterface.h"
70 #include "talk/app/webrtc/videosourceinterface.h"
71 #include "talk/media/base/videocapturer.h"
72 #include "talk/media/base/videorenderer.h"
73 #include "talk/media/devices/videorendererfactory.h"
74 #include "talk/media/webrtc/webrtcvideodecoderfactory.h"
75 #include "talk/media/webrtc/webrtcvideoencoderfactory.h"
76 #include "webrtc/base/bind.h"
77 #include "webrtc/base/checks.h"
78 #include "webrtc/base/event_tracer.h"
79 #include "webrtc/base/logging.h"
80 #include "webrtc/base/logsinks.h"
81 #include "webrtc/base/messagequeue.h"
82 #include "webrtc/base/networkmonitor.h"
83 #include "webrtc/base/ssladapter.h"
84 #include "webrtc/base/stringutils.h"
85 #include "webrtc/system_wrappers/include/field_trial_default.h"
86 #include "webrtc/system_wrappers/include/trace.h"
87 #include "webrtc/voice_engine/include/voe_base.h"
88 
89 #if defined(ANDROID) && !defined(WEBRTC_CHROMIUM_BUILD)
90 #include "talk/app/webrtc/androidvideocapturer.h"
91 #include "talk/app/webrtc/java/jni/androidmediadecoder_jni.h"
92 #include "talk/app/webrtc/java/jni/androidmediaencoder_jni.h"
93 #include "talk/app/webrtc/java/jni/androidvideocapturer_jni.h"
94 #include "talk/app/webrtc/java/jni/androidnetworkmonitor_jni.h"
95 #include "webrtc/modules/video_render/video_render_internal.h"
96 #include "webrtc/system_wrappers/include/logcat_trace_context.h"
97 using webrtc::LogcatTraceContext;
98 #endif
99 
100 using cricket::WebRtcVideoDecoderFactory;
101 using cricket::WebRtcVideoEncoderFactory;
102 using rtc::Bind;
103 using rtc::Thread;
104 using rtc::ThreadManager;
105 using rtc::scoped_ptr;
106 using webrtc::AudioSourceInterface;
107 using webrtc::AudioTrackInterface;
108 using webrtc::AudioTrackVector;
109 using webrtc::CreateSessionDescriptionObserver;
110 using webrtc::DataBuffer;
111 using webrtc::DataChannelInit;
112 using webrtc::DataChannelInterface;
113 using webrtc::DataChannelObserver;
114 using webrtc::IceCandidateInterface;
115 using webrtc::MediaConstraintsInterface;
116 using webrtc::MediaSourceInterface;
117 using webrtc::MediaStreamInterface;
118 using webrtc::MediaStreamTrackInterface;
119 using webrtc::PeerConnectionFactoryInterface;
120 using webrtc::PeerConnectionInterface;
121 using webrtc::PeerConnectionObserver;
122 using webrtc::RtpReceiverInterface;
123 using webrtc::RtpSenderInterface;
124 using webrtc::SessionDescriptionInterface;
125 using webrtc::SetSessionDescriptionObserver;
126 using webrtc::StatsObserver;
127 using webrtc::StatsReport;
128 using webrtc::StatsReports;
129 using webrtc::VideoRendererInterface;
130 using webrtc::VideoSourceInterface;
131 using webrtc::VideoTrackInterface;
132 using webrtc::VideoTrackVector;
133 using webrtc::kVideoCodecVP8;
134 
135 namespace webrtc_jni {
136 
137 // Field trials initialization string
138 static char *field_trials_init_string = NULL;
139 
140 #if defined(ANDROID) && !defined(WEBRTC_CHROMIUM_BUILD)
141 // Set in PeerConnectionFactory_initializeAndroidGlobals().
142 static bool factory_static_initialized = false;
143 static bool video_hw_acceleration_enabled = true;
144 #endif
145 
146 // Return the (singleton) Java Enum object corresponding to |index|;
147 // |state_class_fragment| is something like "MediaSource$State".
JavaEnumFromIndex(JNIEnv * jni,const std::string & state_class_fragment,int index)148 static jobject JavaEnumFromIndex(
149     JNIEnv* jni, const std::string& state_class_fragment, int index) {
150   const std::string state_class = "org/webrtc/" + state_class_fragment;
151   return JavaEnumFromIndex(jni, FindClass(jni, state_class.c_str()),
152                            state_class, index);
153 }
154 
JavaDataChannelInitToNative(JNIEnv * jni,jobject j_init)155 static DataChannelInit JavaDataChannelInitToNative(
156     JNIEnv* jni, jobject j_init) {
157   DataChannelInit init;
158 
159   jclass j_init_class = FindClass(jni, "org/webrtc/DataChannel$Init");
160   jfieldID ordered_id = GetFieldID(jni, j_init_class, "ordered", "Z");
161   jfieldID max_retransmit_time_id =
162       GetFieldID(jni, j_init_class, "maxRetransmitTimeMs", "I");
163   jfieldID max_retransmits_id =
164       GetFieldID(jni, j_init_class, "maxRetransmits", "I");
165   jfieldID protocol_id =
166       GetFieldID(jni, j_init_class, "protocol", "Ljava/lang/String;");
167   jfieldID negotiated_id = GetFieldID(jni, j_init_class, "negotiated", "Z");
168   jfieldID id_id = GetFieldID(jni, j_init_class, "id", "I");
169 
170   init.ordered = GetBooleanField(jni, j_init, ordered_id);
171   init.maxRetransmitTime = GetIntField(jni, j_init, max_retransmit_time_id);
172   init.maxRetransmits = GetIntField(jni, j_init, max_retransmits_id);
173   init.protocol = JavaToStdString(
174       jni, GetStringField(jni, j_init, protocol_id));
175   init.negotiated = GetBooleanField(jni, j_init, negotiated_id);
176   init.id = GetIntField(jni, j_init, id_id);
177 
178   return init;
179 }
180 
181 class ConstraintsWrapper;
182 
183 // Adapter between the C++ PeerConnectionObserver interface and the Java
184 // PeerConnection.Observer interface.  Wraps an instance of the Java interface
185 // and dispatches C++ callbacks to Java.
186 class PCOJava : public PeerConnectionObserver {
187  public:
PCOJava(JNIEnv * jni,jobject j_observer)188   PCOJava(JNIEnv* jni, jobject j_observer)
189       : j_observer_global_(jni, j_observer),
190         j_observer_class_(jni, GetObjectClass(jni, *j_observer_global_)),
191         j_media_stream_class_(jni, FindClass(jni, "org/webrtc/MediaStream")),
192         j_media_stream_ctor_(GetMethodID(
193             jni, *j_media_stream_class_, "<init>", "(J)V")),
194         j_audio_track_class_(jni, FindClass(jni, "org/webrtc/AudioTrack")),
195         j_audio_track_ctor_(GetMethodID(
196             jni, *j_audio_track_class_, "<init>", "(J)V")),
197         j_video_track_class_(jni, FindClass(jni, "org/webrtc/VideoTrack")),
198         j_video_track_ctor_(GetMethodID(
199             jni, *j_video_track_class_, "<init>", "(J)V")),
200         j_data_channel_class_(jni, FindClass(jni, "org/webrtc/DataChannel")),
201         j_data_channel_ctor_(GetMethodID(
202             jni, *j_data_channel_class_, "<init>", "(J)V")) {
203   }
204 
~PCOJava()205   virtual ~PCOJava() {
206     ScopedLocalRefFrame local_ref_frame(jni());
207     while (!remote_streams_.empty())
208       DisposeRemoteStream(remote_streams_.begin());
209   }
210 
OnIceCandidate(const IceCandidateInterface * candidate)211   void OnIceCandidate(const IceCandidateInterface* candidate) override {
212     ScopedLocalRefFrame local_ref_frame(jni());
213     std::string sdp;
214     RTC_CHECK(candidate->ToString(&sdp)) << "got so far: " << sdp;
215     jclass candidate_class = FindClass(jni(), "org/webrtc/IceCandidate");
216     jmethodID ctor = GetMethodID(jni(), candidate_class,
217         "<init>", "(Ljava/lang/String;ILjava/lang/String;)V");
218     jstring j_mid = JavaStringFromStdString(jni(), candidate->sdp_mid());
219     jstring j_sdp = JavaStringFromStdString(jni(), sdp);
220     jobject j_candidate = jni()->NewObject(
221         candidate_class, ctor, j_mid, candidate->sdp_mline_index(), j_sdp);
222     CHECK_EXCEPTION(jni()) << "error during NewObject";
223     jmethodID m = GetMethodID(jni(), *j_observer_class_,
224                               "onIceCandidate", "(Lorg/webrtc/IceCandidate;)V");
225     jni()->CallVoidMethod(*j_observer_global_, m, j_candidate);
226     CHECK_EXCEPTION(jni()) << "error during CallVoidMethod";
227   }
228 
OnSignalingChange(PeerConnectionInterface::SignalingState new_state)229   void OnSignalingChange(
230       PeerConnectionInterface::SignalingState new_state) override {
231     ScopedLocalRefFrame local_ref_frame(jni());
232     jmethodID m = GetMethodID(
233         jni(), *j_observer_class_, "onSignalingChange",
234         "(Lorg/webrtc/PeerConnection$SignalingState;)V");
235     jobject new_state_enum =
236         JavaEnumFromIndex(jni(), "PeerConnection$SignalingState", new_state);
237     jni()->CallVoidMethod(*j_observer_global_, m, new_state_enum);
238     CHECK_EXCEPTION(jni()) << "error during CallVoidMethod";
239   }
240 
OnIceConnectionChange(PeerConnectionInterface::IceConnectionState new_state)241   void OnIceConnectionChange(
242       PeerConnectionInterface::IceConnectionState new_state) override {
243     ScopedLocalRefFrame local_ref_frame(jni());
244     jmethodID m = GetMethodID(
245         jni(), *j_observer_class_, "onIceConnectionChange",
246         "(Lorg/webrtc/PeerConnection$IceConnectionState;)V");
247     jobject new_state_enum = JavaEnumFromIndex(
248         jni(), "PeerConnection$IceConnectionState", new_state);
249     jni()->CallVoidMethod(*j_observer_global_, m, new_state_enum);
250     CHECK_EXCEPTION(jni()) << "error during CallVoidMethod";
251   }
252 
OnIceConnectionReceivingChange(bool receiving)253   void OnIceConnectionReceivingChange(bool receiving) override {
254     ScopedLocalRefFrame local_ref_frame(jni());
255     jmethodID m = GetMethodID(
256         jni(), *j_observer_class_, "onIceConnectionReceivingChange", "(Z)V");
257     jni()->CallVoidMethod(*j_observer_global_, m, receiving);
258     CHECK_EXCEPTION(jni()) << "error during CallVoidMethod";
259   }
260 
OnIceGatheringChange(PeerConnectionInterface::IceGatheringState new_state)261   void OnIceGatheringChange(
262       PeerConnectionInterface::IceGatheringState new_state) override {
263     ScopedLocalRefFrame local_ref_frame(jni());
264     jmethodID m = GetMethodID(
265         jni(), *j_observer_class_, "onIceGatheringChange",
266         "(Lorg/webrtc/PeerConnection$IceGatheringState;)V");
267     jobject new_state_enum = JavaEnumFromIndex(
268         jni(), "PeerConnection$IceGatheringState", new_state);
269     jni()->CallVoidMethod(*j_observer_global_, m, new_state_enum);
270     CHECK_EXCEPTION(jni()) << "error during CallVoidMethod";
271   }
272 
OnAddStream(MediaStreamInterface * stream)273   void OnAddStream(MediaStreamInterface* stream) override {
274     ScopedLocalRefFrame local_ref_frame(jni());
275     // Java MediaStream holds one reference. Corresponding Release() is in
276     // MediaStream_free, triggered by MediaStream.dispose().
277     stream->AddRef();
278     jobject j_stream =
279         jni()->NewObject(*j_media_stream_class_, j_media_stream_ctor_,
280                          reinterpret_cast<jlong>(stream));
281     CHECK_EXCEPTION(jni()) << "error during NewObject";
282 
283     for (const auto& track : stream->GetAudioTracks()) {
284       jstring id = JavaStringFromStdString(jni(), track->id());
285       // Java AudioTrack holds one reference. Corresponding Release() is in
286       // MediaStreamTrack_free, triggered by AudioTrack.dispose().
287       track->AddRef();
288       jobject j_track =
289           jni()->NewObject(*j_audio_track_class_, j_audio_track_ctor_,
290                            reinterpret_cast<jlong>(track.get()), id);
291       CHECK_EXCEPTION(jni()) << "error during NewObject";
292       jfieldID audio_tracks_id = GetFieldID(jni(),
293                                             *j_media_stream_class_,
294                                             "audioTracks",
295                                             "Ljava/util/LinkedList;");
296       jobject audio_tracks = GetObjectField(jni(), j_stream, audio_tracks_id);
297       jmethodID add = GetMethodID(jni(),
298                                   GetObjectClass(jni(), audio_tracks),
299                                   "add",
300                                   "(Ljava/lang/Object;)Z");
301       jboolean added = jni()->CallBooleanMethod(audio_tracks, add, j_track);
302       CHECK_EXCEPTION(jni()) << "error during CallBooleanMethod";
303       RTC_CHECK(added);
304     }
305 
306     for (const auto& track : stream->GetVideoTracks()) {
307       jstring id = JavaStringFromStdString(jni(), track->id());
308       // Java VideoTrack holds one reference. Corresponding Release() is in
309       // MediaStreamTrack_free, triggered by VideoTrack.dispose().
310       track->AddRef();
311       jobject j_track =
312           jni()->NewObject(*j_video_track_class_, j_video_track_ctor_,
313                            reinterpret_cast<jlong>(track.get()), id);
314       CHECK_EXCEPTION(jni()) << "error during NewObject";
315       jfieldID video_tracks_id = GetFieldID(jni(),
316                                             *j_media_stream_class_,
317                                             "videoTracks",
318                                             "Ljava/util/LinkedList;");
319       jobject video_tracks = GetObjectField(jni(), j_stream, video_tracks_id);
320       jmethodID add = GetMethodID(jni(),
321                                   GetObjectClass(jni(), video_tracks),
322                                   "add",
323                                   "(Ljava/lang/Object;)Z");
324       jboolean added = jni()->CallBooleanMethod(video_tracks, add, j_track);
325       CHECK_EXCEPTION(jni()) << "error during CallBooleanMethod";
326       RTC_CHECK(added);
327     }
328     remote_streams_[stream] = NewGlobalRef(jni(), j_stream);
329 
330     jmethodID m = GetMethodID(jni(), *j_observer_class_, "onAddStream",
331                               "(Lorg/webrtc/MediaStream;)V");
332     jni()->CallVoidMethod(*j_observer_global_, m, j_stream);
333     CHECK_EXCEPTION(jni()) << "error during CallVoidMethod";
334   }
335 
OnRemoveStream(MediaStreamInterface * stream)336   void OnRemoveStream(MediaStreamInterface* stream) override {
337     ScopedLocalRefFrame local_ref_frame(jni());
338     NativeToJavaStreamsMap::iterator it = remote_streams_.find(stream);
339     RTC_CHECK(it != remote_streams_.end()) << "unexpected stream: " << std::hex
340                                            << stream;
341     jobject j_stream = it->second;
342     jmethodID m = GetMethodID(jni(), *j_observer_class_, "onRemoveStream",
343                               "(Lorg/webrtc/MediaStream;)V");
344     jni()->CallVoidMethod(*j_observer_global_, m, j_stream);
345     CHECK_EXCEPTION(jni()) << "error during CallVoidMethod";
346     DisposeRemoteStream(it);
347   }
348 
OnDataChannel(DataChannelInterface * channel)349   void OnDataChannel(DataChannelInterface* channel) override {
350     ScopedLocalRefFrame local_ref_frame(jni());
351     jobject j_channel = jni()->NewObject(
352         *j_data_channel_class_, j_data_channel_ctor_, (jlong)channel);
353     CHECK_EXCEPTION(jni()) << "error during NewObject";
354 
355     jmethodID m = GetMethodID(jni(), *j_observer_class_, "onDataChannel",
356                               "(Lorg/webrtc/DataChannel;)V");
357     jni()->CallVoidMethod(*j_observer_global_, m, j_channel);
358 
359     // Channel is now owned by Java object, and will be freed from
360     // DataChannel.dispose().  Important that this be done _after_ the
361     // CallVoidMethod above as Java code might call back into native code and be
362     // surprised to see a refcount of 2.
363     int bumped_count = channel->AddRef();
364     RTC_CHECK(bumped_count == 2) << "Unexpected refcount OnDataChannel";
365 
366     CHECK_EXCEPTION(jni()) << "error during CallVoidMethod";
367   }
368 
OnRenegotiationNeeded()369   void OnRenegotiationNeeded() override {
370     ScopedLocalRefFrame local_ref_frame(jni());
371     jmethodID m =
372         GetMethodID(jni(), *j_observer_class_, "onRenegotiationNeeded", "()V");
373     jni()->CallVoidMethod(*j_observer_global_, m);
374     CHECK_EXCEPTION(jni()) << "error during CallVoidMethod";
375   }
376 
SetConstraints(ConstraintsWrapper * constraints)377   void SetConstraints(ConstraintsWrapper* constraints) {
378     RTC_CHECK(!constraints_.get()) << "constraints already set!";
379     constraints_.reset(constraints);
380   }
381 
constraints()382   const ConstraintsWrapper* constraints() { return constraints_.get(); }
383 
384  private:
385   typedef std::map<MediaStreamInterface*, jobject> NativeToJavaStreamsMap;
386 
DisposeRemoteStream(const NativeToJavaStreamsMap::iterator & it)387   void DisposeRemoteStream(const NativeToJavaStreamsMap::iterator& it) {
388     jobject j_stream = it->second;
389     remote_streams_.erase(it);
390     jni()->CallVoidMethod(
391         j_stream, GetMethodID(jni(), *j_media_stream_class_, "dispose", "()V"));
392     CHECK_EXCEPTION(jni()) << "error during MediaStream.dispose()";
393     DeleteGlobalRef(jni(), j_stream);
394   }
395 
jni()396   JNIEnv* jni() {
397     return AttachCurrentThreadIfNeeded();
398   }
399 
400   const ScopedGlobalRef<jobject> j_observer_global_;
401   const ScopedGlobalRef<jclass> j_observer_class_;
402   const ScopedGlobalRef<jclass> j_media_stream_class_;
403   const jmethodID j_media_stream_ctor_;
404   const ScopedGlobalRef<jclass> j_audio_track_class_;
405   const jmethodID j_audio_track_ctor_;
406   const ScopedGlobalRef<jclass> j_video_track_class_;
407   const jmethodID j_video_track_ctor_;
408   const ScopedGlobalRef<jclass> j_data_channel_class_;
409   const jmethodID j_data_channel_ctor_;
410   // C++ -> Java remote streams. The stored jobects are global refs and must be
411   // manually deleted upon removal. Use DisposeRemoteStream().
412   NativeToJavaStreamsMap remote_streams_;
413   scoped_ptr<ConstraintsWrapper> constraints_;
414 };
415 
416 // Wrapper for a Java MediaConstraints object.  Copies all needed data so when
417 // the constructor returns the Java object is no longer needed.
418 class ConstraintsWrapper : public MediaConstraintsInterface {
419  public:
ConstraintsWrapper(JNIEnv * jni,jobject j_constraints)420   ConstraintsWrapper(JNIEnv* jni, jobject j_constraints) {
421     PopulateConstraintsFromJavaPairList(
422         jni, j_constraints, "mandatory", &mandatory_);
423     PopulateConstraintsFromJavaPairList(
424         jni, j_constraints, "optional", &optional_);
425   }
426 
~ConstraintsWrapper()427   virtual ~ConstraintsWrapper() {}
428 
429   // MediaConstraintsInterface.
GetMandatory() const430   const Constraints& GetMandatory() const override { return mandatory_; }
431 
GetOptional() const432   const Constraints& GetOptional() const override { return optional_; }
433 
434  private:
435   // Helper for translating a List<Pair<String, String>> to a Constraints.
PopulateConstraintsFromJavaPairList(JNIEnv * jni,jobject j_constraints,const char * field_name,Constraints * field)436   static void PopulateConstraintsFromJavaPairList(
437       JNIEnv* jni, jobject j_constraints,
438       const char* field_name, Constraints* field) {
439     jfieldID j_id = GetFieldID(jni,
440         GetObjectClass(jni, j_constraints), field_name, "Ljava/util/List;");
441     jobject j_list = GetObjectField(jni, j_constraints, j_id);
442     jmethodID j_iterator_id = GetMethodID(jni,
443         GetObjectClass(jni, j_list), "iterator", "()Ljava/util/Iterator;");
444     jobject j_iterator = jni->CallObjectMethod(j_list, j_iterator_id);
445     CHECK_EXCEPTION(jni) << "error during CallObjectMethod";
446     jmethodID j_has_next = GetMethodID(jni,
447         GetObjectClass(jni, j_iterator), "hasNext", "()Z");
448     jmethodID j_next = GetMethodID(jni,
449         GetObjectClass(jni, j_iterator), "next", "()Ljava/lang/Object;");
450     while (jni->CallBooleanMethod(j_iterator, j_has_next)) {
451       CHECK_EXCEPTION(jni) << "error during CallBooleanMethod";
452       jobject entry = jni->CallObjectMethod(j_iterator, j_next);
453       CHECK_EXCEPTION(jni) << "error during CallObjectMethod";
454       jmethodID get_key = GetMethodID(jni,
455           GetObjectClass(jni, entry), "getKey", "()Ljava/lang/String;");
456       jstring j_key = reinterpret_cast<jstring>(
457           jni->CallObjectMethod(entry, get_key));
458       CHECK_EXCEPTION(jni) << "error during CallObjectMethod";
459       jmethodID get_value = GetMethodID(jni,
460           GetObjectClass(jni, entry), "getValue", "()Ljava/lang/String;");
461       jstring j_value = reinterpret_cast<jstring>(
462           jni->CallObjectMethod(entry, get_value));
463       CHECK_EXCEPTION(jni) << "error during CallObjectMethod";
464       field->push_back(Constraint(JavaToStdString(jni, j_key),
465                                   JavaToStdString(jni, j_value)));
466     }
467     CHECK_EXCEPTION(jni) << "error during CallBooleanMethod";
468   }
469 
470   Constraints mandatory_;
471   Constraints optional_;
472 };
473 
JavaSdpFromNativeSdp(JNIEnv * jni,const SessionDescriptionInterface * desc)474 static jobject JavaSdpFromNativeSdp(
475     JNIEnv* jni, const SessionDescriptionInterface* desc) {
476   std::string sdp;
477   RTC_CHECK(desc->ToString(&sdp)) << "got so far: " << sdp;
478   jstring j_description = JavaStringFromStdString(jni, sdp);
479 
480   jclass j_type_class = FindClass(
481       jni, "org/webrtc/SessionDescription$Type");
482   jmethodID j_type_from_canonical = GetStaticMethodID(
483       jni, j_type_class, "fromCanonicalForm",
484       "(Ljava/lang/String;)Lorg/webrtc/SessionDescription$Type;");
485   jstring j_type_string = JavaStringFromStdString(jni, desc->type());
486   jobject j_type = jni->CallStaticObjectMethod(
487       j_type_class, j_type_from_canonical, j_type_string);
488   CHECK_EXCEPTION(jni) << "error during CallObjectMethod";
489 
490   jclass j_sdp_class = FindClass(jni, "org/webrtc/SessionDescription");
491   jmethodID j_sdp_ctor = GetMethodID(
492       jni, j_sdp_class, "<init>",
493       "(Lorg/webrtc/SessionDescription$Type;Ljava/lang/String;)V");
494   jobject j_sdp = jni->NewObject(
495       j_sdp_class, j_sdp_ctor, j_type, j_description);
496   CHECK_EXCEPTION(jni) << "error during NewObject";
497   return j_sdp;
498 }
499 
500 template <class T>  // T is one of {Create,Set}SessionDescriptionObserver.
501 class SdpObserverWrapper : public T {
502  public:
SdpObserverWrapper(JNIEnv * jni,jobject j_observer,ConstraintsWrapper * constraints)503   SdpObserverWrapper(JNIEnv* jni, jobject j_observer,
504                      ConstraintsWrapper* constraints)
505       : constraints_(constraints),
506         j_observer_global_(jni, j_observer),
507         j_observer_class_(jni, GetObjectClass(jni, j_observer)) {
508   }
509 
~SdpObserverWrapper()510   virtual ~SdpObserverWrapper() {}
511 
512   // Can't mark override because of templating.
OnSuccess()513   virtual void OnSuccess() {
514     ScopedLocalRefFrame local_ref_frame(jni());
515     jmethodID m = GetMethodID(jni(), *j_observer_class_, "onSetSuccess", "()V");
516     jni()->CallVoidMethod(*j_observer_global_, m);
517     CHECK_EXCEPTION(jni()) << "error during CallVoidMethod";
518   }
519 
520   // Can't mark override because of templating.
OnSuccess(SessionDescriptionInterface * desc)521   virtual void OnSuccess(SessionDescriptionInterface* desc) {
522     ScopedLocalRefFrame local_ref_frame(jni());
523     jmethodID m = GetMethodID(
524         jni(), *j_observer_class_, "onCreateSuccess",
525         "(Lorg/webrtc/SessionDescription;)V");
526     jobject j_sdp = JavaSdpFromNativeSdp(jni(), desc);
527     jni()->CallVoidMethod(*j_observer_global_, m, j_sdp);
528     CHECK_EXCEPTION(jni()) << "error during CallVoidMethod";
529   }
530 
531  protected:
532   // Common implementation for failure of Set & Create types, distinguished by
533   // |op| being "Set" or "Create".
DoOnFailure(const std::string & op,const std::string & error)534   void DoOnFailure(const std::string& op, const std::string& error) {
535     jmethodID m = GetMethodID(jni(), *j_observer_class_, "on" + op + "Failure",
536                               "(Ljava/lang/String;)V");
537     jstring j_error_string = JavaStringFromStdString(jni(), error);
538     jni()->CallVoidMethod(*j_observer_global_, m, j_error_string);
539     CHECK_EXCEPTION(jni()) << "error during CallVoidMethod";
540   }
541 
jni()542   JNIEnv* jni() {
543     return AttachCurrentThreadIfNeeded();
544   }
545 
546  private:
547   scoped_ptr<ConstraintsWrapper> constraints_;
548   const ScopedGlobalRef<jobject> j_observer_global_;
549   const ScopedGlobalRef<jclass> j_observer_class_;
550 };
551 
552 class CreateSdpObserverWrapper
553     : public SdpObserverWrapper<CreateSessionDescriptionObserver> {
554  public:
CreateSdpObserverWrapper(JNIEnv * jni,jobject j_observer,ConstraintsWrapper * constraints)555   CreateSdpObserverWrapper(JNIEnv* jni, jobject j_observer,
556                            ConstraintsWrapper* constraints)
557       : SdpObserverWrapper(jni, j_observer, constraints) {}
558 
OnFailure(const std::string & error)559   void OnFailure(const std::string& error) override {
560     ScopedLocalRefFrame local_ref_frame(jni());
561     SdpObserverWrapper::DoOnFailure(std::string("Create"), error);
562   }
563 };
564 
565 class SetSdpObserverWrapper
566     : public SdpObserverWrapper<SetSessionDescriptionObserver> {
567  public:
SetSdpObserverWrapper(JNIEnv * jni,jobject j_observer,ConstraintsWrapper * constraints)568   SetSdpObserverWrapper(JNIEnv* jni, jobject j_observer,
569                         ConstraintsWrapper* constraints)
570       : SdpObserverWrapper(jni, j_observer, constraints) {}
571 
OnFailure(const std::string & error)572   void OnFailure(const std::string& error) override {
573     ScopedLocalRefFrame local_ref_frame(jni());
574     SdpObserverWrapper::DoOnFailure(std::string("Set"), error);
575   }
576 };
577 
578 // Adapter for a Java DataChannel$Observer presenting a C++ DataChannelObserver
579 // and dispatching the callback from C++ back to Java.
580 class DataChannelObserverWrapper : public DataChannelObserver {
581  public:
DataChannelObserverWrapper(JNIEnv * jni,jobject j_observer)582   DataChannelObserverWrapper(JNIEnv* jni, jobject j_observer)
583       : j_observer_global_(jni, j_observer),
584         j_observer_class_(jni, GetObjectClass(jni, j_observer)),
585         j_buffer_class_(jni, FindClass(jni, "org/webrtc/DataChannel$Buffer")),
586         j_on_buffered_amount_change_mid_(GetMethodID(
587             jni, *j_observer_class_, "onBufferedAmountChange", "(J)V")),
588         j_on_state_change_mid_(
589             GetMethodID(jni, *j_observer_class_, "onStateChange", "()V")),
590         j_on_message_mid_(GetMethodID(jni, *j_observer_class_, "onMessage",
591                                       "(Lorg/webrtc/DataChannel$Buffer;)V")),
592         j_buffer_ctor_(GetMethodID(jni, *j_buffer_class_, "<init>",
593                                    "(Ljava/nio/ByteBuffer;Z)V")) {}
594 
~DataChannelObserverWrapper()595   virtual ~DataChannelObserverWrapper() {}
596 
OnBufferedAmountChange(uint64_t previous_amount)597   void OnBufferedAmountChange(uint64_t previous_amount) override {
598     ScopedLocalRefFrame local_ref_frame(jni());
599     jni()->CallVoidMethod(*j_observer_global_, j_on_buffered_amount_change_mid_,
600                           previous_amount);
601     CHECK_EXCEPTION(jni()) << "error during CallVoidMethod";
602   }
603 
OnStateChange()604   void OnStateChange() override {
605     ScopedLocalRefFrame local_ref_frame(jni());
606     jni()->CallVoidMethod(*j_observer_global_, j_on_state_change_mid_);
607     CHECK_EXCEPTION(jni()) << "error during CallVoidMethod";
608   }
609 
OnMessage(const DataBuffer & buffer)610   void OnMessage(const DataBuffer& buffer) override {
611     ScopedLocalRefFrame local_ref_frame(jni());
612     jobject byte_buffer = jni()->NewDirectByteBuffer(
613         const_cast<char*>(buffer.data.data<char>()), buffer.data.size());
614     jobject j_buffer = jni()->NewObject(*j_buffer_class_, j_buffer_ctor_,
615                                         byte_buffer, buffer.binary);
616     jni()->CallVoidMethod(*j_observer_global_, j_on_message_mid_, j_buffer);
617     CHECK_EXCEPTION(jni()) << "error during CallVoidMethod";
618   }
619 
620  private:
jni()621   JNIEnv* jni() {
622     return AttachCurrentThreadIfNeeded();
623   }
624 
625   const ScopedGlobalRef<jobject> j_observer_global_;
626   const ScopedGlobalRef<jclass> j_observer_class_;
627   const ScopedGlobalRef<jclass> j_buffer_class_;
628   const jmethodID j_on_buffered_amount_change_mid_;
629   const jmethodID j_on_state_change_mid_;
630   const jmethodID j_on_message_mid_;
631   const jmethodID j_buffer_ctor_;
632 };
633 
634 // Adapter for a Java StatsObserver presenting a C++ StatsObserver and
635 // dispatching the callback from C++ back to Java.
636 class StatsObserverWrapper : public StatsObserver {
637  public:
StatsObserverWrapper(JNIEnv * jni,jobject j_observer)638   StatsObserverWrapper(JNIEnv* jni, jobject j_observer)
639       : j_observer_global_(jni, j_observer),
640         j_observer_class_(jni, GetObjectClass(jni, j_observer)),
641         j_stats_report_class_(jni, FindClass(jni, "org/webrtc/StatsReport")),
642         j_stats_report_ctor_(GetMethodID(
643             jni, *j_stats_report_class_, "<init>",
644             "(Ljava/lang/String;Ljava/lang/String;D"
645             "[Lorg/webrtc/StatsReport$Value;)V")),
646         j_value_class_(jni, FindClass(
647             jni, "org/webrtc/StatsReport$Value")),
648         j_value_ctor_(GetMethodID(
649             jni, *j_value_class_, "<init>",
650             "(Ljava/lang/String;Ljava/lang/String;)V")) {
651   }
652 
~StatsObserverWrapper()653   virtual ~StatsObserverWrapper() {}
654 
OnComplete(const StatsReports & reports)655   void OnComplete(const StatsReports& reports) override {
656     ScopedLocalRefFrame local_ref_frame(jni());
657     jobjectArray j_reports = ReportsToJava(jni(), reports);
658     jmethodID m = GetMethodID(jni(), *j_observer_class_, "onComplete",
659                               "([Lorg/webrtc/StatsReport;)V");
660     jni()->CallVoidMethod(*j_observer_global_, m, j_reports);
661     CHECK_EXCEPTION(jni()) << "error during CallVoidMethod";
662   }
663 
664  private:
ReportsToJava(JNIEnv * jni,const StatsReports & reports)665   jobjectArray ReportsToJava(
666       JNIEnv* jni, const StatsReports& reports) {
667     jobjectArray reports_array = jni->NewObjectArray(
668         reports.size(), *j_stats_report_class_, NULL);
669     int i = 0;
670     for (const auto* report : reports) {
671       ScopedLocalRefFrame local_ref_frame(jni);
672       jstring j_id = JavaStringFromStdString(jni, report->id()->ToString());
673       jstring j_type = JavaStringFromStdString(jni, report->TypeToString());
674       jobjectArray j_values = ValuesToJava(jni, report->values());
675       jobject j_report = jni->NewObject(*j_stats_report_class_,
676                                         j_stats_report_ctor_,
677                                         j_id,
678                                         j_type,
679                                         report->timestamp(),
680                                         j_values);
681       jni->SetObjectArrayElement(reports_array, i++, j_report);
682     }
683     return reports_array;
684   }
685 
ValuesToJava(JNIEnv * jni,const StatsReport::Values & values)686   jobjectArray ValuesToJava(JNIEnv* jni, const StatsReport::Values& values) {
687     jobjectArray j_values = jni->NewObjectArray(
688         values.size(), *j_value_class_, NULL);
689     int i = 0;
690     for (const auto& it : values) {
691       ScopedLocalRefFrame local_ref_frame(jni);
692       // Should we use the '.name' enum value here instead of converting the
693       // name to a string?
694       jstring j_name = JavaStringFromStdString(jni, it.second->display_name());
695       jstring j_value = JavaStringFromStdString(jni, it.second->ToString());
696       jobject j_element_value =
697           jni->NewObject(*j_value_class_, j_value_ctor_, j_name, j_value);
698       jni->SetObjectArrayElement(j_values, i++, j_element_value);
699     }
700     return j_values;
701   }
702 
jni()703   JNIEnv* jni() {
704     return AttachCurrentThreadIfNeeded();
705   }
706 
707   const ScopedGlobalRef<jobject> j_observer_global_;
708   const ScopedGlobalRef<jclass> j_observer_class_;
709   const ScopedGlobalRef<jclass> j_stats_report_class_;
710   const jmethodID j_stats_report_ctor_;
711   const ScopedGlobalRef<jclass> j_value_class_;
712   const jmethodID j_value_ctor_;
713 };
714 
715 // Adapter presenting a cricket::VideoRenderer as a
716 // webrtc::VideoRendererInterface.
717 class VideoRendererWrapper : public VideoRendererInterface {
718  public:
Create(cricket::VideoRenderer * renderer)719   static VideoRendererWrapper* Create(cricket::VideoRenderer* renderer) {
720     if (renderer)
721       return new VideoRendererWrapper(renderer);
722     return NULL;
723   }
724 
~VideoRendererWrapper()725   virtual ~VideoRendererWrapper() {}
726 
727   // This wraps VideoRenderer which still has SetSize.
RenderFrame(const cricket::VideoFrame * video_frame)728   void RenderFrame(const cricket::VideoFrame* video_frame) override {
729     ScopedLocalRefFrame local_ref_frame(AttachCurrentThreadIfNeeded());
730     const cricket::VideoFrame* frame =
731       video_frame->GetCopyWithRotationApplied();
732     if (width_ != frame->GetWidth() || height_ != frame->GetHeight()) {
733       width_ = frame->GetWidth();
734       height_ = frame->GetHeight();
735       renderer_->SetSize(width_, height_, 0);
736     }
737     renderer_->RenderFrame(frame);
738   }
739 
740  private:
VideoRendererWrapper(cricket::VideoRenderer * renderer)741   explicit VideoRendererWrapper(cricket::VideoRenderer* renderer)
742     : width_(0), height_(0), renderer_(renderer) {}
743   int width_, height_;
744   scoped_ptr<cricket::VideoRenderer> renderer_;
745 };
746 
747 // Wrapper dispatching webrtc::VideoRendererInterface to a Java VideoRenderer
748 // instance.
749 class JavaVideoRendererWrapper : public VideoRendererInterface {
750  public:
JavaVideoRendererWrapper(JNIEnv * jni,jobject j_callbacks)751   JavaVideoRendererWrapper(JNIEnv* jni, jobject j_callbacks)
752       : j_callbacks_(jni, j_callbacks),
753         j_render_frame_id_(GetMethodID(
754             jni, GetObjectClass(jni, j_callbacks), "renderFrame",
755             "(Lorg/webrtc/VideoRenderer$I420Frame;)V")),
756         j_frame_class_(jni,
757                        FindClass(jni, "org/webrtc/VideoRenderer$I420Frame")),
758         j_i420_frame_ctor_id_(GetMethodID(
759             jni, *j_frame_class_, "<init>", "(III[I[Ljava/nio/ByteBuffer;J)V")),
760         j_texture_frame_ctor_id_(GetMethodID(
761             jni, *j_frame_class_, "<init>",
762             "(IIII[FJ)V")),
763         j_byte_buffer_class_(jni, FindClass(jni, "java/nio/ByteBuffer")) {
764     CHECK_EXCEPTION(jni);
765   }
766 
~JavaVideoRendererWrapper()767   virtual ~JavaVideoRendererWrapper() {}
768 
RenderFrame(const cricket::VideoFrame * video_frame)769   void RenderFrame(const cricket::VideoFrame* video_frame) override {
770     ScopedLocalRefFrame local_ref_frame(jni());
771     jobject j_frame = (video_frame->GetNativeHandle() != nullptr)
772                           ? CricketToJavaTextureFrame(video_frame)
773                           : CricketToJavaI420Frame(video_frame);
774     // |j_callbacks_| is responsible for releasing |j_frame| with
775     // VideoRenderer.renderFrameDone().
776     jni()->CallVoidMethod(*j_callbacks_, j_render_frame_id_, j_frame);
777     CHECK_EXCEPTION(jni());
778   }
779 
780  private:
781   // Make a shallow copy of |frame| to be used with Java. The callee has
782   // ownership of the frame, and the frame should be released with
783   // VideoRenderer.releaseNativeFrame().
javaShallowCopy(const cricket::VideoFrame * frame)784   static jlong javaShallowCopy(const cricket::VideoFrame* frame) {
785     return jlongFromPointer(frame->Copy());
786   }
787 
788   // Return a VideoRenderer.I420Frame referring to the data in |frame|.
CricketToJavaI420Frame(const cricket::VideoFrame * frame)789   jobject CricketToJavaI420Frame(const cricket::VideoFrame* frame) {
790     jintArray strides = jni()->NewIntArray(3);
791     jint* strides_array = jni()->GetIntArrayElements(strides, NULL);
792     strides_array[0] = frame->GetYPitch();
793     strides_array[1] = frame->GetUPitch();
794     strides_array[2] = frame->GetVPitch();
795     jni()->ReleaseIntArrayElements(strides, strides_array, 0);
796     jobjectArray planes = jni()->NewObjectArray(3, *j_byte_buffer_class_, NULL);
797     jobject y_buffer =
798         jni()->NewDirectByteBuffer(const_cast<uint8_t*>(frame->GetYPlane()),
799                                    frame->GetYPitch() * frame->GetHeight());
800     jobject u_buffer = jni()->NewDirectByteBuffer(
801         const_cast<uint8_t*>(frame->GetUPlane()), frame->GetChromaSize());
802     jobject v_buffer = jni()->NewDirectByteBuffer(
803         const_cast<uint8_t*>(frame->GetVPlane()), frame->GetChromaSize());
804     jni()->SetObjectArrayElement(planes, 0, y_buffer);
805     jni()->SetObjectArrayElement(planes, 1, u_buffer);
806     jni()->SetObjectArrayElement(planes, 2, v_buffer);
807     return jni()->NewObject(
808         *j_frame_class_, j_i420_frame_ctor_id_,
809         frame->GetWidth(), frame->GetHeight(),
810         static_cast<int>(frame->GetVideoRotation()),
811         strides, planes, javaShallowCopy(frame));
812   }
813 
814   // Return a VideoRenderer.I420Frame referring texture object in |frame|.
CricketToJavaTextureFrame(const cricket::VideoFrame * frame)815   jobject CricketToJavaTextureFrame(const cricket::VideoFrame* frame) {
816     NativeHandleImpl* handle =
817         reinterpret_cast<NativeHandleImpl*>(frame->GetNativeHandle());
818     jfloatArray sampling_matrix = jni()->NewFloatArray(16);
819     jni()->SetFloatArrayRegion(sampling_matrix, 0, 16, handle->sampling_matrix);
820     return jni()->NewObject(
821         *j_frame_class_, j_texture_frame_ctor_id_,
822         frame->GetWidth(), frame->GetHeight(),
823         static_cast<int>(frame->GetVideoRotation()),
824         handle->oes_texture_id, sampling_matrix, javaShallowCopy(frame));
825   }
826 
jni()827   JNIEnv* jni() {
828     return AttachCurrentThreadIfNeeded();
829   }
830 
831   ScopedGlobalRef<jobject> j_callbacks_;
832   jmethodID j_render_frame_id_;
833   ScopedGlobalRef<jclass> j_frame_class_;
834   jmethodID j_i420_frame_ctor_id_;
835   jmethodID j_texture_frame_ctor_id_;
836   ScopedGlobalRef<jclass> j_byte_buffer_class_;
837 };
838 
839 
ExtractNativeDC(JNIEnv * jni,jobject j_dc)840 static DataChannelInterface* ExtractNativeDC(JNIEnv* jni, jobject j_dc) {
841   jfieldID native_dc_id = GetFieldID(jni,
842       GetObjectClass(jni, j_dc), "nativeDataChannel", "J");
843   jlong j_d = GetLongField(jni, j_dc, native_dc_id);
844   return reinterpret_cast<DataChannelInterface*>(j_d);
845 }
846 
JOW(jlong,DataChannel_registerObserverNative)847 JOW(jlong, DataChannel_registerObserverNative)(
848     JNIEnv* jni, jobject j_dc, jobject j_observer) {
849   scoped_ptr<DataChannelObserverWrapper> observer(
850       new DataChannelObserverWrapper(jni, j_observer));
851   ExtractNativeDC(jni, j_dc)->RegisterObserver(observer.get());
852   return jlongFromPointer(observer.release());
853 }
854 
JOW(void,DataChannel_unregisterObserverNative)855 JOW(void, DataChannel_unregisterObserverNative)(
856     JNIEnv* jni, jobject j_dc, jlong native_observer) {
857   ExtractNativeDC(jni, j_dc)->UnregisterObserver();
858   delete reinterpret_cast<DataChannelObserverWrapper*>(native_observer);
859 }
860 
JOW(jstring,DataChannel_label)861 JOW(jstring, DataChannel_label)(JNIEnv* jni, jobject j_dc) {
862   return JavaStringFromStdString(jni, ExtractNativeDC(jni, j_dc)->label());
863 }
864 
JOW(jobject,DataChannel_state)865 JOW(jobject, DataChannel_state)(JNIEnv* jni, jobject j_dc) {
866   return JavaEnumFromIndex(
867       jni, "DataChannel$State", ExtractNativeDC(jni, j_dc)->state());
868 }
869 
JOW(jlong,DataChannel_bufferedAmount)870 JOW(jlong, DataChannel_bufferedAmount)(JNIEnv* jni, jobject j_dc) {
871   uint64_t buffered_amount = ExtractNativeDC(jni, j_dc)->buffered_amount();
872   RTC_CHECK_LE(buffered_amount, std::numeric_limits<int64_t>::max())
873       << "buffered_amount overflowed jlong!";
874   return static_cast<jlong>(buffered_amount);
875 }
876 
JOW(void,DataChannel_close)877 JOW(void, DataChannel_close)(JNIEnv* jni, jobject j_dc) {
878   ExtractNativeDC(jni, j_dc)->Close();
879 }
880 
JOW(jboolean,DataChannel_sendNative)881 JOW(jboolean, DataChannel_sendNative)(JNIEnv* jni, jobject j_dc,
882                                       jbyteArray data, jboolean binary) {
883   jbyte* bytes = jni->GetByteArrayElements(data, NULL);
884   bool ret = ExtractNativeDC(jni, j_dc)->Send(DataBuffer(
885       rtc::Buffer(bytes, jni->GetArrayLength(data)),
886       binary));
887   jni->ReleaseByteArrayElements(data, bytes, JNI_ABORT);
888   return ret;
889 }
890 
JOW(void,DataChannel_dispose)891 JOW(void, DataChannel_dispose)(JNIEnv* jni, jobject j_dc) {
892   CHECK_RELEASE(ExtractNativeDC(jni, j_dc));
893 }
894 
JOW(void,Logging_nativeEnableTracing)895 JOW(void, Logging_nativeEnableTracing)(
896     JNIEnv* jni, jclass, jstring j_path, jint nativeLevels,
897     jint nativeSeverity) {
898   std::string path = JavaToStdString(jni, j_path);
899   if (nativeLevels != webrtc::kTraceNone) {
900     webrtc::Trace::set_level_filter(nativeLevels);
901 #if defined(ANDROID) && !defined(WEBRTC_CHROMIUM_BUILD)
902     if (path != "logcat:") {
903 #endif
904       RTC_CHECK_EQ(0, webrtc::Trace::SetTraceFile(path.c_str(), false))
905           << "SetTraceFile failed";
906 #if defined(ANDROID) && !defined(WEBRTC_CHROMIUM_BUILD)
907     } else {
908       // Intentionally leak this to avoid needing to reason about its lifecycle.
909       // It keeps no state and functions only as a dispatch point.
910       static LogcatTraceContext* g_trace_callback = new LogcatTraceContext();
911     }
912 #endif
913   }
914   if (nativeSeverity >= rtc::LS_SENSITIVE && nativeSeverity <= rtc::LS_ERROR) {
915     rtc::LogMessage::LogToDebug(
916         static_cast<rtc::LoggingSeverity>(nativeSeverity));
917   }
918 }
919 
JOW(void,Logging_nativeEnableLogThreads)920 JOW(void, Logging_nativeEnableLogThreads)(JNIEnv* jni, jclass) {
921   rtc::LogMessage::LogThreads(true);
922 }
923 
JOW(void,Logging_nativeEnableLogTimeStamps)924 JOW(void, Logging_nativeEnableLogTimeStamps)(JNIEnv* jni, jclass) {
925   rtc::LogMessage::LogTimestamps(true);
926 }
927 
JOW(void,Logging_nativeLog)928 JOW(void, Logging_nativeLog)(
929     JNIEnv* jni, jclass, jint j_severity, jstring j_tag, jstring j_message) {
930   std::string message = JavaToStdString(jni, j_message);
931   std::string tag = JavaToStdString(jni, j_tag);
932   LOG_TAG(static_cast<rtc::LoggingSeverity>(j_severity), tag) << message;
933 }
934 
JOW(void,PeerConnection_freePeerConnection)935 JOW(void, PeerConnection_freePeerConnection)(JNIEnv*, jclass, jlong j_p) {
936   CHECK_RELEASE(reinterpret_cast<PeerConnectionInterface*>(j_p));
937 }
938 
JOW(void,PeerConnection_freeObserver)939 JOW(void, PeerConnection_freeObserver)(JNIEnv*, jclass, jlong j_p) {
940   PCOJava* p = reinterpret_cast<PCOJava*>(j_p);
941   delete p;
942 }
943 
JOW(void,MediaSource_free)944 JOW(void, MediaSource_free)(JNIEnv*, jclass, jlong j_p) {
945   CHECK_RELEASE(reinterpret_cast<MediaSourceInterface*>(j_p));
946 }
947 
JOW(void,VideoCapturer_free)948 JOW(void, VideoCapturer_free)(JNIEnv*, jclass, jlong j_p) {
949   delete reinterpret_cast<cricket::VideoCapturer*>(j_p);
950 }
951 
JOW(void,VideoRenderer_freeGuiVideoRenderer)952 JOW(void, VideoRenderer_freeGuiVideoRenderer)(JNIEnv*, jclass, jlong j_p) {
953   delete reinterpret_cast<VideoRendererWrapper*>(j_p);
954 }
955 
JOW(void,VideoRenderer_freeWrappedVideoRenderer)956 JOW(void, VideoRenderer_freeWrappedVideoRenderer)(JNIEnv*, jclass, jlong j_p) {
957   delete reinterpret_cast<JavaVideoRendererWrapper*>(j_p);
958 }
959 
JOW(void,VideoRenderer_releaseNativeFrame)960 JOW(void, VideoRenderer_releaseNativeFrame)(
961     JNIEnv* jni, jclass, jlong j_frame_ptr) {
962   delete reinterpret_cast<const cricket::VideoFrame*>(j_frame_ptr);
963 }
964 
JOW(void,MediaStreamTrack_free)965 JOW(void, MediaStreamTrack_free)(JNIEnv*, jclass, jlong j_p) {
966   reinterpret_cast<MediaStreamTrackInterface*>(j_p)->Release();
967 }
968 
JOW(jboolean,MediaStream_nativeAddAudioTrack)969 JOW(jboolean, MediaStream_nativeAddAudioTrack)(
970     JNIEnv* jni, jclass, jlong pointer, jlong j_audio_track_pointer) {
971   return reinterpret_cast<MediaStreamInterface*>(pointer)->AddTrack(
972       reinterpret_cast<AudioTrackInterface*>(j_audio_track_pointer));
973 }
974 
JOW(jboolean,MediaStream_nativeAddVideoTrack)975 JOW(jboolean, MediaStream_nativeAddVideoTrack)(
976     JNIEnv* jni, jclass, jlong pointer, jlong j_video_track_pointer) {
977   return reinterpret_cast<MediaStreamInterface*>(pointer)
978       ->AddTrack(reinterpret_cast<VideoTrackInterface*>(j_video_track_pointer));
979 }
980 
JOW(jboolean,MediaStream_nativeRemoveAudioTrack)981 JOW(jboolean, MediaStream_nativeRemoveAudioTrack)(
982     JNIEnv* jni, jclass, jlong pointer, jlong j_audio_track_pointer) {
983   return reinterpret_cast<MediaStreamInterface*>(pointer)->RemoveTrack(
984       reinterpret_cast<AudioTrackInterface*>(j_audio_track_pointer));
985 }
986 
JOW(jboolean,MediaStream_nativeRemoveVideoTrack)987 JOW(jboolean, MediaStream_nativeRemoveVideoTrack)(
988     JNIEnv* jni, jclass, jlong pointer, jlong j_video_track_pointer) {
989   return reinterpret_cast<MediaStreamInterface*>(pointer)->RemoveTrack(
990       reinterpret_cast<VideoTrackInterface*>(j_video_track_pointer));
991 }
992 
JOW(jstring,MediaStream_nativeLabel)993 JOW(jstring, MediaStream_nativeLabel)(JNIEnv* jni, jclass, jlong j_p) {
994   return JavaStringFromStdString(
995       jni, reinterpret_cast<MediaStreamInterface*>(j_p)->label());
996 }
997 
JOW(void,MediaStream_free)998 JOW(void, MediaStream_free)(JNIEnv*, jclass, jlong j_p) {
999   CHECK_RELEASE(reinterpret_cast<MediaStreamInterface*>(j_p));
1000 }
1001 
JOW(jlong,PeerConnectionFactory_nativeCreateObserver)1002 JOW(jlong, PeerConnectionFactory_nativeCreateObserver)(
1003     JNIEnv * jni, jclass, jobject j_observer) {
1004   return (jlong)new PCOJava(jni, j_observer);
1005 }
1006 
1007 #if defined(ANDROID) && !defined(WEBRTC_CHROMIUM_BUILD)
JOW(jboolean,PeerConnectionFactory_initializeAndroidGlobals)1008 JOW(jboolean, PeerConnectionFactory_initializeAndroidGlobals)(
1009     JNIEnv* jni, jclass, jobject context,
1010     jboolean initialize_audio, jboolean initialize_video,
1011     jboolean video_hw_acceleration) {
1012   bool failure = false;
1013   video_hw_acceleration_enabled = video_hw_acceleration;
1014   AndroidNetworkMonitor::SetAndroidContext(jni, context);
1015   if (!factory_static_initialized) {
1016     if (initialize_video) {
1017       failure |= webrtc::SetRenderAndroidVM(GetJVM());
1018       failure |= AndroidVideoCapturerJni::SetAndroidObjects(jni, context);
1019     }
1020     if (initialize_audio)
1021       failure |= webrtc::VoiceEngine::SetAndroidObjects(GetJVM(), context);
1022     factory_static_initialized = true;
1023   }
1024   return !failure;
1025 }
1026 #endif  // defined(ANDROID) && !defined(WEBRTC_CHROMIUM_BUILD)
1027 
JOW(void,PeerConnectionFactory_initializeFieldTrials)1028 JOW(void, PeerConnectionFactory_initializeFieldTrials)(
1029     JNIEnv* jni, jclass, jstring j_trials_init_string) {
1030   field_trials_init_string = NULL;
1031   if (j_trials_init_string != NULL) {
1032     const char* init_string =
1033         jni->GetStringUTFChars(j_trials_init_string, NULL);
1034     int init_string_length = jni->GetStringUTFLength(j_trials_init_string);
1035     field_trials_init_string = new char[init_string_length + 1];
1036     rtc::strcpyn(field_trials_init_string, init_string_length + 1, init_string);
1037     jni->ReleaseStringUTFChars(j_trials_init_string, init_string);
1038     LOG(LS_INFO) << "initializeFieldTrials: " << field_trials_init_string;
1039   }
1040   webrtc::field_trial::InitFieldTrialsFromString(field_trials_init_string);
1041 }
1042 
JOW(void,PeerConnectionFactory_initializeInternalTracer)1043 JOW(void, PeerConnectionFactory_initializeInternalTracer)(JNIEnv* jni, jclass) {
1044   rtc::tracing::SetupInternalTracer();
1045 }
1046 
JOW(jboolean,PeerConnectionFactory_startInternalTracingCapture)1047 JOW(jboolean, PeerConnectionFactory_startInternalTracingCapture)(
1048     JNIEnv* jni, jclass, jstring j_event_tracing_filename) {
1049   if (!j_event_tracing_filename)
1050     return false;
1051 
1052   const char* init_string =
1053       jni->GetStringUTFChars(j_event_tracing_filename, NULL);
1054   LOG(LS_INFO) << "Starting internal tracing to: " << init_string;
1055   bool ret = rtc::tracing::StartInternalCapture(init_string);
1056   jni->ReleaseStringUTFChars(j_event_tracing_filename, init_string);
1057   return ret;
1058 }
1059 
JOW(void,PeerConnectionFactory_stopInternalTracingCapture)1060 JOW(void, PeerConnectionFactory_stopInternalTracingCapture)(
1061     JNIEnv* jni, jclass) {
1062   rtc::tracing::StopInternalCapture();
1063 }
1064 
JOW(void,PeerConnectionFactory_shutdownInternalTracer)1065 JOW(void, PeerConnectionFactory_shutdownInternalTracer)(JNIEnv* jni, jclass) {
1066   rtc::tracing::ShutdownInternalTracer();
1067 }
1068 
1069 // Helper struct for working around the fact that CreatePeerConnectionFactory()
1070 // comes in two flavors: either entirely automagical (constructing its own
1071 // threads and deleting them on teardown, but no external codec factory support)
1072 // or entirely manual (requires caller to delete threads after factory
1073 // teardown).  This struct takes ownership of its ctor's arguments to present a
1074 // single thing for Java to hold and eventually free.
1075 class OwnedFactoryAndThreads {
1076  public:
OwnedFactoryAndThreads(Thread * worker_thread,Thread * signaling_thread,WebRtcVideoEncoderFactory * encoder_factory,WebRtcVideoDecoderFactory * decoder_factory,rtc::NetworkMonitorFactory * network_monitor_factory,PeerConnectionFactoryInterface * factory)1077   OwnedFactoryAndThreads(Thread* worker_thread,
1078                          Thread* signaling_thread,
1079                          WebRtcVideoEncoderFactory* encoder_factory,
1080                          WebRtcVideoDecoderFactory* decoder_factory,
1081                          rtc::NetworkMonitorFactory* network_monitor_factory,
1082                          PeerConnectionFactoryInterface* factory)
1083       : worker_thread_(worker_thread),
1084         signaling_thread_(signaling_thread),
1085         encoder_factory_(encoder_factory),
1086         decoder_factory_(decoder_factory),
1087         network_monitor_factory_(network_monitor_factory),
1088         factory_(factory) {}
1089 
~OwnedFactoryAndThreads()1090   ~OwnedFactoryAndThreads() {
1091     CHECK_RELEASE(factory_);
1092     if (network_monitor_factory_ != nullptr) {
1093       rtc::NetworkMonitorFactory::ReleaseFactory(network_monitor_factory_);
1094     }
1095   }
1096 
factory()1097   PeerConnectionFactoryInterface* factory() { return factory_; }
encoder_factory()1098   WebRtcVideoEncoderFactory* encoder_factory() { return encoder_factory_; }
decoder_factory()1099   WebRtcVideoDecoderFactory* decoder_factory() { return decoder_factory_; }
network_monitor_factory()1100   rtc::NetworkMonitorFactory* network_monitor_factory() {
1101     return network_monitor_factory_;
1102   }
clear_network_monitor_factory()1103   void clear_network_monitor_factory() { network_monitor_factory_ = nullptr; }
1104   void InvokeJavaCallbacksOnFactoryThreads();
1105 
1106  private:
1107   void JavaCallbackOnFactoryThreads();
1108 
1109   const scoped_ptr<Thread> worker_thread_;
1110   const scoped_ptr<Thread> signaling_thread_;
1111   WebRtcVideoEncoderFactory* encoder_factory_;
1112   WebRtcVideoDecoderFactory* decoder_factory_;
1113   rtc::NetworkMonitorFactory* network_monitor_factory_;
1114   PeerConnectionFactoryInterface* factory_;  // Const after ctor except dtor.
1115 };
1116 
JavaCallbackOnFactoryThreads()1117 void OwnedFactoryAndThreads::JavaCallbackOnFactoryThreads() {
1118   JNIEnv* jni = AttachCurrentThreadIfNeeded();
1119   ScopedLocalRefFrame local_ref_frame(jni);
1120   jclass j_factory_class = FindClass(jni, "org/webrtc/PeerConnectionFactory");
1121   jmethodID m = nullptr;
1122   if (Thread::Current() == worker_thread_) {
1123     LOG(LS_INFO) << "Worker thread JavaCallback";
1124     m = GetStaticMethodID(jni, j_factory_class, "onWorkerThreadReady", "()V");
1125   }
1126   if (Thread::Current() == signaling_thread_) {
1127     LOG(LS_INFO) << "Signaling thread JavaCallback";
1128     m = GetStaticMethodID(
1129         jni, j_factory_class, "onSignalingThreadReady", "()V");
1130   }
1131   if (m != nullptr) {
1132     jni->CallStaticVoidMethod(j_factory_class, m);
1133     CHECK_EXCEPTION(jni) << "error during JavaCallback::CallStaticVoidMethod";
1134   }
1135 }
1136 
InvokeJavaCallbacksOnFactoryThreads()1137 void OwnedFactoryAndThreads::InvokeJavaCallbacksOnFactoryThreads() {
1138   LOG(LS_INFO) << "InvokeJavaCallbacksOnFactoryThreads.";
1139   worker_thread_->Invoke<void>(
1140       Bind(&OwnedFactoryAndThreads::JavaCallbackOnFactoryThreads, this));
1141   signaling_thread_->Invoke<void>(
1142       Bind(&OwnedFactoryAndThreads::JavaCallbackOnFactoryThreads, this));
1143 }
1144 
JOW(jlong,PeerConnectionFactory_nativeCreatePeerConnectionFactory)1145 JOW(jlong, PeerConnectionFactory_nativeCreatePeerConnectionFactory)(
1146     JNIEnv* jni, jclass) {
1147   // talk/ assumes pretty widely that the current Thread is ThreadManager'd, but
1148   // ThreadManager only WrapCurrentThread()s the thread where it is first
1149   // created.  Since the semantics around when auto-wrapping happens in
1150   // webrtc/base/ are convoluted, we simply wrap here to avoid having to think
1151   // about ramifications of auto-wrapping there.
1152   rtc::ThreadManager::Instance()->WrapCurrentThread();
1153   webrtc::Trace::CreateTrace();
1154   Thread* worker_thread = new Thread();
1155   worker_thread->SetName("worker_thread", NULL);
1156   Thread* signaling_thread = new Thread();
1157   signaling_thread->SetName("signaling_thread", NULL);
1158   RTC_CHECK(worker_thread->Start() && signaling_thread->Start())
1159       << "Failed to start threads";
1160   WebRtcVideoEncoderFactory* encoder_factory = nullptr;
1161   WebRtcVideoDecoderFactory* decoder_factory = nullptr;
1162   rtc::NetworkMonitorFactory* network_monitor_factory = nullptr;
1163 
1164 #if defined(ANDROID) && !defined(WEBRTC_CHROMIUM_BUILD)
1165   if (video_hw_acceleration_enabled) {
1166     encoder_factory = new MediaCodecVideoEncoderFactory();
1167     decoder_factory = new MediaCodecVideoDecoderFactory();
1168   }
1169   network_monitor_factory = new AndroidNetworkMonitorFactory();
1170   rtc::NetworkMonitorFactory::SetFactory(network_monitor_factory);
1171 #endif
1172   rtc::scoped_refptr<PeerConnectionFactoryInterface> factory(
1173       webrtc::CreatePeerConnectionFactory(worker_thread,
1174                                           signaling_thread,
1175                                           NULL,
1176                                           encoder_factory,
1177                                           decoder_factory));
1178   RTC_CHECK(factory) << "Failed to create the peer connection factory; "
1179                      << "WebRTC/libjingle init likely failed on this device";
1180   OwnedFactoryAndThreads* owned_factory = new OwnedFactoryAndThreads(
1181       worker_thread, signaling_thread,
1182       encoder_factory, decoder_factory,
1183       network_monitor_factory, factory.release());
1184   owned_factory->InvokeJavaCallbacksOnFactoryThreads();
1185   return jlongFromPointer(owned_factory);
1186 }
1187 
JOW(void,PeerConnectionFactory_nativeFreeFactory)1188 JOW(void, PeerConnectionFactory_nativeFreeFactory)(JNIEnv*, jclass, jlong j_p) {
1189   delete reinterpret_cast<OwnedFactoryAndThreads*>(j_p);
1190   if (field_trials_init_string) {
1191     webrtc::field_trial::InitFieldTrialsFromString(NULL);
1192     delete field_trials_init_string;
1193     field_trials_init_string = NULL;
1194   }
1195   webrtc::Trace::ReturnTrace();
1196 }
1197 
factoryFromJava(jlong j_p)1198 static PeerConnectionFactoryInterface* factoryFromJava(jlong j_p) {
1199   return reinterpret_cast<OwnedFactoryAndThreads*>(j_p)->factory();
1200 }
1201 
JOW(void,PeerConnectionFactory_nativeThreadsCallbacks)1202 JOW(void, PeerConnectionFactory_nativeThreadsCallbacks)(
1203     JNIEnv*, jclass, jlong j_p) {
1204   OwnedFactoryAndThreads *factory =
1205       reinterpret_cast<OwnedFactoryAndThreads*>(j_p);
1206   factory->InvokeJavaCallbacksOnFactoryThreads();
1207 }
1208 
JOW(jlong,PeerConnectionFactory_nativeCreateLocalMediaStream)1209 JOW(jlong, PeerConnectionFactory_nativeCreateLocalMediaStream)(
1210     JNIEnv* jni, jclass, jlong native_factory, jstring label) {
1211   rtc::scoped_refptr<PeerConnectionFactoryInterface> factory(
1212       factoryFromJava(native_factory));
1213   rtc::scoped_refptr<MediaStreamInterface> stream(
1214       factory->CreateLocalMediaStream(JavaToStdString(jni, label)));
1215   return (jlong)stream.release();
1216 }
1217 
JOW(jlong,PeerConnectionFactory_nativeCreateVideoSource)1218 JOW(jlong, PeerConnectionFactory_nativeCreateVideoSource)(
1219     JNIEnv* jni, jclass, jlong native_factory, jlong native_capturer,
1220     jobject j_constraints) {
1221   scoped_ptr<ConstraintsWrapper> constraints(
1222       new ConstraintsWrapper(jni, j_constraints));
1223   rtc::scoped_refptr<PeerConnectionFactoryInterface> factory(
1224       factoryFromJava(native_factory));
1225   rtc::scoped_refptr<VideoSourceInterface> source(
1226       factory->CreateVideoSource(
1227           reinterpret_cast<cricket::VideoCapturer*>(native_capturer),
1228           constraints.get()));
1229   return (jlong)source.release();
1230 }
1231 
JOW(jlong,PeerConnectionFactory_nativeCreateVideoTrack)1232 JOW(jlong, PeerConnectionFactory_nativeCreateVideoTrack)(
1233     JNIEnv* jni, jclass, jlong native_factory, jstring id,
1234     jlong native_source) {
1235   rtc::scoped_refptr<PeerConnectionFactoryInterface> factory(
1236       factoryFromJava(native_factory));
1237   rtc::scoped_refptr<VideoTrackInterface> track(
1238       factory->CreateVideoTrack(
1239           JavaToStdString(jni, id),
1240           reinterpret_cast<VideoSourceInterface*>(native_source)));
1241   return (jlong)track.release();
1242 }
1243 
JOW(jlong,PeerConnectionFactory_nativeCreateAudioSource)1244 JOW(jlong, PeerConnectionFactory_nativeCreateAudioSource)(
1245     JNIEnv* jni, jclass, jlong native_factory, jobject j_constraints) {
1246   scoped_ptr<ConstraintsWrapper> constraints(
1247       new ConstraintsWrapper(jni, j_constraints));
1248   rtc::scoped_refptr<PeerConnectionFactoryInterface> factory(
1249       factoryFromJava(native_factory));
1250   rtc::scoped_refptr<AudioSourceInterface> source(
1251       factory->CreateAudioSource(constraints.get()));
1252   return (jlong)source.release();
1253 }
1254 
JOW(jlong,PeerConnectionFactory_nativeCreateAudioTrack)1255 JOW(jlong, PeerConnectionFactory_nativeCreateAudioTrack)(
1256     JNIEnv* jni, jclass, jlong native_factory, jstring id,
1257     jlong native_source) {
1258   rtc::scoped_refptr<PeerConnectionFactoryInterface> factory(
1259       factoryFromJava(native_factory));
1260   rtc::scoped_refptr<AudioTrackInterface> track(factory->CreateAudioTrack(
1261       JavaToStdString(jni, id),
1262       reinterpret_cast<AudioSourceInterface*>(native_source)));
1263   return (jlong)track.release();
1264 }
1265 
JOW(jboolean,PeerConnectionFactory_nativeStartAecDump)1266 JOW(jboolean, PeerConnectionFactory_nativeStartAecDump)(
1267     JNIEnv* jni, jclass, jlong native_factory, jint file) {
1268 #if defined(ANDROID)
1269   rtc::scoped_refptr<PeerConnectionFactoryInterface> factory(
1270       factoryFromJava(native_factory));
1271   return factory->StartAecDump(file);
1272 #else
1273   return false;
1274 #endif
1275 }
1276 
JOW(void,PeerConnectionFactory_nativeStopAecDump)1277 JOW(void, PeerConnectionFactory_nativeStopAecDump)(
1278     JNIEnv* jni, jclass, jlong native_factory) {
1279 #if defined(ANDROID)
1280   rtc::scoped_refptr<PeerConnectionFactoryInterface> factory(
1281       factoryFromJava(native_factory));
1282   factory->StopAecDump();
1283 #endif
1284 }
1285 
JOW(jboolean,PeerConnectionFactory_nativeStartRtcEventLog)1286 JOW(jboolean, PeerConnectionFactory_nativeStartRtcEventLog)(
1287     JNIEnv* jni, jclass, jlong native_factory, jint file) {
1288 #if defined(ANDROID)
1289   rtc::scoped_refptr<PeerConnectionFactoryInterface> factory(
1290       factoryFromJava(native_factory));
1291   return factory->StartRtcEventLog(file);
1292 #else
1293   return false;
1294 #endif
1295 }
1296 
JOW(void,PeerConnectionFactory_nativeStopRtcEventLog)1297 JOW(void, PeerConnectionFactory_nativeStopRtcEventLog)(
1298     JNIEnv* jni, jclass, jlong native_factory) {
1299 #if defined(ANDROID)
1300   rtc::scoped_refptr<PeerConnectionFactoryInterface> factory(
1301       factoryFromJava(native_factory));
1302   factory->StopRtcEventLog();
1303 #endif
1304 }
1305 
JOW(void,PeerConnectionFactory_nativeSetOptions)1306 JOW(void, PeerConnectionFactory_nativeSetOptions)(
1307     JNIEnv* jni, jclass, jlong native_factory, jobject options) {
1308   rtc::scoped_refptr<PeerConnectionFactoryInterface> factory(
1309       factoryFromJava(native_factory));
1310   jclass options_class = jni->GetObjectClass(options);
1311   jfieldID network_ignore_mask_field =
1312       jni->GetFieldID(options_class, "networkIgnoreMask", "I");
1313   int network_ignore_mask =
1314       jni->GetIntField(options, network_ignore_mask_field);
1315 
1316   jfieldID disable_encryption_field =
1317       jni->GetFieldID(options_class, "disableEncryption", "Z");
1318   bool disable_encryption =
1319       jni->GetBooleanField(options, disable_encryption_field);
1320 
1321   jfieldID disable_network_monitor_field =
1322       jni->GetFieldID(options_class, "disableNetworkMonitor", "Z");
1323   bool disable_network_monitor =
1324       jni->GetBooleanField(options, disable_network_monitor_field);
1325 
1326   PeerConnectionFactoryInterface::Options options_to_set;
1327 
1328   // This doesn't necessarily match the c++ version of this struct; feel free
1329   // to add more parameters as necessary.
1330   options_to_set.network_ignore_mask = network_ignore_mask;
1331   options_to_set.disable_encryption = disable_encryption;
1332   options_to_set.disable_network_monitor = disable_network_monitor;
1333   factory->SetOptions(options_to_set);
1334 
1335   if (disable_network_monitor) {
1336     OwnedFactoryAndThreads* owner =
1337         reinterpret_cast<OwnedFactoryAndThreads*>(native_factory);
1338     if (owner->network_monitor_factory()) {
1339       rtc::NetworkMonitorFactory::ReleaseFactory(
1340           owner->network_monitor_factory());
1341       owner->clear_network_monitor_factory();
1342     }
1343   }
1344 }
1345 
JOW(void,PeerConnectionFactory_nativeSetVideoHwAccelerationOptions)1346 JOW(void, PeerConnectionFactory_nativeSetVideoHwAccelerationOptions)(
1347     JNIEnv* jni, jclass, jlong native_factory, jobject local_egl_context,
1348     jobject remote_egl_context) {
1349 #if defined(ANDROID) && !defined(WEBRTC_CHROMIUM_BUILD)
1350   OwnedFactoryAndThreads* owned_factory =
1351       reinterpret_cast<OwnedFactoryAndThreads*>(native_factory);
1352 
1353   jclass j_eglbase14_context_class =
1354       FindClass(jni, "org/webrtc/EglBase14$Context");
1355 
1356   MediaCodecVideoEncoderFactory* encoder_factory =
1357       static_cast<MediaCodecVideoEncoderFactory*>
1358           (owned_factory->encoder_factory());
1359   if (encoder_factory &&
1360       jni->IsInstanceOf(local_egl_context, j_eglbase14_context_class)) {
1361     LOG(LS_INFO) << "Set EGL context for HW encoding.";
1362     encoder_factory->SetEGLContext(jni, local_egl_context);
1363   }
1364 
1365   MediaCodecVideoDecoderFactory* decoder_factory =
1366       static_cast<MediaCodecVideoDecoderFactory*>
1367           (owned_factory->decoder_factory());
1368   if (decoder_factory &&
1369       jni->IsInstanceOf(remote_egl_context, j_eglbase14_context_class)) {
1370     LOG(LS_INFO) << "Set EGL context for HW decoding.";
1371     decoder_factory->SetEGLContext(jni, remote_egl_context);
1372   }
1373 #endif
1374 }
1375 
1376 static std::string
GetJavaEnumName(JNIEnv * jni,const std::string & className,jobject j_enum)1377 GetJavaEnumName(JNIEnv* jni, const std::string& className, jobject j_enum) {
1378   jclass enumClass = FindClass(jni, className.c_str());
1379   jmethodID nameMethod =
1380       GetMethodID(jni, enumClass, "name", "()Ljava/lang/String;");
1381   jstring name =
1382       reinterpret_cast<jstring>(jni->CallObjectMethod(j_enum, nameMethod));
1383   CHECK_EXCEPTION(jni) << "error during CallObjectMethod for "
1384                        << className << ".name";
1385   return JavaToStdString(jni, name);
1386 }
1387 
1388 static PeerConnectionInterface::IceTransportsType
JavaIceTransportsTypeToNativeType(JNIEnv * jni,jobject j_ice_transports_type)1389 JavaIceTransportsTypeToNativeType(JNIEnv* jni, jobject j_ice_transports_type) {
1390   std::string enum_name = GetJavaEnumName(
1391       jni, "org/webrtc/PeerConnection$IceTransportsType",
1392       j_ice_transports_type);
1393 
1394   if (enum_name == "ALL")
1395     return PeerConnectionInterface::kAll;
1396 
1397   if (enum_name == "RELAY")
1398     return PeerConnectionInterface::kRelay;
1399 
1400   if (enum_name == "NOHOST")
1401     return PeerConnectionInterface::kNoHost;
1402 
1403   if (enum_name == "NONE")
1404     return PeerConnectionInterface::kNone;
1405 
1406   RTC_CHECK(false) << "Unexpected IceTransportsType enum_name " << enum_name;
1407   return PeerConnectionInterface::kAll;
1408 }
1409 
1410 static PeerConnectionInterface::BundlePolicy
JavaBundlePolicyToNativeType(JNIEnv * jni,jobject j_bundle_policy)1411 JavaBundlePolicyToNativeType(JNIEnv* jni, jobject j_bundle_policy) {
1412   std::string enum_name = GetJavaEnumName(
1413       jni, "org/webrtc/PeerConnection$BundlePolicy",
1414       j_bundle_policy);
1415 
1416   if (enum_name == "BALANCED")
1417     return PeerConnectionInterface::kBundlePolicyBalanced;
1418 
1419   if (enum_name == "MAXBUNDLE")
1420     return PeerConnectionInterface::kBundlePolicyMaxBundle;
1421 
1422   if (enum_name == "MAXCOMPAT")
1423     return PeerConnectionInterface::kBundlePolicyMaxCompat;
1424 
1425   RTC_CHECK(false) << "Unexpected BundlePolicy enum_name " << enum_name;
1426   return PeerConnectionInterface::kBundlePolicyBalanced;
1427 }
1428 
1429 static PeerConnectionInterface::RtcpMuxPolicy
JavaRtcpMuxPolicyToNativeType(JNIEnv * jni,jobject j_rtcp_mux_policy)1430 JavaRtcpMuxPolicyToNativeType(JNIEnv* jni, jobject j_rtcp_mux_policy) {
1431   std::string enum_name = GetJavaEnumName(
1432       jni, "org/webrtc/PeerConnection$RtcpMuxPolicy",
1433       j_rtcp_mux_policy);
1434 
1435   if (enum_name == "NEGOTIATE")
1436     return PeerConnectionInterface::kRtcpMuxPolicyNegotiate;
1437 
1438   if (enum_name == "REQUIRE")
1439     return PeerConnectionInterface::kRtcpMuxPolicyRequire;
1440 
1441   RTC_CHECK(false) << "Unexpected RtcpMuxPolicy enum_name " << enum_name;
1442   return PeerConnectionInterface::kRtcpMuxPolicyNegotiate;
1443 }
1444 
1445 static PeerConnectionInterface::TcpCandidatePolicy
JavaTcpCandidatePolicyToNativeType(JNIEnv * jni,jobject j_tcp_candidate_policy)1446 JavaTcpCandidatePolicyToNativeType(
1447     JNIEnv* jni, jobject j_tcp_candidate_policy) {
1448   std::string enum_name = GetJavaEnumName(
1449       jni, "org/webrtc/PeerConnection$TcpCandidatePolicy",
1450       j_tcp_candidate_policy);
1451 
1452   if (enum_name == "ENABLED")
1453     return PeerConnectionInterface::kTcpCandidatePolicyEnabled;
1454 
1455   if (enum_name == "DISABLED")
1456     return PeerConnectionInterface::kTcpCandidatePolicyDisabled;
1457 
1458   RTC_CHECK(false) << "Unexpected TcpCandidatePolicy enum_name " << enum_name;
1459   return PeerConnectionInterface::kTcpCandidatePolicyEnabled;
1460 }
1461 
JavaKeyTypeToNativeType(JNIEnv * jni,jobject j_key_type)1462 static rtc::KeyType JavaKeyTypeToNativeType(JNIEnv* jni, jobject j_key_type) {
1463   std::string enum_name = GetJavaEnumName(
1464       jni, "org/webrtc/PeerConnection$KeyType", j_key_type);
1465 
1466   if (enum_name == "RSA")
1467     return rtc::KT_RSA;
1468   if (enum_name == "ECDSA")
1469     return rtc::KT_ECDSA;
1470 
1471   RTC_CHECK(false) << "Unexpected KeyType enum_name " << enum_name;
1472   return rtc::KT_ECDSA;
1473 }
1474 
1475 static PeerConnectionInterface::ContinualGatheringPolicy
JavaContinualGatheringPolicyToNativeType(JNIEnv * jni,jobject j_gathering_policy)1476     JavaContinualGatheringPolicyToNativeType(
1477         JNIEnv* jni, jobject j_gathering_policy) {
1478   std::string enum_name = GetJavaEnumName(
1479       jni, "org/webrtc/PeerConnection$ContinualGatheringPolicy",
1480       j_gathering_policy);
1481   if (enum_name == "GATHER_ONCE")
1482     return PeerConnectionInterface::GATHER_ONCE;
1483 
1484   if (enum_name == "GATHER_CONTINUALLY")
1485     return PeerConnectionInterface::GATHER_CONTINUALLY;
1486 
1487   RTC_CHECK(false) << "Unexpected ContinualGatheringPolicy enum name "
1488                    << enum_name;
1489   return PeerConnectionInterface::GATHER_ONCE;
1490 }
1491 
JavaIceServersToJsepIceServers(JNIEnv * jni,jobject j_ice_servers,PeerConnectionInterface::IceServers * ice_servers)1492 static void JavaIceServersToJsepIceServers(
1493     JNIEnv* jni, jobject j_ice_servers,
1494     PeerConnectionInterface::IceServers* ice_servers) {
1495   jclass list_class = GetObjectClass(jni, j_ice_servers);
1496   jmethodID iterator_id = GetMethodID(
1497       jni, list_class, "iterator", "()Ljava/util/Iterator;");
1498   jobject iterator = jni->CallObjectMethod(j_ice_servers, iterator_id);
1499   CHECK_EXCEPTION(jni) << "error during CallObjectMethod";
1500   jmethodID iterator_has_next = GetMethodID(
1501       jni, GetObjectClass(jni, iterator), "hasNext", "()Z");
1502   jmethodID iterator_next = GetMethodID(
1503       jni, GetObjectClass(jni, iterator), "next", "()Ljava/lang/Object;");
1504   while (jni->CallBooleanMethod(iterator, iterator_has_next)) {
1505     CHECK_EXCEPTION(jni) << "error during CallBooleanMethod";
1506     jobject j_ice_server = jni->CallObjectMethod(iterator, iterator_next);
1507     CHECK_EXCEPTION(jni) << "error during CallObjectMethod";
1508     jclass j_ice_server_class = GetObjectClass(jni, j_ice_server);
1509     jfieldID j_ice_server_uri_id =
1510         GetFieldID(jni, j_ice_server_class, "uri", "Ljava/lang/String;");
1511     jfieldID j_ice_server_username_id =
1512         GetFieldID(jni, j_ice_server_class, "username", "Ljava/lang/String;");
1513     jfieldID j_ice_server_password_id =
1514         GetFieldID(jni, j_ice_server_class, "password", "Ljava/lang/String;");
1515     jstring uri = reinterpret_cast<jstring>(
1516         GetObjectField(jni, j_ice_server, j_ice_server_uri_id));
1517     jstring username = reinterpret_cast<jstring>(
1518         GetObjectField(jni, j_ice_server, j_ice_server_username_id));
1519     jstring password = reinterpret_cast<jstring>(
1520         GetObjectField(jni, j_ice_server, j_ice_server_password_id));
1521     PeerConnectionInterface::IceServer server;
1522     server.uri = JavaToStdString(jni, uri);
1523     server.username = JavaToStdString(jni, username);
1524     server.password = JavaToStdString(jni, password);
1525     ice_servers->push_back(server);
1526   }
1527   CHECK_EXCEPTION(jni) << "error during CallBooleanMethod";
1528 }
1529 
JavaRTCConfigurationToJsepRTCConfiguration(JNIEnv * jni,jobject j_rtc_config,PeerConnectionInterface::RTCConfiguration * rtc_config)1530 static void JavaRTCConfigurationToJsepRTCConfiguration(
1531     JNIEnv* jni,
1532     jobject j_rtc_config,
1533     PeerConnectionInterface::RTCConfiguration* rtc_config) {
1534   jclass j_rtc_config_class = GetObjectClass(jni, j_rtc_config);
1535 
1536   jfieldID j_ice_transports_type_id = GetFieldID(
1537       jni, j_rtc_config_class, "iceTransportsType",
1538       "Lorg/webrtc/PeerConnection$IceTransportsType;");
1539   jobject j_ice_transports_type = GetObjectField(
1540       jni, j_rtc_config, j_ice_transports_type_id);
1541 
1542   jfieldID j_bundle_policy_id = GetFieldID(
1543       jni, j_rtc_config_class, "bundlePolicy",
1544       "Lorg/webrtc/PeerConnection$BundlePolicy;");
1545   jobject j_bundle_policy = GetObjectField(
1546       jni, j_rtc_config, j_bundle_policy_id);
1547 
1548   jfieldID j_rtcp_mux_policy_id = GetFieldID(
1549       jni, j_rtc_config_class, "rtcpMuxPolicy",
1550       "Lorg/webrtc/PeerConnection$RtcpMuxPolicy;");
1551   jobject j_rtcp_mux_policy = GetObjectField(
1552       jni, j_rtc_config, j_rtcp_mux_policy_id);
1553 
1554   jfieldID j_tcp_candidate_policy_id = GetFieldID(
1555       jni, j_rtc_config_class, "tcpCandidatePolicy",
1556       "Lorg/webrtc/PeerConnection$TcpCandidatePolicy;");
1557   jobject j_tcp_candidate_policy = GetObjectField(
1558       jni, j_rtc_config, j_tcp_candidate_policy_id);
1559 
1560   jfieldID j_ice_servers_id = GetFieldID(
1561       jni, j_rtc_config_class, "iceServers", "Ljava/util/List;");
1562   jobject j_ice_servers = GetObjectField(jni, j_rtc_config, j_ice_servers_id);
1563 
1564   jfieldID j_audio_jitter_buffer_max_packets_id =
1565       GetFieldID(jni, j_rtc_config_class, "audioJitterBufferMaxPackets", "I");
1566   jfieldID j_audio_jitter_buffer_fast_accelerate_id = GetFieldID(
1567       jni, j_rtc_config_class, "audioJitterBufferFastAccelerate", "Z");
1568 
1569   jfieldID j_ice_connection_receiving_timeout_id =
1570       GetFieldID(jni, j_rtc_config_class, "iceConnectionReceivingTimeout", "I");
1571 
1572   jfieldID j_ice_backup_candidate_pair_ping_interval_id = GetFieldID(
1573       jni, j_rtc_config_class, "iceBackupCandidatePairPingInterval", "I");
1574 
1575   jfieldID j_continual_gathering_policy_id =
1576       GetFieldID(jni, j_rtc_config_class, "continualGatheringPolicy",
1577                  "Lorg/webrtc/PeerConnection$ContinualGatheringPolicy;");
1578   jobject j_continual_gathering_policy =
1579       GetObjectField(jni, j_rtc_config, j_continual_gathering_policy_id);
1580 
1581   rtc_config->type =
1582       JavaIceTransportsTypeToNativeType(jni, j_ice_transports_type);
1583   rtc_config->bundle_policy =
1584       JavaBundlePolicyToNativeType(jni, j_bundle_policy);
1585   rtc_config->rtcp_mux_policy =
1586       JavaRtcpMuxPolicyToNativeType(jni, j_rtcp_mux_policy);
1587   rtc_config->tcp_candidate_policy =
1588       JavaTcpCandidatePolicyToNativeType(jni, j_tcp_candidate_policy);
1589   JavaIceServersToJsepIceServers(jni, j_ice_servers, &rtc_config->servers);
1590   rtc_config->audio_jitter_buffer_max_packets =
1591       GetIntField(jni, j_rtc_config, j_audio_jitter_buffer_max_packets_id);
1592   rtc_config->audio_jitter_buffer_fast_accelerate = GetBooleanField(
1593       jni, j_rtc_config, j_audio_jitter_buffer_fast_accelerate_id);
1594   rtc_config->ice_connection_receiving_timeout =
1595       GetIntField(jni, j_rtc_config, j_ice_connection_receiving_timeout_id);
1596   rtc_config->ice_backup_candidate_pair_ping_interval = GetIntField(
1597       jni, j_rtc_config, j_ice_backup_candidate_pair_ping_interval_id);
1598   rtc_config->continual_gathering_policy =
1599       JavaContinualGatheringPolicyToNativeType(
1600           jni, j_continual_gathering_policy);
1601 }
1602 
JOW(jlong,PeerConnectionFactory_nativeCreatePeerConnection)1603 JOW(jlong, PeerConnectionFactory_nativeCreatePeerConnection)(
1604     JNIEnv *jni, jclass, jlong factory, jobject j_rtc_config,
1605     jobject j_constraints, jlong observer_p) {
1606   rtc::scoped_refptr<PeerConnectionFactoryInterface> f(
1607       reinterpret_cast<PeerConnectionFactoryInterface*>(
1608           factoryFromJava(factory)));
1609 
1610   PeerConnectionInterface::RTCConfiguration rtc_config;
1611   JavaRTCConfigurationToJsepRTCConfiguration(jni, j_rtc_config, &rtc_config);
1612 
1613   jclass j_rtc_config_class = GetObjectClass(jni, j_rtc_config);
1614   jfieldID j_key_type_id = GetFieldID(jni, j_rtc_config_class, "keyType",
1615                                       "Lorg/webrtc/PeerConnection$KeyType;");
1616   jobject j_key_type = GetObjectField(jni, j_rtc_config, j_key_type_id);
1617 
1618   // Create ECDSA certificate.
1619   if (JavaKeyTypeToNativeType(jni, j_key_type) == rtc::KT_ECDSA) {
1620     scoped_ptr<rtc::SSLIdentity> ssl_identity(
1621         rtc::SSLIdentity::Generate(webrtc::kIdentityName, rtc::KT_ECDSA));
1622     if (ssl_identity.get()) {
1623       rtc_config.certificates.push_back(
1624           rtc::RTCCertificate::Create(std::move(ssl_identity)));
1625       LOG(LS_INFO) << "ECDSA certificate created.";
1626     } else {
1627       // Failing to create certificate should not abort peer connection
1628       // creation. Instead default encryption (currently RSA) will be used.
1629       LOG(LS_WARNING) <<
1630           "Failed to generate SSLIdentity. Default encryption will be used.";
1631     }
1632   }
1633 
1634   PCOJava* observer = reinterpret_cast<PCOJava*>(observer_p);
1635   observer->SetConstraints(new ConstraintsWrapper(jni, j_constraints));
1636   rtc::scoped_refptr<PeerConnectionInterface> pc(f->CreatePeerConnection(
1637       rtc_config, observer->constraints(), NULL, NULL, observer));
1638   return (jlong)pc.release();
1639 }
1640 
ExtractNativePC(JNIEnv * jni,jobject j_pc)1641 static rtc::scoped_refptr<PeerConnectionInterface> ExtractNativePC(
1642     JNIEnv* jni, jobject j_pc) {
1643   jfieldID native_pc_id = GetFieldID(jni,
1644       GetObjectClass(jni, j_pc), "nativePeerConnection", "J");
1645   jlong j_p = GetLongField(jni, j_pc, native_pc_id);
1646   return rtc::scoped_refptr<PeerConnectionInterface>(
1647       reinterpret_cast<PeerConnectionInterface*>(j_p));
1648 }
1649 
JOW(jobject,PeerConnection_getLocalDescription)1650 JOW(jobject, PeerConnection_getLocalDescription)(JNIEnv* jni, jobject j_pc) {
1651   const SessionDescriptionInterface* sdp =
1652       ExtractNativePC(jni, j_pc)->local_description();
1653   return sdp ? JavaSdpFromNativeSdp(jni, sdp) : NULL;
1654 }
1655 
JOW(jobject,PeerConnection_getRemoteDescription)1656 JOW(jobject, PeerConnection_getRemoteDescription)(JNIEnv* jni, jobject j_pc) {
1657   const SessionDescriptionInterface* sdp =
1658       ExtractNativePC(jni, j_pc)->remote_description();
1659   return sdp ? JavaSdpFromNativeSdp(jni, sdp) : NULL;
1660 }
1661 
JOW(jobject,PeerConnection_createDataChannel)1662 JOW(jobject, PeerConnection_createDataChannel)(
1663     JNIEnv* jni, jobject j_pc, jstring j_label, jobject j_init) {
1664   DataChannelInit init = JavaDataChannelInitToNative(jni, j_init);
1665   rtc::scoped_refptr<DataChannelInterface> channel(
1666       ExtractNativePC(jni, j_pc)->CreateDataChannel(
1667           JavaToStdString(jni, j_label), &init));
1668   // Mustn't pass channel.get() directly through NewObject to avoid reading its
1669   // vararg parameter as 64-bit and reading memory that doesn't belong to the
1670   // 32-bit parameter.
1671   jlong nativeChannelPtr = jlongFromPointer(channel.get());
1672   RTC_CHECK(nativeChannelPtr) << "Failed to create DataChannel";
1673   jclass j_data_channel_class = FindClass(jni, "org/webrtc/DataChannel");
1674   jmethodID j_data_channel_ctor = GetMethodID(
1675       jni, j_data_channel_class, "<init>", "(J)V");
1676   jobject j_channel = jni->NewObject(
1677       j_data_channel_class, j_data_channel_ctor, nativeChannelPtr);
1678   CHECK_EXCEPTION(jni) << "error during NewObject";
1679   // Channel is now owned by Java object, and will be freed from there.
1680   int bumped_count = channel->AddRef();
1681   RTC_CHECK(bumped_count == 2) << "Unexpected refcount";
1682   return j_channel;
1683 }
1684 
JOW(void,PeerConnection_createOffer)1685 JOW(void, PeerConnection_createOffer)(
1686     JNIEnv* jni, jobject j_pc, jobject j_observer, jobject j_constraints) {
1687   ConstraintsWrapper* constraints =
1688       new ConstraintsWrapper(jni, j_constraints);
1689   rtc::scoped_refptr<CreateSdpObserverWrapper> observer(
1690       new rtc::RefCountedObject<CreateSdpObserverWrapper>(
1691           jni, j_observer, constraints));
1692   ExtractNativePC(jni, j_pc)->CreateOffer(observer, constraints);
1693 }
1694 
JOW(void,PeerConnection_createAnswer)1695 JOW(void, PeerConnection_createAnswer)(
1696     JNIEnv* jni, jobject j_pc, jobject j_observer, jobject j_constraints) {
1697   ConstraintsWrapper* constraints =
1698       new ConstraintsWrapper(jni, j_constraints);
1699   rtc::scoped_refptr<CreateSdpObserverWrapper> observer(
1700       new rtc::RefCountedObject<CreateSdpObserverWrapper>(
1701           jni, j_observer, constraints));
1702   ExtractNativePC(jni, j_pc)->CreateAnswer(observer, constraints);
1703 }
1704 
1705 // Helper to create a SessionDescriptionInterface from a SessionDescription.
JavaSdpToNativeSdp(JNIEnv * jni,jobject j_sdp)1706 static SessionDescriptionInterface* JavaSdpToNativeSdp(
1707     JNIEnv* jni, jobject j_sdp) {
1708   jfieldID j_type_id = GetFieldID(
1709       jni, GetObjectClass(jni, j_sdp), "type",
1710       "Lorg/webrtc/SessionDescription$Type;");
1711   jobject j_type = GetObjectField(jni, j_sdp, j_type_id);
1712   jmethodID j_canonical_form_id = GetMethodID(
1713       jni, GetObjectClass(jni, j_type), "canonicalForm",
1714       "()Ljava/lang/String;");
1715   jstring j_type_string = (jstring)jni->CallObjectMethod(
1716       j_type, j_canonical_form_id);
1717   CHECK_EXCEPTION(jni) << "error during CallObjectMethod";
1718   std::string std_type = JavaToStdString(jni, j_type_string);
1719 
1720   jfieldID j_description_id = GetFieldID(
1721       jni, GetObjectClass(jni, j_sdp), "description", "Ljava/lang/String;");
1722   jstring j_description = (jstring)GetObjectField(jni, j_sdp, j_description_id);
1723   std::string std_description = JavaToStdString(jni, j_description);
1724 
1725   return webrtc::CreateSessionDescription(
1726       std_type, std_description, NULL);
1727 }
1728 
JOW(void,PeerConnection_setLocalDescription)1729 JOW(void, PeerConnection_setLocalDescription)(
1730     JNIEnv* jni, jobject j_pc,
1731     jobject j_observer, jobject j_sdp) {
1732   rtc::scoped_refptr<SetSdpObserverWrapper> observer(
1733       new rtc::RefCountedObject<SetSdpObserverWrapper>(
1734           jni, j_observer, reinterpret_cast<ConstraintsWrapper*>(NULL)));
1735   ExtractNativePC(jni, j_pc)->SetLocalDescription(
1736       observer, JavaSdpToNativeSdp(jni, j_sdp));
1737 }
1738 
JOW(void,PeerConnection_setRemoteDescription)1739 JOW(void, PeerConnection_setRemoteDescription)(
1740     JNIEnv* jni, jobject j_pc,
1741     jobject j_observer, jobject j_sdp) {
1742   rtc::scoped_refptr<SetSdpObserverWrapper> observer(
1743       new rtc::RefCountedObject<SetSdpObserverWrapper>(
1744           jni, j_observer, reinterpret_cast<ConstraintsWrapper*>(NULL)));
1745   ExtractNativePC(jni, j_pc)->SetRemoteDescription(
1746       observer, JavaSdpToNativeSdp(jni, j_sdp));
1747 }
1748 
JOW(jboolean,PeerConnection_setConfiguration)1749 JOW(jboolean, PeerConnection_setConfiguration)(
1750     JNIEnv* jni, jobject j_pc, jobject j_rtc_config) {
1751   PeerConnectionInterface::RTCConfiguration rtc_config;
1752   JavaRTCConfigurationToJsepRTCConfiguration(jni, j_rtc_config, &rtc_config);
1753   return ExtractNativePC(jni, j_pc)->SetConfiguration(rtc_config);
1754 }
1755 
JOW(jboolean,PeerConnection_nativeAddIceCandidate)1756 JOW(jboolean, PeerConnection_nativeAddIceCandidate)(
1757     JNIEnv* jni, jobject j_pc, jstring j_sdp_mid,
1758     jint j_sdp_mline_index, jstring j_candidate_sdp) {
1759   std::string sdp_mid = JavaToStdString(jni, j_sdp_mid);
1760   std::string sdp = JavaToStdString(jni, j_candidate_sdp);
1761   scoped_ptr<IceCandidateInterface> candidate(
1762       webrtc::CreateIceCandidate(sdp_mid, j_sdp_mline_index, sdp, NULL));
1763   return ExtractNativePC(jni, j_pc)->AddIceCandidate(candidate.get());
1764 }
1765 
JOW(jboolean,PeerConnection_nativeAddLocalStream)1766 JOW(jboolean, PeerConnection_nativeAddLocalStream)(
1767     JNIEnv* jni, jobject j_pc, jlong native_stream) {
1768   return ExtractNativePC(jni, j_pc)->AddStream(
1769       reinterpret_cast<MediaStreamInterface*>(native_stream));
1770 }
1771 
JOW(void,PeerConnection_nativeRemoveLocalStream)1772 JOW(void, PeerConnection_nativeRemoveLocalStream)(
1773     JNIEnv* jni, jobject j_pc, jlong native_stream) {
1774   ExtractNativePC(jni, j_pc)->RemoveStream(
1775       reinterpret_cast<MediaStreamInterface*>(native_stream));
1776 }
1777 
JOW(jobject,PeerConnection_nativeCreateSender)1778 JOW(jobject, PeerConnection_nativeCreateSender)(
1779     JNIEnv* jni, jobject j_pc, jstring j_kind, jstring j_stream_id) {
1780   jclass j_rtp_sender_class = FindClass(jni, "org/webrtc/RtpSender");
1781   jmethodID j_rtp_sender_ctor =
1782       GetMethodID(jni, j_rtp_sender_class, "<init>", "(J)V");
1783 
1784   std::string kind = JavaToStdString(jni, j_kind);
1785   std::string stream_id = JavaToStdString(jni, j_stream_id);
1786   rtc::scoped_refptr<RtpSenderInterface> sender =
1787       ExtractNativePC(jni, j_pc)->CreateSender(kind, stream_id);
1788   if (!sender.get()) {
1789     return nullptr;
1790   }
1791   jlong nativeSenderPtr = jlongFromPointer(sender.get());
1792   jobject j_sender =
1793       jni->NewObject(j_rtp_sender_class, j_rtp_sender_ctor, nativeSenderPtr);
1794   CHECK_EXCEPTION(jni) << "error during NewObject";
1795   // Sender is now owned by the Java object, and will be freed from
1796   // RtpSender.dispose(), called by PeerConnection.dispose() or getSenders().
1797   sender->AddRef();
1798   return j_sender;
1799 }
1800 
JOW(jobject,PeerConnection_nativeGetSenders)1801 JOW(jobject, PeerConnection_nativeGetSenders)(JNIEnv* jni, jobject j_pc) {
1802   jclass j_array_list_class = FindClass(jni, "java/util/ArrayList");
1803   jmethodID j_array_list_ctor =
1804       GetMethodID(jni, j_array_list_class, "<init>", "()V");
1805   jmethodID j_array_list_add =
1806       GetMethodID(jni, j_array_list_class, "add", "(Ljava/lang/Object;)Z");
1807   jobject j_senders = jni->NewObject(j_array_list_class, j_array_list_ctor);
1808   CHECK_EXCEPTION(jni) << "error during NewObject";
1809 
1810   jclass j_rtp_sender_class = FindClass(jni, "org/webrtc/RtpSender");
1811   jmethodID j_rtp_sender_ctor =
1812       GetMethodID(jni, j_rtp_sender_class, "<init>", "(J)V");
1813 
1814   auto senders = ExtractNativePC(jni, j_pc)->GetSenders();
1815   for (const auto& sender : senders) {
1816     jlong nativeSenderPtr = jlongFromPointer(sender.get());
1817     jobject j_sender =
1818         jni->NewObject(j_rtp_sender_class, j_rtp_sender_ctor, nativeSenderPtr);
1819     CHECK_EXCEPTION(jni) << "error during NewObject";
1820     // Sender is now owned by the Java object, and will be freed from
1821     // RtpSender.dispose(), called by PeerConnection.dispose() or getSenders().
1822     sender->AddRef();
1823     jni->CallBooleanMethod(j_senders, j_array_list_add, j_sender);
1824     CHECK_EXCEPTION(jni) << "error during CallBooleanMethod";
1825   }
1826   return j_senders;
1827 }
1828 
JOW(jobject,PeerConnection_nativeGetReceivers)1829 JOW(jobject, PeerConnection_nativeGetReceivers)(JNIEnv* jni, jobject j_pc) {
1830   jclass j_array_list_class = FindClass(jni, "java/util/ArrayList");
1831   jmethodID j_array_list_ctor =
1832       GetMethodID(jni, j_array_list_class, "<init>", "()V");
1833   jmethodID j_array_list_add =
1834       GetMethodID(jni, j_array_list_class, "add", "(Ljava/lang/Object;)Z");
1835   jobject j_receivers = jni->NewObject(j_array_list_class, j_array_list_ctor);
1836   CHECK_EXCEPTION(jni) << "error during NewObject";
1837 
1838   jclass j_rtp_receiver_class = FindClass(jni, "org/webrtc/RtpReceiver");
1839   jmethodID j_rtp_receiver_ctor =
1840       GetMethodID(jni, j_rtp_receiver_class, "<init>", "(J)V");
1841 
1842   auto receivers = ExtractNativePC(jni, j_pc)->GetReceivers();
1843   for (const auto& receiver : receivers) {
1844     jlong nativeReceiverPtr = jlongFromPointer(receiver.get());
1845     jobject j_receiver = jni->NewObject(j_rtp_receiver_class,
1846                                         j_rtp_receiver_ctor, nativeReceiverPtr);
1847     CHECK_EXCEPTION(jni) << "error during NewObject";
1848     // Receiver is now owned by Java object, and will be freed from there.
1849     receiver->AddRef();
1850     jni->CallBooleanMethod(j_receivers, j_array_list_add, j_receiver);
1851     CHECK_EXCEPTION(jni) << "error during CallBooleanMethod";
1852   }
1853   return j_receivers;
1854 }
1855 
JOW(bool,PeerConnection_nativeGetStats)1856 JOW(bool, PeerConnection_nativeGetStats)(
1857     JNIEnv* jni, jobject j_pc, jobject j_observer, jlong native_track) {
1858   rtc::scoped_refptr<StatsObserverWrapper> observer(
1859       new rtc::RefCountedObject<StatsObserverWrapper>(jni, j_observer));
1860   return ExtractNativePC(jni, j_pc)->GetStats(
1861       observer,
1862       reinterpret_cast<MediaStreamTrackInterface*>(native_track),
1863       PeerConnectionInterface::kStatsOutputLevelStandard);
1864 }
1865 
JOW(jobject,PeerConnection_signalingState)1866 JOW(jobject, PeerConnection_signalingState)(JNIEnv* jni, jobject j_pc) {
1867   PeerConnectionInterface::SignalingState state =
1868       ExtractNativePC(jni, j_pc)->signaling_state();
1869   return JavaEnumFromIndex(jni, "PeerConnection$SignalingState", state);
1870 }
1871 
JOW(jobject,PeerConnection_iceConnectionState)1872 JOW(jobject, PeerConnection_iceConnectionState)(JNIEnv* jni, jobject j_pc) {
1873   PeerConnectionInterface::IceConnectionState state =
1874       ExtractNativePC(jni, j_pc)->ice_connection_state();
1875   return JavaEnumFromIndex(jni, "PeerConnection$IceConnectionState", state);
1876 }
1877 
JOW(jobject,PeerConnection_iceGatheringState)1878 JOW(jobject, PeerConnection_iceGatheringState)(JNIEnv* jni, jobject j_pc) {
1879   PeerConnectionInterface::IceGatheringState state =
1880       ExtractNativePC(jni, j_pc)->ice_gathering_state();
1881   return JavaEnumFromIndex(jni, "PeerConnection$IceGatheringState", state);
1882 }
1883 
JOW(void,PeerConnection_close)1884 JOW(void, PeerConnection_close)(JNIEnv* jni, jobject j_pc) {
1885   ExtractNativePC(jni, j_pc)->Close();
1886   return;
1887 }
1888 
JOW(jobject,MediaSource_nativeState)1889 JOW(jobject, MediaSource_nativeState)(JNIEnv* jni, jclass, jlong j_p) {
1890   rtc::scoped_refptr<MediaSourceInterface> p(
1891       reinterpret_cast<MediaSourceInterface*>(j_p));
1892   return JavaEnumFromIndex(jni, "MediaSource$State", p->state());
1893 }
1894 
JOW(jobject,VideoCapturer_nativeCreateVideoCapturer)1895 JOW(jobject, VideoCapturer_nativeCreateVideoCapturer)(
1896     JNIEnv* jni, jclass, jstring j_device_name) {
1897 // Since we can't create platform specific java implementations in Java, we
1898 // defer the creation to C land.
1899 #if defined(ANDROID)
1900   // TODO(nisse): This case is intended to be deleted.
1901   jclass j_video_capturer_class(
1902       FindClass(jni, "org/webrtc/VideoCapturerAndroid"));
1903   const int camera_id = jni->CallStaticIntMethod(
1904       j_video_capturer_class,
1905       GetStaticMethodID(jni, j_video_capturer_class, "lookupDeviceName",
1906                         "(Ljava/lang/String;)I"),
1907       j_device_name);
1908   CHECK_EXCEPTION(jni) << "error during VideoCapturerAndroid.lookupDeviceName";
1909   if (camera_id == -1)
1910     return nullptr;
1911   jobject j_video_capturer = jni->NewObject(
1912       j_video_capturer_class,
1913       GetMethodID(jni, j_video_capturer_class, "<init>", "(I)V"), camera_id);
1914   CHECK_EXCEPTION(jni) << "error during creation of VideoCapturerAndroid";
1915   jfieldID helper_fid = GetFieldID(jni, j_video_capturer_class, "surfaceHelper",
1916                                    "Lorg/webrtc/SurfaceTextureHelper;");
1917 
1918   rtc::scoped_refptr<webrtc::AndroidVideoCapturerDelegate> delegate =
1919       new rtc::RefCountedObject<AndroidVideoCapturerJni>(
1920           jni, j_video_capturer,
1921           GetObjectField(jni, j_video_capturer, helper_fid));
1922   rtc::scoped_ptr<cricket::VideoCapturer> capturer(
1923       new webrtc::AndroidVideoCapturer(delegate));
1924 
1925 #else
1926   std::string device_name = JavaToStdString(jni, j_device_name);
1927   scoped_ptr<cricket::DeviceManagerInterface> device_manager(
1928       cricket::DeviceManagerFactory::Create());
1929   RTC_CHECK(device_manager->Init()) << "DeviceManager::Init() failed";
1930   cricket::Device device;
1931   if (!device_manager->GetVideoCaptureDevice(device_name, &device)) {
1932     LOG(LS_ERROR) << "GetVideoCaptureDevice failed for " << device_name;
1933     return 0;
1934   }
1935   scoped_ptr<cricket::VideoCapturer> capturer(
1936       device_manager->CreateVideoCapturer(device));
1937 
1938   jclass j_video_capturer_class(
1939       FindClass(jni, "org/webrtc/VideoCapturer"));
1940   const jmethodID j_videocapturer_ctor(GetMethodID(
1941       jni, j_video_capturer_class, "<init>", "()V"));
1942   jobject j_video_capturer =
1943       jni->NewObject(j_video_capturer_class,
1944                      j_videocapturer_ctor);
1945   CHECK_EXCEPTION(jni) << "error during creation of VideoCapturer";
1946 
1947 #endif
1948   const jmethodID j_videocapturer_set_native_capturer(GetMethodID(
1949       jni, j_video_capturer_class, "setNativeCapturer", "(J)V"));
1950   jni->CallVoidMethod(j_video_capturer,
1951                       j_videocapturer_set_native_capturer,
1952                       jlongFromPointer(capturer.release()));
1953   CHECK_EXCEPTION(jni) << "error during setNativeCapturer";
1954   return j_video_capturer;
1955 }
1956 
JOW(jlong,VideoRenderer_nativeCreateGuiVideoRenderer)1957 JOW(jlong, VideoRenderer_nativeCreateGuiVideoRenderer)(
1958     JNIEnv* jni, jclass, int x, int y) {
1959   scoped_ptr<VideoRendererWrapper> renderer(VideoRendererWrapper::Create(
1960       cricket::VideoRendererFactory::CreateGuiVideoRenderer(x, y)));
1961   return (jlong)renderer.release();
1962 }
1963 
JOW(jlong,VideoRenderer_nativeWrapVideoRenderer)1964 JOW(jlong, VideoRenderer_nativeWrapVideoRenderer)(
1965     JNIEnv* jni, jclass, jobject j_callbacks) {
1966   scoped_ptr<JavaVideoRendererWrapper> renderer(
1967       new JavaVideoRendererWrapper(jni, j_callbacks));
1968   return (jlong)renderer.release();
1969 }
1970 
JOW(void,VideoRenderer_nativeCopyPlane)1971 JOW(void, VideoRenderer_nativeCopyPlane)(
1972     JNIEnv *jni, jclass, jobject j_src_buffer, jint width, jint height,
1973     jint src_stride, jobject j_dst_buffer, jint dst_stride) {
1974   size_t src_size = jni->GetDirectBufferCapacity(j_src_buffer);
1975   size_t dst_size = jni->GetDirectBufferCapacity(j_dst_buffer);
1976   RTC_CHECK(src_stride >= width) << "Wrong source stride " << src_stride;
1977   RTC_CHECK(dst_stride >= width) << "Wrong destination stride " << dst_stride;
1978   RTC_CHECK(src_size >= src_stride * height)
1979       << "Insufficient source buffer capacity " << src_size;
1980   RTC_CHECK(dst_size >= dst_stride * height)
1981       << "Isufficient destination buffer capacity " << dst_size;
1982   uint8_t *src =
1983       reinterpret_cast<uint8_t*>(jni->GetDirectBufferAddress(j_src_buffer));
1984   uint8_t *dst =
1985       reinterpret_cast<uint8_t*>(jni->GetDirectBufferAddress(j_dst_buffer));
1986   if (src_stride == dst_stride) {
1987     memcpy(dst, src, src_stride * height);
1988   } else {
1989     for (int i = 0; i < height; i++) {
1990       memcpy(dst, src, width);
1991       src += src_stride;
1992       dst += dst_stride;
1993     }
1994   }
1995 }
1996 
JOW(void,VideoSource_stop)1997 JOW(void, VideoSource_stop)(JNIEnv* jni, jclass, jlong j_p) {
1998   reinterpret_cast<VideoSourceInterface*>(j_p)->Stop();
1999 }
2000 
JOW(void,VideoSource_restart)2001 JOW(void, VideoSource_restart)(
2002     JNIEnv* jni, jclass, jlong j_p_source, jlong j_p_format) {
2003   reinterpret_cast<VideoSourceInterface*>(j_p_source)->Restart();
2004 }
2005 
JOW(jstring,MediaStreamTrack_nativeId)2006 JOW(jstring, MediaStreamTrack_nativeId)(JNIEnv* jni, jclass, jlong j_p) {
2007   return JavaStringFromStdString(
2008       jni, reinterpret_cast<MediaStreamTrackInterface*>(j_p)->id());
2009 }
2010 
JOW(jstring,MediaStreamTrack_nativeKind)2011 JOW(jstring, MediaStreamTrack_nativeKind)(JNIEnv* jni, jclass, jlong j_p) {
2012   return JavaStringFromStdString(
2013       jni, reinterpret_cast<MediaStreamTrackInterface*>(j_p)->kind());
2014 }
2015 
JOW(jboolean,MediaStreamTrack_nativeEnabled)2016 JOW(jboolean, MediaStreamTrack_nativeEnabled)(JNIEnv* jni, jclass, jlong j_p) {
2017   return reinterpret_cast<MediaStreamTrackInterface*>(j_p)->enabled();
2018 }
2019 
JOW(jobject,MediaStreamTrack_nativeState)2020 JOW(jobject, MediaStreamTrack_nativeState)(JNIEnv* jni, jclass, jlong j_p) {
2021   return JavaEnumFromIndex(
2022       jni,
2023       "MediaStreamTrack$State",
2024       reinterpret_cast<MediaStreamTrackInterface*>(j_p)->state());
2025 }
2026 
JOW(jboolean,MediaStreamTrack_nativeSetState)2027 JOW(jboolean, MediaStreamTrack_nativeSetState)(
2028     JNIEnv* jni, jclass, jlong j_p, jint j_new_state) {
2029   MediaStreamTrackInterface::TrackState new_state =
2030       (MediaStreamTrackInterface::TrackState)j_new_state;
2031   return reinterpret_cast<MediaStreamTrackInterface*>(j_p)
2032       ->set_state(new_state);
2033 }
2034 
JOW(jboolean,MediaStreamTrack_nativeSetEnabled)2035 JOW(jboolean, MediaStreamTrack_nativeSetEnabled)(
2036     JNIEnv* jni, jclass, jlong j_p, jboolean enabled) {
2037   return reinterpret_cast<MediaStreamTrackInterface*>(j_p)
2038       ->set_enabled(enabled);
2039 }
2040 
JOW(void,VideoTrack_nativeAddRenderer)2041 JOW(void, VideoTrack_nativeAddRenderer)(
2042     JNIEnv* jni, jclass,
2043     jlong j_video_track_pointer, jlong j_renderer_pointer) {
2044   reinterpret_cast<VideoTrackInterface*>(j_video_track_pointer)->AddRenderer(
2045       reinterpret_cast<VideoRendererInterface*>(j_renderer_pointer));
2046 }
2047 
JOW(void,VideoTrack_nativeRemoveRenderer)2048 JOW(void, VideoTrack_nativeRemoveRenderer)(
2049     JNIEnv* jni, jclass,
2050     jlong j_video_track_pointer, jlong j_renderer_pointer) {
2051   reinterpret_cast<VideoTrackInterface*>(j_video_track_pointer)->RemoveRenderer(
2052       reinterpret_cast<VideoRendererInterface*>(j_renderer_pointer));
2053 }
2054 
JOW(jlong,CallSessionFileRotatingLogSink_nativeAddSink)2055 JOW(jlong, CallSessionFileRotatingLogSink_nativeAddSink)(
2056     JNIEnv* jni, jclass,
2057     jstring j_dirPath, jint j_maxFileSize, jint j_severity) {
2058   std::string dir_path = JavaToStdString(jni, j_dirPath);
2059   rtc::CallSessionFileRotatingLogSink* sink =
2060       new rtc::CallSessionFileRotatingLogSink(dir_path, j_maxFileSize);
2061   if (!sink->Init()) {
2062     LOG_V(rtc::LoggingSeverity::LS_WARNING) <<
2063         "Failed to init CallSessionFileRotatingLogSink for path " << dir_path;
2064     delete sink;
2065     return 0;
2066   }
2067   rtc::LogMessage::AddLogToStream(
2068       sink, static_cast<rtc::LoggingSeverity>(j_severity));
2069   return (jlong) sink;
2070 }
2071 
JOW(void,CallSessionFileRotatingLogSink_nativeDeleteSink)2072 JOW(void, CallSessionFileRotatingLogSink_nativeDeleteSink)(
2073     JNIEnv* jni, jclass, jlong j_sink) {
2074   rtc::CallSessionFileRotatingLogSink* sink =
2075       reinterpret_cast<rtc::CallSessionFileRotatingLogSink*>(j_sink);
2076   rtc::LogMessage::RemoveLogToStream(sink);
2077   delete sink;
2078 }
2079 
JOW(jbyteArray,CallSessionFileRotatingLogSink_nativeGetLogData)2080 JOW(jbyteArray, CallSessionFileRotatingLogSink_nativeGetLogData)(
2081     JNIEnv* jni, jclass, jstring j_dirPath) {
2082   std::string dir_path = JavaToStdString(jni, j_dirPath);
2083   rtc::scoped_ptr<rtc::CallSessionFileRotatingStream> stream(
2084       new rtc::CallSessionFileRotatingStream(dir_path));
2085   if (!stream->Open()) {
2086     LOG_V(rtc::LoggingSeverity::LS_WARNING) <<
2087         "Failed to open CallSessionFileRotatingStream for path " << dir_path;
2088     return jni->NewByteArray(0);
2089   }
2090   size_t log_size = 0;
2091   if (!stream->GetSize(&log_size) || log_size == 0) {
2092     LOG_V(rtc::LoggingSeverity::LS_WARNING) <<
2093         "CallSessionFileRotatingStream returns 0 size for path " << dir_path;
2094     return jni->NewByteArray(0);
2095   }
2096 
2097   size_t read = 0;
2098   rtc::scoped_ptr<jbyte> buffer(static_cast<jbyte*>(malloc(log_size)));
2099   stream->ReadAll(buffer.get(), log_size, &read, nullptr);
2100 
2101   jbyteArray result = jni->NewByteArray(read);
2102   jni->SetByteArrayRegion(result, 0, read, buffer.get());
2103 
2104   return result;
2105 }
2106 
JOW(jboolean,RtpSender_nativeSetTrack)2107 JOW(jboolean, RtpSender_nativeSetTrack)(JNIEnv* jni,
2108                                     jclass,
2109                                     jlong j_rtp_sender_pointer,
2110                                     jlong j_track_pointer) {
2111   return reinterpret_cast<RtpSenderInterface*>(j_rtp_sender_pointer)
2112       ->SetTrack(reinterpret_cast<MediaStreamTrackInterface*>(j_track_pointer));
2113 }
2114 
JOW(jlong,RtpSender_nativeGetTrack)2115 JOW(jlong, RtpSender_nativeGetTrack)(JNIEnv* jni,
2116                                   jclass,
2117                                   jlong j_rtp_sender_pointer,
2118                                   jlong j_track_pointer) {
2119   return jlongFromPointer(
2120       reinterpret_cast<RtpSenderInterface*>(j_rtp_sender_pointer)
2121           ->track()
2122           .release());
2123 }
2124 
JOW(jstring,RtpSender_nativeId)2125 JOW(jstring, RtpSender_nativeId)(
2126     JNIEnv* jni, jclass, jlong j_rtp_sender_pointer) {
2127   return JavaStringFromStdString(
2128       jni, reinterpret_cast<RtpSenderInterface*>(j_rtp_sender_pointer)->id());
2129 }
2130 
JOW(void,RtpSender_free)2131 JOW(void, RtpSender_free)(JNIEnv* jni, jclass, jlong j_rtp_sender_pointer) {
2132   reinterpret_cast<RtpSenderInterface*>(j_rtp_sender_pointer)->Release();
2133 }
2134 
JOW(jlong,RtpReceiver_nativeGetTrack)2135 JOW(jlong, RtpReceiver_nativeGetTrack)(JNIEnv* jni,
2136                                     jclass,
2137                                     jlong j_rtp_receiver_pointer,
2138                                     jlong j_track_pointer) {
2139   return jlongFromPointer(
2140       reinterpret_cast<RtpReceiverInterface*>(j_rtp_receiver_pointer)
2141           ->track()
2142           .release());
2143 }
2144 
JOW(jstring,RtpReceiver_nativeId)2145 JOW(jstring, RtpReceiver_nativeId)(
2146     JNIEnv* jni, jclass, jlong j_rtp_receiver_pointer) {
2147   return JavaStringFromStdString(
2148       jni,
2149       reinterpret_cast<RtpReceiverInterface*>(j_rtp_receiver_pointer)->id());
2150 }
2151 
JOW(void,RtpReceiver_free)2152 JOW(void, RtpReceiver_free)(JNIEnv* jni, jclass, jlong j_rtp_receiver_pointer) {
2153   reinterpret_cast<RtpReceiverInterface*>(j_rtp_receiver_pointer)->Release();
2154 }
2155 
2156 }  // namespace webrtc_jni
2157