blob: 54567e5010c563eeff08b4823129ddc6abd3e069 [file] [log] [blame]
Jeff Brown46b9ac02010-04-22 18:58:52 -07001/*
2 * Copyright (C) 2010 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 "KeyEvent-JNI"
18
Steven Moreland2279b252017-07-19 09:50:45 -070019#include <nativehelper/JNIHelp.h>
Jeff Brown46b9ac02010-04-22 18:58:52 -070020
21#include <android_runtime/AndroidRuntime.h>
Ruben Brunk87eac992013-09-09 17:44:59 -070022#include <android_runtime/Log.h>
Jeff Brown9d3b1a42013-07-01 19:07:15 -070023#include <input/Input.h>
Siarhei Vishniakou6a3346d2020-01-23 19:49:39 -060024#include <nativehelper/ScopedPrimitiveArray.h>
Steven Moreland2279b252017-07-19 09:50:45 -070025#include <nativehelper/ScopedUtfChars.h>
Siarhei Vishniakou6a3346d2020-01-23 19:49:39 -060026#include <utils/Log.h>
27#include <optional>
Jeff Brown46b9ac02010-04-22 18:58:52 -070028#include "android_view_KeyEvent.h"
29
Andreas Gampe987f79f2014-11-18 17:29:46 -080030#include "core_jni_helpers.h"
31
Jeff Brown46b9ac02010-04-22 18:58:52 -070032namespace android {
33
Siarhei Vishniakou6a3346d2020-01-23 19:49:39 -060034/**
35 * Convert an std::array of bytes into a Java object.
36 */
37template <size_t N>
38static ScopedLocalRef<jbyteArray> toJbyteArray(JNIEnv* env, const std::array<uint8_t, N>& data) {
39 ScopedLocalRef<jbyteArray> array(env, env->NewByteArray(N));
40 if (array.get() == nullptr) {
41 jniThrowException(env, "java/lang/OutOfMemoryError", nullptr);
42 return array;
43 }
44 static_assert(sizeof(char) == sizeof(uint8_t));
45 env->SetByteArrayRegion(array.get(), 0, N, reinterpret_cast<const signed char*>(data.data()));
46 return array;
47}
48
49/**
50 * Convert a Java object into an std::array of bytes of size N.
51 * If the object is null, or the length is unexpected, return std::nullopt.
52 */
53template <size_t N>
54static std::optional<std::array<uint8_t, N>> fromJobject(JNIEnv* env, jobject object) {
55 if (object == nullptr) {
56 return std::nullopt;
57 }
58 jbyteArray javaArray = reinterpret_cast<jbyteArray>(object);
59 ScopedByteArrayRO bytes(env, javaArray);
60 if (bytes.size() != N) {
61 ALOGE("Could not initialize array from java object, expected length %zu but got %zu", N,
62 bytes.size());
63 return std::nullopt;
64 }
65 std::array<uint8_t, N> array;
66 std::move(bytes.get(), bytes.get() + N, array.begin());
67 return array;
68}
69
Jeff Brown46b9ac02010-04-22 18:58:52 -070070// ----------------------------------------------------------------------------
71
72static struct {
73 jclass clazz;
74
Jeff Brown1f245102010-11-18 20:53:46 -080075 jmethodID obtain;
76 jmethodID recycle;
Jeff Brown46b9ac02010-04-22 18:58:52 -070077
Garfield Tanc8362b22020-01-24 11:32:14 -080078 jfieldID mId;
Jeff Brownc5ed5912010-07-14 18:48:53 -070079 jfieldID mDeviceId;
80 jfieldID mSource;
Siarhei Vishniakou91fa08f2018-06-08 22:49:30 +010081 jfieldID mDisplayId;
Siarhei Vishniakou6a3346d2020-01-23 19:49:39 -060082 jfieldID mHmac;
Jeff Brown46b9ac02010-04-22 18:58:52 -070083 jfieldID mMetaState;
84 jfieldID mAction;
85 jfieldID mKeyCode;
86 jfieldID mScanCode;
87 jfieldID mRepeatCount;
Jeff Brown46b9ac02010-04-22 18:58:52 -070088 jfieldID mFlags;
89 jfieldID mDownTime;
90 jfieldID mEventTime;
91 jfieldID mCharacters;
92} gKeyEventClassInfo;
93
94// ----------------------------------------------------------------------------
95
96jobject android_view_KeyEvent_fromNative(JNIEnv* env, const KeyEvent* event) {
Siarhei Vishniakou6a3346d2020-01-23 19:49:39 -060097 ScopedLocalRef<jbyteArray> hmac = toJbyteArray(env, event->getHmac());
98 jobject eventObj =
99 env->CallStaticObjectMethod(gKeyEventClassInfo.clazz, gKeyEventClassInfo.obtain,
Garfield Tanc8362b22020-01-24 11:32:14 -0800100 event->getId(),
Siarhei Vishniakou6a3346d2020-01-23 19:49:39 -0600101 nanoseconds_to_milliseconds(event->getDownTime()),
102 nanoseconds_to_milliseconds(event->getEventTime()),
103 event->getAction(), event->getKeyCode(),
104 event->getRepeatCount(), event->getMetaState(),
105 event->getDeviceId(), event->getScanCode(),
106 event->getFlags(), event->getSource(),
Siarhei Vishniakoucf075282020-01-15 17:35:58 -0800107 event->getDisplayId(), hmac.get(), nullptr);
Jeff Brown1f245102010-11-18 20:53:46 -0800108 if (env->ExceptionCheck()) {
Steve Block3762c312012-01-06 19:20:56 +0000109 ALOGE("An exception occurred while obtaining a key event.");
Jeff Brown1f245102010-11-18 20:53:46 -0800110 LOGE_EX(env);
111 env->ExceptionClear();
112 return NULL;
113 }
114 return eventObj;
Jeff Brown46b9ac02010-04-22 18:58:52 -0700115}
116
Jeff Brown1f245102010-11-18 20:53:46 -0800117status_t android_view_KeyEvent_toNative(JNIEnv* env, jobject eventObj,
Jeff Brown46b9ac02010-04-22 18:58:52 -0700118 KeyEvent* event) {
Garfield Tanc8362b22020-01-24 11:32:14 -0800119 jint id = env->GetIntField(eventObj, gKeyEventClassInfo.mId);
Jeff Brownc5ed5912010-07-14 18:48:53 -0700120 jint deviceId = env->GetIntField(eventObj, gKeyEventClassInfo.mDeviceId);
121 jint source = env->GetIntField(eventObj, gKeyEventClassInfo.mSource);
Siarhei Vishniakou91fa08f2018-06-08 22:49:30 +0100122 jint displayId = env->GetIntField(eventObj, gKeyEventClassInfo.mDisplayId);
Siarhei Vishniakou6a3346d2020-01-23 19:49:39 -0600123 jobject hmacObj = env->GetObjectField(eventObj, gKeyEventClassInfo.mHmac);
124 std::optional<std::array<uint8_t, 32>> hmac = fromJobject<32>(env, hmacObj);
125 if (!hmac) {
126 hmac = INVALID_HMAC;
127 }
Jeff Brown46b9ac02010-04-22 18:58:52 -0700128 jint metaState = env->GetIntField(eventObj, gKeyEventClassInfo.mMetaState);
129 jint action = env->GetIntField(eventObj, gKeyEventClassInfo.mAction);
130 jint keyCode = env->GetIntField(eventObj, gKeyEventClassInfo.mKeyCode);
131 jint scanCode = env->GetIntField(eventObj, gKeyEventClassInfo.mScanCode);
132 jint repeatCount = env->GetIntField(eventObj, gKeyEventClassInfo.mRepeatCount);
Jeff Brown46b9ac02010-04-22 18:58:52 -0700133 jint flags = env->GetIntField(eventObj, gKeyEventClassInfo.mFlags);
134 jlong downTime = env->GetLongField(eventObj, gKeyEventClassInfo.mDownTime);
135 jlong eventTime = env->GetLongField(eventObj, gKeyEventClassInfo.mEventTime);
136
Garfield Tanc8362b22020-01-24 11:32:14 -0800137 event->initialize(id, deviceId, source, displayId, *hmac, action, flags, keyCode, scanCode,
Siarhei Vishniakou6a3346d2020-01-23 19:49:39 -0600138 metaState, repeatCount, milliseconds_to_nanoseconds(downTime),
139 milliseconds_to_nanoseconds(eventTime));
Jeff Brown1f245102010-11-18 20:53:46 -0800140 return OK;
141}
142
143status_t android_view_KeyEvent_recycle(JNIEnv* env, jobject eventObj) {
144 env->CallVoidMethod(eventObj, gKeyEventClassInfo.recycle);
145 if (env->ExceptionCheck()) {
Steve Block8564c8d2012-01-05 23:22:43 +0000146 ALOGW("An exception occurred while recycling a key event.");
Jeff Brown1f245102010-11-18 20:53:46 -0800147 LOGW_EX(env);
148 env->ExceptionClear();
149 return UNKNOWN_ERROR;
150 }
151 return OK;
Jeff Brown46b9ac02010-04-22 18:58:52 -0700152}
153
Michael Wright337d9d22014-04-22 15:03:48 -0700154static jstring android_view_KeyEvent_nativeKeyCodeToString(JNIEnv* env, jobject clazz,
155 jint keyCode) {
156 return env->NewStringUTF(KeyEvent::getLabel(keyCode));
Dianne Hackborn3c80a4a2010-06-29 19:20:40 -0700157}
158
Michael Wright337d9d22014-04-22 15:03:48 -0700159static jint android_view_KeyEvent_nativeKeyCodeFromString(JNIEnv* env, jobject clazz,
160 jstring label) {
161 ScopedUtfChars keyLabel(env, label);
162 return KeyEvent::getKeyCodeFromLabel(keyLabel.c_str());
Dianne Hackborn3c80a4a2010-06-29 19:20:40 -0700163}
164
Garfield Tanc8362b22020-01-24 11:32:14 -0800165static jint android_view_KeyEvent_nativeNextId() {
166 return static_cast<jint>(InputEvent::nextId());
167}
Jeff Brown1f245102010-11-18 20:53:46 -0800168
Jeff Brown46b9ac02010-04-22 18:58:52 -0700169// ----------------------------------------------------------------------------
170
Dianne Hackborn3c80a4a2010-06-29 19:20:40 -0700171static const JNINativeMethod g_methods[] = {
Garfield Tanc8362b22020-01-24 11:32:14 -0800172 {"nativeKeyCodeToString", "(I)Ljava/lang/String;",
173 (void*)android_view_KeyEvent_nativeKeyCodeToString},
174 {"nativeKeyCodeFromString", "(Ljava/lang/String;)I",
175 (void*)android_view_KeyEvent_nativeKeyCodeFromString},
176 {"nativeNextId", "()I", (void*)android_view_KeyEvent_nativeNextId},
Dianne Hackborn3c80a4a2010-06-29 19:20:40 -0700177};
178
Jeff Brown46b9ac02010-04-22 18:58:52 -0700179int register_android_view_KeyEvent(JNIEnv* env) {
Andreas Gampe987f79f2014-11-18 17:29:46 -0800180 jclass clazz = FindClassOrDie(env, "android/view/KeyEvent");
181 gKeyEventClassInfo.clazz = MakeGlobalRefOrDie(env, clazz);
Jeff Brown1f245102010-11-18 20:53:46 -0800182
Siarhei Vishniakou6a3346d2020-01-23 19:49:39 -0600183 gKeyEventClassInfo.obtain =
184 GetStaticMethodIDOrDie(env, gKeyEventClassInfo.clazz, "obtain",
Garfield Tanc8362b22020-01-24 11:32:14 -0800185 "(IJJIIIIIIIII[BLjava/lang/String;)Landroid/view/KeyEvent;");
Andreas Gampe987f79f2014-11-18 17:29:46 -0800186 gKeyEventClassInfo.recycle = GetMethodIDOrDie(env, gKeyEventClassInfo.clazz,
Jeff Brown1f245102010-11-18 20:53:46 -0800187 "recycle", "()V");
Jeff Brown46b9ac02010-04-22 18:58:52 -0700188
Garfield Tanc8362b22020-01-24 11:32:14 -0800189 gKeyEventClassInfo.mId = GetFieldIDOrDie(env, gKeyEventClassInfo.clazz, "mId", "I");
Andreas Gampe987f79f2014-11-18 17:29:46 -0800190 gKeyEventClassInfo.mDeviceId = GetFieldIDOrDie(env, gKeyEventClassInfo.clazz, "mDeviceId", "I");
191 gKeyEventClassInfo.mSource = GetFieldIDOrDie(env, gKeyEventClassInfo.clazz, "mSource", "I");
Siarhei Vishniakou91fa08f2018-06-08 22:49:30 +0100192 gKeyEventClassInfo.mDisplayId = GetFieldIDOrDie(env, gKeyEventClassInfo.clazz, "mDisplayId",
193 "I");
Siarhei Vishniakou6a3346d2020-01-23 19:49:39 -0600194 gKeyEventClassInfo.mHmac = GetFieldIDOrDie(env, gKeyEventClassInfo.clazz, "mHmac", "[B");
Andreas Gampe987f79f2014-11-18 17:29:46 -0800195 gKeyEventClassInfo.mMetaState = GetFieldIDOrDie(env, gKeyEventClassInfo.clazz, "mMetaState",
196 "I");
197 gKeyEventClassInfo.mAction = GetFieldIDOrDie(env, gKeyEventClassInfo.clazz, "mAction", "I");
198 gKeyEventClassInfo.mKeyCode = GetFieldIDOrDie(env, gKeyEventClassInfo.clazz, "mKeyCode", "I");
199 gKeyEventClassInfo.mScanCode = GetFieldIDOrDie(env, gKeyEventClassInfo.clazz, "mScanCode", "I");
200 gKeyEventClassInfo.mRepeatCount = GetFieldIDOrDie(env, gKeyEventClassInfo.clazz, "mRepeatCount",
201 "I");
202 gKeyEventClassInfo.mFlags = GetFieldIDOrDie(env, gKeyEventClassInfo.clazz, "mFlags", "I");
203 gKeyEventClassInfo.mDownTime = GetFieldIDOrDie(env, gKeyEventClassInfo.clazz, "mDownTime", "J");
204 gKeyEventClassInfo.mEventTime = GetFieldIDOrDie(env, gKeyEventClassInfo.clazz, "mEventTime",
205 "J");
206 gKeyEventClassInfo.mCharacters = GetFieldIDOrDie(env, gKeyEventClassInfo.clazz, "mCharacters",
207 "Ljava/lang/String;");
Jeff Brown46b9ac02010-04-22 18:58:52 -0700208
Andreas Gampe987f79f2014-11-18 17:29:46 -0800209 return RegisterMethodsOrDie(env, "android/view/KeyEvent", g_methods, NELEM(g_methods));
Jeff Brown46b9ac02010-04-22 18:58:52 -0700210}
211
212} // namespace android