1 /*
2 * Copyright (C) 2013 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 "InputQueue"
18
19 #include <fcntl.h>
20 #include <string.h>
21 #include <unistd.h>
22
23 #include <android/input.h>
24 #include <android_runtime/AndroidRuntime.h>
25 #include <android_runtime/android_view_InputQueue.h>
26 #include <input/Input.h>
27 #include <utils/Looper.h>
28 #include <utils/TypeHelpers.h>
29 #include <ScopedLocalRef.h>
30
31 #include "JNIHelp.h"
32 #include "android_os_MessageQueue.h"
33 #include "android_view_KeyEvent.h"
34 #include "android_view_MotionEvent.h"
35
36 namespace android {
37
38 static struct {
39 jmethodID finishInputEvent;
40 } gInputQueueClassInfo;
41
42 enum {
43 MSG_FINISH_INPUT = 1,
44 };
45
InputQueue(jobject inputQueueObj,const sp<Looper> & looper,int dispatchReadFd,int dispatchWriteFd)46 InputQueue::InputQueue(jobject inputQueueObj, const sp<Looper>& looper,
47 int dispatchReadFd, int dispatchWriteFd) :
48 mDispatchReadFd(dispatchReadFd), mDispatchWriteFd(dispatchWriteFd),
49 mDispatchLooper(looper), mHandler(new WeakMessageHandler(this)) {
50 JNIEnv* env = AndroidRuntime::getJNIEnv();
51 mInputQueueWeakGlobal = env->NewGlobalRef(inputQueueObj);
52 }
53
~InputQueue()54 InputQueue::~InputQueue() {
55 mDispatchLooper->removeMessages(mHandler);
56 JNIEnv* env = AndroidRuntime::getJNIEnv();
57 env->DeleteGlobalRef(mInputQueueWeakGlobal);
58 close(mDispatchReadFd);
59 close(mDispatchWriteFd);
60 }
61
attachLooper(Looper * looper,int ident,ALooper_callbackFunc callback,void * data)62 void InputQueue::attachLooper(Looper* looper, int ident,
63 ALooper_callbackFunc callback, void* data) {
64 Mutex::Autolock _l(mLock);
65 for (size_t i = 0; i < mAppLoopers.size(); i++) {
66 if (looper == mAppLoopers[i]) {
67 return;
68 }
69 }
70 mAppLoopers.push(looper);
71 looper->addFd(mDispatchReadFd, ident, ALOOPER_EVENT_INPUT, callback, data);
72 }
73
detachLooper()74 void InputQueue::detachLooper() {
75 Mutex::Autolock _l(mLock);
76 detachLooperLocked();
77 }
78
detachLooperLocked()79 void InputQueue::detachLooperLocked() {
80 for (size_t i = 0; i < mAppLoopers.size(); i++) {
81 mAppLoopers[i]->removeFd(mDispatchReadFd);
82 }
83 mAppLoopers.clear();
84 }
85
hasEvents()86 bool InputQueue::hasEvents() {
87 Mutex::Autolock _l(mLock);
88 return mPendingEvents.size() > 0;
89 }
90
getEvent(InputEvent ** outEvent)91 status_t InputQueue::getEvent(InputEvent** outEvent) {
92 Mutex::Autolock _l(mLock);
93 *outEvent = NULL;
94 if (!mPendingEvents.isEmpty()) {
95 *outEvent = mPendingEvents[0];
96 mPendingEvents.removeAt(0);
97 }
98
99 if (mPendingEvents.isEmpty()) {
100 char byteread[16];
101 ssize_t nRead;
102 do {
103 nRead = TEMP_FAILURE_RETRY(read(mDispatchReadFd, &byteread, sizeof(byteread)));
104 if (nRead < 0 && errno != EAGAIN) {
105 ALOGW("Failed to read from native dispatch pipe: %s", strerror(errno));
106 }
107 } while (nRead > 0);
108 }
109
110 return *outEvent != NULL ? OK : WOULD_BLOCK;
111 }
112
preDispatchEvent(InputEvent * e)113 bool InputQueue::preDispatchEvent(InputEvent* e) {
114 if (e->getType() == AINPUT_EVENT_TYPE_KEY) {
115 KeyEvent* keyEvent = static_cast<KeyEvent*>(e);
116 if (keyEvent->getFlags() & AKEY_EVENT_FLAG_PREDISPATCH) {
117 finishEvent(e, false);
118 return true;
119 }
120 }
121 return false;
122 }
123
finishEvent(InputEvent * event,bool handled)124 void InputQueue::finishEvent(InputEvent* event, bool handled) {
125 Mutex::Autolock _l(mLock);
126 mFinishedEvents.push(key_value_pair_t<InputEvent*, bool>(event, handled));
127 if (mFinishedEvents.size() == 1) {
128 mDispatchLooper->sendMessage(this, Message(MSG_FINISH_INPUT));
129 }
130 }
131
handleMessage(const Message & message)132 void InputQueue::handleMessage(const Message& message) {
133 switch(message.what) {
134 case MSG_FINISH_INPUT:
135 JNIEnv* env = AndroidRuntime::getJNIEnv();
136 ScopedLocalRef<jobject> inputQueueObj(env, jniGetReferent(env, mInputQueueWeakGlobal));
137 if (!inputQueueObj.get()) {
138 ALOGW("InputQueue was finalized without being disposed");
139 return;
140 }
141 while (true) {
142 InputEvent* event;
143 bool handled;
144 {
145 Mutex::Autolock _l(mLock);
146 if (mFinishedEvents.isEmpty()) {
147 break;
148 }
149 event = mFinishedEvents[0].getKey();
150 handled = mFinishedEvents[0].getValue();
151 mFinishedEvents.removeAt(0);
152 }
153 env->CallVoidMethod(inputQueueObj.get(), gInputQueueClassInfo.finishInputEvent,
154 reinterpret_cast<jlong>(event), handled);
155 recycleInputEvent(event);
156 }
157 break;
158 }
159 }
160
recycleInputEvent(InputEvent * event)161 void InputQueue::recycleInputEvent(InputEvent* event) {
162 mPooledInputEventFactory.recycle(event);
163 }
164
createKeyEvent()165 KeyEvent* InputQueue::createKeyEvent() {
166 return mPooledInputEventFactory.createKeyEvent();
167 }
168
createMotionEvent()169 MotionEvent* InputQueue::createMotionEvent() {
170 return mPooledInputEventFactory.createMotionEvent();
171 }
172
enqueueEvent(InputEvent * event)173 void InputQueue::enqueueEvent(InputEvent* event) {
174 Mutex::Autolock _l(mLock);
175 mPendingEvents.push(event);
176 if (mPendingEvents.size() == 1) {
177 char dummy = 0;
178 int res = TEMP_FAILURE_RETRY(write(mDispatchWriteFd, &dummy, sizeof(dummy)));
179 if (res < 0 && errno != EAGAIN) {
180 ALOGW("Failed writing to dispatch fd: %s", strerror(errno));
181 }
182 }
183 }
184
createQueue(jobject inputQueueObj,const sp<Looper> & looper)185 InputQueue* InputQueue::createQueue(jobject inputQueueObj, const sp<Looper>& looper) {
186 int pipeFds[2];
187 if (pipe(pipeFds)) {
188 ALOGW("Could not create native input dispatching pipe: %s", strerror(errno));
189 return NULL;
190 }
191 fcntl(pipeFds[0], F_SETFL, O_NONBLOCK);
192 fcntl(pipeFds[1], F_SETFL, O_NONBLOCK);
193 return new InputQueue(inputQueueObj, looper, pipeFds[0], pipeFds[1]);
194 }
195
nativeInit(JNIEnv * env,jobject clazz,jobject queueWeak,jobject jMsgQueue)196 static jlong nativeInit(JNIEnv* env, jobject clazz, jobject queueWeak, jobject jMsgQueue) {
197 sp<MessageQueue> messageQueue = android_os_MessageQueue_getMessageQueue(env, jMsgQueue);
198 if (messageQueue == NULL) {
199 jniThrowRuntimeException(env, "MessageQueue is not initialized.");
200 return 0;
201 }
202 sp<InputQueue> queue = InputQueue::createQueue(queueWeak, messageQueue->getLooper());
203 if (!queue.get()) {
204 jniThrowRuntimeException(env, "InputQueue failed to initialize");
205 return 0;
206 }
207 queue->incStrong(&gInputQueueClassInfo);
208 return reinterpret_cast<jlong>(queue.get());
209 }
210
nativeDispose(JNIEnv * env,jobject clazz,jlong ptr)211 static void nativeDispose(JNIEnv* env, jobject clazz, jlong ptr) {
212 sp<InputQueue> queue = reinterpret_cast<InputQueue*>(ptr);
213 queue->detachLooper();
214 queue->decStrong(&gInputQueueClassInfo);
215 }
216
nativeSendKeyEvent(JNIEnv * env,jobject clazz,jlong ptr,jobject eventObj,jboolean predispatch)217 static jlong nativeSendKeyEvent(JNIEnv* env, jobject clazz, jlong ptr, jobject eventObj,
218 jboolean predispatch) {
219 InputQueue* queue = reinterpret_cast<InputQueue*>(ptr);
220 KeyEvent* event = queue->createKeyEvent();
221 status_t status = android_view_KeyEvent_toNative(env, eventObj, event);
222 if (status) {
223 queue->recycleInputEvent(event);
224 jniThrowRuntimeException(env, "Could not read contents of KeyEvent object.");
225 return -1;
226 }
227
228 if (predispatch) {
229 event->setFlags(event->getFlags() | AKEY_EVENT_FLAG_PREDISPATCH);
230 }
231
232 queue->enqueueEvent(event);
233 return reinterpret_cast<jlong>(event);
234 }
235
nativeSendMotionEvent(JNIEnv * env,jobject clazz,jlong ptr,jobject eventObj)236 static jlong nativeSendMotionEvent(JNIEnv* env, jobject clazz, jlong ptr, jobject eventObj) {
237 sp<InputQueue> queue = reinterpret_cast<InputQueue*>(ptr);
238 MotionEvent* originalEvent = android_view_MotionEvent_getNativePtr(env, eventObj);
239 if (!originalEvent) {
240 jniThrowRuntimeException(env, "Could not obtain MotionEvent pointer.");
241 return -1;
242 }
243 MotionEvent* event = queue->createMotionEvent();
244 event->copyFrom(originalEvent, true /* keepHistory */);
245 queue->enqueueEvent(event);
246 return reinterpret_cast<jlong>(event);
247 }
248
249 static const JNINativeMethod g_methods[] = {
250 { "nativeInit", "(Ljava/lang/ref/WeakReference;Landroid/os/MessageQueue;)J",
251 (void*) nativeInit },
252 { "nativeDispose", "(J)V", (void*) nativeDispose },
253 { "nativeSendKeyEvent", "(JLandroid/view/KeyEvent;Z)J", (void*) nativeSendKeyEvent },
254 { "nativeSendMotionEvent", "(JLandroid/view/MotionEvent;)J", (void*) nativeSendMotionEvent },
255 };
256
257 static const char* const kInputQueuePathName = "android/view/InputQueue";
258
259 #define FIND_CLASS(var, className) \
260 do { \
261 var = env->FindClass(className); \
262 LOG_FATAL_IF(! var, "Unable to find class %s", className); \
263 } while(0)
264
265 #define GET_METHOD_ID(var, clazz, methodName, fieldDescriptor) \
266 do { \
267 var = env->GetMethodID(clazz, methodName, fieldDescriptor); \
268 LOG_FATAL_IF(! var, "Unable to find method" methodName); \
269 } while(0)
270
register_android_view_InputQueue(JNIEnv * env)271 int register_android_view_InputQueue(JNIEnv* env)
272 {
273 jclass clazz;
274 FIND_CLASS(clazz, kInputQueuePathName);
275 GET_METHOD_ID(gInputQueueClassInfo.finishInputEvent, clazz, "finishInputEvent", "(JZ)V");
276
277 return AndroidRuntime::registerNativeMethods(
278 env, kInputQueuePathName,
279 g_methods, NELEM(g_methods));
280 }
281
282 } // namespace android
283