1 /*
2  * Copyright (C) 2011 The Android Open Source Project
3  *
4  * Licensed under the Apache License, Version 2.0 (the "License");
5  * you may not use this file except in compliance with the License.
6  * You may obtain a copy of the License at
7  *
8  *      http://www.apache.org/licenses/LICENSE-2.0
9  *
10  * Unless required by applicable law or agreed to in writing, software
11  * distributed under the License is distributed on an "AS IS" BASIS,
12  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13  * See the License for the specific language governing permissions and
14  * limitations under the License.
15  */
16 
17 #define LOG_TAG "InputWindowHandle"
18 
19 #include <nativehelper/JNIHelp.h>
20 #include "jni.h"
21 #include <android_runtime/AndroidRuntime.h>
22 #include <utils/threads.h>
23 
24 #include <android_view_InputChannel.h>
25 #include <android/graphics/Region.h>
26 #include <ui/Region.h>
27 
28 #include "com_android_server_input_InputWindowHandle.h"
29 #include "com_android_server_input_InputApplicationHandle.h"
30 
31 namespace android {
32 
33 static struct {
34     jfieldID ptr;
35     jfieldID inputApplicationHandle;
36     jfieldID inputChannel;
37     jfieldID name;
38     jfieldID layoutParamsFlags;
39     jfieldID layoutParamsType;
40     jfieldID dispatchingTimeoutNanos;
41     jfieldID frameLeft;
42     jfieldID frameTop;
43     jfieldID frameRight;
44     jfieldID frameBottom;
45     jfieldID scaleFactor;
46     jfieldID touchableRegion;
47     jfieldID visible;
48     jfieldID canReceiveKeys;
49     jfieldID hasFocus;
50     jfieldID hasWallpaper;
51     jfieldID paused;
52     jfieldID layer;
53     jfieldID ownerPid;
54     jfieldID ownerUid;
55     jfieldID inputFeatures;
56     jfieldID displayId;
57 } gInputWindowHandleClassInfo;
58 
59 static Mutex gHandleMutex;
60 
61 
62 // --- NativeInputWindowHandle ---
63 
NativeInputWindowHandle(const sp<InputApplicationHandle> & inputApplicationHandle,jweak objWeak)64 NativeInputWindowHandle::NativeInputWindowHandle(
65         const sp<InputApplicationHandle>& inputApplicationHandle, jweak objWeak) :
66         InputWindowHandle(inputApplicationHandle),
67         mObjWeak(objWeak) {
68 }
69 
~NativeInputWindowHandle()70 NativeInputWindowHandle::~NativeInputWindowHandle() {
71     JNIEnv* env = AndroidRuntime::getJNIEnv();
72     env->DeleteWeakGlobalRef(mObjWeak);
73 }
74 
getInputWindowHandleObjLocalRef(JNIEnv * env)75 jobject NativeInputWindowHandle::getInputWindowHandleObjLocalRef(JNIEnv* env) {
76     return env->NewLocalRef(mObjWeak);
77 }
78 
updateInfo()79 bool NativeInputWindowHandle::updateInfo() {
80     JNIEnv* env = AndroidRuntime::getJNIEnv();
81     jobject obj = env->NewLocalRef(mObjWeak);
82     if (!obj) {
83         releaseInfo();
84         return false;
85     }
86 
87     if (!mInfo) {
88         mInfo = new InputWindowInfo();
89     } else {
90         mInfo->touchableRegion.clear();
91     }
92 
93     jobject inputChannelObj = env->GetObjectField(obj,
94             gInputWindowHandleClassInfo.inputChannel);
95     if (inputChannelObj) {
96         mInfo->inputChannel = android_view_InputChannel_getInputChannel(env, inputChannelObj);
97         env->DeleteLocalRef(inputChannelObj);
98     } else {
99         mInfo->inputChannel.clear();
100     }
101 
102     jstring nameObj = jstring(env->GetObjectField(obj,
103             gInputWindowHandleClassInfo.name));
104     if (nameObj) {
105         const char* nameStr = env->GetStringUTFChars(nameObj, NULL);
106         mInfo->name = nameStr;
107         env->ReleaseStringUTFChars(nameObj, nameStr);
108         env->DeleteLocalRef(nameObj);
109     } else {
110         mInfo->name = "<null>";
111     }
112 
113     mInfo->layoutParamsFlags = env->GetIntField(obj,
114             gInputWindowHandleClassInfo.layoutParamsFlags);
115     mInfo->layoutParamsType = env->GetIntField(obj,
116             gInputWindowHandleClassInfo.layoutParamsType);
117     mInfo->dispatchingTimeout = env->GetLongField(obj,
118             gInputWindowHandleClassInfo.dispatchingTimeoutNanos);
119     mInfo->frameLeft = env->GetIntField(obj,
120             gInputWindowHandleClassInfo.frameLeft);
121     mInfo->frameTop = env->GetIntField(obj,
122             gInputWindowHandleClassInfo.frameTop);
123     mInfo->frameRight = env->GetIntField(obj,
124             gInputWindowHandleClassInfo.frameRight);
125     mInfo->frameBottom = env->GetIntField(obj,
126             gInputWindowHandleClassInfo.frameBottom);
127     mInfo->scaleFactor = env->GetFloatField(obj,
128             gInputWindowHandleClassInfo.scaleFactor);
129 
130     jobject regionObj = env->GetObjectField(obj,
131             gInputWindowHandleClassInfo.touchableRegion);
132     if (regionObj) {
133         SkRegion* region = android_graphics_Region_getSkRegion(env, regionObj);
134         for (SkRegion::Iterator it(*region); !it.done(); it.next()) {
135             const SkIRect& rect = it.rect();
136             mInfo->addTouchableRegion(Rect(rect.fLeft, rect.fTop, rect.fRight, rect.fBottom));
137         }
138         env->DeleteLocalRef(regionObj);
139     }
140 
141     mInfo->visible = env->GetBooleanField(obj,
142             gInputWindowHandleClassInfo.visible);
143     mInfo->canReceiveKeys = env->GetBooleanField(obj,
144             gInputWindowHandleClassInfo.canReceiveKeys);
145     mInfo->hasFocus = env->GetBooleanField(obj,
146             gInputWindowHandleClassInfo.hasFocus);
147     mInfo->hasWallpaper = env->GetBooleanField(obj,
148             gInputWindowHandleClassInfo.hasWallpaper);
149     mInfo->paused = env->GetBooleanField(obj,
150             gInputWindowHandleClassInfo.paused);
151     mInfo->layer = env->GetIntField(obj,
152             gInputWindowHandleClassInfo.layer);
153     mInfo->ownerPid = env->GetIntField(obj,
154             gInputWindowHandleClassInfo.ownerPid);
155     mInfo->ownerUid = env->GetIntField(obj,
156             gInputWindowHandleClassInfo.ownerUid);
157     mInfo->inputFeatures = env->GetIntField(obj,
158             gInputWindowHandleClassInfo.inputFeatures);
159     mInfo->displayId = env->GetIntField(obj,
160             gInputWindowHandleClassInfo.displayId);
161 
162     env->DeleteLocalRef(obj);
163     return true;
164 }
165 
166 
167 // --- Global functions ---
168 
android_server_InputWindowHandle_getHandle(JNIEnv * env,jobject inputWindowHandleObj)169 sp<NativeInputWindowHandle> android_server_InputWindowHandle_getHandle(
170         JNIEnv* env, jobject inputWindowHandleObj) {
171     if (!inputWindowHandleObj) {
172         return NULL;
173     }
174 
175     AutoMutex _l(gHandleMutex);
176 
177     jlong ptr = env->GetLongField(inputWindowHandleObj, gInputWindowHandleClassInfo.ptr);
178     NativeInputWindowHandle* handle;
179     if (ptr) {
180         handle = reinterpret_cast<NativeInputWindowHandle*>(ptr);
181     } else {
182         jobject inputApplicationHandleObj = env->GetObjectField(inputWindowHandleObj,
183                 gInputWindowHandleClassInfo.inputApplicationHandle);
184         sp<InputApplicationHandle> inputApplicationHandle =
185                 android_server_InputApplicationHandle_getHandle(env, inputApplicationHandleObj);
186         env->DeleteLocalRef(inputApplicationHandleObj);
187 
188         jweak objWeak = env->NewWeakGlobalRef(inputWindowHandleObj);
189         handle = new NativeInputWindowHandle(inputApplicationHandle, objWeak);
190         handle->incStrong((void*)android_server_InputWindowHandle_getHandle);
191         env->SetLongField(inputWindowHandleObj, gInputWindowHandleClassInfo.ptr,
192                 reinterpret_cast<jlong>(handle));
193     }
194     return handle;
195 }
196 
197 
198 // --- JNI ---
199 
android_server_InputWindowHandle_nativeDispose(JNIEnv * env,jobject obj)200 static void android_server_InputWindowHandle_nativeDispose(JNIEnv* env, jobject obj) {
201     AutoMutex _l(gHandleMutex);
202 
203     jlong ptr = env->GetLongField(obj, gInputWindowHandleClassInfo.ptr);
204     if (ptr) {
205         env->SetLongField(obj, gInputWindowHandleClassInfo.ptr, 0);
206 
207         NativeInputWindowHandle* handle = reinterpret_cast<NativeInputWindowHandle*>(ptr);
208         handle->decStrong((void*)android_server_InputWindowHandle_getHandle);
209     }
210 }
211 
212 
213 static const JNINativeMethod gInputWindowHandleMethods[] = {
214     /* name, signature, funcPtr */
215     { "nativeDispose", "()V",
216             (void*) android_server_InputWindowHandle_nativeDispose },
217 };
218 
219 #define FIND_CLASS(var, className) \
220         var = env->FindClass(className); \
221         LOG_FATAL_IF(! (var), "Unable to find class " className);
222 
223 #define GET_FIELD_ID(var, clazz, fieldName, fieldDescriptor) \
224         var = env->GetFieldID(clazz, fieldName, fieldDescriptor); \
225         LOG_FATAL_IF(! (var), "Unable to find field " fieldName);
226 
register_android_server_InputWindowHandle(JNIEnv * env)227 int register_android_server_InputWindowHandle(JNIEnv* env) {
228     int res = jniRegisterNativeMethods(env, "com/android/server/input/InputWindowHandle",
229             gInputWindowHandleMethods, NELEM(gInputWindowHandleMethods));
230     (void) res;  // Faked use when LOG_NDEBUG.
231     LOG_FATAL_IF(res < 0, "Unable to register native methods.");
232 
233     jclass clazz;
234     FIND_CLASS(clazz, "com/android/server/input/InputWindowHandle");
235 
236     GET_FIELD_ID(gInputWindowHandleClassInfo.ptr, clazz,
237             "ptr", "J");
238 
239     GET_FIELD_ID(gInputWindowHandleClassInfo.inputApplicationHandle,
240             clazz,
241             "inputApplicationHandle", "Lcom/android/server/input/InputApplicationHandle;");
242 
243     GET_FIELD_ID(gInputWindowHandleClassInfo.inputChannel, clazz,
244             "inputChannel", "Landroid/view/InputChannel;");
245 
246     GET_FIELD_ID(gInputWindowHandleClassInfo.name, clazz,
247             "name", "Ljava/lang/String;");
248 
249     GET_FIELD_ID(gInputWindowHandleClassInfo.layoutParamsFlags, clazz,
250             "layoutParamsFlags", "I");
251 
252     GET_FIELD_ID(gInputWindowHandleClassInfo.layoutParamsType, clazz,
253             "layoutParamsType", "I");
254 
255     GET_FIELD_ID(gInputWindowHandleClassInfo.dispatchingTimeoutNanos, clazz,
256             "dispatchingTimeoutNanos", "J");
257 
258     GET_FIELD_ID(gInputWindowHandleClassInfo.frameLeft, clazz,
259             "frameLeft", "I");
260 
261     GET_FIELD_ID(gInputWindowHandleClassInfo.frameTop, clazz,
262             "frameTop", "I");
263 
264     GET_FIELD_ID(gInputWindowHandleClassInfo.frameRight, clazz,
265             "frameRight", "I");
266 
267     GET_FIELD_ID(gInputWindowHandleClassInfo.frameBottom, clazz,
268             "frameBottom", "I");
269 
270     GET_FIELD_ID(gInputWindowHandleClassInfo.scaleFactor, clazz,
271             "scaleFactor", "F");
272 
273     GET_FIELD_ID(gInputWindowHandleClassInfo.touchableRegion, clazz,
274             "touchableRegion", "Landroid/graphics/Region;");
275 
276     GET_FIELD_ID(gInputWindowHandleClassInfo.visible, clazz,
277             "visible", "Z");
278 
279     GET_FIELD_ID(gInputWindowHandleClassInfo.canReceiveKeys, clazz,
280             "canReceiveKeys", "Z");
281 
282     GET_FIELD_ID(gInputWindowHandleClassInfo.hasFocus, clazz,
283             "hasFocus", "Z");
284 
285     GET_FIELD_ID(gInputWindowHandleClassInfo.hasWallpaper, clazz,
286             "hasWallpaper", "Z");
287 
288     GET_FIELD_ID(gInputWindowHandleClassInfo.paused, clazz,
289             "paused", "Z");
290 
291     GET_FIELD_ID(gInputWindowHandleClassInfo.layer, clazz,
292             "layer", "I");
293 
294     GET_FIELD_ID(gInputWindowHandleClassInfo.ownerPid, clazz,
295             "ownerPid", "I");
296 
297     GET_FIELD_ID(gInputWindowHandleClassInfo.ownerUid, clazz,
298             "ownerUid", "I");
299 
300     GET_FIELD_ID(gInputWindowHandleClassInfo.inputFeatures, clazz,
301             "inputFeatures", "I");
302 
303     GET_FIELD_ID(gInputWindowHandleClassInfo.displayId, clazz,
304             "displayId", "I");
305     return 0;
306 }
307 
308 } /* namespace android */
309