blob: 33b08907be4301c5c794677b988bfa88be070dad [file] [log] [blame]
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001/*
2 * libjingle
jlmiller@webrtc.org5f93d0a2015-01-20 21:36:13 +00003 * Copyright 2013 Google Inc.
henrike@webrtc.org28e20752013-07-10 00:45:36 +00004 *
5 * Redistribution and use in source and binary forms, with or without
6 * modification, are permitted provided that the following conditions are met:
7 *
8 * 1. Redistributions of source code must retain the above copyright notice,
9 * this list of conditions and the following disclaimer.
10 * 2. Redistributions in binary form must reproduce the above copyright notice,
11 * this list of conditions and the following disclaimer in the documentation
12 * and/or other materials provided with the distribution.
13 * 3. The name of the author may not be used to endorse or promote products
14 * derived from this software without specific prior written permission.
15 *
16 * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR IMPLIED
17 * WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
18 * MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO
19 * EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
20 * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
21 * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS;
22 * OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY,
23 * WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR
24 * OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF
25 * ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
26 */
27
28// 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
buildbot@webrtc.orga09a9992014-08-13 17:26:08 +000059#include <limits>
henrike@webrtc.org28e20752013-07-10 00:45:36 +000060
perkj@webrtc.org96e4db92015-02-13 12:46:51 +000061#include "talk/app/webrtc/java/jni/classreferenceholder.h"
62#include "talk/app/webrtc/java/jni/jni_helpers.h"
glaznev@webrtc.org18c92472015-02-18 18:42:55 +000063#include "talk/app/webrtc/java/jni/native_handle_impl.h"
glaznev97579a42015-09-01 11:31:27 -070064#include "talk/app/webrtc/dtlsidentitystore.h"
henrike@webrtc.org28e20752013-07-10 00:45:36 +000065#include "talk/app/webrtc/mediaconstraintsinterface.h"
66#include "talk/app/webrtc/peerconnectioninterface.h"
deadbeef4139c0f2015-10-06 12:29:25 -070067#include "talk/app/webrtc/rtpreceiverinterface.h"
68#include "talk/app/webrtc/rtpsenderinterface.h"
henrike@webrtc.org28e20752013-07-10 00:45:36 +000069#include "talk/app/webrtc/videosourceinterface.h"
henrike@webrtc.org28e20752013-07-10 00:45:36 +000070#include "talk/media/base/videocapturer.h"
71#include "talk/media/base/videorenderer.h"
72#include "talk/media/devices/videorendererfactory.h"
glaznev@webrtc.orgefe4b9a2014-07-22 17:44:53 +000073#include "talk/media/webrtc/webrtcvideodecoderfactory.h"
fischman@webrtc.org540acde2014-02-13 03:56:14 +000074#include "talk/media/webrtc/webrtcvideoencoderfactory.h"
buildbot@webrtc.orga09a9992014-08-13 17:26:08 +000075#include "webrtc/base/bind.h"
andresp@webrtc.org4d19e052014-09-09 11:45:44 +000076#include "webrtc/base/checks.h"
buildbot@webrtc.orga09a9992014-08-13 17:26:08 +000077#include "webrtc/base/logging.h"
Jiayang Liue63d2a12015-09-01 16:11:18 -070078#include "webrtc/base/logsinks.h"
buildbot@webrtc.orga09a9992014-08-13 17:26:08 +000079#include "webrtc/base/messagequeue.h"
80#include "webrtc/base/ssladapter.h"
glaznev@webrtc.org44ae4c82015-02-09 23:25:58 +000081#include "webrtc/base/stringutils.h"
glaznev@webrtc.org44ae4c82015-02-09 23:25:58 +000082#include "webrtc/system_wrappers/interface/field_trial_default.h"
henrike@webrtc.org9de257d2013-07-17 14:42:53 +000083#include "webrtc/system_wrappers/interface/trace.h"
henrike@webrtc.org9de257d2013-07-17 14:42:53 +000084#include "webrtc/voice_engine/include/voe_base.h"
henrike@webrtc.org28e20752013-07-10 00:45:36 +000085
glaznev@webrtc.org99678452014-09-15 17:52:42 +000086#if defined(ANDROID) && !defined(WEBRTC_CHROMIUM_BUILD)
perkj@webrtc.org83bc7212015-02-11 11:26:56 +000087#include "talk/app/webrtc/androidvideocapturer.h"
glaznev@webrtc.org18c92472015-02-18 18:42:55 +000088#include "talk/app/webrtc/java/jni/androidmediadecoder_jni.h"
89#include "talk/app/webrtc/java/jni/androidmediaencoder_jni.h"
perkj@webrtc.org96e4db92015-02-13 12:46:51 +000090#include "talk/app/webrtc/java/jni/androidvideocapturer_jni.h"
andresp@webrtc.org85ef7702014-09-17 11:44:51 +000091#include "webrtc/modules/video_render/video_render_internal.h"
fischman@webrtc.org7e4d0df2013-10-01 02:40:43 +000092#include "webrtc/system_wrappers/interface/logcat_trace_context.h"
93using webrtc::LogcatTraceContext;
94#endif
95
Alex Glaznev4d2f4d12015-09-01 15:04:13 -070096using cricket::WebRtcVideoDecoderFactory;
97using cricket::WebRtcVideoEncoderFactory;
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +000098using rtc::Bind;
99using rtc::Thread;
100using rtc::ThreadManager;
101using rtc::scoped_ptr;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000102using webrtc::AudioSourceInterface;
103using webrtc::AudioTrackInterface;
104using webrtc::AudioTrackVector;
105using webrtc::CreateSessionDescriptionObserver;
henrike@webrtc.org723d6832013-07-12 16:04:50 +0000106using webrtc::DataBuffer;
107using webrtc::DataChannelInit;
108using webrtc::DataChannelInterface;
109using webrtc::DataChannelObserver;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000110using webrtc::IceCandidateInterface;
111using webrtc::MediaConstraintsInterface;
112using webrtc::MediaSourceInterface;
113using webrtc::MediaStreamInterface;
114using webrtc::MediaStreamTrackInterface;
115using webrtc::PeerConnectionFactoryInterface;
116using webrtc::PeerConnectionInterface;
117using webrtc::PeerConnectionObserver;
deadbeef4139c0f2015-10-06 12:29:25 -0700118using webrtc::RtpReceiverInterface;
119using webrtc::RtpSenderInterface;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000120using webrtc::SessionDescriptionInterface;
121using webrtc::SetSessionDescriptionObserver;
122using webrtc::StatsObserver;
123using webrtc::StatsReport;
tommi@webrtc.orge2e199b2014-12-15 13:22:54 +0000124using webrtc::StatsReports;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000125using webrtc::VideoRendererInterface;
126using webrtc::VideoSourceInterface;
127using webrtc::VideoTrackInterface;
128using webrtc::VideoTrackVector;
fischman@webrtc.org540acde2014-02-13 03:56:14 +0000129using webrtc::kVideoCodecVP8;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000130
perkj@webrtc.org96e4db92015-02-13 12:46:51 +0000131namespace webrtc_jni {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000132
glaznev@webrtc.org44ae4c82015-02-09 23:25:58 +0000133// Field trials initialization string
134static char *field_trials_init_string = NULL;
135
glaznev@webrtc.org99678452014-09-15 17:52:42 +0000136#if defined(ANDROID) && !defined(WEBRTC_CHROMIUM_BUILD)
137// Set in PeerConnectionFactory_initializeAndroidGlobals().
138static bool factory_static_initialized = false;
Alex Glaznev4d2f4d12015-09-01 15:04:13 -0700139static bool video_hw_acceleration_enabled = true;
glaznev@webrtc.org99678452014-09-15 17:52:42 +0000140#endif
141
perkj@webrtc.org83bc7212015-02-11 11:26:56 +0000142extern "C" jint JNIEXPORT JNICALL JNI_OnLoad(JavaVM *jvm, void *reserved) {
perkj@webrtc.org96e4db92015-02-13 12:46:51 +0000143 jint ret = InitGlobalJniVariables(jvm);
144 if (ret < 0)
145 return -1;
perkj@webrtc.org83bc7212015-02-11 11:26:56 +0000146
henrikg91d6ede2015-09-17 00:24:34 -0700147 RTC_CHECK(rtc::InitializeSSL()) << "Failed to InitializeSSL()";
perkj@webrtc.org96e4db92015-02-13 12:46:51 +0000148 LoadGlobalClassReferenceHolder();
perkj@webrtc.org83bc7212015-02-11 11:26:56 +0000149
perkj@webrtc.org96e4db92015-02-13 12:46:51 +0000150 return ret;
perkj@webrtc.org83bc7212015-02-11 11:26:56 +0000151}
152
153extern "C" void JNIEXPORT JNICALL JNI_OnUnLoad(JavaVM *jvm, void *reserved) {
perkj@webrtc.org96e4db92015-02-13 12:46:51 +0000154 FreeGlobalClassReferenceHolder();
henrikg91d6ede2015-09-17 00:24:34 -0700155 RTC_CHECK(rtc::CleanupSSL()) << "Failed to CleanupSSL()";
fischman@webrtc.org540acde2014-02-13 03:56:14 +0000156}
157
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000158// Return the (singleton) Java Enum object corresponding to |index|;
159// |state_class_fragment| is something like "MediaSource$State".
glaznev@webrtc.orgb28474c2015-02-23 17:44:27 +0000160static jobject JavaEnumFromIndex(
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000161 JNIEnv* jni, const std::string& state_class_fragment, int index) {
perkj@webrtc.org96e4db92015-02-13 12:46:51 +0000162 const std::string state_class = "org/webrtc/" + state_class_fragment;
163 return JavaEnumFromIndex(jni, FindClass(jni, state_class.c_str()),
164 state_class, index);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000165}
166
henrike@webrtc.org723d6832013-07-12 16:04:50 +0000167static DataChannelInit JavaDataChannelInitToNative(
168 JNIEnv* jni, jobject j_init) {
169 DataChannelInit init;
170
171 jclass j_init_class = FindClass(jni, "org/webrtc/DataChannel$Init");
172 jfieldID ordered_id = GetFieldID(jni, j_init_class, "ordered", "Z");
173 jfieldID max_retransmit_time_id =
174 GetFieldID(jni, j_init_class, "maxRetransmitTimeMs", "I");
175 jfieldID max_retransmits_id =
176 GetFieldID(jni, j_init_class, "maxRetransmits", "I");
177 jfieldID protocol_id =
178 GetFieldID(jni, j_init_class, "protocol", "Ljava/lang/String;");
179 jfieldID negotiated_id = GetFieldID(jni, j_init_class, "negotiated", "Z");
180 jfieldID id_id = GetFieldID(jni, j_init_class, "id", "I");
181
182 init.ordered = GetBooleanField(jni, j_init, ordered_id);
183 init.maxRetransmitTime = GetIntField(jni, j_init, max_retransmit_time_id);
184 init.maxRetransmits = GetIntField(jni, j_init, max_retransmits_id);
185 init.protocol = JavaToStdString(
186 jni, GetStringField(jni, j_init, protocol_id));
187 init.negotiated = GetBooleanField(jni, j_init, negotiated_id);
188 init.id = GetIntField(jni, j_init, id_id);
189
190 return init;
191}
192
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000193class ConstraintsWrapper;
194
195// Adapter between the C++ PeerConnectionObserver interface and the Java
196// PeerConnection.Observer interface. Wraps an instance of the Java interface
197// and dispatches C++ callbacks to Java.
198class PCOJava : public PeerConnectionObserver {
199 public:
200 PCOJava(JNIEnv* jni, jobject j_observer)
201 : j_observer_global_(jni, j_observer),
henrike@webrtc.org723d6832013-07-12 16:04:50 +0000202 j_observer_class_(jni, GetObjectClass(jni, *j_observer_global_)),
203 j_media_stream_class_(jni, FindClass(jni, "org/webrtc/MediaStream")),
204 j_media_stream_ctor_(GetMethodID(
205 jni, *j_media_stream_class_, "<init>", "(J)V")),
206 j_audio_track_class_(jni, FindClass(jni, "org/webrtc/AudioTrack")),
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000207 j_audio_track_ctor_(GetMethodID(
henrike@webrtc.org723d6832013-07-12 16:04:50 +0000208 jni, *j_audio_track_class_, "<init>", "(J)V")),
209 j_video_track_class_(jni, FindClass(jni, "org/webrtc/VideoTrack")),
210 j_video_track_ctor_(GetMethodID(
211 jni, *j_video_track_class_, "<init>", "(J)V")),
212 j_data_channel_class_(jni, FindClass(jni, "org/webrtc/DataChannel")),
213 j_data_channel_ctor_(GetMethodID(
214 jni, *j_data_channel_class_, "<init>", "(J)V")) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000215 }
216
Magnus Jedvert1c3dd382015-08-27 13:39:58 +0200217 virtual ~PCOJava() {
218 ScopedLocalRefFrame local_ref_frame(jni());
219 while (!remote_streams_.empty())
220 DisposeRemoteStream(remote_streams_.begin());
221 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000222
kjellander@webrtc.org14665ff2015-03-04 12:58:35 +0000223 void OnIceCandidate(const IceCandidateInterface* candidate) override {
fischman@webrtc.org41776152014-01-09 00:31:17 +0000224 ScopedLocalRefFrame local_ref_frame(jni());
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000225 std::string sdp;
henrikg91d6ede2015-09-17 00:24:34 -0700226 RTC_CHECK(candidate->ToString(&sdp)) << "got so far: " << sdp;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000227 jclass candidate_class = FindClass(jni(), "org/webrtc/IceCandidate");
228 jmethodID ctor = GetMethodID(jni(), candidate_class,
229 "<init>", "(Ljava/lang/String;ILjava/lang/String;)V");
fischman@webrtc.org41776152014-01-09 00:31:17 +0000230 jstring j_mid = JavaStringFromStdString(jni(), candidate->sdp_mid());
231 jstring j_sdp = JavaStringFromStdString(jni(), sdp);
232 jobject j_candidate = jni()->NewObject(
233 candidate_class, ctor, j_mid, candidate->sdp_mline_index(), j_sdp);
andresp@webrtc.org4d19e052014-09-09 11:45:44 +0000234 CHECK_EXCEPTION(jni()) << "error during NewObject";
henrike@webrtc.org723d6832013-07-12 16:04:50 +0000235 jmethodID m = GetMethodID(jni(), *j_observer_class_,
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000236 "onIceCandidate", "(Lorg/webrtc/IceCandidate;)V");
fischman@webrtc.org41776152014-01-09 00:31:17 +0000237 jni()->CallVoidMethod(*j_observer_global_, m, j_candidate);
andresp@webrtc.org4d19e052014-09-09 11:45:44 +0000238 CHECK_EXCEPTION(jni()) << "error during CallVoidMethod";
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000239 }
240
kjellander@webrtc.org14665ff2015-03-04 12:58:35 +0000241 void OnSignalingChange(
242 PeerConnectionInterface::SignalingState new_state) override {
fischman@webrtc.org41776152014-01-09 00:31:17 +0000243 ScopedLocalRefFrame local_ref_frame(jni());
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000244 jmethodID m = GetMethodID(
henrike@webrtc.org723d6832013-07-12 16:04:50 +0000245 jni(), *j_observer_class_, "onSignalingChange",
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000246 "(Lorg/webrtc/PeerConnection$SignalingState;)V");
fischman@webrtc.org41776152014-01-09 00:31:17 +0000247 jobject new_state_enum =
248 JavaEnumFromIndex(jni(), "PeerConnection$SignalingState", new_state);
249 jni()->CallVoidMethod(*j_observer_global_, m, new_state_enum);
andresp@webrtc.org4d19e052014-09-09 11:45:44 +0000250 CHECK_EXCEPTION(jni()) << "error during CallVoidMethod";
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000251 }
252
kjellander@webrtc.org14665ff2015-03-04 12:58:35 +0000253 void OnIceConnectionChange(
254 PeerConnectionInterface::IceConnectionState new_state) override {
fischman@webrtc.org41776152014-01-09 00:31:17 +0000255 ScopedLocalRefFrame local_ref_frame(jni());
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000256 jmethodID m = GetMethodID(
henrike@webrtc.org723d6832013-07-12 16:04:50 +0000257 jni(), *j_observer_class_, "onIceConnectionChange",
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000258 "(Lorg/webrtc/PeerConnection$IceConnectionState;)V");
fischman@webrtc.org41776152014-01-09 00:31:17 +0000259 jobject new_state_enum = JavaEnumFromIndex(
260 jni(), "PeerConnection$IceConnectionState", new_state);
261 jni()->CallVoidMethod(*j_observer_global_, m, new_state_enum);
andresp@webrtc.org4d19e052014-09-09 11:45:44 +0000262 CHECK_EXCEPTION(jni()) << "error during CallVoidMethod";
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000263 }
264
Peter Thatcher54360512015-07-08 11:08:35 -0700265 void OnIceConnectionReceivingChange(bool receiving) override {
266 ScopedLocalRefFrame local_ref_frame(jni());
267 jmethodID m = GetMethodID(
268 jni(), *j_observer_class_, "onIceConnectionReceivingChange", "(Z)V");
269 jni()->CallVoidMethod(*j_observer_global_, m, receiving);
270 CHECK_EXCEPTION(jni()) << "error during CallVoidMethod";
271 }
272
kjellander@webrtc.org14665ff2015-03-04 12:58:35 +0000273 void OnIceGatheringChange(
274 PeerConnectionInterface::IceGatheringState new_state) override {
fischman@webrtc.org41776152014-01-09 00:31:17 +0000275 ScopedLocalRefFrame local_ref_frame(jni());
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000276 jmethodID m = GetMethodID(
henrike@webrtc.org723d6832013-07-12 16:04:50 +0000277 jni(), *j_observer_class_, "onIceGatheringChange",
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000278 "(Lorg/webrtc/PeerConnection$IceGatheringState;)V");
fischman@webrtc.org41776152014-01-09 00:31:17 +0000279 jobject new_state_enum = JavaEnumFromIndex(
280 jni(), "PeerConnection$IceGatheringState", new_state);
281 jni()->CallVoidMethod(*j_observer_global_, m, new_state_enum);
andresp@webrtc.org4d19e052014-09-09 11:45:44 +0000282 CHECK_EXCEPTION(jni()) << "error during CallVoidMethod";
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000283 }
284
kjellander@webrtc.org14665ff2015-03-04 12:58:35 +0000285 void OnAddStream(MediaStreamInterface* stream) override {
fischman@webrtc.org41776152014-01-09 00:31:17 +0000286 ScopedLocalRefFrame local_ref_frame(jni());
Magnus Jedvert1c3dd382015-08-27 13:39:58 +0200287 // Java MediaStream holds one reference. Corresponding Release() is in
288 // MediaStream_free, triggered by MediaStream.dispose().
289 stream->AddRef();
290 jobject j_stream =
291 jni()->NewObject(*j_media_stream_class_, j_media_stream_ctor_,
292 reinterpret_cast<jlong>(stream));
andresp@webrtc.org4d19e052014-09-09 11:45:44 +0000293 CHECK_EXCEPTION(jni()) << "error during NewObject";
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000294
Magnus Jedvert1c3dd382015-08-27 13:39:58 +0200295 for (const auto& track : stream->GetAudioTracks()) {
fischman@webrtc.org41776152014-01-09 00:31:17 +0000296 jstring id = JavaStringFromStdString(jni(), track->id());
Magnus Jedvert1c3dd382015-08-27 13:39:58 +0200297 // Java AudioTrack holds one reference. Corresponding Release() is in
298 // MediaStreamTrack_free, triggered by AudioTrack.dispose().
299 track->AddRef();
300 jobject j_track =
301 jni()->NewObject(*j_audio_track_class_, j_audio_track_ctor_,
302 reinterpret_cast<jlong>(track.get()), id);
andresp@webrtc.org4d19e052014-09-09 11:45:44 +0000303 CHECK_EXCEPTION(jni()) << "error during NewObject";
fischman@webrtc.org32001ef2013-08-12 23:26:21 +0000304 jfieldID audio_tracks_id = GetFieldID(jni(),
305 *j_media_stream_class_,
306 "audioTracks",
307 "Ljava/util/LinkedList;");
fischman@webrtc.org41776152014-01-09 00:31:17 +0000308 jobject audio_tracks = GetObjectField(jni(), j_stream, audio_tracks_id);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000309 jmethodID add = GetMethodID(jni(),
fischman@webrtc.org41776152014-01-09 00:31:17 +0000310 GetObjectClass(jni(), audio_tracks),
311 "add",
312 "(Ljava/lang/Object;)Z");
313 jboolean added = jni()->CallBooleanMethod(audio_tracks, add, j_track);
andresp@webrtc.org4d19e052014-09-09 11:45:44 +0000314 CHECK_EXCEPTION(jni()) << "error during CallBooleanMethod";
henrikg91d6ede2015-09-17 00:24:34 -0700315 RTC_CHECK(added);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000316 }
317
Magnus Jedvert1c3dd382015-08-27 13:39:58 +0200318 for (const auto& track : stream->GetVideoTracks()) {
fischman@webrtc.org41776152014-01-09 00:31:17 +0000319 jstring id = JavaStringFromStdString(jni(), track->id());
Magnus Jedvert1c3dd382015-08-27 13:39:58 +0200320 // Java VideoTrack holds one reference. Corresponding Release() is in
321 // MediaStreamTrack_free, triggered by VideoTrack.dispose().
322 track->AddRef();
323 jobject j_track =
324 jni()->NewObject(*j_video_track_class_, j_video_track_ctor_,
325 reinterpret_cast<jlong>(track.get()), id);
andresp@webrtc.org4d19e052014-09-09 11:45:44 +0000326 CHECK_EXCEPTION(jni()) << "error during NewObject";
fischman@webrtc.org32001ef2013-08-12 23:26:21 +0000327 jfieldID video_tracks_id = GetFieldID(jni(),
328 *j_media_stream_class_,
329 "videoTracks",
330 "Ljava/util/LinkedList;");
fischman@webrtc.org41776152014-01-09 00:31:17 +0000331 jobject video_tracks = GetObjectField(jni(), j_stream, video_tracks_id);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000332 jmethodID add = GetMethodID(jni(),
fischman@webrtc.org41776152014-01-09 00:31:17 +0000333 GetObjectClass(jni(), video_tracks),
334 "add",
335 "(Ljava/lang/Object;)Z");
336 jboolean added = jni()->CallBooleanMethod(video_tracks, add, j_track);
andresp@webrtc.org4d19e052014-09-09 11:45:44 +0000337 CHECK_EXCEPTION(jni()) << "error during CallBooleanMethod";
henrikg91d6ede2015-09-17 00:24:34 -0700338 RTC_CHECK(added);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000339 }
Magnus Jedvert1c3dd382015-08-27 13:39:58 +0200340 remote_streams_[stream] = NewGlobalRef(jni(), j_stream);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000341
henrike@webrtc.org723d6832013-07-12 16:04:50 +0000342 jmethodID m = GetMethodID(jni(), *j_observer_class_, "onAddStream",
343 "(Lorg/webrtc/MediaStream;)V");
fischman@webrtc.org41776152014-01-09 00:31:17 +0000344 jni()->CallVoidMethod(*j_observer_global_, m, j_stream);
andresp@webrtc.org4d19e052014-09-09 11:45:44 +0000345 CHECK_EXCEPTION(jni()) << "error during CallVoidMethod";
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000346 }
347
kjellander@webrtc.org14665ff2015-03-04 12:58:35 +0000348 void OnRemoveStream(MediaStreamInterface* stream) override {
fischman@webrtc.org41776152014-01-09 00:31:17 +0000349 ScopedLocalRefFrame local_ref_frame(jni());
Magnus Jedvert1c3dd382015-08-27 13:39:58 +0200350 NativeToJavaStreamsMap::iterator it = remote_streams_.find(stream);
henrikg91d6ede2015-09-17 00:24:34 -0700351 RTC_CHECK(it != remote_streams_.end()) << "unexpected stream: " << std::hex
352 << stream;
Magnus Jedvert1c3dd382015-08-27 13:39:58 +0200353 jobject j_stream = it->second;
henrike@webrtc.org723d6832013-07-12 16:04:50 +0000354 jmethodID m = GetMethodID(jni(), *j_observer_class_, "onRemoveStream",
355 "(Lorg/webrtc/MediaStream;)V");
Magnus Jedvert1c3dd382015-08-27 13:39:58 +0200356 jni()->CallVoidMethod(*j_observer_global_, m, j_stream);
andresp@webrtc.org4d19e052014-09-09 11:45:44 +0000357 CHECK_EXCEPTION(jni()) << "error during CallVoidMethod";
Magnus Jedvert1c3dd382015-08-27 13:39:58 +0200358 DisposeRemoteStream(it);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000359 }
360
kjellander@webrtc.org14665ff2015-03-04 12:58:35 +0000361 void OnDataChannel(DataChannelInterface* channel) override {
fischman@webrtc.org41776152014-01-09 00:31:17 +0000362 ScopedLocalRefFrame local_ref_frame(jni());
363 jobject j_channel = jni()->NewObject(
364 *j_data_channel_class_, j_data_channel_ctor_, (jlong)channel);
andresp@webrtc.org4d19e052014-09-09 11:45:44 +0000365 CHECK_EXCEPTION(jni()) << "error during NewObject";
henrike@webrtc.org723d6832013-07-12 16:04:50 +0000366
367 jmethodID m = GetMethodID(jni(), *j_observer_class_, "onDataChannel",
368 "(Lorg/webrtc/DataChannel;)V");
fischman@webrtc.org41776152014-01-09 00:31:17 +0000369 jni()->CallVoidMethod(*j_observer_global_, m, j_channel);
fischman@webrtc.org32001ef2013-08-12 23:26:21 +0000370
371 // Channel is now owned by Java object, and will be freed from
372 // DataChannel.dispose(). Important that this be done _after_ the
373 // CallVoidMethod above as Java code might call back into native code and be
374 // surprised to see a refcount of 2.
375 int bumped_count = channel->AddRef();
henrikg91d6ede2015-09-17 00:24:34 -0700376 RTC_CHECK(bumped_count == 2) << "Unexpected refcount OnDataChannel";
fischman@webrtc.org32001ef2013-08-12 23:26:21 +0000377
andresp@webrtc.org4d19e052014-09-09 11:45:44 +0000378 CHECK_EXCEPTION(jni()) << "error during CallVoidMethod";
henrike@webrtc.org723d6832013-07-12 16:04:50 +0000379 }
380
kjellander@webrtc.org14665ff2015-03-04 12:58:35 +0000381 void OnRenegotiationNeeded() override {
fischman@webrtc.orgd7568a02014-01-13 22:04:12 +0000382 ScopedLocalRefFrame local_ref_frame(jni());
383 jmethodID m =
384 GetMethodID(jni(), *j_observer_class_, "onRenegotiationNeeded", "()V");
385 jni()->CallVoidMethod(*j_observer_global_, m);
andresp@webrtc.org4d19e052014-09-09 11:45:44 +0000386 CHECK_EXCEPTION(jni()) << "error during CallVoidMethod";
fischman@webrtc.orgd7568a02014-01-13 22:04:12 +0000387 }
388
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000389 void SetConstraints(ConstraintsWrapper* constraints) {
henrikg91d6ede2015-09-17 00:24:34 -0700390 RTC_CHECK(!constraints_.get()) << "constraints already set!";
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000391 constraints_.reset(constraints);
392 }
393
394 const ConstraintsWrapper* constraints() { return constraints_.get(); }
395
396 private:
Magnus Jedvert1c3dd382015-08-27 13:39:58 +0200397 typedef std::map<MediaStreamInterface*, jobject> NativeToJavaStreamsMap;
398
399 void DisposeRemoteStream(const NativeToJavaStreamsMap::iterator& it) {
400 jobject j_stream = it->second;
401 remote_streams_.erase(it);
402 jni()->CallVoidMethod(
403 j_stream, GetMethodID(jni(), *j_media_stream_class_, "dispose", "()V"));
404 CHECK_EXCEPTION(jni()) << "error during MediaStream.dispose()";
405 DeleteGlobalRef(jni(), j_stream);
406 }
407
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000408 JNIEnv* jni() {
409 return AttachCurrentThreadIfNeeded();
410 }
411
henrike@webrtc.org723d6832013-07-12 16:04:50 +0000412 const ScopedGlobalRef<jobject> j_observer_global_;
413 const ScopedGlobalRef<jclass> j_observer_class_;
414 const ScopedGlobalRef<jclass> j_media_stream_class_;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000415 const jmethodID j_media_stream_ctor_;
henrike@webrtc.org723d6832013-07-12 16:04:50 +0000416 const ScopedGlobalRef<jclass> j_audio_track_class_;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000417 const jmethodID j_audio_track_ctor_;
henrike@webrtc.org723d6832013-07-12 16:04:50 +0000418 const ScopedGlobalRef<jclass> j_video_track_class_;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000419 const jmethodID j_video_track_ctor_;
henrike@webrtc.org723d6832013-07-12 16:04:50 +0000420 const ScopedGlobalRef<jclass> j_data_channel_class_;
421 const jmethodID j_data_channel_ctor_;
Magnus Jedvert1c3dd382015-08-27 13:39:58 +0200422 // C++ -> Java remote streams. The stored jobects are global refs and must be
423 // manually deleted upon removal. Use DisposeRemoteStream().
424 NativeToJavaStreamsMap remote_streams_;
fischman@webrtc.org540acde2014-02-13 03:56:14 +0000425 scoped_ptr<ConstraintsWrapper> constraints_;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000426};
427
428// Wrapper for a Java MediaConstraints object. Copies all needed data so when
429// the constructor returns the Java object is no longer needed.
430class ConstraintsWrapper : public MediaConstraintsInterface {
431 public:
432 ConstraintsWrapper(JNIEnv* jni, jobject j_constraints) {
433 PopulateConstraintsFromJavaPairList(
434 jni, j_constraints, "mandatory", &mandatory_);
435 PopulateConstraintsFromJavaPairList(
436 jni, j_constraints, "optional", &optional_);
437 }
438
439 virtual ~ConstraintsWrapper() {}
440
441 // MediaConstraintsInterface.
kjellander@webrtc.org14665ff2015-03-04 12:58:35 +0000442 const Constraints& GetMandatory() const override { return mandatory_; }
henrike@webrtc.org723d6832013-07-12 16:04:50 +0000443
kjellander@webrtc.org14665ff2015-03-04 12:58:35 +0000444 const Constraints& GetOptional() const override { return optional_; }
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000445
446 private:
447 // Helper for translating a List<Pair<String, String>> to a Constraints.
448 static void PopulateConstraintsFromJavaPairList(
449 JNIEnv* jni, jobject j_constraints,
450 const char* field_name, Constraints* field) {
451 jfieldID j_id = GetFieldID(jni,
452 GetObjectClass(jni, j_constraints), field_name, "Ljava/util/List;");
453 jobject j_list = GetObjectField(jni, j_constraints, j_id);
454 jmethodID j_iterator_id = GetMethodID(jni,
455 GetObjectClass(jni, j_list), "iterator", "()Ljava/util/Iterator;");
456 jobject j_iterator = jni->CallObjectMethod(j_list, j_iterator_id);
andresp@webrtc.org4d19e052014-09-09 11:45:44 +0000457 CHECK_EXCEPTION(jni) << "error during CallObjectMethod";
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000458 jmethodID j_has_next = GetMethodID(jni,
459 GetObjectClass(jni, j_iterator), "hasNext", "()Z");
460 jmethodID j_next = GetMethodID(jni,
461 GetObjectClass(jni, j_iterator), "next", "()Ljava/lang/Object;");
462 while (jni->CallBooleanMethod(j_iterator, j_has_next)) {
andresp@webrtc.org4d19e052014-09-09 11:45:44 +0000463 CHECK_EXCEPTION(jni) << "error during CallBooleanMethod";
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000464 jobject entry = jni->CallObjectMethod(j_iterator, j_next);
andresp@webrtc.org4d19e052014-09-09 11:45:44 +0000465 CHECK_EXCEPTION(jni) << "error during CallObjectMethod";
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000466 jmethodID get_key = GetMethodID(jni,
467 GetObjectClass(jni, entry), "getKey", "()Ljava/lang/String;");
468 jstring j_key = reinterpret_cast<jstring>(
469 jni->CallObjectMethod(entry, get_key));
andresp@webrtc.org4d19e052014-09-09 11:45:44 +0000470 CHECK_EXCEPTION(jni) << "error during CallObjectMethod";
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000471 jmethodID get_value = GetMethodID(jni,
472 GetObjectClass(jni, entry), "getValue", "()Ljava/lang/String;");
473 jstring j_value = reinterpret_cast<jstring>(
474 jni->CallObjectMethod(entry, get_value));
andresp@webrtc.org4d19e052014-09-09 11:45:44 +0000475 CHECK_EXCEPTION(jni) << "error during CallObjectMethod";
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000476 field->push_back(Constraint(JavaToStdString(jni, j_key),
477 JavaToStdString(jni, j_value)));
478 }
andresp@webrtc.org4d19e052014-09-09 11:45:44 +0000479 CHECK_EXCEPTION(jni) << "error during CallBooleanMethod";
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000480 }
481
482 Constraints mandatory_;
483 Constraints optional_;
484};
485
486static jobject JavaSdpFromNativeSdp(
487 JNIEnv* jni, const SessionDescriptionInterface* desc) {
488 std::string sdp;
henrikg91d6ede2015-09-17 00:24:34 -0700489 RTC_CHECK(desc->ToString(&sdp)) << "got so far: " << sdp;
fischman@webrtc.org41776152014-01-09 00:31:17 +0000490 jstring j_description = JavaStringFromStdString(jni, sdp);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000491
492 jclass j_type_class = FindClass(
493 jni, "org/webrtc/SessionDescription$Type");
494 jmethodID j_type_from_canonical = GetStaticMethodID(
495 jni, j_type_class, "fromCanonicalForm",
496 "(Ljava/lang/String;)Lorg/webrtc/SessionDescription$Type;");
fischman@webrtc.org41776152014-01-09 00:31:17 +0000497 jstring j_type_string = JavaStringFromStdString(jni, desc->type());
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000498 jobject j_type = jni->CallStaticObjectMethod(
fischman@webrtc.org41776152014-01-09 00:31:17 +0000499 j_type_class, j_type_from_canonical, j_type_string);
andresp@webrtc.org4d19e052014-09-09 11:45:44 +0000500 CHECK_EXCEPTION(jni) << "error during CallObjectMethod";
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000501
502 jclass j_sdp_class = FindClass(jni, "org/webrtc/SessionDescription");
503 jmethodID j_sdp_ctor = GetMethodID(
504 jni, j_sdp_class, "<init>",
505 "(Lorg/webrtc/SessionDescription$Type;Ljava/lang/String;)V");
506 jobject j_sdp = jni->NewObject(
fischman@webrtc.org41776152014-01-09 00:31:17 +0000507 j_sdp_class, j_sdp_ctor, j_type, j_description);
andresp@webrtc.org4d19e052014-09-09 11:45:44 +0000508 CHECK_EXCEPTION(jni) << "error during NewObject";
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000509 return j_sdp;
510}
511
512template <class T> // T is one of {Create,Set}SessionDescriptionObserver.
513class SdpObserverWrapper : public T {
514 public:
515 SdpObserverWrapper(JNIEnv* jni, jobject j_observer,
516 ConstraintsWrapper* constraints)
517 : constraints_(constraints),
henrike@webrtc.org723d6832013-07-12 16:04:50 +0000518 j_observer_global_(jni, j_observer),
519 j_observer_class_(jni, GetObjectClass(jni, j_observer)) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000520 }
521
henrike@webrtc.org723d6832013-07-12 16:04:50 +0000522 virtual ~SdpObserverWrapper() {}
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000523
kjellander@webrtc.org14665ff2015-03-04 12:58:35 +0000524 // Can't mark override because of templating.
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000525 virtual void OnSuccess() {
fischman@webrtc.org41776152014-01-09 00:31:17 +0000526 ScopedLocalRefFrame local_ref_frame(jni());
henrike@webrtc.org723d6832013-07-12 16:04:50 +0000527 jmethodID m = GetMethodID(jni(), *j_observer_class_, "onSetSuccess", "()V");
528 jni()->CallVoidMethod(*j_observer_global_, m);
andresp@webrtc.org4d19e052014-09-09 11:45:44 +0000529 CHECK_EXCEPTION(jni()) << "error during CallVoidMethod";
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000530 }
531
kjellander@webrtc.org14665ff2015-03-04 12:58:35 +0000532 // Can't mark override because of templating.
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000533 virtual void OnSuccess(SessionDescriptionInterface* desc) {
fischman@webrtc.org41776152014-01-09 00:31:17 +0000534 ScopedLocalRefFrame local_ref_frame(jni());
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000535 jmethodID m = GetMethodID(
henrike@webrtc.org723d6832013-07-12 16:04:50 +0000536 jni(), *j_observer_class_, "onCreateSuccess",
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000537 "(Lorg/webrtc/SessionDescription;)V");
fischman@webrtc.org41776152014-01-09 00:31:17 +0000538 jobject j_sdp = JavaSdpFromNativeSdp(jni(), desc);
539 jni()->CallVoidMethod(*j_observer_global_, m, j_sdp);
andresp@webrtc.org4d19e052014-09-09 11:45:44 +0000540 CHECK_EXCEPTION(jni()) << "error during CallVoidMethod";
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000541 }
542
543 protected:
544 // Common implementation for failure of Set & Create types, distinguished by
545 // |op| being "Set" or "Create".
546 void OnFailure(const std::string& op, const std::string& error) {
henrike@webrtc.org723d6832013-07-12 16:04:50 +0000547 jmethodID m = GetMethodID(jni(), *j_observer_class_, "on" + op + "Failure",
548 "(Ljava/lang/String;)V");
fischman@webrtc.org41776152014-01-09 00:31:17 +0000549 jstring j_error_string = JavaStringFromStdString(jni(), error);
550 jni()->CallVoidMethod(*j_observer_global_, m, j_error_string);
andresp@webrtc.org4d19e052014-09-09 11:45:44 +0000551 CHECK_EXCEPTION(jni()) << "error during CallVoidMethod";
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000552 }
553
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000554 JNIEnv* jni() {
555 return AttachCurrentThreadIfNeeded();
556 }
557
fischman@webrtc.org41776152014-01-09 00:31:17 +0000558 private:
fischman@webrtc.org540acde2014-02-13 03:56:14 +0000559 scoped_ptr<ConstraintsWrapper> constraints_;
henrike@webrtc.org723d6832013-07-12 16:04:50 +0000560 const ScopedGlobalRef<jobject> j_observer_global_;
561 const ScopedGlobalRef<jclass> j_observer_class_;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000562};
563
564class CreateSdpObserverWrapper
565 : public SdpObserverWrapper<CreateSessionDescriptionObserver> {
566 public:
567 CreateSdpObserverWrapper(JNIEnv* jni, jobject j_observer,
568 ConstraintsWrapper* constraints)
569 : SdpObserverWrapper(jni, j_observer, constraints) {}
570
kjellander@webrtc.org14665ff2015-03-04 12:58:35 +0000571 void OnFailure(const std::string& error) override {
fischman@webrtc.org41776152014-01-09 00:31:17 +0000572 ScopedLocalRefFrame local_ref_frame(jni());
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000573 SdpObserverWrapper::OnFailure(std::string("Create"), error);
574 }
575};
576
577class SetSdpObserverWrapper
578 : public SdpObserverWrapper<SetSessionDescriptionObserver> {
579 public:
580 SetSdpObserverWrapper(JNIEnv* jni, jobject j_observer,
581 ConstraintsWrapper* constraints)
582 : SdpObserverWrapper(jni, j_observer, constraints) {}
583
kjellander@webrtc.org14665ff2015-03-04 12:58:35 +0000584 void OnFailure(const std::string& error) override {
fischman@webrtc.org41776152014-01-09 00:31:17 +0000585 ScopedLocalRefFrame local_ref_frame(jni());
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000586 SdpObserverWrapper::OnFailure(std::string("Set"), error);
587 }
588};
589
henrike@webrtc.org723d6832013-07-12 16:04:50 +0000590// Adapter for a Java DataChannel$Observer presenting a C++ DataChannelObserver
591// and dispatching the callback from C++ back to Java.
592class DataChannelObserverWrapper : public DataChannelObserver {
593 public:
594 DataChannelObserverWrapper(JNIEnv* jni, jobject j_observer)
595 : j_observer_global_(jni, j_observer),
596 j_observer_class_(jni, GetObjectClass(jni, j_observer)),
pbos@webrtc.orgb648b9d2014-08-26 11:08:06 +0000597 j_buffer_class_(jni, FindClass(jni, "org/webrtc/DataChannel$Buffer")),
bemasc0edd50c2015-07-01 13:34:33 -0700598 j_on_buffered_amount_change_mid_(GetMethodID(
599 jni, *j_observer_class_, "onBufferedAmountChange", "(J)V")),
600 j_on_state_change_mid_(
601 GetMethodID(jni, *j_observer_class_, "onStateChange", "()V")),
henrike@webrtc.org723d6832013-07-12 16:04:50 +0000602 j_on_message_mid_(GetMethodID(jni, *j_observer_class_, "onMessage",
603 "(Lorg/webrtc/DataChannel$Buffer;)V")),
bemasc0edd50c2015-07-01 13:34:33 -0700604 j_buffer_ctor_(GetMethodID(jni, *j_buffer_class_, "<init>",
605 "(Ljava/nio/ByteBuffer;Z)V")) {}
henrike@webrtc.org723d6832013-07-12 16:04:50 +0000606
607 virtual ~DataChannelObserverWrapper() {}
608
bemasc0edd50c2015-07-01 13:34:33 -0700609 void OnBufferedAmountChange(uint64 previous_amount) override {
610 ScopedLocalRefFrame local_ref_frame(jni());
611 jni()->CallVoidMethod(*j_observer_global_, j_on_buffered_amount_change_mid_,
612 previous_amount);
613 CHECK_EXCEPTION(jni()) << "error during CallVoidMethod";
614 }
615
kjellander@webrtc.org14665ff2015-03-04 12:58:35 +0000616 void OnStateChange() override {
fischman@webrtc.org41776152014-01-09 00:31:17 +0000617 ScopedLocalRefFrame local_ref_frame(jni());
henrike@webrtc.org723d6832013-07-12 16:04:50 +0000618 jni()->CallVoidMethod(*j_observer_global_, j_on_state_change_mid_);
andresp@webrtc.org4d19e052014-09-09 11:45:44 +0000619 CHECK_EXCEPTION(jni()) << "error during CallVoidMethod";
henrike@webrtc.org723d6832013-07-12 16:04:50 +0000620 }
621
kjellander@webrtc.org14665ff2015-03-04 12:58:35 +0000622 void OnMessage(const DataBuffer& buffer) override {
fischman@webrtc.org41776152014-01-09 00:31:17 +0000623 ScopedLocalRefFrame local_ref_frame(jni());
kwiberg@webrtc.orgeebcab52015-03-24 09:19:06 +0000624 jobject byte_buffer = jni()->NewDirectByteBuffer(
Karl Wiberg94784372015-04-20 14:03:07 +0200625 const_cast<char*>(buffer.data.data<char>()), buffer.data.size());
henrike@webrtc.org723d6832013-07-12 16:04:50 +0000626 jobject j_buffer = jni()->NewObject(*j_buffer_class_, j_buffer_ctor_,
627 byte_buffer, buffer.binary);
628 jni()->CallVoidMethod(*j_observer_global_, j_on_message_mid_, j_buffer);
andresp@webrtc.org4d19e052014-09-09 11:45:44 +0000629 CHECK_EXCEPTION(jni()) << "error during CallVoidMethod";
henrike@webrtc.org723d6832013-07-12 16:04:50 +0000630 }
631
632 private:
633 JNIEnv* jni() {
634 return AttachCurrentThreadIfNeeded();
635 }
636
637 const ScopedGlobalRef<jobject> j_observer_global_;
638 const ScopedGlobalRef<jclass> j_observer_class_;
639 const ScopedGlobalRef<jclass> j_buffer_class_;
bemasc0edd50c2015-07-01 13:34:33 -0700640 const jmethodID j_on_buffered_amount_change_mid_;
henrike@webrtc.org723d6832013-07-12 16:04:50 +0000641 const jmethodID j_on_state_change_mid_;
642 const jmethodID j_on_message_mid_;
643 const jmethodID j_buffer_ctor_;
644};
645
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000646// Adapter for a Java StatsObserver presenting a C++ StatsObserver and
647// dispatching the callback from C++ back to Java.
648class StatsObserverWrapper : public StatsObserver {
649 public:
650 StatsObserverWrapper(JNIEnv* jni, jobject j_observer)
henrike@webrtc.org723d6832013-07-12 16:04:50 +0000651 : j_observer_global_(jni, j_observer),
652 j_observer_class_(jni, GetObjectClass(jni, j_observer)),
653 j_stats_report_class_(jni, FindClass(jni, "org/webrtc/StatsReport")),
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000654 j_stats_report_ctor_(GetMethodID(
henrike@webrtc.org723d6832013-07-12 16:04:50 +0000655 jni, *j_stats_report_class_, "<init>",
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000656 "(Ljava/lang/String;Ljava/lang/String;D"
657 "[Lorg/webrtc/StatsReport$Value;)V")),
henrike@webrtc.org723d6832013-07-12 16:04:50 +0000658 j_value_class_(jni, FindClass(
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000659 jni, "org/webrtc/StatsReport$Value")),
660 j_value_ctor_(GetMethodID(
henrike@webrtc.org723d6832013-07-12 16:04:50 +0000661 jni, *j_value_class_, "<init>",
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000662 "(Ljava/lang/String;Ljava/lang/String;)V")) {
663 }
664
henrike@webrtc.org723d6832013-07-12 16:04:50 +0000665 virtual ~StatsObserverWrapper() {}
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000666
kjellander@webrtc.org14665ff2015-03-04 12:58:35 +0000667 void OnComplete(const StatsReports& reports) override {
fischman@webrtc.org41776152014-01-09 00:31:17 +0000668 ScopedLocalRefFrame local_ref_frame(jni());
669 jobjectArray j_reports = ReportsToJava(jni(), reports);
henrike@webrtc.org723d6832013-07-12 16:04:50 +0000670 jmethodID m = GetMethodID(jni(), *j_observer_class_, "onComplete",
671 "([Lorg/webrtc/StatsReport;)V");
fischman@webrtc.org41776152014-01-09 00:31:17 +0000672 jni()->CallVoidMethod(*j_observer_global_, m, j_reports);
andresp@webrtc.org4d19e052014-09-09 11:45:44 +0000673 CHECK_EXCEPTION(jni()) << "error during CallVoidMethod";
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000674 }
675
676 private:
677 jobjectArray ReportsToJava(
tommi@webrtc.orge2e199b2014-12-15 13:22:54 +0000678 JNIEnv* jni, const StatsReports& reports) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000679 jobjectArray reports_array = jni->NewObjectArray(
henrike@webrtc.org723d6832013-07-12 16:04:50 +0000680 reports.size(), *j_stats_report_class_, NULL);
tommi@webrtc.orge2e199b2014-12-15 13:22:54 +0000681 int i = 0;
682 for (const auto* report : reports) {
fischman@webrtc.org41776152014-01-09 00:31:17 +0000683 ScopedLocalRefFrame local_ref_frame(jni);
tommi@webrtc.orgd3900292015-03-12 16:35:55 +0000684 jstring j_id = JavaStringFromStdString(jni, report->id()->ToString());
tommi@webrtc.org8e327c42015-01-19 20:41:26 +0000685 jstring j_type = JavaStringFromStdString(jni, report->TypeToString());
686 jobjectArray j_values = ValuesToJava(jni, report->values());
fischman@webrtc.org41776152014-01-09 00:31:17 +0000687 jobject j_report = jni->NewObject(*j_stats_report_class_,
688 j_stats_report_ctor_,
689 j_id,
690 j_type,
tommi@webrtc.org8e327c42015-01-19 20:41:26 +0000691 report->timestamp(),
fischman@webrtc.org41776152014-01-09 00:31:17 +0000692 j_values);
tommi@webrtc.orge2e199b2014-12-15 13:22:54 +0000693 jni->SetObjectArrayElement(reports_array, i++, j_report);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000694 }
695 return reports_array;
696 }
697
698 jobjectArray ValuesToJava(JNIEnv* jni, const StatsReport::Values& values) {
699 jobjectArray j_values = jni->NewObjectArray(
henrike@webrtc.org723d6832013-07-12 16:04:50 +0000700 values.size(), *j_value_class_, NULL);
tommi@webrtc.org92f40182015-03-04 15:25:19 +0000701 int i = 0;
702 for (const auto& it : values) {
fischman@webrtc.org41776152014-01-09 00:31:17 +0000703 ScopedLocalRefFrame local_ref_frame(jni);
tommi@webrtc.orgc57310b2014-12-12 17:41:28 +0000704 // Should we use the '.name' enum value here instead of converting the
705 // name to a string?
tommi@webrtc.org92f40182015-03-04 15:25:19 +0000706 jstring j_name = JavaStringFromStdString(jni, it.second->display_name());
707 jstring j_value = JavaStringFromStdString(jni, it.second->ToString());
fischman@webrtc.org41776152014-01-09 00:31:17 +0000708 jobject j_element_value =
709 jni->NewObject(*j_value_class_, j_value_ctor_, j_name, j_value);
tommi@webrtc.org92f40182015-03-04 15:25:19 +0000710 jni->SetObjectArrayElement(j_values, i++, j_element_value);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000711 }
712 return j_values;
713 }
714
715 JNIEnv* jni() {
716 return AttachCurrentThreadIfNeeded();
717 }
718
henrike@webrtc.org723d6832013-07-12 16:04:50 +0000719 const ScopedGlobalRef<jobject> j_observer_global_;
720 const ScopedGlobalRef<jclass> j_observer_class_;
721 const ScopedGlobalRef<jclass> j_stats_report_class_;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000722 const jmethodID j_stats_report_ctor_;
henrike@webrtc.org723d6832013-07-12 16:04:50 +0000723 const ScopedGlobalRef<jclass> j_value_class_;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000724 const jmethodID j_value_ctor_;
725};
726
727// Adapter presenting a cricket::VideoRenderer as a
728// webrtc::VideoRendererInterface.
729class VideoRendererWrapper : public VideoRendererInterface {
730 public:
731 static VideoRendererWrapper* Create(cricket::VideoRenderer* renderer) {
732 if (renderer)
733 return new VideoRendererWrapper(renderer);
734 return NULL;
735 }
736
737 virtual ~VideoRendererWrapper() {}
738
guoweis@webrtc.org00c509a2015-03-12 21:37:26 +0000739 // This wraps VideoRenderer which still has SetSize.
740 void RenderFrame(const cricket::VideoFrame* video_frame) override {
fischman@webrtc.org41776152014-01-09 00:31:17 +0000741 ScopedLocalRefFrame local_ref_frame(AttachCurrentThreadIfNeeded());
guoweis@webrtc.org00c509a2015-03-12 21:37:26 +0000742 const cricket::VideoFrame* frame =
743 video_frame->GetCopyWithRotationApplied();
744 if (width_ != frame->GetWidth() || height_ != frame->GetHeight()) {
745 width_ = frame->GetWidth();
746 height_ = frame->GetHeight();
747 renderer_->SetSize(width_, height_, 0);
748 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000749 renderer_->RenderFrame(frame);
750 }
751
752 private:
753 explicit VideoRendererWrapper(cricket::VideoRenderer* renderer)
Henrik Kjellander7c027b62015-04-22 13:21:30 +0200754 : width_(0), height_(0), renderer_(renderer) {}
guoweis@webrtc.org00c509a2015-03-12 21:37:26 +0000755 int width_, height_;
fischman@webrtc.org540acde2014-02-13 03:56:14 +0000756 scoped_ptr<cricket::VideoRenderer> renderer_;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000757};
758
759// Wrapper dispatching webrtc::VideoRendererInterface to a Java VideoRenderer
760// instance.
761class JavaVideoRendererWrapper : public VideoRendererInterface {
762 public:
763 JavaVideoRendererWrapper(JNIEnv* jni, jobject j_callbacks)
henrike@webrtc.org723d6832013-07-12 16:04:50 +0000764 : j_callbacks_(jni, j_callbacks),
henrike@webrtc.org723d6832013-07-12 16:04:50 +0000765 j_render_frame_id_(GetMethodID(
766 jni, GetObjectClass(jni, j_callbacks), "renderFrame",
767 "(Lorg/webrtc/VideoRenderer$I420Frame;)V")),
768 j_frame_class_(jni,
769 FindClass(jni, "org/webrtc/VideoRenderer$I420Frame")),
glaznev@webrtc.org99678452014-09-15 17:52:42 +0000770 j_i420_frame_ctor_id_(GetMethodID(
Magnus Jedverta6cba3a2015-08-29 15:57:43 +0200771 jni, *j_frame_class_, "<init>", "(III[I[Ljava/nio/ByteBuffer;J)V")),
glaznev@webrtc.org99678452014-09-15 17:52:42 +0000772 j_texture_frame_ctor_id_(GetMethodID(
773 jni, *j_frame_class_, "<init>",
Magnus Jedverta6cba3a2015-08-29 15:57:43 +0200774 "(IIILjava/lang/Object;IJ)V")),
Magnus Jedvert7ef9d912015-08-25 09:32:06 +0200775 j_byte_buffer_class_(jni, FindClass(jni, "java/nio/ByteBuffer")) {
andresp@webrtc.org4d19e052014-09-09 11:45:44 +0000776 CHECK_EXCEPTION(jni);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000777 }
778
henrike@webrtc.org723d6832013-07-12 16:04:50 +0000779 virtual ~JavaVideoRendererWrapper() {}
780
guoweis@webrtc.org00c509a2015-03-12 21:37:26 +0000781 void RenderFrame(const cricket::VideoFrame* video_frame) override {
fischman@webrtc.org41776152014-01-09 00:31:17 +0000782 ScopedLocalRefFrame local_ref_frame(jni());
Magnus Jedvert7ef9d912015-08-25 09:32:06 +0200783 jobject j_frame = (video_frame->GetNativeHandle() != nullptr)
784 ? CricketToJavaTextureFrame(video_frame)
785 : CricketToJavaI420Frame(video_frame);
Magnus Jedvert1d640e52015-09-29 16:33:16 +0200786 // |j_callbacks_| is responsible for releasing |j_frame| with
787 // VideoRenderer.renderFrameDone().
Magnus Jedvert7ef9d912015-08-25 09:32:06 +0200788 jni()->CallVoidMethod(*j_callbacks_, j_render_frame_id_, j_frame);
789 CHECK_EXCEPTION(jni());
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000790 }
791
792 private:
Magnus Jedvert1d640e52015-09-29 16:33:16 +0200793 // Make a shallow copy of |frame| to be used with Java. The callee has
794 // ownership of the frame, and the frame should be released with
795 // VideoRenderer.releaseNativeFrame().
796 static jlong javaShallowCopy(const cricket::VideoFrame* frame) {
797 return jlongFromPointer(frame->Copy());
798 }
799
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000800 // Return a VideoRenderer.I420Frame referring to the data in |frame|.
glaznev@webrtc.org99678452014-09-15 17:52:42 +0000801 jobject CricketToJavaI420Frame(const cricket::VideoFrame* frame) {
fischman@webrtc.org41776152014-01-09 00:31:17 +0000802 jintArray strides = jni()->NewIntArray(3);
803 jint* strides_array = jni()->GetIntArrayElements(strides, NULL);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000804 strides_array[0] = frame->GetYPitch();
805 strides_array[1] = frame->GetUPitch();
806 strides_array[2] = frame->GetVPitch();
fischman@webrtc.org41776152014-01-09 00:31:17 +0000807 jni()->ReleaseIntArrayElements(strides, strides_array, 0);
808 jobjectArray planes = jni()->NewObjectArray(3, *j_byte_buffer_class_, NULL);
809 jobject y_buffer = jni()->NewDirectByteBuffer(
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000810 const_cast<uint8*>(frame->GetYPlane()),
fischman@webrtc.org41776152014-01-09 00:31:17 +0000811 frame->GetYPitch() * frame->GetHeight());
812 jobject u_buffer = jni()->NewDirectByteBuffer(
813 const_cast<uint8*>(frame->GetUPlane()), frame->GetChromaSize());
814 jobject v_buffer = jni()->NewDirectByteBuffer(
815 const_cast<uint8*>(frame->GetVPlane()), frame->GetChromaSize());
816 jni()->SetObjectArrayElement(planes, 0, y_buffer);
817 jni()->SetObjectArrayElement(planes, 1, u_buffer);
818 jni()->SetObjectArrayElement(planes, 2, v_buffer);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000819 return jni()->NewObject(
glaznev@webrtc.org99678452014-09-15 17:52:42 +0000820 *j_frame_class_, j_i420_frame_ctor_id_,
guoweis@webrtc.org840da7b2015-03-18 16:58:13 +0000821 frame->GetWidth(), frame->GetHeight(),
822 static_cast<int>(frame->GetVideoRotation()),
Magnus Jedvert1d640e52015-09-29 16:33:16 +0200823 strides, planes, javaShallowCopy(frame));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000824 }
825
glaznev@webrtc.org99678452014-09-15 17:52:42 +0000826 // Return a VideoRenderer.I420Frame referring texture object in |frame|.
827 jobject CricketToJavaTextureFrame(const cricket::VideoFrame* frame) {
828 NativeHandleImpl* handle =
829 reinterpret_cast<NativeHandleImpl*>(frame->GetNativeHandle());
830 jobject texture_object = reinterpret_cast<jobject>(handle->GetHandle());
831 int texture_id = handle->GetTextureId();
832 return jni()->NewObject(
833 *j_frame_class_, j_texture_frame_ctor_id_,
guoweis@webrtc.org840da7b2015-03-18 16:58:13 +0000834 frame->GetWidth(), frame->GetHeight(),
835 static_cast<int>(frame->GetVideoRotation()),
Magnus Jedvert1d640e52015-09-29 16:33:16 +0200836 texture_object, texture_id, javaShallowCopy(frame));
glaznev@webrtc.org99678452014-09-15 17:52:42 +0000837 }
838
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000839 JNIEnv* jni() {
840 return AttachCurrentThreadIfNeeded();
841 }
842
henrike@webrtc.org723d6832013-07-12 16:04:50 +0000843 ScopedGlobalRef<jobject> j_callbacks_;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000844 jmethodID j_render_frame_id_;
henrike@webrtc.org723d6832013-07-12 16:04:50 +0000845 ScopedGlobalRef<jclass> j_frame_class_;
glaznev@webrtc.org99678452014-09-15 17:52:42 +0000846 jmethodID j_i420_frame_ctor_id_;
847 jmethodID j_texture_frame_ctor_id_;
henrike@webrtc.org723d6832013-07-12 16:04:50 +0000848 ScopedGlobalRef<jclass> j_byte_buffer_class_;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000849};
850
fischman@webrtc.org540acde2014-02-13 03:56:14 +0000851
fischman@webrtc.org32001ef2013-08-12 23:26:21 +0000852static DataChannelInterface* ExtractNativeDC(JNIEnv* jni, jobject j_dc) {
henrike@webrtc.org723d6832013-07-12 16:04:50 +0000853 jfieldID native_dc_id = GetFieldID(jni,
854 GetObjectClass(jni, j_dc), "nativeDataChannel", "J");
855 jlong j_d = GetLongField(jni, j_dc, native_dc_id);
fischman@webrtc.org32001ef2013-08-12 23:26:21 +0000856 return reinterpret_cast<DataChannelInterface*>(j_d);
henrike@webrtc.org723d6832013-07-12 16:04:50 +0000857}
858
859JOW(jlong, DataChannel_registerObserverNative)(
860 JNIEnv* jni, jobject j_dc, jobject j_observer) {
fischman@webrtc.org540acde2014-02-13 03:56:14 +0000861 scoped_ptr<DataChannelObserverWrapper> observer(
henrike@webrtc.org723d6832013-07-12 16:04:50 +0000862 new DataChannelObserverWrapper(jni, j_observer));
863 ExtractNativeDC(jni, j_dc)->RegisterObserver(observer.get());
fischman@webrtc.orgf41f06b2013-12-11 21:07:18 +0000864 return jlongFromPointer(observer.release());
henrike@webrtc.org723d6832013-07-12 16:04:50 +0000865}
866
867JOW(void, DataChannel_unregisterObserverNative)(
868 JNIEnv* jni, jobject j_dc, jlong native_observer) {
869 ExtractNativeDC(jni, j_dc)->UnregisterObserver();
870 delete reinterpret_cast<DataChannelObserverWrapper*>(native_observer);
871}
872
873JOW(jstring, DataChannel_label)(JNIEnv* jni, jobject j_dc) {
874 return JavaStringFromStdString(jni, ExtractNativeDC(jni, j_dc)->label());
875}
876
877JOW(jobject, DataChannel_state)(JNIEnv* jni, jobject j_dc) {
878 return JavaEnumFromIndex(
879 jni, "DataChannel$State", ExtractNativeDC(jni, j_dc)->state());
880}
881
882JOW(jlong, DataChannel_bufferedAmount)(JNIEnv* jni, jobject j_dc) {
883 uint64 buffered_amount = ExtractNativeDC(jni, j_dc)->buffered_amount();
henrikg91d6ede2015-09-17 00:24:34 -0700884 RTC_CHECK_LE(buffered_amount, std::numeric_limits<int64>::max())
andresp@webrtc.org4d19e052014-09-09 11:45:44 +0000885 << "buffered_amount overflowed jlong!";
henrike@webrtc.org723d6832013-07-12 16:04:50 +0000886 return static_cast<jlong>(buffered_amount);
887}
888
889JOW(void, DataChannel_close)(JNIEnv* jni, jobject j_dc) {
890 ExtractNativeDC(jni, j_dc)->Close();
891}
892
893JOW(jboolean, DataChannel_sendNative)(JNIEnv* jni, jobject j_dc,
894 jbyteArray data, jboolean binary) {
895 jbyte* bytes = jni->GetByteArrayElements(data, NULL);
896 bool ret = ExtractNativeDC(jni, j_dc)->Send(DataBuffer(
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +0000897 rtc::Buffer(bytes, jni->GetArrayLength(data)),
henrike@webrtc.org723d6832013-07-12 16:04:50 +0000898 binary));
899 jni->ReleaseByteArrayElements(data, bytes, JNI_ABORT);
900 return ret;
901}
902
903JOW(void, DataChannel_dispose)(JNIEnv* jni, jobject j_dc) {
fischman@webrtc.org32001ef2013-08-12 23:26:21 +0000904 CHECK_RELEASE(ExtractNativeDC(jni, j_dc));
henrike@webrtc.org723d6832013-07-12 16:04:50 +0000905}
906
fischman@webrtc.orgc883fdc2013-08-06 19:00:53 +0000907JOW(void, Logging_nativeEnableTracing)(
908 JNIEnv* jni, jclass, jstring j_path, jint nativeLevels,
909 jint nativeSeverity) {
910 std::string path = JavaToStdString(jni, j_path);
911 if (nativeLevels != webrtc::kTraceNone) {
andrew@webrtc.org90805182013-09-05 16:40:43 +0000912 webrtc::Trace::set_level_filter(nativeLevels);
glaznev@webrtc.orge6581242014-09-19 16:53:46 +0000913#if defined(ANDROID) && !defined(WEBRTC_CHROMIUM_BUILD)
fischman@webrtc.org7e4d0df2013-10-01 02:40:43 +0000914 if (path != "logcat:") {
915#endif
henrikg91d6ede2015-09-17 00:24:34 -0700916 RTC_CHECK_EQ(0, webrtc::Trace::SetTraceFile(path.c_str(), false))
andresp@webrtc.org4d19e052014-09-09 11:45:44 +0000917 << "SetTraceFile failed";
glaznev@webrtc.orge6581242014-09-19 16:53:46 +0000918#if defined(ANDROID) && !defined(WEBRTC_CHROMIUM_BUILD)
fischman@webrtc.org7e4d0df2013-10-01 02:40:43 +0000919 } else {
920 // Intentionally leak this to avoid needing to reason about its lifecycle.
921 // It keeps no state and functions only as a dispatch point.
922 static LogcatTraceContext* g_trace_callback = new LogcatTraceContext();
923 }
924#endif
fischman@webrtc.orgc883fdc2013-08-06 19:00:53 +0000925 }
Tommi0eefb4d2015-05-23 09:54:07 +0200926 if (nativeSeverity >= rtc::LS_SENSITIVE && nativeSeverity <= rtc::LS_ERROR) {
927 rtc::LogMessage::LogToDebug(
928 static_cast<rtc::LoggingSeverity>(nativeSeverity));
929 }
fischman@webrtc.orgc883fdc2013-08-06 19:00:53 +0000930}
931
Jiayang Liue63d2a12015-09-01 16:11:18 -0700932JOW(void, Logging_nativeEnableLogThreads)(JNIEnv* jni, jclass) {
933 rtc::LogMessage::LogThreads(true);
934}
935
936JOW(void, Logging_nativeEnableLogTimeStamps)(JNIEnv* jni, jclass) {
937 rtc::LogMessage::LogTimestamps(true);
938}
939
940JOW(void, Logging_nativeLog)(
jiayl66f0da22015-09-14 15:06:39 -0700941 JNIEnv* jni, jclass, jint j_severity, jstring j_tag, jstring j_message) {
Jiayang Liue63d2a12015-09-01 16:11:18 -0700942 std::string message = JavaToStdString(jni, j_message);
jiayl66f0da22015-09-14 15:06:39 -0700943 std::string tag = JavaToStdString(jni, j_tag);
944 LOG_TAG(static_cast<rtc::LoggingSeverity>(j_severity), tag) << message;
Jiayang Liue63d2a12015-09-01 16:11:18 -0700945}
946
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000947JOW(void, PeerConnection_freePeerConnection)(JNIEnv*, jclass, jlong j_p) {
fischman@webrtc.org32001ef2013-08-12 23:26:21 +0000948 CHECK_RELEASE(reinterpret_cast<PeerConnectionInterface*>(j_p));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000949}
950
951JOW(void, PeerConnection_freeObserver)(JNIEnv*, jclass, jlong j_p) {
952 PCOJava* p = reinterpret_cast<PCOJava*>(j_p);
953 delete p;
954}
955
956JOW(void, MediaSource_free)(JNIEnv*, jclass, jlong j_p) {
fischman@webrtc.org32001ef2013-08-12 23:26:21 +0000957 CHECK_RELEASE(reinterpret_cast<MediaSourceInterface*>(j_p));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000958}
959
960JOW(void, VideoCapturer_free)(JNIEnv*, jclass, jlong j_p) {
961 delete reinterpret_cast<cricket::VideoCapturer*>(j_p);
962}
963
glaznev@webrtc.orga59c5012014-09-17 03:26:59 +0000964JOW(void, VideoRenderer_freeGuiVideoRenderer)(JNIEnv*, jclass, jlong j_p) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000965 delete reinterpret_cast<VideoRendererWrapper*>(j_p);
966}
967
glaznev@webrtc.orga59c5012014-09-17 03:26:59 +0000968JOW(void, VideoRenderer_freeWrappedVideoRenderer)(JNIEnv*, jclass, jlong j_p) {
969 delete reinterpret_cast<JavaVideoRendererWrapper*>(j_p);
970}
971
Magnus Jedverta6cba3a2015-08-29 15:57:43 +0200972JOW(void, VideoRenderer_releaseNativeFrame)(
973 JNIEnv* jni, jclass, jlong j_frame_ptr) {
974 delete reinterpret_cast<const cricket::VideoFrame*>(j_frame_ptr);
975}
976
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000977JOW(void, MediaStreamTrack_free)(JNIEnv*, jclass, jlong j_p) {
deadbeef4139c0f2015-10-06 12:29:25 -0700978 reinterpret_cast<MediaStreamTrackInterface*>(j_p)->Release();
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000979}
980
981JOW(jboolean, MediaStream_nativeAddAudioTrack)(
982 JNIEnv* jni, jclass, jlong pointer, jlong j_audio_track_pointer) {
fischman@webrtc.org32001ef2013-08-12 23:26:21 +0000983 return reinterpret_cast<MediaStreamInterface*>(pointer)->AddTrack(
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000984 reinterpret_cast<AudioTrackInterface*>(j_audio_track_pointer));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000985}
986
987JOW(jboolean, MediaStream_nativeAddVideoTrack)(
988 JNIEnv* jni, jclass, jlong pointer, jlong j_video_track_pointer) {
fischman@webrtc.org32001ef2013-08-12 23:26:21 +0000989 return reinterpret_cast<MediaStreamInterface*>(pointer)
990 ->AddTrack(reinterpret_cast<VideoTrackInterface*>(j_video_track_pointer));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000991}
992
993JOW(jboolean, MediaStream_nativeRemoveAudioTrack)(
994 JNIEnv* jni, jclass, jlong pointer, jlong j_audio_track_pointer) {
fischman@webrtc.org32001ef2013-08-12 23:26:21 +0000995 return reinterpret_cast<MediaStreamInterface*>(pointer)->RemoveTrack(
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000996 reinterpret_cast<AudioTrackInterface*>(j_audio_track_pointer));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000997}
998
999JOW(jboolean, MediaStream_nativeRemoveVideoTrack)(
1000 JNIEnv* jni, jclass, jlong pointer, jlong j_video_track_pointer) {
fischman@webrtc.org32001ef2013-08-12 23:26:21 +00001001 return reinterpret_cast<MediaStreamInterface*>(pointer)->RemoveTrack(
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001002 reinterpret_cast<VideoTrackInterface*>(j_video_track_pointer));
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001003}
1004
1005JOW(jstring, MediaStream_nativeLabel)(JNIEnv* jni, jclass, jlong j_p) {
1006 return JavaStringFromStdString(
1007 jni, reinterpret_cast<MediaStreamInterface*>(j_p)->label());
1008}
1009
1010JOW(void, MediaStream_free)(JNIEnv*, jclass, jlong j_p) {
fischman@webrtc.org32001ef2013-08-12 23:26:21 +00001011 CHECK_RELEASE(reinterpret_cast<MediaStreamInterface*>(j_p));
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001012}
1013
1014JOW(jlong, PeerConnectionFactory_nativeCreateObserver)(
1015 JNIEnv * jni, jclass, jobject j_observer) {
1016 return (jlong)new PCOJava(jni, j_observer);
1017}
1018
glaznev@webrtc.org1d53f642014-09-11 16:58:25 +00001019#if defined(ANDROID) && !defined(WEBRTC_CHROMIUM_BUILD)
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001020JOW(jboolean, PeerConnectionFactory_initializeAndroidGlobals)(
fischman@webrtc.orga150bc92014-05-14 22:00:50 +00001021 JNIEnv* jni, jclass, jobject context,
glaznev@webrtc.org99678452014-09-15 17:52:42 +00001022 jboolean initialize_audio, jboolean initialize_video,
Alex Glaznev4d2f4d12015-09-01 15:04:13 -07001023 jboolean video_hw_acceleration) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001024 bool failure = false;
Alex Glaznev4d2f4d12015-09-01 15:04:13 -07001025 video_hw_acceleration_enabled = video_hw_acceleration;
glaznev@webrtc.org99678452014-09-15 17:52:42 +00001026 if (!factory_static_initialized) {
andresp@webrtc.org85ef7702014-09-17 11:44:51 +00001027 if (initialize_video) {
perkj@webrtc.org96e4db92015-02-13 12:46:51 +00001028 failure |= webrtc::SetRenderAndroidVM(GetJVM());
perkj@webrtc.org83bc7212015-02-11 11:26:56 +00001029 failure |= AndroidVideoCapturerJni::SetAndroidObjects(jni, context);
andresp@webrtc.org85ef7702014-09-17 11:44:51 +00001030 }
glaznev@webrtc.org99678452014-09-15 17:52:42 +00001031 if (initialize_audio)
henrika@webrtc.org474d1eb2015-03-09 12:39:53 +00001032 failure |= webrtc::VoiceEngine::SetAndroidObjects(GetJVM(), context);
glaznev@webrtc.org99678452014-09-15 17:52:42 +00001033 factory_static_initialized = true;
1034 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001035 return !failure;
1036}
glaznev@webrtc.org1d53f642014-09-11 16:58:25 +00001037#endif // defined(ANDROID) && !defined(WEBRTC_CHROMIUM_BUILD)
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001038
glaznev@webrtc.org44ae4c82015-02-09 23:25:58 +00001039JOW(void, PeerConnectionFactory_initializeFieldTrials)(
1040 JNIEnv* jni, jclass, jstring j_trials_init_string) {
1041 field_trials_init_string = NULL;
1042 if (j_trials_init_string != NULL) {
1043 const char* init_string =
1044 jni->GetStringUTFChars(j_trials_init_string, NULL);
1045 int init_string_length = jni->GetStringUTFLength(j_trials_init_string);
1046 field_trials_init_string = new char[init_string_length + 1];
1047 rtc::strcpyn(field_trials_init_string, init_string_length + 1, init_string);
1048 jni->ReleaseStringUTFChars(j_trials_init_string, init_string);
perkj@webrtc.org96e4db92015-02-13 12:46:51 +00001049 LOG(LS_INFO) << "initializeFieldTrials: " << field_trials_init_string;
glaznev@webrtc.org44ae4c82015-02-09 23:25:58 +00001050 }
1051 webrtc::field_trial::InitFieldTrialsFromString(field_trials_init_string);
1052}
1053
fischman@webrtc.org540acde2014-02-13 03:56:14 +00001054// Helper struct for working around the fact that CreatePeerConnectionFactory()
1055// comes in two flavors: either entirely automagical (constructing its own
1056// threads and deleting them on teardown, but no external codec factory support)
1057// or entirely manual (requires caller to delete threads after factory
1058// teardown). This struct takes ownership of its ctor's arguments to present a
1059// single thing for Java to hold and eventually free.
1060class OwnedFactoryAndThreads {
1061 public:
1062 OwnedFactoryAndThreads(Thread* worker_thread,
1063 Thread* signaling_thread,
Alex Glaznev4d2f4d12015-09-01 15:04:13 -07001064 WebRtcVideoEncoderFactory* encoder_factory,
1065 WebRtcVideoDecoderFactory* decoder_factory,
fischman@webrtc.org540acde2014-02-13 03:56:14 +00001066 PeerConnectionFactoryInterface* factory)
1067 : worker_thread_(worker_thread),
1068 signaling_thread_(signaling_thread),
Alex Glaznev4d2f4d12015-09-01 15:04:13 -07001069 encoder_factory_(encoder_factory),
1070 decoder_factory_(decoder_factory),
fischman@webrtc.org540acde2014-02-13 03:56:14 +00001071 factory_(factory) {}
1072
1073 ~OwnedFactoryAndThreads() { CHECK_RELEASE(factory_); }
1074
1075 PeerConnectionFactoryInterface* factory() { return factory_; }
Alex Glaznev4d2f4d12015-09-01 15:04:13 -07001076 WebRtcVideoEncoderFactory* encoder_factory() { return encoder_factory_; }
1077 WebRtcVideoDecoderFactory* decoder_factory() { return decoder_factory_; }
fischman@webrtc.org540acde2014-02-13 03:56:14 +00001078
1079 private:
1080 const scoped_ptr<Thread> worker_thread_;
1081 const scoped_ptr<Thread> signaling_thread_;
Alex Glaznev4d2f4d12015-09-01 15:04:13 -07001082 WebRtcVideoEncoderFactory* encoder_factory_;
1083 WebRtcVideoDecoderFactory* decoder_factory_;
fischman@webrtc.org540acde2014-02-13 03:56:14 +00001084 PeerConnectionFactoryInterface* factory_; // Const after ctor except dtor.
1085};
1086
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001087JOW(jlong, PeerConnectionFactory_nativeCreatePeerConnectionFactory)(
1088 JNIEnv* jni, jclass) {
fischman@webrtc.org2c98af72014-05-14 17:33:32 +00001089 // talk/ assumes pretty widely that the current Thread is ThreadManager'd, but
1090 // ThreadManager only WrapCurrentThread()s the thread where it is first
1091 // created. Since the semantics around when auto-wrapping happens in
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +00001092 // webrtc/base/ are convoluted, we simply wrap here to avoid having to think
fischman@webrtc.org2c98af72014-05-14 17:33:32 +00001093 // about ramifications of auto-wrapping there.
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +00001094 rtc::ThreadManager::Instance()->WrapCurrentThread();
fischman@webrtc.org32001ef2013-08-12 23:26:21 +00001095 webrtc::Trace::CreateTrace();
fischman@webrtc.org540acde2014-02-13 03:56:14 +00001096 Thread* worker_thread = new Thread();
1097 worker_thread->SetName("worker_thread", NULL);
1098 Thread* signaling_thread = new Thread();
1099 signaling_thread->SetName("signaling_thread", NULL);
henrikg91d6ede2015-09-17 00:24:34 -07001100 RTC_CHECK(worker_thread->Start() && signaling_thread->Start())
andresp@webrtc.org4d19e052014-09-09 11:45:44 +00001101 << "Failed to start threads";
Alex Glaznev4d2f4d12015-09-01 15:04:13 -07001102 WebRtcVideoEncoderFactory* encoder_factory = nullptr;
1103 WebRtcVideoDecoderFactory* decoder_factory = nullptr;
glaznev@webrtc.org99678452014-09-15 17:52:42 +00001104#if defined(ANDROID) && !defined(WEBRTC_CHROMIUM_BUILD)
Alex Glaznev4d2f4d12015-09-01 15:04:13 -07001105 if (video_hw_acceleration_enabled) {
1106 encoder_factory = new MediaCodecVideoEncoderFactory();
1107 decoder_factory = new MediaCodecVideoDecoderFactory();
glaznev@webrtc.orgdea51732014-12-01 20:02:13 +00001108 }
fischman@webrtc.org540acde2014-02-13 03:56:14 +00001109#endif
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +00001110 rtc::scoped_refptr<PeerConnectionFactoryInterface> factory(
fischman@webrtc.org540acde2014-02-13 03:56:14 +00001111 webrtc::CreatePeerConnectionFactory(worker_thread,
1112 signaling_thread,
1113 NULL,
Alex Glaznev4d2f4d12015-09-01 15:04:13 -07001114 encoder_factory,
1115 decoder_factory));
Patrik Höglunde1c5ec72015-09-17 17:20:38 +02001116 RTC_CHECK(factory) << "Failed to create the peer connection factory; "
1117 << "WebRTC/libjingle init likely failed on this device";
fischman@webrtc.org540acde2014-02-13 03:56:14 +00001118 OwnedFactoryAndThreads* owned_factory = new OwnedFactoryAndThreads(
Alex Glaznev4d2f4d12015-09-01 15:04:13 -07001119 worker_thread, signaling_thread,
1120 encoder_factory, decoder_factory,
1121 factory.release());
fischman@webrtc.org540acde2014-02-13 03:56:14 +00001122 return jlongFromPointer(owned_factory);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001123}
1124
1125JOW(void, PeerConnectionFactory_freeFactory)(JNIEnv*, jclass, jlong j_p) {
fischman@webrtc.org540acde2014-02-13 03:56:14 +00001126 delete reinterpret_cast<OwnedFactoryAndThreads*>(j_p);
glaznev@webrtc.org44ae4c82015-02-09 23:25:58 +00001127 if (field_trials_init_string) {
1128 webrtc::field_trial::InitFieldTrialsFromString(NULL);
1129 delete field_trials_init_string;
1130 field_trials_init_string = NULL;
1131 }
fischman@webrtc.org32001ef2013-08-12 23:26:21 +00001132 webrtc::Trace::ReturnTrace();
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001133}
1134
fischman@webrtc.org540acde2014-02-13 03:56:14 +00001135static PeerConnectionFactoryInterface* factoryFromJava(jlong j_p) {
1136 return reinterpret_cast<OwnedFactoryAndThreads*>(j_p)->factory();
1137}
1138
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001139JOW(jlong, PeerConnectionFactory_nativeCreateLocalMediaStream)(
1140 JNIEnv* jni, jclass, jlong native_factory, jstring label) {
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +00001141 rtc::scoped_refptr<PeerConnectionFactoryInterface> factory(
fischman@webrtc.org540acde2014-02-13 03:56:14 +00001142 factoryFromJava(native_factory));
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +00001143 rtc::scoped_refptr<MediaStreamInterface> stream(
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001144 factory->CreateLocalMediaStream(JavaToStdString(jni, label)));
1145 return (jlong)stream.release();
1146}
1147
1148JOW(jlong, PeerConnectionFactory_nativeCreateVideoSource)(
1149 JNIEnv* jni, jclass, jlong native_factory, jlong native_capturer,
1150 jobject j_constraints) {
fischman@webrtc.org540acde2014-02-13 03:56:14 +00001151 scoped_ptr<ConstraintsWrapper> constraints(
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001152 new ConstraintsWrapper(jni, j_constraints));
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +00001153 rtc::scoped_refptr<PeerConnectionFactoryInterface> factory(
fischman@webrtc.org540acde2014-02-13 03:56:14 +00001154 factoryFromJava(native_factory));
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +00001155 rtc::scoped_refptr<VideoSourceInterface> source(
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001156 factory->CreateVideoSource(
1157 reinterpret_cast<cricket::VideoCapturer*>(native_capturer),
1158 constraints.get()));
1159 return (jlong)source.release();
1160}
1161
1162JOW(jlong, PeerConnectionFactory_nativeCreateVideoTrack)(
1163 JNIEnv* jni, jclass, jlong native_factory, jstring id,
1164 jlong native_source) {
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +00001165 rtc::scoped_refptr<PeerConnectionFactoryInterface> factory(
fischman@webrtc.org540acde2014-02-13 03:56:14 +00001166 factoryFromJava(native_factory));
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +00001167 rtc::scoped_refptr<VideoTrackInterface> track(
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001168 factory->CreateVideoTrack(
1169 JavaToStdString(jni, id),
1170 reinterpret_cast<VideoSourceInterface*>(native_source)));
1171 return (jlong)track.release();
1172}
1173
fischman@webrtc.org3eda6432014-02-13 04:01:04 +00001174JOW(jlong, PeerConnectionFactory_nativeCreateAudioSource)(
1175 JNIEnv* jni, jclass, jlong native_factory, jobject j_constraints) {
1176 scoped_ptr<ConstraintsWrapper> constraints(
1177 new ConstraintsWrapper(jni, j_constraints));
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +00001178 rtc::scoped_refptr<PeerConnectionFactoryInterface> factory(
fischman@webrtc.org540acde2014-02-13 03:56:14 +00001179 factoryFromJava(native_factory));
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +00001180 rtc::scoped_refptr<AudioSourceInterface> source(
fischman@webrtc.org3eda6432014-02-13 04:01:04 +00001181 factory->CreateAudioSource(constraints.get()));
1182 return (jlong)source.release();
1183}
1184
1185JOW(jlong, PeerConnectionFactory_nativeCreateAudioTrack)(
1186 JNIEnv* jni, jclass, jlong native_factory, jstring id,
1187 jlong native_source) {
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +00001188 rtc::scoped_refptr<PeerConnectionFactoryInterface> factory(
fischman@webrtc.org3eda6432014-02-13 04:01:04 +00001189 factoryFromJava(native_factory));
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +00001190 rtc::scoped_refptr<AudioTrackInterface> track(factory->CreateAudioTrack(
fischman@webrtc.org3eda6432014-02-13 04:01:04 +00001191 JavaToStdString(jni, id),
1192 reinterpret_cast<AudioSourceInterface*>(native_source)));
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001193 return (jlong)track.release();
1194}
1195
phoglund@webrtc.org006521d2015-02-12 09:23:59 +00001196JOW(void, PeerConnectionFactory_nativeSetOptions)(
1197 JNIEnv* jni, jclass, jlong native_factory, jobject options) {
1198 rtc::scoped_refptr<PeerConnectionFactoryInterface> factory(
1199 factoryFromJava(native_factory));
1200 jclass options_class = jni->GetObjectClass(options);
1201 jfieldID network_ignore_mask_field =
1202 jni->GetFieldID(options_class, "networkIgnoreMask", "I");
1203 int network_ignore_mask =
1204 jni->GetIntField(options, network_ignore_mask_field);
Jiayang Liu61093862015-07-08 15:25:45 -07001205
1206 jfieldID disable_encryption_field =
1207 jni->GetFieldID(options_class, "disableEncryption", "Z");
1208 bool disable_encryption =
1209 jni->GetBooleanField(options, disable_encryption_field);
1210
phoglund@webrtc.org006521d2015-02-12 09:23:59 +00001211 PeerConnectionFactoryInterface::Options options_to_set;
1212
1213 // This doesn't necessarily match the c++ version of this struct; feel free
1214 // to add more parameters as necessary.
1215 options_to_set.network_ignore_mask = network_ignore_mask;
Jiayang Liu61093862015-07-08 15:25:45 -07001216 options_to_set.disable_encryption = disable_encryption;
phoglund@webrtc.org006521d2015-02-12 09:23:59 +00001217 factory->SetOptions(options_to_set);
1218}
1219
Alex Glaznev4d2f4d12015-09-01 15:04:13 -07001220JOW(void, PeerConnectionFactory_nativeSetVideoHwAccelerationOptions)(
1221 JNIEnv* jni, jclass, jlong native_factory, jobject render_egl_context) {
1222#if defined(ANDROID) && !defined(WEBRTC_CHROMIUM_BUILD)
1223 OwnedFactoryAndThreads* owned_factory =
1224 reinterpret_cast<OwnedFactoryAndThreads*>(native_factory);
1225 MediaCodecVideoDecoderFactory* decoder_factory =
1226 static_cast<MediaCodecVideoDecoderFactory*>
1227 (owned_factory->decoder_factory());
1228 if (decoder_factory) {
1229 LOG(LS_INFO) << "Set EGL context for HW acceleration.";
1230 decoder_factory->SetEGLContext(jni, render_egl_context);
1231 }
1232#endif
1233}
1234
1235
Jiayang Liucac1b382015-04-30 12:35:24 -07001236static std::string
1237GetJavaEnumName(JNIEnv* jni, const std::string& className, jobject j_enum) {
1238 jclass enumClass = FindClass(jni, className.c_str());
1239 jmethodID nameMethod =
1240 GetMethodID(jni, enumClass, "name", "()Ljava/lang/String;");
1241 jstring name =
1242 reinterpret_cast<jstring>(jni->CallObjectMethod(j_enum, nameMethod));
1243 CHECK_EXCEPTION(jni) << "error during CallObjectMethod for "
1244 << className << ".name";
1245 return JavaToStdString(jni, name);
1246}
1247
1248static PeerConnectionInterface::IceTransportsType
1249JavaIceTransportsTypeToNativeType(JNIEnv* jni, jobject j_ice_transports_type) {
1250 std::string enum_name = GetJavaEnumName(
1251 jni, "org/webrtc/PeerConnection$IceTransportsType",
1252 j_ice_transports_type);
1253
1254 if (enum_name == "ALL")
1255 return PeerConnectionInterface::kAll;
1256
1257 if (enum_name == "RELAY")
1258 return PeerConnectionInterface::kRelay;
1259
1260 if (enum_name == "NOHOST")
1261 return PeerConnectionInterface::kNoHost;
1262
1263 if (enum_name == "NONE")
1264 return PeerConnectionInterface::kNone;
1265
henrikg91d6ede2015-09-17 00:24:34 -07001266 RTC_CHECK(false) << "Unexpected IceTransportsType enum_name " << enum_name;
Jiayang Liucac1b382015-04-30 12:35:24 -07001267 return PeerConnectionInterface::kAll;
1268}
1269
1270static PeerConnectionInterface::BundlePolicy
1271JavaBundlePolicyToNativeType(JNIEnv* jni, jobject j_bundle_policy) {
1272 std::string enum_name = GetJavaEnumName(
1273 jni, "org/webrtc/PeerConnection$BundlePolicy",
1274 j_bundle_policy);
1275
1276 if (enum_name == "BALANCED")
1277 return PeerConnectionInterface::kBundlePolicyBalanced;
1278
1279 if (enum_name == "MAXBUNDLE")
1280 return PeerConnectionInterface::kBundlePolicyMaxBundle;
1281
1282 if (enum_name == "MAXCOMPAT")
1283 return PeerConnectionInterface::kBundlePolicyMaxCompat;
1284
henrikg91d6ede2015-09-17 00:24:34 -07001285 RTC_CHECK(false) << "Unexpected BundlePolicy enum_name " << enum_name;
Jiayang Liucac1b382015-04-30 12:35:24 -07001286 return PeerConnectionInterface::kBundlePolicyBalanced;
1287}
1288
Peter Thatcheraf55ccc2015-05-21 07:48:41 -07001289static PeerConnectionInterface::RtcpMuxPolicy
1290JavaRtcpMuxPolicyToNativeType(JNIEnv* jni, jobject j_rtcp_mux_policy) {
1291 std::string enum_name = GetJavaEnumName(
1292 jni, "org/webrtc/PeerConnection$RtcpMuxPolicy",
1293 j_rtcp_mux_policy);
1294
1295 if (enum_name == "NEGOTIATE")
1296 return PeerConnectionInterface::kRtcpMuxPolicyNegotiate;
1297
1298 if (enum_name == "REQUIRE")
1299 return PeerConnectionInterface::kRtcpMuxPolicyRequire;
1300
henrikg91d6ede2015-09-17 00:24:34 -07001301 RTC_CHECK(false) << "Unexpected RtcpMuxPolicy enum_name " << enum_name;
Peter Thatcheraf55ccc2015-05-21 07:48:41 -07001302 return PeerConnectionInterface::kRtcpMuxPolicyNegotiate;
1303}
1304
Jiayang Liucac1b382015-04-30 12:35:24 -07001305static PeerConnectionInterface::TcpCandidatePolicy
1306JavaTcpCandidatePolicyToNativeType(
1307 JNIEnv* jni, jobject j_tcp_candidate_policy) {
1308 std::string enum_name = GetJavaEnumName(
1309 jni, "org/webrtc/PeerConnection$TcpCandidatePolicy",
1310 j_tcp_candidate_policy);
1311
1312 if (enum_name == "ENABLED")
1313 return PeerConnectionInterface::kTcpCandidatePolicyEnabled;
1314
1315 if (enum_name == "DISABLED")
1316 return PeerConnectionInterface::kTcpCandidatePolicyDisabled;
1317
henrikg91d6ede2015-09-17 00:24:34 -07001318 RTC_CHECK(false) << "Unexpected TcpCandidatePolicy enum_name " << enum_name;
Jiayang Liucac1b382015-04-30 12:35:24 -07001319 return PeerConnectionInterface::kTcpCandidatePolicyEnabled;
1320}
1321
glaznev97579a42015-09-01 11:31:27 -07001322static rtc::KeyType JavaKeyTypeToNativeType(JNIEnv* jni, jobject j_key_type) {
1323 std::string enum_name = GetJavaEnumName(
1324 jni, "org/webrtc/PeerConnection$KeyType", j_key_type);
1325
1326 if (enum_name == "RSA")
1327 return rtc::KT_RSA;
1328 if (enum_name == "ECDSA")
1329 return rtc::KT_ECDSA;
1330
henrikg91d6ede2015-09-17 00:24:34 -07001331 RTC_CHECK(false) << "Unexpected KeyType enum_name " << enum_name;
glaznev97579a42015-09-01 11:31:27 -07001332 return rtc::KT_ECDSA;
1333}
1334
honghaiz1f429e32015-09-28 07:57:34 -07001335static PeerConnectionInterface::ContinualGatheringPolicy
1336 JavaContinualGatheringPolicyToNativeType(
1337 JNIEnv* jni, jobject j_gathering_policy) {
1338 std::string enum_name = GetJavaEnumName(
1339 jni, "org/webrtc/PeerConnection$ContinualGatheringPolicy",
1340 j_gathering_policy);
1341 if (enum_name == "GATHER_ONCE")
1342 return PeerConnectionInterface::GATHER_ONCE;
1343
1344 if (enum_name == "GATHER_CONTINUALLY")
1345 return PeerConnectionInterface::GATHER_CONTINUALLY;
1346
1347 RTC_CHECK(false) << "Unexpected ContinualGatheringPolicy enum name "
1348 << enum_name;
1349 return PeerConnectionInterface::GATHER_ONCE;
1350}
1351
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001352static void JavaIceServersToJsepIceServers(
1353 JNIEnv* jni, jobject j_ice_servers,
1354 PeerConnectionInterface::IceServers* ice_servers) {
1355 jclass list_class = GetObjectClass(jni, j_ice_servers);
1356 jmethodID iterator_id = GetMethodID(
1357 jni, list_class, "iterator", "()Ljava/util/Iterator;");
1358 jobject iterator = jni->CallObjectMethod(j_ice_servers, iterator_id);
andresp@webrtc.org4d19e052014-09-09 11:45:44 +00001359 CHECK_EXCEPTION(jni) << "error during CallObjectMethod";
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001360 jmethodID iterator_has_next = GetMethodID(
1361 jni, GetObjectClass(jni, iterator), "hasNext", "()Z");
1362 jmethodID iterator_next = GetMethodID(
1363 jni, GetObjectClass(jni, iterator), "next", "()Ljava/lang/Object;");
1364 while (jni->CallBooleanMethod(iterator, iterator_has_next)) {
andresp@webrtc.org4d19e052014-09-09 11:45:44 +00001365 CHECK_EXCEPTION(jni) << "error during CallBooleanMethod";
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001366 jobject j_ice_server = jni->CallObjectMethod(iterator, iterator_next);
andresp@webrtc.org4d19e052014-09-09 11:45:44 +00001367 CHECK_EXCEPTION(jni) << "error during CallObjectMethod";
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001368 jclass j_ice_server_class = GetObjectClass(jni, j_ice_server);
1369 jfieldID j_ice_server_uri_id =
1370 GetFieldID(jni, j_ice_server_class, "uri", "Ljava/lang/String;");
1371 jfieldID j_ice_server_username_id =
1372 GetFieldID(jni, j_ice_server_class, "username", "Ljava/lang/String;");
1373 jfieldID j_ice_server_password_id =
1374 GetFieldID(jni, j_ice_server_class, "password", "Ljava/lang/String;");
1375 jstring uri = reinterpret_cast<jstring>(
1376 GetObjectField(jni, j_ice_server, j_ice_server_uri_id));
1377 jstring username = reinterpret_cast<jstring>(
1378 GetObjectField(jni, j_ice_server, j_ice_server_username_id));
1379 jstring password = reinterpret_cast<jstring>(
1380 GetObjectField(jni, j_ice_server, j_ice_server_password_id));
1381 PeerConnectionInterface::IceServer server;
1382 server.uri = JavaToStdString(jni, uri);
1383 server.username = JavaToStdString(jni, username);
1384 server.password = JavaToStdString(jni, password);
1385 ice_servers->push_back(server);
1386 }
andresp@webrtc.org4d19e052014-09-09 11:45:44 +00001387 CHECK_EXCEPTION(jni) << "error during CallBooleanMethod";
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001388}
1389
deadbeefa67696b2015-09-29 11:56:26 -07001390static void JavaRTCConfigurationToJsepRTCConfiguration(
1391 JNIEnv* jni,
1392 jobject j_rtc_config,
1393 PeerConnectionInterface::RTCConfiguration* rtc_config) {
Jiayang Liucac1b382015-04-30 12:35:24 -07001394 jclass j_rtc_config_class = GetObjectClass(jni, j_rtc_config);
1395
1396 jfieldID j_ice_transports_type_id = GetFieldID(
1397 jni, j_rtc_config_class, "iceTransportsType",
1398 "Lorg/webrtc/PeerConnection$IceTransportsType;");
1399 jobject j_ice_transports_type = GetObjectField(
1400 jni, j_rtc_config, j_ice_transports_type_id);
1401
1402 jfieldID j_bundle_policy_id = GetFieldID(
1403 jni, j_rtc_config_class, "bundlePolicy",
1404 "Lorg/webrtc/PeerConnection$BundlePolicy;");
1405 jobject j_bundle_policy = GetObjectField(
1406 jni, j_rtc_config, j_bundle_policy_id);
1407
Peter Thatcheraf55ccc2015-05-21 07:48:41 -07001408 jfieldID j_rtcp_mux_policy_id = GetFieldID(
1409 jni, j_rtc_config_class, "rtcpMuxPolicy",
1410 "Lorg/webrtc/PeerConnection$RtcpMuxPolicy;");
1411 jobject j_rtcp_mux_policy = GetObjectField(
1412 jni, j_rtc_config, j_rtcp_mux_policy_id);
1413
Jiayang Liucac1b382015-04-30 12:35:24 -07001414 jfieldID j_tcp_candidate_policy_id = GetFieldID(
1415 jni, j_rtc_config_class, "tcpCandidatePolicy",
1416 "Lorg/webrtc/PeerConnection$TcpCandidatePolicy;");
1417 jobject j_tcp_candidate_policy = GetObjectField(
1418 jni, j_rtc_config, j_tcp_candidate_policy_id);
1419
1420 jfieldID j_ice_servers_id = GetFieldID(
glaznev97579a42015-09-01 11:31:27 -07001421 jni, j_rtc_config_class, "iceServers", "Ljava/util/List;");
Jiayang Liucac1b382015-04-30 12:35:24 -07001422 jobject j_ice_servers = GetObjectField(jni, j_rtc_config, j_ice_servers_id);
1423
honghaiz4edc39c2015-09-01 09:53:56 -07001424 jfieldID j_audio_jitter_buffer_max_packets_id =
1425 GetFieldID(jni, j_rtc_config_class, "audioJitterBufferMaxPackets", "I");
Henrik Lundin5263b3c2015-06-01 10:29:41 +02001426 jfieldID j_audio_jitter_buffer_fast_accelerate_id = GetFieldID(
1427 jni, j_rtc_config_class, "audioJitterBufferFastAccelerate", "Z");
Jiayang Liucac1b382015-04-30 12:35:24 -07001428
honghaiz4edc39c2015-09-01 09:53:56 -07001429 jfieldID j_ice_connection_receiving_timeout_id =
1430 GetFieldID(jni, j_rtc_config_class, "iceConnectionReceivingTimeout", "I");
1431
honghaiz1f429e32015-09-28 07:57:34 -07001432 jfieldID j_continual_gathering_policy_id =
1433 GetFieldID(jni, j_rtc_config_class, "continualGatheringPolicy",
1434 "Lorg/webrtc/PeerConnection$ContinualGatheringPolicy;");
1435 jobject j_continual_gathering_policy =
1436 GetObjectField(jni, j_rtc_config, j_continual_gathering_policy_id);
1437
deadbeefa67696b2015-09-29 11:56:26 -07001438 rtc_config->type =
deadbeef7603c762015-09-23 17:37:11 -07001439 JavaIceTransportsTypeToNativeType(jni, j_ice_transports_type);
deadbeefa67696b2015-09-29 11:56:26 -07001440 rtc_config->bundle_policy =
1441 JavaBundlePolicyToNativeType(jni, j_bundle_policy);
1442 rtc_config->rtcp_mux_policy =
deadbeef7603c762015-09-23 17:37:11 -07001443 JavaRtcpMuxPolicyToNativeType(jni, j_rtcp_mux_policy);
deadbeefa67696b2015-09-29 11:56:26 -07001444 rtc_config->tcp_candidate_policy =
deadbeef7603c762015-09-23 17:37:11 -07001445 JavaTcpCandidatePolicyToNativeType(jni, j_tcp_candidate_policy);
deadbeefa67696b2015-09-29 11:56:26 -07001446 JavaIceServersToJsepIceServers(jni, j_ice_servers, &rtc_config->servers);
1447 rtc_config->audio_jitter_buffer_max_packets =
deadbeef7603c762015-09-23 17:37:11 -07001448 GetIntField(jni, j_rtc_config, j_audio_jitter_buffer_max_packets_id);
deadbeefa67696b2015-09-29 11:56:26 -07001449 rtc_config->audio_jitter_buffer_fast_accelerate = GetBooleanField(
deadbeef7603c762015-09-23 17:37:11 -07001450 jni, j_rtc_config, j_audio_jitter_buffer_fast_accelerate_id);
deadbeefa67696b2015-09-29 11:56:26 -07001451 rtc_config->ice_connection_receiving_timeout =
deadbeef7603c762015-09-23 17:37:11 -07001452 GetIntField(jni, j_rtc_config, j_ice_connection_receiving_timeout_id);
deadbeefa67696b2015-09-29 11:56:26 -07001453 rtc_config->continual_gathering_policy =
honghaiz1f429e32015-09-28 07:57:34 -07001454 JavaContinualGatheringPolicyToNativeType(
1455 jni, j_continual_gathering_policy);
deadbeefa67696b2015-09-29 11:56:26 -07001456}
1457
1458JOW(jlong, PeerConnectionFactory_nativeCreatePeerConnection)(
1459 JNIEnv *jni, jclass, jlong factory, jobject j_rtc_config,
1460 jobject j_constraints, jlong observer_p) {
1461 rtc::scoped_refptr<PeerConnectionFactoryInterface> f(
1462 reinterpret_cast<PeerConnectionFactoryInterface*>(
1463 factoryFromJava(factory)));
1464
1465 PeerConnectionInterface::RTCConfiguration rtc_config;
1466 JavaRTCConfigurationToJsepRTCConfiguration(jni, j_rtc_config, &rtc_config);
1467
1468 jclass j_rtc_config_class = GetObjectClass(jni, j_rtc_config);
1469 jfieldID j_key_type_id = GetFieldID(jni, j_rtc_config_class, "keyType",
1470 "Lorg/webrtc/PeerConnection$KeyType;");
1471 jobject j_key_type = GetObjectField(jni, j_rtc_config, j_key_type_id);
Jiayang Liucac1b382015-04-30 12:35:24 -07001472
glaznev97579a42015-09-01 11:31:27 -07001473 // Create ECDSA certificate.
1474 if (JavaKeyTypeToNativeType(jni, j_key_type) == rtc::KT_ECDSA) {
1475 scoped_ptr<rtc::SSLIdentity> ssl_identity(
1476 rtc::SSLIdentity::Generate(webrtc::kIdentityName, rtc::KT_ECDSA));
1477 if (ssl_identity.get()) {
1478 rtc_config.certificates.push_back(
1479 rtc::RTCCertificate::Create(ssl_identity.Pass()));
1480 LOG(LS_INFO) << "ECDSA certificate created.";
1481 } else {
1482 // Failing to create certificate should not abort peer connection
1483 // creation. Instead default encryption (currently RSA) will be used.
1484 LOG(LS_WARNING) <<
1485 "Failed to generate SSLIdentity. Default encryption will be used.";
1486 }
1487 }
1488
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001489 PCOJava* observer = reinterpret_cast<PCOJava*>(observer_p);
1490 observer->SetConstraints(new ConstraintsWrapper(jni, j_constraints));
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +00001491 rtc::scoped_refptr<PeerConnectionInterface> pc(f->CreatePeerConnection(
Jiayang Liucac1b382015-04-30 12:35:24 -07001492 rtc_config, observer->constraints(), NULL, NULL, observer));
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001493 return (jlong)pc.release();
1494}
1495
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +00001496static rtc::scoped_refptr<PeerConnectionInterface> ExtractNativePC(
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001497 JNIEnv* jni, jobject j_pc) {
1498 jfieldID native_pc_id = GetFieldID(jni,
1499 GetObjectClass(jni, j_pc), "nativePeerConnection", "J");
1500 jlong j_p = GetLongField(jni, j_pc, native_pc_id);
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +00001501 return rtc::scoped_refptr<PeerConnectionInterface>(
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001502 reinterpret_cast<PeerConnectionInterface*>(j_p));
1503}
1504
1505JOW(jobject, PeerConnection_getLocalDescription)(JNIEnv* jni, jobject j_pc) {
1506 const SessionDescriptionInterface* sdp =
1507 ExtractNativePC(jni, j_pc)->local_description();
1508 return sdp ? JavaSdpFromNativeSdp(jni, sdp) : NULL;
1509}
1510
1511JOW(jobject, PeerConnection_getRemoteDescription)(JNIEnv* jni, jobject j_pc) {
1512 const SessionDescriptionInterface* sdp =
1513 ExtractNativePC(jni, j_pc)->remote_description();
1514 return sdp ? JavaSdpFromNativeSdp(jni, sdp) : NULL;
1515}
1516
henrike@webrtc.org723d6832013-07-12 16:04:50 +00001517JOW(jobject, PeerConnection_createDataChannel)(
1518 JNIEnv* jni, jobject j_pc, jstring j_label, jobject j_init) {
1519 DataChannelInit init = JavaDataChannelInitToNative(jni, j_init);
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +00001520 rtc::scoped_refptr<DataChannelInterface> channel(
henrike@webrtc.org723d6832013-07-12 16:04:50 +00001521 ExtractNativePC(jni, j_pc)->CreateDataChannel(
1522 JavaToStdString(jni, j_label), &init));
fischman@webrtc.org87881672013-09-03 18:58:12 +00001523 // Mustn't pass channel.get() directly through NewObject to avoid reading its
1524 // vararg parameter as 64-bit and reading memory that doesn't belong to the
1525 // 32-bit parameter.
1526 jlong nativeChannelPtr = jlongFromPointer(channel.get());
henrikg91d6ede2015-09-17 00:24:34 -07001527 RTC_CHECK(nativeChannelPtr) << "Failed to create DataChannel";
henrike@webrtc.org723d6832013-07-12 16:04:50 +00001528 jclass j_data_channel_class = FindClass(jni, "org/webrtc/DataChannel");
1529 jmethodID j_data_channel_ctor = GetMethodID(
1530 jni, j_data_channel_class, "<init>", "(J)V");
1531 jobject j_channel = jni->NewObject(
fischman@webrtc.org87881672013-09-03 18:58:12 +00001532 j_data_channel_class, j_data_channel_ctor, nativeChannelPtr);
andresp@webrtc.org4d19e052014-09-09 11:45:44 +00001533 CHECK_EXCEPTION(jni) << "error during NewObject";
henrike@webrtc.org723d6832013-07-12 16:04:50 +00001534 // Channel is now owned by Java object, and will be freed from there.
fischman@webrtc.org32001ef2013-08-12 23:26:21 +00001535 int bumped_count = channel->AddRef();
henrikg91d6ede2015-09-17 00:24:34 -07001536 RTC_CHECK(bumped_count == 2) << "Unexpected refcount";
henrike@webrtc.org723d6832013-07-12 16:04:50 +00001537 return j_channel;
1538}
1539
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001540JOW(void, PeerConnection_createOffer)(
1541 JNIEnv* jni, jobject j_pc, jobject j_observer, jobject j_constraints) {
1542 ConstraintsWrapper* constraints =
1543 new ConstraintsWrapper(jni, j_constraints);
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +00001544 rtc::scoped_refptr<CreateSdpObserverWrapper> observer(
1545 new rtc::RefCountedObject<CreateSdpObserverWrapper>(
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001546 jni, j_observer, constraints));
1547 ExtractNativePC(jni, j_pc)->CreateOffer(observer, constraints);
1548}
1549
1550JOW(void, PeerConnection_createAnswer)(
1551 JNIEnv* jni, jobject j_pc, jobject j_observer, jobject j_constraints) {
1552 ConstraintsWrapper* constraints =
1553 new ConstraintsWrapper(jni, j_constraints);
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +00001554 rtc::scoped_refptr<CreateSdpObserverWrapper> observer(
1555 new rtc::RefCountedObject<CreateSdpObserverWrapper>(
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001556 jni, j_observer, constraints));
1557 ExtractNativePC(jni, j_pc)->CreateAnswer(observer, constraints);
1558}
1559
1560// Helper to create a SessionDescriptionInterface from a SessionDescription.
1561static SessionDescriptionInterface* JavaSdpToNativeSdp(
1562 JNIEnv* jni, jobject j_sdp) {
1563 jfieldID j_type_id = GetFieldID(
1564 jni, GetObjectClass(jni, j_sdp), "type",
1565 "Lorg/webrtc/SessionDescription$Type;");
1566 jobject j_type = GetObjectField(jni, j_sdp, j_type_id);
1567 jmethodID j_canonical_form_id = GetMethodID(
1568 jni, GetObjectClass(jni, j_type), "canonicalForm",
1569 "()Ljava/lang/String;");
1570 jstring j_type_string = (jstring)jni->CallObjectMethod(
1571 j_type, j_canonical_form_id);
andresp@webrtc.org4d19e052014-09-09 11:45:44 +00001572 CHECK_EXCEPTION(jni) << "error during CallObjectMethod";
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001573 std::string std_type = JavaToStdString(jni, j_type_string);
1574
1575 jfieldID j_description_id = GetFieldID(
1576 jni, GetObjectClass(jni, j_sdp), "description", "Ljava/lang/String;");
1577 jstring j_description = (jstring)GetObjectField(jni, j_sdp, j_description_id);
1578 std::string std_description = JavaToStdString(jni, j_description);
1579
1580 return webrtc::CreateSessionDescription(
1581 std_type, std_description, NULL);
1582}
1583
1584JOW(void, PeerConnection_setLocalDescription)(
1585 JNIEnv* jni, jobject j_pc,
1586 jobject j_observer, jobject j_sdp) {
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +00001587 rtc::scoped_refptr<SetSdpObserverWrapper> observer(
1588 new rtc::RefCountedObject<SetSdpObserverWrapper>(
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001589 jni, j_observer, reinterpret_cast<ConstraintsWrapper*>(NULL)));
1590 ExtractNativePC(jni, j_pc)->SetLocalDescription(
1591 observer, JavaSdpToNativeSdp(jni, j_sdp));
1592}
1593
1594JOW(void, PeerConnection_setRemoteDescription)(
1595 JNIEnv* jni, jobject j_pc,
1596 jobject j_observer, jobject j_sdp) {
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +00001597 rtc::scoped_refptr<SetSdpObserverWrapper> observer(
1598 new rtc::RefCountedObject<SetSdpObserverWrapper>(
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001599 jni, j_observer, reinterpret_cast<ConstraintsWrapper*>(NULL)));
1600 ExtractNativePC(jni, j_pc)->SetRemoteDescription(
1601 observer, JavaSdpToNativeSdp(jni, j_sdp));
1602}
1603
deadbeefa67696b2015-09-29 11:56:26 -07001604JOW(jboolean, PeerConnection_setConfiguration)(
1605 JNIEnv* jni, jobject j_pc, jobject j_rtc_config) {
1606 PeerConnectionInterface::RTCConfiguration rtc_config;
1607 JavaRTCConfigurationToJsepRTCConfiguration(jni, j_rtc_config, &rtc_config);
1608 return ExtractNativePC(jni, j_pc)->SetConfiguration(rtc_config);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001609}
1610
1611JOW(jboolean, PeerConnection_nativeAddIceCandidate)(
1612 JNIEnv* jni, jobject j_pc, jstring j_sdp_mid,
1613 jint j_sdp_mline_index, jstring j_candidate_sdp) {
1614 std::string sdp_mid = JavaToStdString(jni, j_sdp_mid);
1615 std::string sdp = JavaToStdString(jni, j_candidate_sdp);
fischman@webrtc.org540acde2014-02-13 03:56:14 +00001616 scoped_ptr<IceCandidateInterface> candidate(
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001617 webrtc::CreateIceCandidate(sdp_mid, j_sdp_mline_index, sdp, NULL));
1618 return ExtractNativePC(jni, j_pc)->AddIceCandidate(candidate.get());
1619}
1620
1621JOW(jboolean, PeerConnection_nativeAddLocalStream)(
perkj@webrtc.orgc2dd5ee2014-11-04 11:31:29 +00001622 JNIEnv* jni, jobject j_pc, jlong native_stream) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001623 return ExtractNativePC(jni, j_pc)->AddStream(
perkj@webrtc.orgc2dd5ee2014-11-04 11:31:29 +00001624 reinterpret_cast<MediaStreamInterface*>(native_stream));
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001625}
1626
1627JOW(void, PeerConnection_nativeRemoveLocalStream)(
1628 JNIEnv* jni, jobject j_pc, jlong native_stream) {
1629 ExtractNativePC(jni, j_pc)->RemoveStream(
1630 reinterpret_cast<MediaStreamInterface*>(native_stream));
1631}
1632
deadbeef4139c0f2015-10-06 12:29:25 -07001633JOW(jobject, PeerConnection_nativeGetSenders)(JNIEnv* jni, jobject j_pc) {
1634 jclass j_array_list_class = FindClass(jni, "java/util/ArrayList");
1635 jmethodID j_array_list_ctor =
1636 GetMethodID(jni, j_array_list_class, "<init>", "()V");
1637 jmethodID j_array_list_add =
1638 GetMethodID(jni, j_array_list_class, "add", "(Ljava/lang/Object;)Z");
1639 jobject j_senders = jni->NewObject(j_array_list_class, j_array_list_ctor);
1640 CHECK_EXCEPTION(jni) << "error during NewObject";
1641
1642 jclass j_rtp_sender_class = FindClass(jni, "org/webrtc/RtpSender");
1643 jmethodID j_rtp_sender_ctor =
1644 GetMethodID(jni, j_rtp_sender_class, "<init>", "(J)V");
1645
1646 auto senders = ExtractNativePC(jni, j_pc)->GetSenders();
1647 for (const auto& sender : senders) {
1648 jlong nativeSenderPtr = jlongFromPointer(sender.get());
1649 jobject j_sender =
1650 jni->NewObject(j_rtp_sender_class, j_rtp_sender_ctor, nativeSenderPtr);
1651 CHECK_EXCEPTION(jni) << "error during NewObject";
1652 // Sender is now owned by Java object, and will be freed from there.
1653 sender->AddRef();
1654 jni->CallBooleanMethod(j_senders, j_array_list_add, j_sender);
1655 CHECK_EXCEPTION(jni) << "error during CallBooleanMethod";
1656 }
1657 return j_senders;
1658}
1659
1660JOW(jobject, PeerConnection_nativeGetReceivers)(JNIEnv* jni, jobject j_pc) {
1661 jclass j_array_list_class = FindClass(jni, "java/util/ArrayList");
1662 jmethodID j_array_list_ctor =
1663 GetMethodID(jni, j_array_list_class, "<init>", "()V");
1664 jmethodID j_array_list_add =
1665 GetMethodID(jni, j_array_list_class, "add", "(Ljava/lang/Object;)Z");
1666 jobject j_receivers = jni->NewObject(j_array_list_class, j_array_list_ctor);
1667 CHECK_EXCEPTION(jni) << "error during NewObject";
1668
1669 jclass j_rtp_receiver_class = FindClass(jni, "org/webrtc/RtpReceiver");
1670 jmethodID j_rtp_receiver_ctor =
1671 GetMethodID(jni, j_rtp_receiver_class, "<init>", "(J)V");
1672
1673 auto receivers = ExtractNativePC(jni, j_pc)->GetReceivers();
1674 for (const auto& receiver : receivers) {
1675 jlong nativeReceiverPtr = jlongFromPointer(receiver.get());
1676 jobject j_receiver = jni->NewObject(j_rtp_receiver_class,
1677 j_rtp_receiver_ctor, nativeReceiverPtr);
1678 CHECK_EXCEPTION(jni) << "error during NewObject";
1679 // Receiver is now owned by Java object, and will be freed from there.
1680 receiver->AddRef();
1681 jni->CallBooleanMethod(j_receivers, j_array_list_add, j_receiver);
1682 CHECK_EXCEPTION(jni) << "error during CallBooleanMethod";
1683 }
1684 return j_receivers;
1685}
1686
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001687JOW(bool, PeerConnection_nativeGetStats)(
1688 JNIEnv* jni, jobject j_pc, jobject j_observer, jlong native_track) {
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +00001689 rtc::scoped_refptr<StatsObserverWrapper> observer(
1690 new rtc::RefCountedObject<StatsObserverWrapper>(jni, j_observer));
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001691 return ExtractNativePC(jni, j_pc)->GetStats(
jiayl@webrtc.orgdb41b4d2014-03-03 21:30:06 +00001692 observer,
1693 reinterpret_cast<MediaStreamTrackInterface*>(native_track),
1694 PeerConnectionInterface::kStatsOutputLevelStandard);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001695}
1696
1697JOW(jobject, PeerConnection_signalingState)(JNIEnv* jni, jobject j_pc) {
1698 PeerConnectionInterface::SignalingState state =
1699 ExtractNativePC(jni, j_pc)->signaling_state();
1700 return JavaEnumFromIndex(jni, "PeerConnection$SignalingState", state);
1701}
1702
1703JOW(jobject, PeerConnection_iceConnectionState)(JNIEnv* jni, jobject j_pc) {
1704 PeerConnectionInterface::IceConnectionState state =
1705 ExtractNativePC(jni, j_pc)->ice_connection_state();
1706 return JavaEnumFromIndex(jni, "PeerConnection$IceConnectionState", state);
1707}
1708
braveyao@webrtc.orgfedb9ea2015-01-21 07:57:06 +00001709JOW(jobject, PeerConnection_iceGatheringState)(JNIEnv* jni, jobject j_pc) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001710 PeerConnectionInterface::IceGatheringState state =
1711 ExtractNativePC(jni, j_pc)->ice_gathering_state();
braveyao@webrtc.orgfedb9ea2015-01-21 07:57:06 +00001712 return JavaEnumFromIndex(jni, "PeerConnection$IceGatheringState", state);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001713}
1714
1715JOW(void, PeerConnection_close)(JNIEnv* jni, jobject j_pc) {
1716 ExtractNativePC(jni, j_pc)->Close();
1717 return;
1718}
1719
1720JOW(jobject, MediaSource_nativeState)(JNIEnv* jni, jclass, jlong j_p) {
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +00001721 rtc::scoped_refptr<MediaSourceInterface> p(
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001722 reinterpret_cast<MediaSourceInterface*>(j_p));
1723 return JavaEnumFromIndex(jni, "MediaSource$State", p->state());
1724}
1725
perkj@webrtc.org83bc7212015-02-11 11:26:56 +00001726JOW(jobject, VideoCapturer_nativeCreateVideoCapturer)(
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001727 JNIEnv* jni, jclass, jstring j_device_name) {
perkj@webrtc.org83bc7212015-02-11 11:26:56 +00001728// Since we can't create platform specific java implementations in Java, we
1729// defer the creation to C land.
1730#if defined(ANDROID)
1731 jclass j_video_capturer_class(
1732 FindClass(jni, "org/webrtc/VideoCapturerAndroid"));
perkje0bce242015-10-05 16:21:54 +02001733 const int camera_id = jni->CallStaticIntMethod(
1734 j_video_capturer_class,
1735 GetStaticMethodID(jni, j_video_capturer_class, "lookupDeviceName",
1736 "(Ljava/lang/String;)I"),
1737 j_device_name);
1738 CHECK_EXCEPTION(jni) << "error during VideoCapturerAndroid.lookupDeviceName";
1739 if (camera_id == -1)
perkj@webrtc.org3db042e2015-02-19 08:43:38 +00001740 return nullptr;
perkje0bce242015-10-05 16:21:54 +02001741 jobject j_video_capturer = jni->NewObject(
1742 j_video_capturer_class,
1743 GetMethodID(jni, j_video_capturer_class, "<init>", "(I)V"), camera_id);
1744 CHECK_EXCEPTION(jni) << "error during creation of VideoCapturerAndroid";
1745 rtc::scoped_refptr<webrtc::AndroidVideoCapturerDelegate> delegate =
1746 new rtc::RefCountedObject<AndroidVideoCapturerJni>(jni, j_video_capturer);
1747 rtc::scoped_ptr<cricket::VideoCapturer> capturer(
Per33544192015-04-02 12:30:51 +02001748 new webrtc::AndroidVideoCapturer(delegate));
perkj@webrtc.org83bc7212015-02-11 11:26:56 +00001749
1750#else
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001751 std::string device_name = JavaToStdString(jni, j_device_name);
fischman@webrtc.org540acde2014-02-13 03:56:14 +00001752 scoped_ptr<cricket::DeviceManagerInterface> device_manager(
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001753 cricket::DeviceManagerFactory::Create());
henrikg91d6ede2015-09-17 00:24:34 -07001754 RTC_CHECK(device_manager->Init()) << "DeviceManager::Init() failed";
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001755 cricket::Device device;
1756 if (!device_manager->GetVideoCaptureDevice(device_name, &device)) {
fischman@webrtc.org4e65e072013-10-03 18:23:13 +00001757 LOG(LS_ERROR) << "GetVideoCaptureDevice failed for " << device_name;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001758 return 0;
1759 }
fischman@webrtc.org540acde2014-02-13 03:56:14 +00001760 scoped_ptr<cricket::VideoCapturer> capturer(
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001761 device_manager->CreateVideoCapturer(device));
perkj@webrtc.org83bc7212015-02-11 11:26:56 +00001762
1763 jclass j_video_capturer_class(
1764 FindClass(jni, "org/webrtc/VideoCapturer"));
1765 const jmethodID j_videocapturer_ctor(GetMethodID(
1766 jni, j_video_capturer_class, "<init>", "()V"));
1767 jobject j_video_capturer =
1768 jni->NewObject(j_video_capturer_class,
1769 j_videocapturer_ctor);
1770 CHECK_EXCEPTION(jni) << "error during creation of VideoCapturer";
1771
1772#endif
1773 const jmethodID j_videocapturer_set_native_capturer(GetMethodID(
1774 jni, j_video_capturer_class, "setNativeCapturer", "(J)V"));
1775 jni->CallVoidMethod(j_video_capturer,
1776 j_videocapturer_set_native_capturer,
perkje0bce242015-10-05 16:21:54 +02001777 jlongFromPointer(capturer.release()));
perkj@webrtc.org83bc7212015-02-11 11:26:56 +00001778 CHECK_EXCEPTION(jni) << "error during setNativeCapturer";
1779 return j_video_capturer;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001780}
1781
1782JOW(jlong, VideoRenderer_nativeCreateGuiVideoRenderer)(
1783 JNIEnv* jni, jclass, int x, int y) {
fischman@webrtc.org540acde2014-02-13 03:56:14 +00001784 scoped_ptr<VideoRendererWrapper> renderer(VideoRendererWrapper::Create(
1785 cricket::VideoRendererFactory::CreateGuiVideoRenderer(x, y)));
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001786 return (jlong)renderer.release();
1787}
1788
1789JOW(jlong, VideoRenderer_nativeWrapVideoRenderer)(
1790 JNIEnv* jni, jclass, jobject j_callbacks) {
fischman@webrtc.org540acde2014-02-13 03:56:14 +00001791 scoped_ptr<JavaVideoRendererWrapper> renderer(
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001792 new JavaVideoRendererWrapper(jni, j_callbacks));
1793 return (jlong)renderer.release();
1794}
1795
glaznev@webrtc.orgf6932292015-02-05 17:29:59 +00001796JOW(void, VideoRenderer_nativeCopyPlane)(
1797 JNIEnv *jni, jclass, jobject j_src_buffer, jint width, jint height,
1798 jint src_stride, jobject j_dst_buffer, jint dst_stride) {
1799 size_t src_size = jni->GetDirectBufferCapacity(j_src_buffer);
1800 size_t dst_size = jni->GetDirectBufferCapacity(j_dst_buffer);
henrikg91d6ede2015-09-17 00:24:34 -07001801 RTC_CHECK(src_stride >= width) << "Wrong source stride " << src_stride;
1802 RTC_CHECK(dst_stride >= width) << "Wrong destination stride " << dst_stride;
1803 RTC_CHECK(src_size >= src_stride * height)
glaznev@webrtc.orgf6932292015-02-05 17:29:59 +00001804 << "Insufficient source buffer capacity " << src_size;
henrikg91d6ede2015-09-17 00:24:34 -07001805 RTC_CHECK(dst_size >= dst_stride * height)
glaznev@webrtc.orgf6932292015-02-05 17:29:59 +00001806 << "Isufficient destination buffer capacity " << dst_size;
1807 uint8_t *src =
1808 reinterpret_cast<uint8_t*>(jni->GetDirectBufferAddress(j_src_buffer));
1809 uint8_t *dst =
1810 reinterpret_cast<uint8_t*>(jni->GetDirectBufferAddress(j_dst_buffer));
1811 if (src_stride == dst_stride) {
1812 memcpy(dst, src, src_stride * height);
1813 } else {
1814 for (int i = 0; i < height; i++) {
1815 memcpy(dst, src, width);
1816 src += src_stride;
1817 dst += dst_stride;
1818 }
1819 }
1820}
1821
perkj@webrtc.org8f605e82015-02-17 13:53:56 +00001822JOW(void, VideoSource_stop)(JNIEnv* jni, jclass, jlong j_p) {
1823 reinterpret_cast<VideoSourceInterface*>(j_p)->Stop();
fischman@webrtc.org4e65e072013-10-03 18:23:13 +00001824}
1825
1826JOW(void, VideoSource_restart)(
1827 JNIEnv* jni, jclass, jlong j_p_source, jlong j_p_format) {
perkj@webrtc.org8f605e82015-02-17 13:53:56 +00001828 reinterpret_cast<VideoSourceInterface*>(j_p_source)->Restart();
fischman@webrtc.org4e65e072013-10-03 18:23:13 +00001829}
1830
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001831JOW(jstring, MediaStreamTrack_nativeId)(JNIEnv* jni, jclass, jlong j_p) {
fischman@webrtc.org32001ef2013-08-12 23:26:21 +00001832 return JavaStringFromStdString(
1833 jni, reinterpret_cast<MediaStreamTrackInterface*>(j_p)->id());
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001834}
1835
1836JOW(jstring, MediaStreamTrack_nativeKind)(JNIEnv* jni, jclass, jlong j_p) {
fischman@webrtc.org32001ef2013-08-12 23:26:21 +00001837 return JavaStringFromStdString(
1838 jni, reinterpret_cast<MediaStreamTrackInterface*>(j_p)->kind());
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001839}
1840
1841JOW(jboolean, MediaStreamTrack_nativeEnabled)(JNIEnv* jni, jclass, jlong j_p) {
fischman@webrtc.org32001ef2013-08-12 23:26:21 +00001842 return reinterpret_cast<MediaStreamTrackInterface*>(j_p)->enabled();
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001843}
1844
1845JOW(jobject, MediaStreamTrack_nativeState)(JNIEnv* jni, jclass, jlong j_p) {
fischman@webrtc.org32001ef2013-08-12 23:26:21 +00001846 return JavaEnumFromIndex(
1847 jni,
1848 "MediaStreamTrack$State",
1849 reinterpret_cast<MediaStreamTrackInterface*>(j_p)->state());
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001850}
1851
1852JOW(jboolean, MediaStreamTrack_nativeSetState)(
1853 JNIEnv* jni, jclass, jlong j_p, jint j_new_state) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001854 MediaStreamTrackInterface::TrackState new_state =
1855 (MediaStreamTrackInterface::TrackState)j_new_state;
fischman@webrtc.org32001ef2013-08-12 23:26:21 +00001856 return reinterpret_cast<MediaStreamTrackInterface*>(j_p)
1857 ->set_state(new_state);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001858}
1859
1860JOW(jboolean, MediaStreamTrack_nativeSetEnabled)(
1861 JNIEnv* jni, jclass, jlong j_p, jboolean enabled) {
fischman@webrtc.org32001ef2013-08-12 23:26:21 +00001862 return reinterpret_cast<MediaStreamTrackInterface*>(j_p)
1863 ->set_enabled(enabled);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001864}
1865
1866JOW(void, VideoTrack_nativeAddRenderer)(
1867 JNIEnv* jni, jclass,
1868 jlong j_video_track_pointer, jlong j_renderer_pointer) {
fischman@webrtc.org32001ef2013-08-12 23:26:21 +00001869 reinterpret_cast<VideoTrackInterface*>(j_video_track_pointer)->AddRenderer(
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001870 reinterpret_cast<VideoRendererInterface*>(j_renderer_pointer));
1871}
1872
1873JOW(void, VideoTrack_nativeRemoveRenderer)(
1874 JNIEnv* jni, jclass,
1875 jlong j_video_track_pointer, jlong j_renderer_pointer) {
fischman@webrtc.org32001ef2013-08-12 23:26:21 +00001876 reinterpret_cast<VideoTrackInterface*>(j_video_track_pointer)->RemoveRenderer(
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001877 reinterpret_cast<VideoRendererInterface*>(j_renderer_pointer));
1878}
perkj@webrtc.org96e4db92015-02-13 12:46:51 +00001879
Jiayang Liue63d2a12015-09-01 16:11:18 -07001880JOW(jlong, CallSessionFileRotatingLogSink_nativeAddSink)(
1881 JNIEnv* jni, jclass,
1882 jstring j_dirPath, jint j_maxFileSize, jint j_severity) {
1883 std::string dir_path = JavaToStdString(jni, j_dirPath);
1884 rtc::CallSessionFileRotatingLogSink* sink =
1885 new rtc::CallSessionFileRotatingLogSink(dir_path, j_maxFileSize);
1886 if (!sink->Init()) {
1887 LOG_V(rtc::LoggingSeverity::LS_WARNING) <<
1888 "Failed to init CallSessionFileRotatingLogSink for path " << dir_path;
1889 delete sink;
1890 return 0;
1891 }
1892 rtc::LogMessage::AddLogToStream(
1893 sink, static_cast<rtc::LoggingSeverity>(j_severity));
1894 return (jlong) sink;
1895}
1896
1897JOW(void, CallSessionFileRotatingLogSink_nativeDeleteSink)(
1898 JNIEnv* jni, jclass, jlong j_sink) {
1899 rtc::CallSessionFileRotatingLogSink* sink =
1900 reinterpret_cast<rtc::CallSessionFileRotatingLogSink*>(j_sink);
1901 rtc::LogMessage::RemoveLogToStream(sink);
1902 delete sink;
1903}
1904
1905JOW(jbyteArray, CallSessionFileRotatingLogSink_nativeGetLogData)(
1906 JNIEnv* jni, jclass, jstring j_dirPath) {
1907 std::string dir_path = JavaToStdString(jni, j_dirPath);
1908 rtc::scoped_ptr<rtc::CallSessionFileRotatingStream> stream(
1909 new rtc::CallSessionFileRotatingStream(dir_path));
1910 if (!stream->Open()) {
1911 LOG_V(rtc::LoggingSeverity::LS_WARNING) <<
1912 "Failed to open CallSessionFileRotatingStream for path " << dir_path;
1913 return jni->NewByteArray(0);
1914 }
1915 size_t log_size = 0;
1916 if (!stream->GetSize(&log_size) || log_size == 0) {
1917 LOG_V(rtc::LoggingSeverity::LS_WARNING) <<
1918 "CallSessionFileRotatingStream returns 0 size for path " << dir_path;
1919 return jni->NewByteArray(0);
1920 }
1921
1922 size_t read = 0;
1923 rtc::scoped_ptr<jbyte> buffer(static_cast<jbyte*>(malloc(log_size)));
1924 stream->ReadAll(buffer.get(), log_size, &read, nullptr);
1925
1926 jbyteArray result = jni->NewByteArray(read);
1927 jni->SetByteArrayRegion(result, 0, read, buffer.get());
1928
1929 return result;
1930}
1931
deadbeef4139c0f2015-10-06 12:29:25 -07001932JOW(void, RtpSender_nativeSetTrack)(JNIEnv* jni,
1933 jclass,
1934 jlong j_rtp_sender_pointer,
1935 jlong j_track_pointer) {
1936 reinterpret_cast<RtpSenderInterface*>(j_rtp_sender_pointer)
1937 ->SetTrack(reinterpret_cast<MediaStreamTrackInterface*>(j_track_pointer));
1938}
1939
1940JOW(jlong, RtpSender_nativeGetTrack)(JNIEnv* jni,
1941 jclass,
1942 jlong j_rtp_sender_pointer,
1943 jlong j_track_pointer) {
1944 return jlongFromPointer(
1945 reinterpret_cast<RtpSenderInterface*>(j_rtp_sender_pointer)
1946 ->track()
1947 .release());
1948}
1949
1950JOW(jstring, RtpSender_nativeId)(
1951 JNIEnv* jni, jclass, jlong j_rtp_sender_pointer) {
1952 return JavaStringFromStdString(
1953 jni, reinterpret_cast<RtpSenderInterface*>(j_rtp_sender_pointer)->id());
1954}
1955
1956JOW(void, RtpSender_free)(JNIEnv* jni, jclass, jlong j_rtp_sender_pointer) {
1957 reinterpret_cast<RtpSenderInterface*>(j_rtp_sender_pointer)->Release();
1958}
1959
1960JOW(jlong, RtpReceiver_nativeGetTrack)(JNIEnv* jni,
1961 jclass,
1962 jlong j_rtp_receiver_pointer,
1963 jlong j_track_pointer) {
1964 return jlongFromPointer(
1965 reinterpret_cast<RtpReceiverInterface*>(j_rtp_receiver_pointer)
1966 ->track()
1967 .release());
1968}
1969
1970JOW(jstring, RtpReceiver_nativeId)(
1971 JNIEnv* jni, jclass, jlong j_rtp_receiver_pointer) {
1972 return JavaStringFromStdString(
1973 jni,
1974 reinterpret_cast<RtpReceiverInterface*>(j_rtp_receiver_pointer)->id());
1975}
1976
1977JOW(void, RtpReceiver_free)(JNIEnv* jni, jclass, jlong j_rtp_receiver_pointer) {
1978 reinterpret_cast<RtpReceiverInterface*>(j_rtp_receiver_pointer)->Release();
1979}
1980
perkj@webrtc.org96e4db92015-02-13 12:46:51 +00001981} // namespace webrtc_jni