blob: 574c977cd0498f48f05334e5132bb6afddaa288a [file] [log] [blame]
Henrik Kjellanderff761fb2015-11-04 08:31:52 +01001/*
2 * Copyright (c) 2015 The WebRTC project authors. All Rights Reserved.
3 *
4 * Use of this source code is governed by a BSD-style license
5 * that can be found in the LICENSE file in the root of the source
6 * tree. An additional intellectual property rights grant can be found
7 * in the file PATENTS. All contributing project authors may
8 * be found in the AUTHORS file in the root of the source tree.
9 */
10
11#ifndef WEBRTC_MODULES_UTILITY_INCLUDE_JVM_ANDROID_H_
12#define WEBRTC_MODULES_UTILITY_INCLUDE_JVM_ANDROID_H_
13
14#include <jni.h>
kwiberg84be5112016-04-27 01:19:58 -070015
16#include <memory>
Henrik Kjellanderff761fb2015-11-04 08:31:52 +010017#include <string>
18
Henrik Kjellanderff761fb2015-11-04 08:31:52 +010019#include "webrtc/base/thread_checker.h"
20#include "webrtc/modules/utility/include/helpers_android.h"
21
22namespace webrtc {
23
24// The JNI interface pointer (JNIEnv) is valid only in the current thread.
25// Should another thread need to access the Java VM, it must first call
26// AttachCurrentThread() to attach itself to the VM and obtain a JNI interface
27// pointer. The native thread remains attached to the VM until it calls
28// DetachCurrentThread() to detach.
29class AttachCurrentThreadIfNeeded {
30 public:
31 AttachCurrentThreadIfNeeded();
32 ~AttachCurrentThreadIfNeeded();
33
34 private:
35 rtc::ThreadChecker thread_checker_;
36 bool attached_;
37};
38
39// This class is created by the NativeRegistration class and is used to wrap
40// the actual Java object handle (jobject) on which we can call methods from
41// C++ in to Java. See example in JVM for more details.
42// TODO(henrika): extend support for type of function calls.
43class GlobalRef {
44 public:
45 GlobalRef(JNIEnv* jni, jobject object);
46 ~GlobalRef();
47
48 jboolean CallBooleanMethod(jmethodID methodID, ...);
49 jint CallIntMethod(jmethodID methodID, ...);
50 void CallVoidMethod(jmethodID methodID, ...);
51
52 private:
53 JNIEnv* const jni_;
54 const jobject j_object_;
55};
56
57// Wraps the jclass object on which we can call GetMethodId() functions to
58// query method IDs.
59class JavaClass {
60 public:
61 JavaClass(JNIEnv* jni, jclass clazz) : jni_(jni), j_class_(clazz) {}
62 ~JavaClass() {}
63
64 jmethodID GetMethodId(const char* name, const char* signature);
65 jmethodID GetStaticMethodId(const char* name, const char* signature);
66 jobject CallStaticObjectMethod(jmethodID methodID, ...);
67
68 protected:
69 JNIEnv* const jni_;
70 jclass const j_class_;
71};
72
73// Adds support of the NewObject factory method to the JavaClass class.
74// See example in JVM for more details on how to use it.
75class NativeRegistration : public JavaClass {
76 public:
77 NativeRegistration(JNIEnv* jni, jclass clazz);
78 ~NativeRegistration();
79
kwiberg84be5112016-04-27 01:19:58 -070080 std::unique_ptr<GlobalRef> NewObject(
Henrik Kjellanderff761fb2015-11-04 08:31:52 +010081 const char* name, const char* signature, ...);
82
83 private:
84 JNIEnv* const jni_;
85};
86
87// This class is created by the JVM class and is used to expose methods that
88// needs the JNI interface pointer but its main purpose is to create a
89// NativeRegistration object given name of a Java class and a list of native
90// methods. See example in JVM for more details.
91class JNIEnvironment {
92 public:
93 explicit JNIEnvironment(JNIEnv* jni);
94 ~JNIEnvironment();
95
96 // Registers native methods with the Java class specified by |name|.
97 // Note that the class name must be one of the names in the static
98 // |loaded_classes| array defined in jvm_android.cc.
99 // This method must be called on the construction thread.
kwiberg84be5112016-04-27 01:19:58 -0700100 std::unique_ptr<NativeRegistration> RegisterNatives(
Henrik Kjellanderff761fb2015-11-04 08:31:52 +0100101 const char* name, const JNINativeMethod *methods, int num_methods);
102
103 // Converts from Java string to std::string.
104 // This method must be called on the construction thread.
105 std::string JavaToStdString(const jstring& j_string);
106
107 private:
108 rtc::ThreadChecker thread_checker_;
109 JNIEnv* const jni_;
110};
111
112// Main class for working with Java from C++ using JNI in WebRTC.
113//
114// Example usage:
115//
116// // At initialization (e.g. in JNI_OnLoad), call JVM::Initialize.
117// JNIEnv* jni = ::base::android::AttachCurrentThread();
118// JavaVM* jvm = NULL;
119// jni->GetJavaVM(&jvm);
120// jobject context = ::base::android::GetApplicationContext();
121// webrtc::JVM::Initialize(jvm, context);
122//
123// // Header (.h) file of example class called User.
kwiberg84be5112016-04-27 01:19:58 -0700124// std::unique_ptr<JNIEnvironment> env;
125// std::unique_ptr<NativeRegistration> reg;
126// std::unique_ptr<GlobalRef> obj;
Henrik Kjellanderff761fb2015-11-04 08:31:52 +0100127//
128// // Construction (in .cc file) of User class.
129// User::User() {
130// // Calling thread must be attached to the JVM.
131// env = JVM::GetInstance()->environment();
132// reg = env->RegisterNatives("org/webrtc/WebRtcTest", ,);
133// obj = reg->NewObject("<init>", ,);
134// }
135//
136// // Each User method can now use |reg| and |obj| and call Java functions
137// // in WebRtcTest.java, e.g. boolean init() {}.
138// bool User::Foo() {
139// jmethodID id = reg->GetMethodId("init", "()Z");
140// return obj->CallBooleanMethod(id);
141// }
142//
143// // And finally, e.g. in JNI_OnUnLoad, call JVM::Uninitialize.
144// JVM::Uninitialize();
145class JVM {
146 public:
147 // Stores global handles to the Java VM interface and the application context.
148 // Should be called once on a thread that is attached to the JVM.
149 static void Initialize(JavaVM* jvm, jobject context);
150 // Clears handles stored in Initialize(). Must be called on same thread as
151 // Initialize().
152 static void Uninitialize();
153 // Gives access to the global Java VM interface pointer, which then can be
154 // used to create a valid JNIEnvironment object or to get a JavaClass object.
155 static JVM* GetInstance();
156
157 // Creates a JNIEnvironment object.
158 // This method returns a NULL pointer if AttachCurrentThread() has not been
159 // called successfully. Use the AttachCurrentThreadIfNeeded class if needed.
kwiberg84be5112016-04-27 01:19:58 -0700160 std::unique_ptr<JNIEnvironment> environment();
Henrik Kjellanderff761fb2015-11-04 08:31:52 +0100161
162 // Returns a JavaClass object given class |name|.
163 // Note that the class name must be one of the names in the static
164 // |loaded_classes| array defined in jvm_android.cc.
165 // This method must be called on the construction thread.
166 JavaClass GetClass(const char* name);
167
168 // TODO(henrika): can we make these private?
169 JavaVM* jvm() const { return jvm_; }
170 jobject context() const { return context_; }
171
172 protected:
173 JVM(JavaVM* jvm, jobject context);
174 ~JVM();
175
176 private:
177 JNIEnv* jni() const { return GetEnv(jvm_); }
178
179 rtc::ThreadChecker thread_checker_;
180 JavaVM* const jvm_;
181 jobject context_;
182};
183
184} // namespace webrtc
185
186#endif // WEBRTC_MODULES_UTILITY_INCLUDE_JVM_ANDROID_H_