blob: 4e55e4774f74df57c3703e1eb4c97a8bb098b03d [file] [log] [blame]
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001/*
2 * libjingle
fischman@webrtc.org33584f92013-07-25 16:43:30 +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
fischman@webrtc.org32001ef2013-08-12 23:26:21 +000059#include <asm/unistd.h>
henrike@webrtc.org723d6832013-07-12 16:04:50 +000060#include <limits>
henrike@webrtc.org28e20752013-07-10 00:45:36 +000061#include <map>
fischman@webrtc.org32001ef2013-08-12 23:26:21 +000062#include <sys/prctl.h>
63#include <sys/syscall.h>
henrike@webrtc.org28e20752013-07-10 00:45:36 +000064
65#include "talk/app/webrtc/mediaconstraintsinterface.h"
66#include "talk/app/webrtc/peerconnectioninterface.h"
67#include "talk/app/webrtc/videosourceinterface.h"
68#include "talk/base/logging.h"
69#include "talk/base/ssladapter.h"
70#include "talk/media/base/videocapturer.h"
71#include "talk/media/base/videorenderer.h"
72#include "talk/media/devices/videorendererfactory.h"
73#include "talk/media/webrtc/webrtcvideocapturer.h"
fischman@webrtc.org3d496fb2013-07-30 17:14:35 +000074#include "third_party/icu/source/common/unicode/unistr.h"
henrike@webrtc.org9de257d2013-07-17 14:42:53 +000075#include "webrtc/system_wrappers/interface/trace.h"
76#include "webrtc/video_engine/include/vie_base.h"
77#include "webrtc/voice_engine/include/voe_base.h"
henrike@webrtc.org28e20752013-07-10 00:45:36 +000078
79using icu::UnicodeString;
80using webrtc::AudioSourceInterface;
81using webrtc::AudioTrackInterface;
82using webrtc::AudioTrackVector;
83using webrtc::CreateSessionDescriptionObserver;
henrike@webrtc.org723d6832013-07-12 16:04:50 +000084using webrtc::DataBuffer;
85using webrtc::DataChannelInit;
86using webrtc::DataChannelInterface;
87using webrtc::DataChannelObserver;
henrike@webrtc.org28e20752013-07-10 00:45:36 +000088using webrtc::IceCandidateInterface;
89using webrtc::MediaConstraintsInterface;
90using webrtc::MediaSourceInterface;
91using webrtc::MediaStreamInterface;
92using webrtc::MediaStreamTrackInterface;
93using webrtc::PeerConnectionFactoryInterface;
94using webrtc::PeerConnectionInterface;
95using webrtc::PeerConnectionObserver;
96using webrtc::SessionDescriptionInterface;
97using webrtc::SetSessionDescriptionObserver;
98using webrtc::StatsObserver;
99using webrtc::StatsReport;
100using webrtc::VideoRendererInterface;
101using webrtc::VideoSourceInterface;
102using webrtc::VideoTrackInterface;
103using webrtc::VideoTrackVector;
104using webrtc::VideoRendererInterface;
105
106// Abort the process if |x| is false, emitting |msg|.
107#define CHECK(x, msg) \
108 if (x) {} else { \
109 LOG(LS_ERROR) << __FILE__ << ":" << __LINE__ << ": " << msg; \
110 abort(); \
111 }
112// Abort the process if |jni| has a Java exception pending, emitting |msg|.
113#define CHECK_EXCEPTION(jni, msg) \
114 if (0) {} else { \
115 if (jni->ExceptionCheck()) { \
116 jni->ExceptionDescribe(); \
117 jni->ExceptionClear(); \
118 CHECK(0, msg); \
119 } \
120 }
121
fischman@webrtc.org32001ef2013-08-12 23:26:21 +0000122// Helper that calls ptr->Release() and logs a useful message if that didn't
123// actually delete *ptr because of extra refcounts.
124#define CHECK_RELEASE(ptr) \
125 do { \
126 int count = (ptr)->Release(); \
127 if (count != 0) { \
128 LOG(LS_ERROR) << "Refcount unexpectedly not 0: " << (ptr) \
129 << ": " << count; \
130 } \
131 CHECK(!count, "Unexpected refcount"); \
132 } while (0)
133
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000134namespace {
135
136static JavaVM* g_jvm = NULL; // Set in JNI_OnLoad().
137
138static pthread_once_t g_jni_ptr_once = PTHREAD_ONCE_INIT;
139static pthread_key_t g_jni_ptr; // Key for per-thread JNIEnv* data.
140
fischman@webrtc.org32001ef2013-08-12 23:26:21 +0000141// Return thread ID as a string.
142static std::string GetThreadId() {
143 char buf[21]; // Big enough to hold a kuint64max plus terminating NULL.
144 CHECK(snprintf(buf, sizeof(buf), "%llu", syscall(__NR_gettid)) <= sizeof(buf),
145 "Thread id is bigger than uint64??");
146 return std::string(buf);
147}
148
149// Return the current thread's name.
150static std::string GetThreadName() {
151 char name[17];
152 CHECK(prctl(PR_GET_NAME, name) == 0, "prctl(PR_GET_NAME) failed");
153 name[16] = '\0';
154 return std::string(name);
155}
156
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000157static void ThreadDestructor(void* unused) {
158 jint status = g_jvm->DetachCurrentThread();
159 CHECK(status == JNI_OK, "Failed to detach thread: " << status);
160}
161
162static void CreateJNIPtrKey() {
163 CHECK(!pthread_key_create(&g_jni_ptr, &ThreadDestructor),
164 "pthread_key_create");
165}
166
167// Deal with difference in signatures between Oracle's jni.h and Android's.
168static JNIEnv* AttachCurrentThreadIfNeeded() {
169 CHECK(!pthread_once(&g_jni_ptr_once, &CreateJNIPtrKey),
170 "pthread_once");
171 JNIEnv* jni = reinterpret_cast<JNIEnv*>(pthread_getspecific(g_jni_ptr));
172 if (jni == NULL) {
173#ifdef _JAVASOFT_JNI_H_ // Oracle's jni.h violates the JNI spec!
174 void* env;
175#else
176 JNIEnv* env;
177#endif
fischman@webrtc.org32001ef2013-08-12 23:26:21 +0000178 char* name = strdup((GetThreadName() + " - " + GetThreadId()).c_str());
179 JavaVMAttachArgs args;
180 args.version = JNI_VERSION_1_6;
181 args.name = name;
182 args.group = NULL;
183 CHECK(!g_jvm->AttachCurrentThread(&env, &args), "Failed to attach thread");
184 free(name);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000185 CHECK(env, "AttachCurrentThread handed back NULL!");
186 jni = reinterpret_cast<JNIEnv*>(env);
187 CHECK(!pthread_setspecific(g_jni_ptr, jni), "pthread_setspecific");
188 }
189 return jni;
190}
191
192// Android's FindClass() is trickier than usual because the app-specific
193// ClassLoader is not consulted when there is no app-specific frame on the
194// stack. Consequently, we only look up classes once in JNI_OnLoad.
195// http://developer.android.com/training/articles/perf-jni.html#faq_FindClass
196class ClassReferenceHolder {
197 public:
198 explicit ClassReferenceHolder(JNIEnv* jni) {
199 LoadClass(jni, "java/nio/ByteBuffer");
200 LoadClass(jni, "org/webrtc/AudioTrack");
henrike@webrtc.org723d6832013-07-12 16:04:50 +0000201 LoadClass(jni, "org/webrtc/DataChannel");
202 LoadClass(jni, "org/webrtc/DataChannel$Buffer");
203 LoadClass(jni, "org/webrtc/DataChannel$Init");
204 LoadClass(jni, "org/webrtc/DataChannel$State");
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000205 LoadClass(jni, "org/webrtc/IceCandidate");
206 LoadClass(jni, "org/webrtc/MediaSource$State");
207 LoadClass(jni, "org/webrtc/MediaStream");
208 LoadClass(jni, "org/webrtc/MediaStreamTrack$State");
209 LoadClass(jni, "org/webrtc/PeerConnection$SignalingState");
210 LoadClass(jni, "org/webrtc/PeerConnection$IceConnectionState");
211 LoadClass(jni, "org/webrtc/PeerConnection$IceGatheringState");
212 LoadClass(jni, "org/webrtc/SessionDescription");
213 LoadClass(jni, "org/webrtc/SessionDescription$Type");
214 LoadClass(jni, "org/webrtc/StatsReport");
215 LoadClass(jni, "org/webrtc/StatsReport$Value");
216 LoadClass(jni, "org/webrtc/VideoRenderer$I420Frame");
217 LoadClass(jni, "org/webrtc/VideoTrack");
218 }
219
220 ~ClassReferenceHolder() {
221 CHECK(classes_.empty(), "Must call FreeReferences() before dtor!");
222 }
223
224 void FreeReferences(JNIEnv* jni) {
225 for (std::map<std::string, jclass>::const_iterator it = classes_.begin();
226 it != classes_.end(); ++it) {
227 jni->DeleteGlobalRef(it->second);
228 }
229 classes_.clear();
230 }
231
232 jclass GetClass(const std::string& name) {
233 std::map<std::string, jclass>::iterator it = classes_.find(name);
234 CHECK(it != classes_.end(), "Unexpected GetClass() call for: " << name);
235 return it->second;
236 }
237
238 private:
239 void LoadClass(JNIEnv* jni, const std::string& name) {
240 jclass localRef = jni->FindClass(name.c_str());
241 CHECK_EXCEPTION(jni, "error during FindClass: " << name);
242 CHECK(localRef, name);
243 jclass globalRef = reinterpret_cast<jclass>(jni->NewGlobalRef(localRef));
244 CHECK_EXCEPTION(jni, "error during NewGlobalRef: " << name);
245 CHECK(globalRef, name);
246 bool inserted = classes_.insert(std::make_pair(name, globalRef)).second;
247 CHECK(inserted, "Duplicate class name: " << name);
248 }
249
250 std::map<std::string, jclass> classes_;
251};
252
253// Allocated in JNI_OnLoad(), freed in JNI_OnUnLoad().
254static ClassReferenceHolder* g_class_reference_holder = NULL;
255
256// JNIEnv-helper methods that CHECK success: no Java exception thrown and found
257// object/class/method/field is non-null.
258jmethodID GetMethodID(
259 JNIEnv* jni, jclass c, const std::string& name, const char* signature) {
260 jmethodID m = jni->GetMethodID(c, name.c_str(), signature);
261 CHECK_EXCEPTION(jni,
262 "error during GetMethodID: " << name << ", " << signature);
263 CHECK(m, name << ", " << signature);
264 return m;
265}
266
267jmethodID GetStaticMethodID(
268 JNIEnv* jni, jclass c, const char* name, const char* signature) {
269 jmethodID m = jni->GetStaticMethodID(c, name, signature);
270 CHECK_EXCEPTION(jni,
271 "error during GetStaticMethodID: "
272 << name << ", " << signature);
273 CHECK(m, name << ", " << signature);
274 return m;
275}
276
277jfieldID GetFieldID(
278 JNIEnv* jni, jclass c, const char* name, const char* signature) {
279 jfieldID f = jni->GetFieldID(c, name, signature);
280 CHECK_EXCEPTION(jni, "error during GetFieldID");
281 CHECK(f, name << ", " << signature);
282 return f;
283}
284
285jclass FindClass(JNIEnv* jni, const char* name) {
286 return g_class_reference_holder->GetClass(name);
287}
288
289jclass GetObjectClass(JNIEnv* jni, jobject object) {
290 jclass c = jni->GetObjectClass(object);
291 CHECK_EXCEPTION(jni, "error during GetObjectClass");
292 CHECK(c, "");
293 return c;
294}
295
296jobject GetObjectField(JNIEnv* jni, jobject object, jfieldID id) {
297 jobject o = jni->GetObjectField(object, id);
298 CHECK_EXCEPTION(jni, "error during GetObjectField");
299 CHECK(o, "");
300 return o;
301}
302
henrike@webrtc.org723d6832013-07-12 16:04:50 +0000303jstring GetStringField(JNIEnv* jni, jobject object, jfieldID id) {
304 return static_cast<jstring>(GetObjectField(jni, object, id));
305}
306
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000307jlong GetLongField(JNIEnv* jni, jobject object, jfieldID id) {
308 jlong l = jni->GetLongField(object, id);
309 CHECK_EXCEPTION(jni, "error during GetLongField");
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000310 return l;
311}
312
henrike@webrtc.org723d6832013-07-12 16:04:50 +0000313jint GetIntField(JNIEnv* jni, jobject object, jfieldID id) {
314 jint i = jni->GetIntField(object, id);
315 CHECK_EXCEPTION(jni, "error during GetIntField");
316 return i;
317}
318
319bool GetBooleanField(JNIEnv* jni, jobject object, jfieldID id) {
320 jboolean b = jni->GetBooleanField(object, id);
321 CHECK_EXCEPTION(jni, "error during GetBooleanField");
322 return b;
323}
324
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000325jobject NewGlobalRef(JNIEnv* jni, jobject o) {
326 jobject ret = jni->NewGlobalRef(o);
327 CHECK_EXCEPTION(jni, "error during NewGlobalRef");
328 CHECK(ret, "");
329 return ret;
330}
331
332void DeleteGlobalRef(JNIEnv* jni, jobject o) {
333 jni->DeleteGlobalRef(o);
334 CHECK_EXCEPTION(jni, "error during DeleteGlobalRef");
335}
336
337// Given a jweak reference, allocate a (strong) local reference scoped to the
338// lifetime of this object if the weak reference is still valid, or NULL
339// otherwise.
340class WeakRef {
341 public:
342 WeakRef(JNIEnv* jni, jweak ref)
343 : jni_(jni), obj_(jni_->NewLocalRef(ref)) {
344 CHECK_EXCEPTION(jni, "error during NewLocalRef");
345 }
346 ~WeakRef() {
347 if (obj_) {
348 jni_->DeleteLocalRef(obj_);
349 CHECK_EXCEPTION(jni_, "error during DeleteLocalRef");
350 }
351 }
352 jobject obj() { return obj_; }
353
354 private:
355 JNIEnv* const jni_;
356 jobject const obj_;
357};
358
359// Given a local ref, take ownership of it and delete the ref when this goes out
360// of scope.
361template<class T> // T is jclass, jobject, jintArray, etc.
362class ScopedLocalRef {
363 public:
364 ScopedLocalRef(JNIEnv* jni, T obj)
365 : jni_(jni), obj_(obj) {}
366 ~ScopedLocalRef() {
367 jni_->DeleteLocalRef(obj_);
368 }
369 T operator*() const {
370 return obj_;
371 }
372 private:
373 JNIEnv* jni_;
374 T obj_;
375};
376
377// Scoped holder for global Java refs.
henrike@webrtc.org723d6832013-07-12 16:04:50 +0000378template<class T> // T is jclass, jobject, jintArray, etc.
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000379class ScopedGlobalRef {
380 public:
henrike@webrtc.org723d6832013-07-12 16:04:50 +0000381 explicit ScopedGlobalRef(JNIEnv* jni, T obj)
382 : obj_(static_cast<T>(jni->NewGlobalRef(obj))) {}
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000383 ~ScopedGlobalRef() {
384 DeleteGlobalRef(AttachCurrentThreadIfNeeded(), obj_);
385 }
henrike@webrtc.org723d6832013-07-12 16:04:50 +0000386 T operator*() const {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000387 return obj_;
388 }
389 private:
henrike@webrtc.org723d6832013-07-12 16:04:50 +0000390 T obj_;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000391};
392
393// Return the (singleton) Java Enum object corresponding to |index|;
394// |state_class_fragment| is something like "MediaSource$State".
395jobject JavaEnumFromIndex(
396 JNIEnv* jni, const std::string& state_class_fragment, int index) {
397 std::string state_class_name = "org/webrtc/" + state_class_fragment;
398 jclass state_class = FindClass(jni, state_class_name.c_str());
399 jmethodID state_values_id = GetStaticMethodID(
400 jni, state_class, "values", ("()[L" + state_class_name + ";").c_str());
401 ScopedLocalRef<jobjectArray> state_values(
402 jni,
403 (jobjectArray)jni->CallStaticObjectMethod(state_class, state_values_id));
404 CHECK_EXCEPTION(jni, "error during CallStaticObjectMethod");
405 jobject ret = jni->GetObjectArrayElement(*state_values, index);
406 CHECK_EXCEPTION(jni, "error during GetObjectArrayElement");
407 return ret;
408}
409
410// Given a UTF-8 encoded |native| string return a new (UTF-16) jstring.
411static jstring JavaStringFromStdString(JNIEnv* jni, const std::string& native) {
412 UnicodeString ustr(UnicodeString::fromUTF8(native));
413 jstring jstr = jni->NewString(ustr.getBuffer(), ustr.length());
414 CHECK_EXCEPTION(jni, "error during NewString");
415 return jstr;
416}
417
418// Given a (UTF-16) jstring return a new UTF-8 native string.
419static std::string JavaToStdString(JNIEnv* jni, const jstring& j_string) {
420 const jchar* jchars = jni->GetStringChars(j_string, NULL);
421 CHECK_EXCEPTION(jni, "Error during GetStringChars");
422 UnicodeString ustr(jchars, jni->GetStringLength(j_string));
423 CHECK_EXCEPTION(jni, "Error during GetStringLength");
424 jni->ReleaseStringChars(j_string, jchars);
425 CHECK_EXCEPTION(jni, "Error during ReleaseStringChars");
426 std::string ret;
427 return ustr.toUTF8String(ret);
428}
429
henrike@webrtc.org723d6832013-07-12 16:04:50 +0000430static DataChannelInit JavaDataChannelInitToNative(
431 JNIEnv* jni, jobject j_init) {
432 DataChannelInit init;
433
434 jclass j_init_class = FindClass(jni, "org/webrtc/DataChannel$Init");
435 jfieldID ordered_id = GetFieldID(jni, j_init_class, "ordered", "Z");
436 jfieldID max_retransmit_time_id =
437 GetFieldID(jni, j_init_class, "maxRetransmitTimeMs", "I");
438 jfieldID max_retransmits_id =
439 GetFieldID(jni, j_init_class, "maxRetransmits", "I");
440 jfieldID protocol_id =
441 GetFieldID(jni, j_init_class, "protocol", "Ljava/lang/String;");
442 jfieldID negotiated_id = GetFieldID(jni, j_init_class, "negotiated", "Z");
443 jfieldID id_id = GetFieldID(jni, j_init_class, "id", "I");
444
445 init.ordered = GetBooleanField(jni, j_init, ordered_id);
446 init.maxRetransmitTime = GetIntField(jni, j_init, max_retransmit_time_id);
447 init.maxRetransmits = GetIntField(jni, j_init, max_retransmits_id);
448 init.protocol = JavaToStdString(
449 jni, GetStringField(jni, j_init, protocol_id));
450 init.negotiated = GetBooleanField(jni, j_init, negotiated_id);
451 init.id = GetIntField(jni, j_init, id_id);
452
453 return init;
454}
455
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000456class ConstraintsWrapper;
457
458// Adapter between the C++ PeerConnectionObserver interface and the Java
459// PeerConnection.Observer interface. Wraps an instance of the Java interface
460// and dispatches C++ callbacks to Java.
461class PCOJava : public PeerConnectionObserver {
462 public:
463 PCOJava(JNIEnv* jni, jobject j_observer)
464 : j_observer_global_(jni, j_observer),
henrike@webrtc.org723d6832013-07-12 16:04:50 +0000465 j_observer_class_(jni, GetObjectClass(jni, *j_observer_global_)),
466 j_media_stream_class_(jni, FindClass(jni, "org/webrtc/MediaStream")),
467 j_media_stream_ctor_(GetMethodID(
468 jni, *j_media_stream_class_, "<init>", "(J)V")),
469 j_audio_track_class_(jni, FindClass(jni, "org/webrtc/AudioTrack")),
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000470 j_audio_track_ctor_(GetMethodID(
henrike@webrtc.org723d6832013-07-12 16:04:50 +0000471 jni, *j_audio_track_class_, "<init>", "(J)V")),
472 j_video_track_class_(jni, FindClass(jni, "org/webrtc/VideoTrack")),
473 j_video_track_ctor_(GetMethodID(
474 jni, *j_video_track_class_, "<init>", "(J)V")),
475 j_data_channel_class_(jni, FindClass(jni, "org/webrtc/DataChannel")),
476 j_data_channel_ctor_(GetMethodID(
477 jni, *j_data_channel_class_, "<init>", "(J)V")) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000478 }
479
480 virtual ~PCOJava() {}
481
henrike@webrtc.org723d6832013-07-12 16:04:50 +0000482 virtual void OnIceCandidate(const IceCandidateInterface* candidate) OVERRIDE {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000483 std::string sdp;
484 CHECK(candidate->ToString(&sdp), "got so far: " << sdp);
485 jclass candidate_class = FindClass(jni(), "org/webrtc/IceCandidate");
486 jmethodID ctor = GetMethodID(jni(), candidate_class,
487 "<init>", "(Ljava/lang/String;ILjava/lang/String;)V");
488 ScopedLocalRef<jstring> j_mid(
489 jni(), JavaStringFromStdString(jni(), candidate->sdp_mid()));
490 ScopedLocalRef<jstring> j_sdp(jni(), JavaStringFromStdString(jni(), sdp));
491 ScopedLocalRef<jobject> j_candidate(jni(), jni()->NewObject(
492 candidate_class, ctor, *j_mid, candidate->sdp_mline_index(), *j_sdp));
493 CHECK_EXCEPTION(jni(), "error during NewObject");
henrike@webrtc.org723d6832013-07-12 16:04:50 +0000494 jmethodID m = GetMethodID(jni(), *j_observer_class_,
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000495 "onIceCandidate", "(Lorg/webrtc/IceCandidate;)V");
496 jni()->CallVoidMethod(*j_observer_global_, m, *j_candidate);
497 CHECK_EXCEPTION(jni(), "error during CallVoidMethod");
498 }
499
henrike@webrtc.org723d6832013-07-12 16:04:50 +0000500 virtual void OnError() OVERRIDE {
501 jmethodID m = GetMethodID(jni(), *j_observer_class_, "onError", "(V)V");
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000502 jni()->CallVoidMethod(*j_observer_global_, m);
503 CHECK_EXCEPTION(jni(), "error during CallVoidMethod");
504 }
505
506 virtual void OnSignalingChange(
henrike@webrtc.org723d6832013-07-12 16:04:50 +0000507 PeerConnectionInterface::SignalingState new_state) OVERRIDE {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000508 jmethodID m = GetMethodID(
henrike@webrtc.org723d6832013-07-12 16:04:50 +0000509 jni(), *j_observer_class_, "onSignalingChange",
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000510 "(Lorg/webrtc/PeerConnection$SignalingState;)V");
511 ScopedLocalRef<jobject> new_state_enum(jni(), JavaEnumFromIndex(
512 jni(), "PeerConnection$SignalingState", new_state));
513 jni()->CallVoidMethod(*j_observer_global_, m, *new_state_enum);
514 CHECK_EXCEPTION(jni(), "error during CallVoidMethod");
515 }
516
517 virtual void OnIceConnectionChange(
henrike@webrtc.org723d6832013-07-12 16:04:50 +0000518 PeerConnectionInterface::IceConnectionState new_state) OVERRIDE {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000519 jmethodID m = GetMethodID(
henrike@webrtc.org723d6832013-07-12 16:04:50 +0000520 jni(), *j_observer_class_, "onIceConnectionChange",
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000521 "(Lorg/webrtc/PeerConnection$IceConnectionState;)V");
522 ScopedLocalRef<jobject> new_state_enum(jni(), JavaEnumFromIndex(
523 jni(), "PeerConnection$IceConnectionState", new_state));
524 jni()->CallVoidMethod(*j_observer_global_, m, *new_state_enum);
525 CHECK_EXCEPTION(jni(), "error during CallVoidMethod");
526 }
527
528 virtual void OnIceGatheringChange(
henrike@webrtc.org723d6832013-07-12 16:04:50 +0000529 PeerConnectionInterface::IceGatheringState new_state) OVERRIDE {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000530 jmethodID m = GetMethodID(
henrike@webrtc.org723d6832013-07-12 16:04:50 +0000531 jni(), *j_observer_class_, "onIceGatheringChange",
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000532 "(Lorg/webrtc/PeerConnection$IceGatheringState;)V");
533 ScopedLocalRef<jobject> new_state_enum(jni(), JavaEnumFromIndex(
534 jni(), "PeerConnection$IceGatheringState", new_state));
535 jni()->CallVoidMethod(*j_observer_global_, m, *new_state_enum);
536 CHECK_EXCEPTION(jni(), "error during CallVoidMethod");
537 }
538
henrike@webrtc.org723d6832013-07-12 16:04:50 +0000539 virtual void OnAddStream(MediaStreamInterface* stream) OVERRIDE {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000540 ScopedLocalRef<jobject> j_stream(jni(), jni()->NewObject(
henrike@webrtc.org723d6832013-07-12 16:04:50 +0000541 *j_media_stream_class_, j_media_stream_ctor_, (jlong)stream));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000542 CHECK_EXCEPTION(jni(), "error during NewObject");
543
544 AudioTrackVector audio_tracks = stream->GetAudioTracks();
545 for (size_t i = 0; i < audio_tracks.size(); ++i) {
546 AudioTrackInterface* track = audio_tracks[i];
547 ScopedLocalRef<jstring> id(
548 jni(), JavaStringFromStdString(jni(), track->id()));
549 ScopedLocalRef<jobject> j_track(jni(), jni()->NewObject(
henrike@webrtc.org723d6832013-07-12 16:04:50 +0000550 *j_audio_track_class_, j_audio_track_ctor_, (jlong)track, *id));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000551 CHECK_EXCEPTION(jni(), "error during NewObject");
fischman@webrtc.org32001ef2013-08-12 23:26:21 +0000552 jfieldID audio_tracks_id = GetFieldID(jni(),
553 *j_media_stream_class_,
554 "audioTracks",
555 "Ljava/util/LinkedList;");
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000556 ScopedLocalRef<jobject> audio_tracks(jni(), GetObjectField(
557 jni(), *j_stream, audio_tracks_id));
558 jmethodID add = GetMethodID(jni(),
559 GetObjectClass(jni(), *audio_tracks), "add", "(Ljava/lang/Object;)Z");
560 jboolean added = jni()->CallBooleanMethod(*audio_tracks, add, *j_track);
561 CHECK_EXCEPTION(jni(), "error during CallBooleanMethod");
562 CHECK(added, "");
563 }
564
565 VideoTrackVector video_tracks = stream->GetVideoTracks();
566 for (size_t i = 0; i < video_tracks.size(); ++i) {
567 VideoTrackInterface* track = video_tracks[i];
568 ScopedLocalRef<jstring> id(
569 jni(), JavaStringFromStdString(jni(), track->id()));
570 ScopedLocalRef<jobject> j_track(jni(), jni()->NewObject(
henrike@webrtc.org723d6832013-07-12 16:04:50 +0000571 *j_video_track_class_, j_video_track_ctor_, (jlong)track, *id));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000572 CHECK_EXCEPTION(jni(), "error during NewObject");
fischman@webrtc.org32001ef2013-08-12 23:26:21 +0000573 jfieldID video_tracks_id = GetFieldID(jni(),
574 *j_media_stream_class_,
575 "videoTracks",
576 "Ljava/util/LinkedList;");
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000577 ScopedLocalRef<jobject> video_tracks(jni(), GetObjectField(
578 jni(), *j_stream, video_tracks_id));
579 jmethodID add = GetMethodID(jni(),
580 GetObjectClass(jni(), *video_tracks), "add", "(Ljava/lang/Object;)Z");
581 jboolean added = jni()->CallBooleanMethod(*video_tracks, add, *j_track);
582 CHECK_EXCEPTION(jni(), "error during CallBooleanMethod");
583 CHECK(added, "");
584 }
585 streams_[stream] = jni()->NewWeakGlobalRef(*j_stream);
586 CHECK_EXCEPTION(jni(), "error during NewWeakGlobalRef");
587
henrike@webrtc.org723d6832013-07-12 16:04:50 +0000588 jmethodID m = GetMethodID(jni(), *j_observer_class_, "onAddStream",
589 "(Lorg/webrtc/MediaStream;)V");
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000590 jni()->CallVoidMethod(*j_observer_global_, m, *j_stream);
591 CHECK_EXCEPTION(jni(), "error during CallVoidMethod");
592 }
593
henrike@webrtc.org723d6832013-07-12 16:04:50 +0000594 virtual void OnRemoveStream(MediaStreamInterface* stream) OVERRIDE {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000595 NativeToJavaStreamsMap::iterator it = streams_.find(stream);
596 CHECK(it != streams_.end(), "unexpected stream: " << std::hex << stream);
597
598 WeakRef s(jni(), it->second);
599 streams_.erase(it);
600 if (!s.obj())
601 return;
602
henrike@webrtc.org723d6832013-07-12 16:04:50 +0000603 jmethodID m = GetMethodID(jni(), *j_observer_class_, "onRemoveStream",
604 "(Lorg/webrtc/MediaStream;)V");
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000605 jni()->CallVoidMethod(*j_observer_global_, m, s.obj());
606 CHECK_EXCEPTION(jni(), "error during CallVoidMethod");
607 }
608
henrike@webrtc.org723d6832013-07-12 16:04:50 +0000609 virtual void OnDataChannel(DataChannelInterface* channel) OVERRIDE {
610 ScopedLocalRef<jobject> j_channel(jni(), jni()->NewObject(
611 *j_data_channel_class_, j_data_channel_ctor_, (jlong)channel));
612 CHECK_EXCEPTION(jni(), "error during NewObject");
henrike@webrtc.org723d6832013-07-12 16:04:50 +0000613
614 jmethodID m = GetMethodID(jni(), *j_observer_class_, "onDataChannel",
615 "(Lorg/webrtc/DataChannel;)V");
616 jni()->CallVoidMethod(*j_observer_global_, m, *j_channel);
fischman@webrtc.org32001ef2013-08-12 23:26:21 +0000617
618 // Channel is now owned by Java object, and will be freed from
619 // DataChannel.dispose(). Important that this be done _after_ the
620 // CallVoidMethod above as Java code might call back into native code and be
621 // surprised to see a refcount of 2.
622 int bumped_count = channel->AddRef();
623 CHECK(bumped_count == 2, "Unexpected refcount OnDataChannel");
624
henrike@webrtc.org723d6832013-07-12 16:04:50 +0000625 CHECK_EXCEPTION(jni(), "error during CallVoidMethod");
626 }
627
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000628 void SetConstraints(ConstraintsWrapper* constraints) {
629 CHECK(!constraints_.get(), "constraints already set!");
630 constraints_.reset(constraints);
631 }
632
633 const ConstraintsWrapper* constraints() { return constraints_.get(); }
634
635 private:
636 JNIEnv* jni() {
637 return AttachCurrentThreadIfNeeded();
638 }
639
henrike@webrtc.org723d6832013-07-12 16:04:50 +0000640 const ScopedGlobalRef<jobject> j_observer_global_;
641 const ScopedGlobalRef<jclass> j_observer_class_;
642 const ScopedGlobalRef<jclass> j_media_stream_class_;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000643 const jmethodID j_media_stream_ctor_;
henrike@webrtc.org723d6832013-07-12 16:04:50 +0000644 const ScopedGlobalRef<jclass> j_audio_track_class_;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000645 const jmethodID j_audio_track_ctor_;
henrike@webrtc.org723d6832013-07-12 16:04:50 +0000646 const ScopedGlobalRef<jclass> j_video_track_class_;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000647 const jmethodID j_video_track_ctor_;
henrike@webrtc.org723d6832013-07-12 16:04:50 +0000648 const ScopedGlobalRef<jclass> j_data_channel_class_;
649 const jmethodID j_data_channel_ctor_;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000650 typedef std::map<void*, jweak> NativeToJavaStreamsMap;
651 NativeToJavaStreamsMap streams_; // C++ -> Java streams.
652 talk_base::scoped_ptr<ConstraintsWrapper> constraints_;
653};
654
655// Wrapper for a Java MediaConstraints object. Copies all needed data so when
656// the constructor returns the Java object is no longer needed.
657class ConstraintsWrapper : public MediaConstraintsInterface {
658 public:
659 ConstraintsWrapper(JNIEnv* jni, jobject j_constraints) {
660 PopulateConstraintsFromJavaPairList(
661 jni, j_constraints, "mandatory", &mandatory_);
662 PopulateConstraintsFromJavaPairList(
663 jni, j_constraints, "optional", &optional_);
664 }
665
666 virtual ~ConstraintsWrapper() {}
667
668 // MediaConstraintsInterface.
henrike@webrtc.org723d6832013-07-12 16:04:50 +0000669 virtual const Constraints& GetMandatory() const OVERRIDE {
670 return mandatory_;
671 }
672
673 virtual const Constraints& GetOptional() const OVERRIDE {
674 return optional_;
675 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000676
677 private:
678 // Helper for translating a List<Pair<String, String>> to a Constraints.
679 static void PopulateConstraintsFromJavaPairList(
680 JNIEnv* jni, jobject j_constraints,
681 const char* field_name, Constraints* field) {
682 jfieldID j_id = GetFieldID(jni,
683 GetObjectClass(jni, j_constraints), field_name, "Ljava/util/List;");
684 jobject j_list = GetObjectField(jni, j_constraints, j_id);
685 jmethodID j_iterator_id = GetMethodID(jni,
686 GetObjectClass(jni, j_list), "iterator", "()Ljava/util/Iterator;");
687 jobject j_iterator = jni->CallObjectMethod(j_list, j_iterator_id);
688 CHECK_EXCEPTION(jni, "error during CallObjectMethod");
689 jmethodID j_has_next = GetMethodID(jni,
690 GetObjectClass(jni, j_iterator), "hasNext", "()Z");
691 jmethodID j_next = GetMethodID(jni,
692 GetObjectClass(jni, j_iterator), "next", "()Ljava/lang/Object;");
693 while (jni->CallBooleanMethod(j_iterator, j_has_next)) {
694 CHECK_EXCEPTION(jni, "error during CallBooleanMethod");
695 jobject entry = jni->CallObjectMethod(j_iterator, j_next);
696 CHECK_EXCEPTION(jni, "error during CallObjectMethod");
697 jmethodID get_key = GetMethodID(jni,
698 GetObjectClass(jni, entry), "getKey", "()Ljava/lang/String;");
699 jstring j_key = reinterpret_cast<jstring>(
700 jni->CallObjectMethod(entry, get_key));
701 CHECK_EXCEPTION(jni, "error during CallObjectMethod");
702 jmethodID get_value = GetMethodID(jni,
703 GetObjectClass(jni, entry), "getValue", "()Ljava/lang/String;");
704 jstring j_value = reinterpret_cast<jstring>(
705 jni->CallObjectMethod(entry, get_value));
706 CHECK_EXCEPTION(jni, "error during CallObjectMethod");
707 field->push_back(Constraint(JavaToStdString(jni, j_key),
708 JavaToStdString(jni, j_value)));
709 }
710 CHECK_EXCEPTION(jni, "error during CallBooleanMethod");
711 }
712
713 Constraints mandatory_;
714 Constraints optional_;
715};
716
717static jobject JavaSdpFromNativeSdp(
718 JNIEnv* jni, const SessionDescriptionInterface* desc) {
719 std::string sdp;
720 CHECK(desc->ToString(&sdp), "got so far: " << sdp);
721 ScopedLocalRef<jstring> j_description(jni, JavaStringFromStdString(jni, sdp));
722
723 jclass j_type_class = FindClass(
724 jni, "org/webrtc/SessionDescription$Type");
725 jmethodID j_type_from_canonical = GetStaticMethodID(
726 jni, j_type_class, "fromCanonicalForm",
727 "(Ljava/lang/String;)Lorg/webrtc/SessionDescription$Type;");
728 ScopedLocalRef<jstring> j_type_string(
729 jni, JavaStringFromStdString(jni, desc->type()));
730 jobject j_type = jni->CallStaticObjectMethod(
731 j_type_class, j_type_from_canonical, *j_type_string);
732 CHECK_EXCEPTION(jni, "error during CallObjectMethod");
733
734 jclass j_sdp_class = FindClass(jni, "org/webrtc/SessionDescription");
735 jmethodID j_sdp_ctor = GetMethodID(
736 jni, j_sdp_class, "<init>",
737 "(Lorg/webrtc/SessionDescription$Type;Ljava/lang/String;)V");
738 jobject j_sdp = jni->NewObject(
739 j_sdp_class, j_sdp_ctor, j_type, *j_description);
740 CHECK_EXCEPTION(jni, "error during NewObject");
741 return j_sdp;
742}
743
744template <class T> // T is one of {Create,Set}SessionDescriptionObserver.
745class SdpObserverWrapper : public T {
746 public:
747 SdpObserverWrapper(JNIEnv* jni, jobject j_observer,
748 ConstraintsWrapper* constraints)
749 : constraints_(constraints),
henrike@webrtc.org723d6832013-07-12 16:04:50 +0000750 j_observer_global_(jni, j_observer),
751 j_observer_class_(jni, GetObjectClass(jni, j_observer)) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000752 }
753
henrike@webrtc.org723d6832013-07-12 16:04:50 +0000754 virtual ~SdpObserverWrapper() {}
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000755
henrike@webrtc.org723d6832013-07-12 16:04:50 +0000756 // Can't mark OVERRIDE because of templating.
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000757 virtual void OnSuccess() {
henrike@webrtc.org723d6832013-07-12 16:04:50 +0000758 jmethodID m = GetMethodID(jni(), *j_observer_class_, "onSetSuccess", "()V");
759 jni()->CallVoidMethod(*j_observer_global_, m);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000760 CHECK_EXCEPTION(jni(), "error during CallVoidMethod");
761 }
762
henrike@webrtc.org723d6832013-07-12 16:04:50 +0000763 // Can't mark OVERRIDE because of templating.
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000764 virtual void OnSuccess(SessionDescriptionInterface* desc) {
765 jmethodID m = GetMethodID(
henrike@webrtc.org723d6832013-07-12 16:04:50 +0000766 jni(), *j_observer_class_, "onCreateSuccess",
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000767 "(Lorg/webrtc/SessionDescription;)V");
768 ScopedLocalRef<jobject> j_sdp(jni(), JavaSdpFromNativeSdp(jni(), desc));
henrike@webrtc.org723d6832013-07-12 16:04:50 +0000769 jni()->CallVoidMethod(*j_observer_global_, m, *j_sdp);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000770 CHECK_EXCEPTION(jni(), "error during CallVoidMethod");
771 }
772
773 protected:
774 // Common implementation for failure of Set & Create types, distinguished by
775 // |op| being "Set" or "Create".
776 void OnFailure(const std::string& op, const std::string& error) {
henrike@webrtc.org723d6832013-07-12 16:04:50 +0000777 jmethodID m = GetMethodID(jni(), *j_observer_class_, "on" + op + "Failure",
778 "(Ljava/lang/String;)V");
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000779 ScopedLocalRef<jstring> j_error_string(
780 jni(), JavaStringFromStdString(jni(), error));
henrike@webrtc.org723d6832013-07-12 16:04:50 +0000781 jni()->CallVoidMethod(*j_observer_global_, m, *j_error_string);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000782 CHECK_EXCEPTION(jni(), "error during CallVoidMethod");
783 }
784
785 private:
786 JNIEnv* jni() {
787 return AttachCurrentThreadIfNeeded();
788 }
789
790 talk_base::scoped_ptr<ConstraintsWrapper> constraints_;
henrike@webrtc.org723d6832013-07-12 16:04:50 +0000791 const ScopedGlobalRef<jobject> j_observer_global_;
792 const ScopedGlobalRef<jclass> j_observer_class_;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000793};
794
795class CreateSdpObserverWrapper
796 : public SdpObserverWrapper<CreateSessionDescriptionObserver> {
797 public:
798 CreateSdpObserverWrapper(JNIEnv* jni, jobject j_observer,
799 ConstraintsWrapper* constraints)
800 : SdpObserverWrapper(jni, j_observer, constraints) {}
801
henrike@webrtc.org723d6832013-07-12 16:04:50 +0000802 virtual void OnFailure(const std::string& error) OVERRIDE {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000803 SdpObserverWrapper::OnFailure(std::string("Create"), error);
804 }
805};
806
807class SetSdpObserverWrapper
808 : public SdpObserverWrapper<SetSessionDescriptionObserver> {
809 public:
810 SetSdpObserverWrapper(JNIEnv* jni, jobject j_observer,
811 ConstraintsWrapper* constraints)
812 : SdpObserverWrapper(jni, j_observer, constraints) {}
813
henrike@webrtc.org723d6832013-07-12 16:04:50 +0000814 virtual void OnFailure(const std::string& error) OVERRIDE {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000815 SdpObserverWrapper::OnFailure(std::string("Set"), error);
816 }
817};
818
henrike@webrtc.org723d6832013-07-12 16:04:50 +0000819// Adapter for a Java DataChannel$Observer presenting a C++ DataChannelObserver
820// and dispatching the callback from C++ back to Java.
821class DataChannelObserverWrapper : public DataChannelObserver {
822 public:
823 DataChannelObserverWrapper(JNIEnv* jni, jobject j_observer)
824 : j_observer_global_(jni, j_observer),
825 j_observer_class_(jni, GetObjectClass(jni, j_observer)),
826 j_on_state_change_mid_(GetMethodID(jni, *j_observer_class_,
827 "onStateChange", "()V")),
828 j_on_message_mid_(GetMethodID(jni, *j_observer_class_, "onMessage",
829 "(Lorg/webrtc/DataChannel$Buffer;)V")),
830 j_buffer_class_(jni, FindClass(jni, "org/webrtc/DataChannel$Buffer")),
831 j_buffer_ctor_(GetMethodID(jni, *j_buffer_class_,
832 "<init>", "(Ljava/nio/ByteBuffer;Z)V")) {
833 }
834
835 virtual ~DataChannelObserverWrapper() {}
836
837 virtual void OnStateChange() OVERRIDE {
838 jni()->CallVoidMethod(*j_observer_global_, j_on_state_change_mid_);
839 CHECK_EXCEPTION(jni(), "error during CallVoidMethod");
840 }
841
842 virtual void OnMessage(const DataBuffer& buffer) OVERRIDE {
843 jobject byte_buffer =
844 jni()->NewDirectByteBuffer(const_cast<char*>(buffer.data.data()),
845 buffer.data.length());
846 jobject j_buffer = jni()->NewObject(*j_buffer_class_, j_buffer_ctor_,
847 byte_buffer, buffer.binary);
848 jni()->CallVoidMethod(*j_observer_global_, j_on_message_mid_, j_buffer);
849 CHECK_EXCEPTION(jni(), "error during CallVoidMethod");
850 }
851
852 private:
853 JNIEnv* jni() {
854 return AttachCurrentThreadIfNeeded();
855 }
856
857 const ScopedGlobalRef<jobject> j_observer_global_;
858 const ScopedGlobalRef<jclass> j_observer_class_;
859 const ScopedGlobalRef<jclass> j_buffer_class_;
860 const jmethodID j_on_state_change_mid_;
861 const jmethodID j_on_message_mid_;
862 const jmethodID j_buffer_ctor_;
863};
864
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000865// Adapter for a Java StatsObserver presenting a C++ StatsObserver and
866// dispatching the callback from C++ back to Java.
867class StatsObserverWrapper : public StatsObserver {
868 public:
869 StatsObserverWrapper(JNIEnv* jni, jobject j_observer)
henrike@webrtc.org723d6832013-07-12 16:04:50 +0000870 : j_observer_global_(jni, j_observer),
871 j_observer_class_(jni, GetObjectClass(jni, j_observer)),
872 j_stats_report_class_(jni, FindClass(jni, "org/webrtc/StatsReport")),
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000873 j_stats_report_ctor_(GetMethodID(
henrike@webrtc.org723d6832013-07-12 16:04:50 +0000874 jni, *j_stats_report_class_, "<init>",
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000875 "(Ljava/lang/String;Ljava/lang/String;D"
876 "[Lorg/webrtc/StatsReport$Value;)V")),
henrike@webrtc.org723d6832013-07-12 16:04:50 +0000877 j_value_class_(jni, FindClass(
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000878 jni, "org/webrtc/StatsReport$Value")),
879 j_value_ctor_(GetMethodID(
henrike@webrtc.org723d6832013-07-12 16:04:50 +0000880 jni, *j_value_class_, "<init>",
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000881 "(Ljava/lang/String;Ljava/lang/String;)V")) {
882 }
883
henrike@webrtc.org723d6832013-07-12 16:04:50 +0000884 virtual ~StatsObserverWrapper() {}
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000885
henrike@webrtc.org723d6832013-07-12 16:04:50 +0000886 virtual void OnComplete(const std::vector<StatsReport>& reports) OVERRIDE {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000887 ScopedLocalRef<jobjectArray> j_reports(jni(),
888 ReportsToJava(jni(), reports));
henrike@webrtc.org723d6832013-07-12 16:04:50 +0000889 jmethodID m = GetMethodID(jni(), *j_observer_class_, "onComplete",
890 "([Lorg/webrtc/StatsReport;)V");
891 jni()->CallVoidMethod(*j_observer_global_, m, *j_reports);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000892 CHECK_EXCEPTION(jni(), "error during CallVoidMethod");
893 }
894
895 private:
896 jobjectArray ReportsToJava(
897 JNIEnv* jni, const std::vector<StatsReport>& reports) {
898 jobjectArray reports_array = jni->NewObjectArray(
henrike@webrtc.org723d6832013-07-12 16:04:50 +0000899 reports.size(), *j_stats_report_class_, NULL);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000900 for (int i = 0; i < reports.size(); ++i) {
901 const StatsReport& report = reports[i];
902 ScopedLocalRef<jstring> j_id(
903 jni, JavaStringFromStdString(jni, report.id));
904 ScopedLocalRef<jstring> j_type(
905 jni, JavaStringFromStdString(jni, report.type));
906 ScopedLocalRef<jobjectArray> j_values(
907 jni, ValuesToJava(jni, report.values));
908 ScopedLocalRef<jobject> j_report(jni, jni->NewObject(
henrike@webrtc.org723d6832013-07-12 16:04:50 +0000909 *j_stats_report_class_, j_stats_report_ctor_, *j_id, *j_type,
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000910 report.timestamp, *j_values));
911 jni->SetObjectArrayElement(reports_array, i, *j_report);
912 }
913 return reports_array;
914 }
915
916 jobjectArray ValuesToJava(JNIEnv* jni, const StatsReport::Values& values) {
917 jobjectArray j_values = jni->NewObjectArray(
henrike@webrtc.org723d6832013-07-12 16:04:50 +0000918 values.size(), *j_value_class_, NULL);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000919 for (int i = 0; i < values.size(); ++i) {
920 const StatsReport::Value& value = values[i];
921 ScopedLocalRef<jstring> j_name(
922 jni, JavaStringFromStdString(jni, value.name));
923 ScopedLocalRef<jstring> j_value(
924 jni, JavaStringFromStdString(jni, value.value));
925 ScopedLocalRef<jobject> j_element_value(jni, jni->NewObject(
henrike@webrtc.org723d6832013-07-12 16:04:50 +0000926 *j_value_class_, j_value_ctor_, *j_name, *j_value));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000927 jni->SetObjectArrayElement(j_values, i, *j_element_value);
928 }
929 return j_values;
930 }
931
932 JNIEnv* jni() {
933 return AttachCurrentThreadIfNeeded();
934 }
935
henrike@webrtc.org723d6832013-07-12 16:04:50 +0000936 const ScopedGlobalRef<jobject> j_observer_global_;
937 const ScopedGlobalRef<jclass> j_observer_class_;
938 const ScopedGlobalRef<jclass> j_stats_report_class_;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000939 const jmethodID j_stats_report_ctor_;
henrike@webrtc.org723d6832013-07-12 16:04:50 +0000940 const ScopedGlobalRef<jclass> j_value_class_;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000941 const jmethodID j_value_ctor_;
942};
943
944// Adapter presenting a cricket::VideoRenderer as a
945// webrtc::VideoRendererInterface.
946class VideoRendererWrapper : public VideoRendererInterface {
947 public:
948 static VideoRendererWrapper* Create(cricket::VideoRenderer* renderer) {
949 if (renderer)
950 return new VideoRendererWrapper(renderer);
951 return NULL;
952 }
953
954 virtual ~VideoRendererWrapper() {}
955
henrike@webrtc.org723d6832013-07-12 16:04:50 +0000956 virtual void SetSize(int width, int height) OVERRIDE {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000957 const bool kNotReserved = false; // What does this param mean??
958 renderer_->SetSize(width, height, kNotReserved);
959 }
960
henrike@webrtc.org723d6832013-07-12 16:04:50 +0000961 virtual void RenderFrame(const cricket::VideoFrame* frame) OVERRIDE {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000962 renderer_->RenderFrame(frame);
963 }
964
965 private:
966 explicit VideoRendererWrapper(cricket::VideoRenderer* renderer)
967 : renderer_(renderer) {}
968
969 talk_base::scoped_ptr<cricket::VideoRenderer> renderer_;
970};
971
972// Wrapper dispatching webrtc::VideoRendererInterface to a Java VideoRenderer
973// instance.
974class JavaVideoRendererWrapper : public VideoRendererInterface {
975 public:
976 JavaVideoRendererWrapper(JNIEnv* jni, jobject j_callbacks)
henrike@webrtc.org723d6832013-07-12 16:04:50 +0000977 : j_callbacks_(jni, j_callbacks),
978 j_set_size_id_(GetMethodID(
979 jni, GetObjectClass(jni, j_callbacks), "setSize", "(II)V")),
980 j_render_frame_id_(GetMethodID(
981 jni, GetObjectClass(jni, j_callbacks), "renderFrame",
982 "(Lorg/webrtc/VideoRenderer$I420Frame;)V")),
983 j_frame_class_(jni,
984 FindClass(jni, "org/webrtc/VideoRenderer$I420Frame")),
985 j_frame_ctor_id_(GetMethodID(
986 jni, *j_frame_class_, "<init>", "(II[I[Ljava/nio/ByteBuffer;)V")),
987 j_byte_buffer_class_(jni, FindClass(jni, "java/nio/ByteBuffer")) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000988 CHECK_EXCEPTION(jni, "");
989 }
990
henrike@webrtc.org723d6832013-07-12 16:04:50 +0000991 virtual ~JavaVideoRendererWrapper() {}
992
993 virtual void SetSize(int width, int height) OVERRIDE {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000994 jni()->CallVoidMethod(*j_callbacks_, j_set_size_id_, width, height);
995 CHECK_EXCEPTION(jni(), "");
996 }
997
henrike@webrtc.org723d6832013-07-12 16:04:50 +0000998 virtual void RenderFrame(const cricket::VideoFrame* frame) OVERRIDE {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000999 ScopedLocalRef<jobject> j_frame(jni(), CricketToJavaFrame(frame));
1000 jni()->CallVoidMethod(*j_callbacks_, j_render_frame_id_, *j_frame);
1001 CHECK_EXCEPTION(jni(), "");
1002 }
1003
1004 private:
1005 // Return a VideoRenderer.I420Frame referring to the data in |frame|.
1006 jobject CricketToJavaFrame(const cricket::VideoFrame* frame) {
1007 ScopedLocalRef<jintArray> strides(jni(), jni()->NewIntArray(3));
1008 jint* strides_array = jni()->GetIntArrayElements(*strides, NULL);
1009 strides_array[0] = frame->GetYPitch();
1010 strides_array[1] = frame->GetUPitch();
1011 strides_array[2] = frame->GetVPitch();
1012 jni()->ReleaseIntArrayElements(*strides, strides_array, 0);
1013 ScopedLocalRef<jobjectArray> planes(
henrike@webrtc.org723d6832013-07-12 16:04:50 +00001014 jni(), jni()->NewObjectArray(3, *j_byte_buffer_class_, NULL));
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001015 ScopedLocalRef<jobject> y_buffer(jni(), jni()->NewDirectByteBuffer(
1016 const_cast<uint8*>(frame->GetYPlane()),
1017 frame->GetYPitch() * frame->GetHeight()));
1018 ScopedLocalRef<jobject> u_buffer(jni(), jni()->NewDirectByteBuffer(
1019 const_cast<uint8*>(frame->GetUPlane()), frame->GetChromaSize()));
1020 ScopedLocalRef<jobject> v_buffer(jni(), jni()->NewDirectByteBuffer(
1021 const_cast<uint8*>(frame->GetVPlane()), frame->GetChromaSize()));
1022 jni()->SetObjectArrayElement(*planes, 0, *y_buffer);
1023 jni()->SetObjectArrayElement(*planes, 1, *u_buffer);
1024 jni()->SetObjectArrayElement(*planes, 2, *v_buffer);
1025 return jni()->NewObject(
henrike@webrtc.org723d6832013-07-12 16:04:50 +00001026 *j_frame_class_, j_frame_ctor_id_,
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001027 frame->GetWidth(), frame->GetHeight(), *strides, *planes);
1028 }
1029
1030 JNIEnv* jni() {
1031 return AttachCurrentThreadIfNeeded();
1032 }
1033
henrike@webrtc.org723d6832013-07-12 16:04:50 +00001034 ScopedGlobalRef<jobject> j_callbacks_;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001035 jmethodID j_set_size_id_;
1036 jmethodID j_render_frame_id_;
henrike@webrtc.org723d6832013-07-12 16:04:50 +00001037 ScopedGlobalRef<jclass> j_frame_class_;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001038 jmethodID j_frame_ctor_id_;
henrike@webrtc.org723d6832013-07-12 16:04:50 +00001039 ScopedGlobalRef<jclass> j_byte_buffer_class_;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001040};
1041
1042} // anonymous namespace
1043
1044
1045// Convenience macro defining JNI-accessible methods in the org.webrtc package.
1046// Eliminates unnecessary boilerplate and line-wraps, reducing visual clutter.
1047#define JOW(rettype, name) extern "C" rettype JNIEXPORT JNICALL \
1048 Java_org_webrtc_##name
1049
1050extern "C" jint JNIEXPORT JNICALL JNI_OnLoad(JavaVM *jvm, void *reserved) {
1051 CHECK(!g_jvm, "JNI_OnLoad called more than once!");
1052 g_jvm = jvm;
1053 CHECK(g_jvm, "JNI_OnLoad handed NULL?");
1054
1055 CHECK(talk_base::InitializeSSL(), "Failed to InitializeSSL()");
1056
1057 JNIEnv* jni;
1058 if (jvm->GetEnv(reinterpret_cast<void**>(&jni), JNI_VERSION_1_6) != JNI_OK)
1059 return -1;
1060 g_class_reference_holder = new ClassReferenceHolder(jni);
1061
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001062 return JNI_VERSION_1_6;
1063}
1064
1065extern "C" void JNIEXPORT JNICALL JNI_OnUnLoad(JavaVM *jvm, void *reserved) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001066 delete g_class_reference_holder;
1067 g_class_reference_holder = NULL;
1068 CHECK(talk_base::CleanupSSL(), "Failed to CleanupSSL()");
1069}
1070
fischman@webrtc.org32001ef2013-08-12 23:26:21 +00001071static DataChannelInterface* ExtractNativeDC(JNIEnv* jni, jobject j_dc) {
henrike@webrtc.org723d6832013-07-12 16:04:50 +00001072 jfieldID native_dc_id = GetFieldID(jni,
1073 GetObjectClass(jni, j_dc), "nativeDataChannel", "J");
1074 jlong j_d = GetLongField(jni, j_dc, native_dc_id);
fischman@webrtc.org32001ef2013-08-12 23:26:21 +00001075 return reinterpret_cast<DataChannelInterface*>(j_d);
henrike@webrtc.org723d6832013-07-12 16:04:50 +00001076}
1077
1078JOW(jlong, DataChannel_registerObserverNative)(
1079 JNIEnv* jni, jobject j_dc, jobject j_observer) {
1080 talk_base::scoped_ptr<DataChannelObserverWrapper> observer(
1081 new DataChannelObserverWrapper(jni, j_observer));
1082 ExtractNativeDC(jni, j_dc)->RegisterObserver(observer.get());
1083 return reinterpret_cast<jlong>(observer.release());
1084}
1085
1086JOW(void, DataChannel_unregisterObserverNative)(
1087 JNIEnv* jni, jobject j_dc, jlong native_observer) {
1088 ExtractNativeDC(jni, j_dc)->UnregisterObserver();
1089 delete reinterpret_cast<DataChannelObserverWrapper*>(native_observer);
1090}
1091
1092JOW(jstring, DataChannel_label)(JNIEnv* jni, jobject j_dc) {
1093 return JavaStringFromStdString(jni, ExtractNativeDC(jni, j_dc)->label());
1094}
1095
1096JOW(jobject, DataChannel_state)(JNIEnv* jni, jobject j_dc) {
1097 return JavaEnumFromIndex(
1098 jni, "DataChannel$State", ExtractNativeDC(jni, j_dc)->state());
1099}
1100
1101JOW(jlong, DataChannel_bufferedAmount)(JNIEnv* jni, jobject j_dc) {
1102 uint64 buffered_amount = ExtractNativeDC(jni, j_dc)->buffered_amount();
1103 CHECK(buffered_amount <= std::numeric_limits<int64>::max(),
1104 "buffered_amount overflowed jlong!");
1105 return static_cast<jlong>(buffered_amount);
1106}
1107
1108JOW(void, DataChannel_close)(JNIEnv* jni, jobject j_dc) {
1109 ExtractNativeDC(jni, j_dc)->Close();
1110}
1111
1112JOW(jboolean, DataChannel_sendNative)(JNIEnv* jni, jobject j_dc,
1113 jbyteArray data, jboolean binary) {
1114 jbyte* bytes = jni->GetByteArrayElements(data, NULL);
1115 bool ret = ExtractNativeDC(jni, j_dc)->Send(DataBuffer(
1116 talk_base::Buffer(bytes, jni->GetArrayLength(data)),
1117 binary));
1118 jni->ReleaseByteArrayElements(data, bytes, JNI_ABORT);
1119 return ret;
1120}
1121
1122JOW(void, DataChannel_dispose)(JNIEnv* jni, jobject j_dc) {
fischman@webrtc.org32001ef2013-08-12 23:26:21 +00001123 CHECK_RELEASE(ExtractNativeDC(jni, j_dc));
henrike@webrtc.org723d6832013-07-12 16:04:50 +00001124}
1125
fischman@webrtc.orgc883fdc2013-08-06 19:00:53 +00001126JOW(void, Logging_nativeEnableTracing)(
1127 JNIEnv* jni, jclass, jstring j_path, jint nativeLevels,
1128 jint nativeSeverity) {
1129 std::string path = JavaToStdString(jni, j_path);
1130 if (nativeLevels != webrtc::kTraceNone) {
1131 CHECK(!webrtc::Trace::SetTraceFile(path.c_str(), false),
1132 "SetTraceFile failed");
1133 CHECK(!webrtc::Trace::SetLevelFilter(nativeLevels),
1134 "SetLevelFilter failed");
1135 }
1136 talk_base::LogMessage::LogToDebug(nativeSeverity);
1137}
1138
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001139JOW(void, PeerConnection_freePeerConnection)(JNIEnv*, jclass, jlong j_p) {
fischman@webrtc.org32001ef2013-08-12 23:26:21 +00001140 CHECK_RELEASE(reinterpret_cast<PeerConnectionInterface*>(j_p));
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001141}
1142
1143JOW(void, PeerConnection_freeObserver)(JNIEnv*, jclass, jlong j_p) {
1144 PCOJava* p = reinterpret_cast<PCOJava*>(j_p);
1145 delete p;
1146}
1147
1148JOW(void, MediaSource_free)(JNIEnv*, jclass, jlong j_p) {
fischman@webrtc.org32001ef2013-08-12 23:26:21 +00001149 CHECK_RELEASE(reinterpret_cast<MediaSourceInterface*>(j_p));
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001150}
1151
1152JOW(void, VideoCapturer_free)(JNIEnv*, jclass, jlong j_p) {
1153 delete reinterpret_cast<cricket::VideoCapturer*>(j_p);
1154}
1155
1156JOW(void, VideoRenderer_free)(JNIEnv*, jclass, jlong j_p) {
1157 delete reinterpret_cast<VideoRendererWrapper*>(j_p);
1158}
1159
1160JOW(void, MediaStreamTrack_free)(JNIEnv*, jclass, jlong j_p) {
fischman@webrtc.org32001ef2013-08-12 23:26:21 +00001161 CHECK_RELEASE(reinterpret_cast<MediaStreamTrackInterface*>(j_p));
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001162}
1163
1164JOW(jboolean, MediaStream_nativeAddAudioTrack)(
1165 JNIEnv* jni, jclass, jlong pointer, jlong j_audio_track_pointer) {
fischman@webrtc.org32001ef2013-08-12 23:26:21 +00001166 return reinterpret_cast<MediaStreamInterface*>(pointer)->AddTrack(
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001167 reinterpret_cast<AudioTrackInterface*>(j_audio_track_pointer));
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001168}
1169
1170JOW(jboolean, MediaStream_nativeAddVideoTrack)(
1171 JNIEnv* jni, jclass, jlong pointer, jlong j_video_track_pointer) {
fischman@webrtc.org32001ef2013-08-12 23:26:21 +00001172 return reinterpret_cast<MediaStreamInterface*>(pointer)
1173 ->AddTrack(reinterpret_cast<VideoTrackInterface*>(j_video_track_pointer));
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001174}
1175
1176JOW(jboolean, MediaStream_nativeRemoveAudioTrack)(
1177 JNIEnv* jni, jclass, jlong pointer, jlong j_audio_track_pointer) {
fischman@webrtc.org32001ef2013-08-12 23:26:21 +00001178 return reinterpret_cast<MediaStreamInterface*>(pointer)->RemoveTrack(
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001179 reinterpret_cast<AudioTrackInterface*>(j_audio_track_pointer));
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001180}
1181
1182JOW(jboolean, MediaStream_nativeRemoveVideoTrack)(
1183 JNIEnv* jni, jclass, jlong pointer, jlong j_video_track_pointer) {
fischman@webrtc.org32001ef2013-08-12 23:26:21 +00001184 return reinterpret_cast<MediaStreamInterface*>(pointer)->RemoveTrack(
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001185 reinterpret_cast<VideoTrackInterface*>(j_video_track_pointer));
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001186}
1187
1188JOW(jstring, MediaStream_nativeLabel)(JNIEnv* jni, jclass, jlong j_p) {
1189 return JavaStringFromStdString(
1190 jni, reinterpret_cast<MediaStreamInterface*>(j_p)->label());
1191}
1192
1193JOW(void, MediaStream_free)(JNIEnv*, jclass, jlong j_p) {
fischman@webrtc.org32001ef2013-08-12 23:26:21 +00001194 CHECK_RELEASE(reinterpret_cast<MediaStreamInterface*>(j_p));
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001195}
1196
1197JOW(jlong, PeerConnectionFactory_nativeCreateObserver)(
1198 JNIEnv * jni, jclass, jobject j_observer) {
1199 return (jlong)new PCOJava(jni, j_observer);
1200}
1201
1202#ifdef ANDROID
1203JOW(jboolean, PeerConnectionFactory_initializeAndroidGlobals)(
1204 JNIEnv* jni, jclass, jobject context) {
1205 CHECK(g_jvm, "JNI_OnLoad failed to run?");
1206 bool failure = false;
1207 failure |= webrtc::VideoEngine::SetAndroidObjects(g_jvm, context);
1208 failure |= webrtc::VoiceEngine::SetAndroidObjects(g_jvm, jni, context);
1209 return !failure;
1210}
1211#endif // ANDROID
1212
1213JOW(jlong, PeerConnectionFactory_nativeCreatePeerConnectionFactory)(
1214 JNIEnv* jni, jclass) {
fischman@webrtc.org32001ef2013-08-12 23:26:21 +00001215 webrtc::Trace::CreateTrace();
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001216 talk_base::scoped_refptr<PeerConnectionFactoryInterface> factory(
1217 webrtc::CreatePeerConnectionFactory());
1218 return (jlong)factory.release();
1219}
1220
1221JOW(void, PeerConnectionFactory_freeFactory)(JNIEnv*, jclass, jlong j_p) {
fischman@webrtc.org32001ef2013-08-12 23:26:21 +00001222 CHECK_RELEASE(reinterpret_cast<PeerConnectionFactoryInterface*>(j_p));
1223 webrtc::Trace::ReturnTrace();
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001224}
1225
1226JOW(jlong, PeerConnectionFactory_nativeCreateLocalMediaStream)(
1227 JNIEnv* jni, jclass, jlong native_factory, jstring label) {
1228 talk_base::scoped_refptr<PeerConnectionFactoryInterface> factory(
1229 reinterpret_cast<PeerConnectionFactoryInterface*>(native_factory));
1230 talk_base::scoped_refptr<MediaStreamInterface> stream(
1231 factory->CreateLocalMediaStream(JavaToStdString(jni, label)));
1232 return (jlong)stream.release();
1233}
1234
1235JOW(jlong, PeerConnectionFactory_nativeCreateVideoSource)(
1236 JNIEnv* jni, jclass, jlong native_factory, jlong native_capturer,
1237 jobject j_constraints) {
1238 talk_base::scoped_ptr<ConstraintsWrapper> constraints(
1239 new ConstraintsWrapper(jni, j_constraints));
1240 talk_base::scoped_refptr<PeerConnectionFactoryInterface> factory(
1241 reinterpret_cast<PeerConnectionFactoryInterface*>(native_factory));
1242 talk_base::scoped_refptr<VideoSourceInterface> source(
1243 factory->CreateVideoSource(
1244 reinterpret_cast<cricket::VideoCapturer*>(native_capturer),
1245 constraints.get()));
1246 return (jlong)source.release();
1247}
1248
1249JOW(jlong, PeerConnectionFactory_nativeCreateVideoTrack)(
1250 JNIEnv* jni, jclass, jlong native_factory, jstring id,
1251 jlong native_source) {
1252 talk_base::scoped_refptr<PeerConnectionFactoryInterface> factory(
1253 reinterpret_cast<PeerConnectionFactoryInterface*>(native_factory));
1254 talk_base::scoped_refptr<VideoTrackInterface> track(
1255 factory->CreateVideoTrack(
1256 JavaToStdString(jni, id),
1257 reinterpret_cast<VideoSourceInterface*>(native_source)));
1258 return (jlong)track.release();
1259}
1260
1261JOW(jlong, PeerConnectionFactory_nativeCreateAudioTrack)(
1262 JNIEnv* jni, jclass, jlong native_factory, jstring id) {
1263 talk_base::scoped_refptr<PeerConnectionFactoryInterface> factory(
1264 reinterpret_cast<PeerConnectionFactoryInterface*>(native_factory));
1265 talk_base::scoped_refptr<AudioTrackInterface> track(
1266 factory->CreateAudioTrack(JavaToStdString(jni, id), NULL));
1267 return (jlong)track.release();
1268}
1269
1270static void JavaIceServersToJsepIceServers(
1271 JNIEnv* jni, jobject j_ice_servers,
1272 PeerConnectionInterface::IceServers* ice_servers) {
1273 jclass list_class = GetObjectClass(jni, j_ice_servers);
1274 jmethodID iterator_id = GetMethodID(
1275 jni, list_class, "iterator", "()Ljava/util/Iterator;");
1276 jobject iterator = jni->CallObjectMethod(j_ice_servers, iterator_id);
1277 CHECK_EXCEPTION(jni, "error during CallObjectMethod");
1278 jmethodID iterator_has_next = GetMethodID(
1279 jni, GetObjectClass(jni, iterator), "hasNext", "()Z");
1280 jmethodID iterator_next = GetMethodID(
1281 jni, GetObjectClass(jni, iterator), "next", "()Ljava/lang/Object;");
1282 while (jni->CallBooleanMethod(iterator, iterator_has_next)) {
1283 CHECK_EXCEPTION(jni, "error during CallBooleanMethod");
1284 jobject j_ice_server = jni->CallObjectMethod(iterator, iterator_next);
1285 CHECK_EXCEPTION(jni, "error during CallObjectMethod");
1286 jclass j_ice_server_class = GetObjectClass(jni, j_ice_server);
1287 jfieldID j_ice_server_uri_id =
1288 GetFieldID(jni, j_ice_server_class, "uri", "Ljava/lang/String;");
1289 jfieldID j_ice_server_username_id =
1290 GetFieldID(jni, j_ice_server_class, "username", "Ljava/lang/String;");
1291 jfieldID j_ice_server_password_id =
1292 GetFieldID(jni, j_ice_server_class, "password", "Ljava/lang/String;");
1293 jstring uri = reinterpret_cast<jstring>(
1294 GetObjectField(jni, j_ice_server, j_ice_server_uri_id));
1295 jstring username = reinterpret_cast<jstring>(
1296 GetObjectField(jni, j_ice_server, j_ice_server_username_id));
1297 jstring password = reinterpret_cast<jstring>(
1298 GetObjectField(jni, j_ice_server, j_ice_server_password_id));
1299 PeerConnectionInterface::IceServer server;
1300 server.uri = JavaToStdString(jni, uri);
1301 server.username = JavaToStdString(jni, username);
1302 server.password = JavaToStdString(jni, password);
1303 ice_servers->push_back(server);
1304 }
1305 CHECK_EXCEPTION(jni, "error during CallBooleanMethod");
1306}
1307
1308JOW(jlong, PeerConnectionFactory_nativeCreatePeerConnection)(
1309 JNIEnv *jni, jclass, jlong factory, jobject j_ice_servers,
1310 jobject j_constraints, jlong observer_p) {
1311 talk_base::scoped_refptr<PeerConnectionFactoryInterface> f(
1312 reinterpret_cast<PeerConnectionFactoryInterface*>(factory));
1313 PeerConnectionInterface::IceServers servers;
1314 JavaIceServersToJsepIceServers(jni, j_ice_servers, &servers);
1315 PCOJava* observer = reinterpret_cast<PCOJava*>(observer_p);
1316 observer->SetConstraints(new ConstraintsWrapper(jni, j_constraints));
1317 talk_base::scoped_refptr<PeerConnectionInterface> pc(f->CreatePeerConnection(
1318 servers, observer->constraints(), NULL, observer));
1319 return (jlong)pc.release();
1320}
1321
1322static talk_base::scoped_refptr<PeerConnectionInterface> ExtractNativePC(
1323 JNIEnv* jni, jobject j_pc) {
1324 jfieldID native_pc_id = GetFieldID(jni,
1325 GetObjectClass(jni, j_pc), "nativePeerConnection", "J");
1326 jlong j_p = GetLongField(jni, j_pc, native_pc_id);
1327 return talk_base::scoped_refptr<PeerConnectionInterface>(
1328 reinterpret_cast<PeerConnectionInterface*>(j_p));
1329}
1330
1331JOW(jobject, PeerConnection_getLocalDescription)(JNIEnv* jni, jobject j_pc) {
1332 const SessionDescriptionInterface* sdp =
1333 ExtractNativePC(jni, j_pc)->local_description();
1334 return sdp ? JavaSdpFromNativeSdp(jni, sdp) : NULL;
1335}
1336
1337JOW(jobject, PeerConnection_getRemoteDescription)(JNIEnv* jni, jobject j_pc) {
1338 const SessionDescriptionInterface* sdp =
1339 ExtractNativePC(jni, j_pc)->remote_description();
1340 return sdp ? JavaSdpFromNativeSdp(jni, sdp) : NULL;
1341}
1342
henrike@webrtc.org723d6832013-07-12 16:04:50 +00001343JOW(jobject, PeerConnection_createDataChannel)(
1344 JNIEnv* jni, jobject j_pc, jstring j_label, jobject j_init) {
1345 DataChannelInit init = JavaDataChannelInitToNative(jni, j_init);
1346 talk_base::scoped_refptr<DataChannelInterface> channel(
1347 ExtractNativePC(jni, j_pc)->CreateDataChannel(
1348 JavaToStdString(jni, j_label), &init));
1349 jclass j_data_channel_class = FindClass(jni, "org/webrtc/DataChannel");
1350 jmethodID j_data_channel_ctor = GetMethodID(
1351 jni, j_data_channel_class, "<init>", "(J)V");
1352 jobject j_channel = jni->NewObject(
1353 j_data_channel_class, j_data_channel_ctor, channel.get());
1354 CHECK_EXCEPTION(jni, "error during NewObject");
1355 // Channel is now owned by Java object, and will be freed from there.
fischman@webrtc.org32001ef2013-08-12 23:26:21 +00001356 int bumped_count = channel->AddRef();
1357 CHECK(bumped_count == 2, "Unexpected refcount");
henrike@webrtc.org723d6832013-07-12 16:04:50 +00001358 return j_channel;
1359}
1360
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001361JOW(void, PeerConnection_createOffer)(
1362 JNIEnv* jni, jobject j_pc, jobject j_observer, jobject j_constraints) {
1363 ConstraintsWrapper* constraints =
1364 new ConstraintsWrapper(jni, j_constraints);
1365 talk_base::scoped_refptr<CreateSdpObserverWrapper> observer(
1366 new talk_base::RefCountedObject<CreateSdpObserverWrapper>(
1367 jni, j_observer, constraints));
1368 ExtractNativePC(jni, j_pc)->CreateOffer(observer, constraints);
1369}
1370
1371JOW(void, PeerConnection_createAnswer)(
1372 JNIEnv* jni, jobject j_pc, jobject j_observer, jobject j_constraints) {
1373 ConstraintsWrapper* constraints =
1374 new ConstraintsWrapper(jni, j_constraints);
1375 talk_base::scoped_refptr<CreateSdpObserverWrapper> observer(
1376 new talk_base::RefCountedObject<CreateSdpObserverWrapper>(
1377 jni, j_observer, constraints));
1378 ExtractNativePC(jni, j_pc)->CreateAnswer(observer, constraints);
1379}
1380
1381// Helper to create a SessionDescriptionInterface from a SessionDescription.
1382static SessionDescriptionInterface* JavaSdpToNativeSdp(
1383 JNIEnv* jni, jobject j_sdp) {
1384 jfieldID j_type_id = GetFieldID(
1385 jni, GetObjectClass(jni, j_sdp), "type",
1386 "Lorg/webrtc/SessionDescription$Type;");
1387 jobject j_type = GetObjectField(jni, j_sdp, j_type_id);
1388 jmethodID j_canonical_form_id = GetMethodID(
1389 jni, GetObjectClass(jni, j_type), "canonicalForm",
1390 "()Ljava/lang/String;");
1391 jstring j_type_string = (jstring)jni->CallObjectMethod(
1392 j_type, j_canonical_form_id);
1393 CHECK_EXCEPTION(jni, "error during CallObjectMethod");
1394 std::string std_type = JavaToStdString(jni, j_type_string);
1395
1396 jfieldID j_description_id = GetFieldID(
1397 jni, GetObjectClass(jni, j_sdp), "description", "Ljava/lang/String;");
1398 jstring j_description = (jstring)GetObjectField(jni, j_sdp, j_description_id);
1399 std::string std_description = JavaToStdString(jni, j_description);
1400
1401 return webrtc::CreateSessionDescription(
1402 std_type, std_description, NULL);
1403}
1404
1405JOW(void, PeerConnection_setLocalDescription)(
1406 JNIEnv* jni, jobject j_pc,
1407 jobject j_observer, jobject j_sdp) {
1408 talk_base::scoped_refptr<SetSdpObserverWrapper> observer(
1409 new talk_base::RefCountedObject<SetSdpObserverWrapper>(
1410 jni, j_observer, reinterpret_cast<ConstraintsWrapper*>(NULL)));
1411 ExtractNativePC(jni, j_pc)->SetLocalDescription(
1412 observer, JavaSdpToNativeSdp(jni, j_sdp));
1413}
1414
1415JOW(void, PeerConnection_setRemoteDescription)(
1416 JNIEnv* jni, jobject j_pc,
1417 jobject j_observer, jobject j_sdp) {
1418 talk_base::scoped_refptr<SetSdpObserverWrapper> observer(
1419 new talk_base::RefCountedObject<SetSdpObserverWrapper>(
1420 jni, j_observer, reinterpret_cast<ConstraintsWrapper*>(NULL)));
1421 ExtractNativePC(jni, j_pc)->SetRemoteDescription(
1422 observer, JavaSdpToNativeSdp(jni, j_sdp));
1423}
1424
1425JOW(jboolean, PeerConnection_updateIce)(
1426 JNIEnv* jni, jobject j_pc, jobject j_ice_servers, jobject j_constraints) {
1427 PeerConnectionInterface::IceServers ice_servers;
1428 JavaIceServersToJsepIceServers(jni, j_ice_servers, &ice_servers);
1429 talk_base::scoped_ptr<ConstraintsWrapper> constraints(
1430 new ConstraintsWrapper(jni, j_constraints));
1431 return ExtractNativePC(jni, j_pc)->UpdateIce(ice_servers, constraints.get());
1432}
1433
1434JOW(jboolean, PeerConnection_nativeAddIceCandidate)(
1435 JNIEnv* jni, jobject j_pc, jstring j_sdp_mid,
1436 jint j_sdp_mline_index, jstring j_candidate_sdp) {
1437 std::string sdp_mid = JavaToStdString(jni, j_sdp_mid);
1438 std::string sdp = JavaToStdString(jni, j_candidate_sdp);
1439 talk_base::scoped_ptr<IceCandidateInterface> candidate(
1440 webrtc::CreateIceCandidate(sdp_mid, j_sdp_mline_index, sdp, NULL));
1441 return ExtractNativePC(jni, j_pc)->AddIceCandidate(candidate.get());
1442}
1443
1444JOW(jboolean, PeerConnection_nativeAddLocalStream)(
1445 JNIEnv* jni, jobject j_pc, jlong native_stream, jobject j_constraints) {
1446 talk_base::scoped_ptr<ConstraintsWrapper> constraints(
1447 new ConstraintsWrapper(jni, j_constraints));
1448 return ExtractNativePC(jni, j_pc)->AddStream(
1449 reinterpret_cast<MediaStreamInterface*>(native_stream),
1450 constraints.get());
1451}
1452
1453JOW(void, PeerConnection_nativeRemoveLocalStream)(
1454 JNIEnv* jni, jobject j_pc, jlong native_stream) {
1455 ExtractNativePC(jni, j_pc)->RemoveStream(
1456 reinterpret_cast<MediaStreamInterface*>(native_stream));
1457}
1458
1459JOW(bool, PeerConnection_nativeGetStats)(
1460 JNIEnv* jni, jobject j_pc, jobject j_observer, jlong native_track) {
1461 talk_base::scoped_refptr<StatsObserverWrapper> observer(
1462 new talk_base::RefCountedObject<StatsObserverWrapper>(jni, j_observer));
1463 return ExtractNativePC(jni, j_pc)->GetStats(
1464 observer, reinterpret_cast<MediaStreamTrackInterface*>(native_track));
1465}
1466
1467JOW(jobject, PeerConnection_signalingState)(JNIEnv* jni, jobject j_pc) {
1468 PeerConnectionInterface::SignalingState state =
1469 ExtractNativePC(jni, j_pc)->signaling_state();
1470 return JavaEnumFromIndex(jni, "PeerConnection$SignalingState", state);
1471}
1472
1473JOW(jobject, PeerConnection_iceConnectionState)(JNIEnv* jni, jobject j_pc) {
1474 PeerConnectionInterface::IceConnectionState state =
1475 ExtractNativePC(jni, j_pc)->ice_connection_state();
1476 return JavaEnumFromIndex(jni, "PeerConnection$IceConnectionState", state);
1477}
1478
1479JOW(jobject, PeerGathering_iceGatheringState)(JNIEnv* jni, jobject j_pc) {
1480 PeerConnectionInterface::IceGatheringState state =
1481 ExtractNativePC(jni, j_pc)->ice_gathering_state();
1482 return JavaEnumFromIndex(jni, "PeerGathering$IceGatheringState", state);
1483}
1484
1485JOW(void, PeerConnection_close)(JNIEnv* jni, jobject j_pc) {
1486 ExtractNativePC(jni, j_pc)->Close();
1487 return;
1488}
1489
1490JOW(jobject, MediaSource_nativeState)(JNIEnv* jni, jclass, jlong j_p) {
1491 talk_base::scoped_refptr<MediaSourceInterface> p(
1492 reinterpret_cast<MediaSourceInterface*>(j_p));
1493 return JavaEnumFromIndex(jni, "MediaSource$State", p->state());
1494}
1495
1496JOW(jlong, VideoCapturer_nativeCreateVideoCapturer)(
1497 JNIEnv* jni, jclass, jstring j_device_name) {
1498 std::string device_name = JavaToStdString(jni, j_device_name);
1499 talk_base::scoped_ptr<cricket::DeviceManagerInterface> device_manager(
1500 cricket::DeviceManagerFactory::Create());
1501 CHECK(device_manager->Init(), "DeviceManager::Init() failed");
1502 cricket::Device device;
1503 if (!device_manager->GetVideoCaptureDevice(device_name, &device)) {
1504 LOG(LS_ERROR) << "GetVideoCaptureDevice failed";
1505 return 0;
1506 }
1507 talk_base::scoped_ptr<cricket::VideoCapturer> capturer(
1508 device_manager->CreateVideoCapturer(device));
1509 return (jlong)capturer.release();
1510}
1511
1512JOW(jlong, VideoRenderer_nativeCreateGuiVideoRenderer)(
1513 JNIEnv* jni, jclass, int x, int y) {
1514 talk_base::scoped_ptr<VideoRendererWrapper> renderer(
1515 VideoRendererWrapper::Create(
1516 cricket::VideoRendererFactory::CreateGuiVideoRenderer(x, y)));
1517 return (jlong)renderer.release();
1518}
1519
1520JOW(jlong, VideoRenderer_nativeWrapVideoRenderer)(
1521 JNIEnv* jni, jclass, jobject j_callbacks) {
1522 talk_base::scoped_ptr<JavaVideoRendererWrapper> renderer(
1523 new JavaVideoRendererWrapper(jni, j_callbacks));
1524 return (jlong)renderer.release();
1525}
1526
1527JOW(jstring, MediaStreamTrack_nativeId)(JNIEnv* jni, jclass, jlong j_p) {
fischman@webrtc.org32001ef2013-08-12 23:26:21 +00001528 return JavaStringFromStdString(
1529 jni, reinterpret_cast<MediaStreamTrackInterface*>(j_p)->id());
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001530}
1531
1532JOW(jstring, MediaStreamTrack_nativeKind)(JNIEnv* jni, jclass, jlong j_p) {
fischman@webrtc.org32001ef2013-08-12 23:26:21 +00001533 return JavaStringFromStdString(
1534 jni, reinterpret_cast<MediaStreamTrackInterface*>(j_p)->kind());
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001535}
1536
1537JOW(jboolean, MediaStreamTrack_nativeEnabled)(JNIEnv* jni, jclass, jlong j_p) {
fischman@webrtc.org32001ef2013-08-12 23:26:21 +00001538 return reinterpret_cast<MediaStreamTrackInterface*>(j_p)->enabled();
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001539}
1540
1541JOW(jobject, MediaStreamTrack_nativeState)(JNIEnv* jni, jclass, jlong j_p) {
fischman@webrtc.org32001ef2013-08-12 23:26:21 +00001542 return JavaEnumFromIndex(
1543 jni,
1544 "MediaStreamTrack$State",
1545 reinterpret_cast<MediaStreamTrackInterface*>(j_p)->state());
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001546}
1547
1548JOW(jboolean, MediaStreamTrack_nativeSetState)(
1549 JNIEnv* jni, jclass, jlong j_p, jint j_new_state) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001550 MediaStreamTrackInterface::TrackState new_state =
1551 (MediaStreamTrackInterface::TrackState)j_new_state;
fischman@webrtc.org32001ef2013-08-12 23:26:21 +00001552 return reinterpret_cast<MediaStreamTrackInterface*>(j_p)
1553 ->set_state(new_state);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001554}
1555
1556JOW(jboolean, MediaStreamTrack_nativeSetEnabled)(
1557 JNIEnv* jni, jclass, jlong j_p, jboolean enabled) {
fischman@webrtc.org32001ef2013-08-12 23:26:21 +00001558 return reinterpret_cast<MediaStreamTrackInterface*>(j_p)
1559 ->set_enabled(enabled);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001560}
1561
1562JOW(void, VideoTrack_nativeAddRenderer)(
1563 JNIEnv* jni, jclass,
1564 jlong j_video_track_pointer, jlong j_renderer_pointer) {
fischman@webrtc.org32001ef2013-08-12 23:26:21 +00001565 reinterpret_cast<VideoTrackInterface*>(j_video_track_pointer)->AddRenderer(
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001566 reinterpret_cast<VideoRendererInterface*>(j_renderer_pointer));
1567}
1568
1569JOW(void, VideoTrack_nativeRemoveRenderer)(
1570 JNIEnv* jni, jclass,
1571 jlong j_video_track_pointer, jlong j_renderer_pointer) {
fischman@webrtc.org32001ef2013-08-12 23:26:21 +00001572 reinterpret_cast<VideoTrackInterface*>(j_video_track_pointer)->RemoveRenderer(
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001573 reinterpret_cast<VideoRendererInterface*>(j_renderer_pointer));
1574}