blob: 649a14087ea6e73226474525f90c881095b909e3 [file] [log] [blame]
Jeff Brown5912f952013-07-01 19:10:31 -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 "Input"
18//#define LOG_NDEBUG 0
19
chaviw09c8d2d2020-08-24 15:48:26 -070020#include <attestation/HmacKeyManager.h>
Garfield Tan84b087e2020-01-23 10:49:05 -080021#include <cutils/compiler.h>
Siarhei Vishniakouc68fdec2020-10-22 14:58:14 -050022#include <inttypes.h>
Jeff Brown5912f952013-07-01 19:10:31 -070023#include <limits.h>
Garfield Tan84b087e2020-01-23 10:49:05 -080024#include <string.h>
Jeff Brown5912f952013-07-01 19:10:31 -070025
Siarhei Vishniakouc68fdec2020-10-22 14:58:14 -050026#include <android-base/stringprintf.h>
Jeff Brown5912f952013-07-01 19:10:31 -070027#include <input/Input.h>
Siarhei Vishniakou7feb2ea2019-11-25 15:11:23 -080028#include <input/InputDevice.h>
Michael Wright872db4f2014-04-22 15:03:51 -070029#include <input/InputEventLabels.h>
Jeff Brown5912f952013-07-01 19:10:31 -070030
Brett Chabotfaa986c2020-11-04 17:39:36 -080031#ifdef __linux__
Jeff Brown5912f952013-07-01 19:10:31 -070032#include <binder/Parcel.h>
Brett Chabotfaa986c2020-11-04 17:39:36 -080033#endif
Brett Chabot58208522020-09-09 13:55:24 -070034#ifdef __ANDROID__
Garfield Tan84b087e2020-01-23 10:49:05 -080035#include <sys/random.h>
Jeff Brown5912f952013-07-01 19:10:31 -070036#endif
37
Siarhei Vishniakouc68fdec2020-10-22 14:58:14 -050038using android::base::StringPrintf;
39
Jeff Brown5912f952013-07-01 19:10:31 -070040namespace android {
41
Siarhei Vishniakou16a2e302019-01-14 19:21:45 -080042const char* motionClassificationToString(MotionClassification classification) {
43 switch (classification) {
44 case MotionClassification::NONE:
45 return "NONE";
46 case MotionClassification::AMBIGUOUS_GESTURE:
47 return "AMBIGUOUS_GESTURE";
48 case MotionClassification::DEEP_PRESS:
49 return "DEEP_PRESS";
50 }
51}
52
Garfield Tan84b087e2020-01-23 10:49:05 -080053// --- IdGenerator ---
54IdGenerator::IdGenerator(Source source) : mSource(source) {}
55
56int32_t IdGenerator::nextId() const {
57 constexpr uint32_t SEQUENCE_NUMBER_MASK = ~SOURCE_MASK;
58 int32_t id = 0;
59
60// Avoid building against syscall getrandom(2) on host, which will fail build on Mac. Host doesn't
61// use sequence number so just always return mSource.
62#ifdef __ANDROID__
63 constexpr size_t BUF_LEN = sizeof(id);
64 size_t totalBytes = 0;
65 while (totalBytes < BUF_LEN) {
66 ssize_t bytes = TEMP_FAILURE_RETRY(getrandom(&id, BUF_LEN, GRND_NONBLOCK));
67 if (CC_UNLIKELY(bytes < 0)) {
68 ALOGW("Failed to fill in random number for sequence number: %s.", strerror(errno));
69 id = 0;
70 break;
71 }
72 totalBytes += bytes;
73 }
74#endif // __ANDROID__
75
76 return (id & SEQUENCE_NUMBER_MASK) | static_cast<int32_t>(mSource);
77}
78
Jeff Brown5912f952013-07-01 19:10:31 -070079// --- InputEvent ---
80
Siarhei Vishniakou7feb2ea2019-11-25 15:11:23 -080081const char* inputEventTypeToString(int32_t type) {
82 switch (type) {
83 case AINPUT_EVENT_TYPE_KEY: {
84 return "KEY";
85 }
86 case AINPUT_EVENT_TYPE_MOTION: {
87 return "MOTION";
88 }
89 case AINPUT_EVENT_TYPE_FOCUS: {
90 return "FOCUS";
91 }
Prabir Pradhan3f37b7b2020-11-10 16:50:18 -080092 case AINPUT_EVENT_TYPE_CAPTURE: {
93 return "CAPTURE";
94 }
arthurhung7632c332020-12-30 16:58:01 +080095 case AINPUT_EVENT_TYPE_DRAG: {
96 return "DRAG";
97 }
Siarhei Vishniakou7feb2ea2019-11-25 15:11:23 -080098 }
99 return "UNKNOWN";
100}
101
Siarhei Vishniakou54d3e182020-01-15 17:38:38 -0800102VerifiedKeyEvent verifiedKeyEventFromKeyEvent(const KeyEvent& event) {
103 return {{VerifiedInputEvent::Type::KEY, event.getDeviceId(), event.getEventTime(),
104 event.getSource(), event.getDisplayId()},
105 event.getAction(),
106 event.getDownTime(),
107 event.getFlags() & VERIFIED_KEY_EVENT_FLAGS,
108 event.getKeyCode(),
109 event.getScanCode(),
110 event.getMetaState(),
111 event.getRepeatCount()};
112}
113
114VerifiedMotionEvent verifiedMotionEventFromMotionEvent(const MotionEvent& event) {
115 return {{VerifiedInputEvent::Type::MOTION, event.getDeviceId(), event.getEventTime(),
116 event.getSource(), event.getDisplayId()},
117 event.getRawX(0),
118 event.getRawY(0),
119 event.getActionMasked(),
120 event.getDownTime(),
121 event.getFlags() & VERIFIED_MOTION_EVENT_FLAGS,
122 event.getMetaState(),
123 event.getButtonState()};
124}
125
Garfield Tan4cc839f2020-01-24 11:26:14 -0800126void InputEvent::initialize(int32_t id, int32_t deviceId, uint32_t source, int32_t displayId,
Siarhei Vishniakou9c858ac2020-01-23 14:20:11 -0600127 std::array<uint8_t, 32> hmac) {
Garfield Tan4cc839f2020-01-24 11:26:14 -0800128 mId = id;
Jeff Brown5912f952013-07-01 19:10:31 -0700129 mDeviceId = deviceId;
130 mSource = source;
Siarhei Vishniakoua62a8dd2018-06-08 21:17:33 +0100131 mDisplayId = displayId;
Siarhei Vishniakou9c858ac2020-01-23 14:20:11 -0600132 mHmac = hmac;
Jeff Brown5912f952013-07-01 19:10:31 -0700133}
134
135void InputEvent::initialize(const InputEvent& from) {
Garfield Tan4cc839f2020-01-24 11:26:14 -0800136 mId = from.mId;
Jeff Brown5912f952013-07-01 19:10:31 -0700137 mDeviceId = from.mDeviceId;
138 mSource = from.mSource;
Siarhei Vishniakoua62a8dd2018-06-08 21:17:33 +0100139 mDisplayId = from.mDisplayId;
Siarhei Vishniakou9c858ac2020-01-23 14:20:11 -0600140 mHmac = from.mHmac;
Jeff Brown5912f952013-07-01 19:10:31 -0700141}
142
Garfield Tan4cc839f2020-01-24 11:26:14 -0800143int32_t InputEvent::nextId() {
144 static IdGenerator idGen(IdGenerator::Source::OTHER);
145 return idGen.nextId();
146}
147
Jeff Brown5912f952013-07-01 19:10:31 -0700148// --- KeyEvent ---
149
Michael Wright872db4f2014-04-22 15:03:51 -0700150const char* KeyEvent::getLabel(int32_t keyCode) {
Chris Ye4958d062020-08-20 13:21:10 -0700151 return InputEventLookup::getLabelByKeyCode(keyCode);
Jeff Brown5912f952013-07-01 19:10:31 -0700152}
153
Michael Wright872db4f2014-04-22 15:03:51 -0700154int32_t KeyEvent::getKeyCodeFromLabel(const char* label) {
Chris Ye4958d062020-08-20 13:21:10 -0700155 return InputEventLookup::getKeyCodeByLabel(label);
Jeff Brown5912f952013-07-01 19:10:31 -0700156}
157
Garfield Tan4cc839f2020-01-24 11:26:14 -0800158void KeyEvent::initialize(int32_t id, int32_t deviceId, uint32_t source, int32_t displayId,
Siarhei Vishniakou9c858ac2020-01-23 14:20:11 -0600159 std::array<uint8_t, 32> hmac, int32_t action, int32_t flags,
160 int32_t keyCode, int32_t scanCode, int32_t metaState, int32_t repeatCount,
161 nsecs_t downTime, nsecs_t eventTime) {
Garfield Tan4cc839f2020-01-24 11:26:14 -0800162 InputEvent::initialize(id, deviceId, source, displayId, hmac);
Jeff Brown5912f952013-07-01 19:10:31 -0700163 mAction = action;
164 mFlags = flags;
165 mKeyCode = keyCode;
166 mScanCode = scanCode;
167 mMetaState = metaState;
168 mRepeatCount = repeatCount;
169 mDownTime = downTime;
170 mEventTime = eventTime;
171}
172
173void KeyEvent::initialize(const KeyEvent& from) {
174 InputEvent::initialize(from);
175 mAction = from.mAction;
176 mFlags = from.mFlags;
177 mKeyCode = from.mKeyCode;
178 mScanCode = from.mScanCode;
179 mMetaState = from.mMetaState;
180 mRepeatCount = from.mRepeatCount;
181 mDownTime = from.mDownTime;
182 mEventTime = from.mEventTime;
183}
184
Siarhei Vishniakoud44dddf2020-03-25 16:16:40 -0700185const char* KeyEvent::actionToString(int32_t action) {
186 // Convert KeyEvent action to string
187 switch (action) {
188 case AKEY_EVENT_ACTION_DOWN:
189 return "DOWN";
190 case AKEY_EVENT_ACTION_UP:
191 return "UP";
192 case AKEY_EVENT_ACTION_MULTIPLE:
193 return "MULTIPLE";
194 }
195 return "UNKNOWN";
196}
Jeff Brown5912f952013-07-01 19:10:31 -0700197
198// --- PointerCoords ---
199
200float PointerCoords::getAxisValue(int32_t axis) const {
Michael Wright38dcdff2014-03-19 12:06:10 -0700201 if (axis < 0 || axis > 63 || !BitSet64::hasBit(bits, axis)){
Jeff Brown5912f952013-07-01 19:10:31 -0700202 return 0;
203 }
Michael Wright38dcdff2014-03-19 12:06:10 -0700204 return values[BitSet64::getIndexOfBit(bits, axis)];
Jeff Brown5912f952013-07-01 19:10:31 -0700205}
206
207status_t PointerCoords::setAxisValue(int32_t axis, float value) {
208 if (axis < 0 || axis > 63) {
209 return NAME_NOT_FOUND;
210 }
211
Michael Wright38dcdff2014-03-19 12:06:10 -0700212 uint32_t index = BitSet64::getIndexOfBit(bits, axis);
213 if (!BitSet64::hasBit(bits, axis)) {
Jeff Brown5912f952013-07-01 19:10:31 -0700214 if (value == 0) {
215 return OK; // axes with value 0 do not need to be stored
216 }
Michael Wright38dcdff2014-03-19 12:06:10 -0700217
218 uint32_t count = BitSet64::count(bits);
Jeff Brown5912f952013-07-01 19:10:31 -0700219 if (count >= MAX_AXES) {
220 tooManyAxes(axis);
221 return NO_MEMORY;
222 }
Michael Wright38dcdff2014-03-19 12:06:10 -0700223 BitSet64::markBit(bits, axis);
Jeff Brown5912f952013-07-01 19:10:31 -0700224 for (uint32_t i = count; i > index; i--) {
225 values[i] = values[i - 1];
226 }
227 }
Michael Wright38dcdff2014-03-19 12:06:10 -0700228
Jeff Brown5912f952013-07-01 19:10:31 -0700229 values[index] = value;
230 return OK;
231}
232
233static inline void scaleAxisValue(PointerCoords& c, int axis, float scaleFactor) {
234 float value = c.getAxisValue(axis);
235 if (value != 0) {
236 c.setAxisValue(axis, value * scaleFactor);
237 }
238}
239
Robert Carre07e1032018-11-26 12:55:53 -0800240void PointerCoords::scale(float globalScaleFactor, float windowXScale, float windowYScale) {
Jeff Brown5912f952013-07-01 19:10:31 -0700241 // No need to scale pressure or size since they are normalized.
242 // No need to scale orientation since it is meaningless to do so.
Robert Carre07e1032018-11-26 12:55:53 -0800243
244 // If there is a global scale factor, it is included in the windowX/YScale
245 // so we don't need to apply it twice to the X/Y axes.
246 // However we don't want to apply any windowXYScale not included in the global scale
247 // to the TOUCH_MAJOR/MINOR coordinates.
248 scaleAxisValue(*this, AMOTION_EVENT_AXIS_X, windowXScale);
249 scaleAxisValue(*this, AMOTION_EVENT_AXIS_Y, windowYScale);
250 scaleAxisValue(*this, AMOTION_EVENT_AXIS_TOUCH_MAJOR, globalScaleFactor);
251 scaleAxisValue(*this, AMOTION_EVENT_AXIS_TOUCH_MINOR, globalScaleFactor);
252 scaleAxisValue(*this, AMOTION_EVENT_AXIS_TOOL_MAJOR, globalScaleFactor);
253 scaleAxisValue(*this, AMOTION_EVENT_AXIS_TOOL_MINOR, globalScaleFactor);
254}
255
256void PointerCoords::scale(float globalScaleFactor) {
257 scale(globalScaleFactor, globalScaleFactor, globalScaleFactor);
Jeff Brown5912f952013-07-01 19:10:31 -0700258}
259
Jeff Brownf086ddb2014-02-11 14:28:48 -0800260void PointerCoords::applyOffset(float xOffset, float yOffset) {
261 setAxisValue(AMOTION_EVENT_AXIS_X, getX() + xOffset);
262 setAxisValue(AMOTION_EVENT_AXIS_Y, getY() + yOffset);
263}
264
Brett Chabotfaa986c2020-11-04 17:39:36 -0800265#ifdef __linux__
Jeff Brown5912f952013-07-01 19:10:31 -0700266status_t PointerCoords::readFromParcel(Parcel* parcel) {
267 bits = parcel->readInt64();
268
Michael Wright38dcdff2014-03-19 12:06:10 -0700269 uint32_t count = BitSet64::count(bits);
Jeff Brown5912f952013-07-01 19:10:31 -0700270 if (count > MAX_AXES) {
271 return BAD_VALUE;
272 }
273
274 for (uint32_t i = 0; i < count; i++) {
275 values[i] = parcel->readFloat();
276 }
277 return OK;
278}
279
280status_t PointerCoords::writeToParcel(Parcel* parcel) const {
281 parcel->writeInt64(bits);
282
Michael Wright38dcdff2014-03-19 12:06:10 -0700283 uint32_t count = BitSet64::count(bits);
Jeff Brown5912f952013-07-01 19:10:31 -0700284 for (uint32_t i = 0; i < count; i++) {
285 parcel->writeFloat(values[i]);
286 }
287 return OK;
288}
Brett Chabotfaa986c2020-11-04 17:39:36 -0800289#endif
Jeff Brown5912f952013-07-01 19:10:31 -0700290
291void PointerCoords::tooManyAxes(int axis) {
292 ALOGW("Could not set value for axis %d because the PointerCoords structure is full and "
293 "cannot contain more than %d axis values.", axis, int(MAX_AXES));
294}
295
296bool PointerCoords::operator==(const PointerCoords& other) const {
297 if (bits != other.bits) {
298 return false;
299 }
Michael Wright38dcdff2014-03-19 12:06:10 -0700300 uint32_t count = BitSet64::count(bits);
Jeff Brown5912f952013-07-01 19:10:31 -0700301 for (uint32_t i = 0; i < count; i++) {
302 if (values[i] != other.values[i]) {
303 return false;
304 }
305 }
306 return true;
307}
308
309void PointerCoords::copyFrom(const PointerCoords& other) {
310 bits = other.bits;
Michael Wright38dcdff2014-03-19 12:06:10 -0700311 uint32_t count = BitSet64::count(bits);
Jeff Brown5912f952013-07-01 19:10:31 -0700312 for (uint32_t i = 0; i < count; i++) {
313 values[i] = other.values[i];
314 }
315}
316
chaviwc01e1372020-07-01 12:37:31 -0700317void PointerCoords::transform(const ui::Transform& transform) {
318 vec2 newCoords = transform.transform(getX(), getY());
319 setAxisValue(AMOTION_EVENT_AXIS_X, newCoords.x);
320 setAxisValue(AMOTION_EVENT_AXIS_Y, newCoords.y);
321}
Jeff Brown5912f952013-07-01 19:10:31 -0700322
323// --- PointerProperties ---
324
325bool PointerProperties::operator==(const PointerProperties& other) const {
326 return id == other.id
327 && toolType == other.toolType;
328}
329
330void PointerProperties::copyFrom(const PointerProperties& other) {
331 id = other.id;
332 toolType = other.toolType;
333}
334
335
336// --- MotionEvent ---
337
Garfield Tan4cc839f2020-01-24 11:26:14 -0800338void MotionEvent::initialize(int32_t id, int32_t deviceId, uint32_t source, int32_t displayId,
Siarhei Vishniakou9c858ac2020-01-23 14:20:11 -0600339 std::array<uint8_t, 32> hmac, int32_t action, int32_t actionButton,
340 int32_t flags, int32_t edgeFlags, int32_t metaState,
chaviw9eaa22c2020-07-01 16:21:27 -0700341 int32_t buttonState, MotionClassification classification,
342 const ui::Transform& transform, float xPrecision, float yPrecision,
Evan Rosky84f07f02021-04-16 10:42:42 -0700343 float rawXCursorPosition, float rawYCursorPosition,
344 int32_t displayWidth, int32_t displayHeight, nsecs_t downTime,
chaviw9eaa22c2020-07-01 16:21:27 -0700345 nsecs_t eventTime, size_t pointerCount,
Siarhei Vishniakou9c858ac2020-01-23 14:20:11 -0600346 const PointerProperties* pointerProperties,
Garfield Tan00f511d2019-06-12 16:55:40 -0700347 const PointerCoords* pointerCoords) {
Garfield Tan4cc839f2020-01-24 11:26:14 -0800348 InputEvent::initialize(id, deviceId, source, displayId, hmac);
Jeff Brown5912f952013-07-01 19:10:31 -0700349 mAction = action;
Michael Wright7b159c92015-05-14 14:48:03 +0100350 mActionButton = actionButton;
Jeff Brown5912f952013-07-01 19:10:31 -0700351 mFlags = flags;
352 mEdgeFlags = edgeFlags;
353 mMetaState = metaState;
354 mButtonState = buttonState;
Siarhei Vishniakou49e59222018-12-28 18:17:15 -0800355 mClassification = classification;
chaviw9eaa22c2020-07-01 16:21:27 -0700356 mTransform = transform;
Jeff Brown5912f952013-07-01 19:10:31 -0700357 mXPrecision = xPrecision;
358 mYPrecision = yPrecision;
Garfield Tan937bb832019-07-25 17:48:31 -0700359 mRawXCursorPosition = rawXCursorPosition;
360 mRawYCursorPosition = rawYCursorPosition;
Evan Rosky84f07f02021-04-16 10:42:42 -0700361 mDisplayWidth = displayWidth;
362 mDisplayHeight = displayHeight;
Jeff Brown5912f952013-07-01 19:10:31 -0700363 mDownTime = downTime;
364 mPointerProperties.clear();
365 mPointerProperties.appendArray(pointerProperties, pointerCount);
366 mSampleEventTimes.clear();
367 mSamplePointerCoords.clear();
368 addSample(eventTime, pointerCoords);
369}
370
371void MotionEvent::copyFrom(const MotionEvent* other, bool keepHistory) {
Garfield Tan4cc839f2020-01-24 11:26:14 -0800372 InputEvent::initialize(other->mId, other->mDeviceId, other->mSource, other->mDisplayId,
373 other->mHmac);
Jeff Brown5912f952013-07-01 19:10:31 -0700374 mAction = other->mAction;
Michael Wright7b159c92015-05-14 14:48:03 +0100375 mActionButton = other->mActionButton;
Jeff Brown5912f952013-07-01 19:10:31 -0700376 mFlags = other->mFlags;
377 mEdgeFlags = other->mEdgeFlags;
378 mMetaState = other->mMetaState;
379 mButtonState = other->mButtonState;
Siarhei Vishniakou49e59222018-12-28 18:17:15 -0800380 mClassification = other->mClassification;
chaviw9eaa22c2020-07-01 16:21:27 -0700381 mTransform = other->mTransform;
Jeff Brown5912f952013-07-01 19:10:31 -0700382 mXPrecision = other->mXPrecision;
383 mYPrecision = other->mYPrecision;
Garfield Tan937bb832019-07-25 17:48:31 -0700384 mRawXCursorPosition = other->mRawXCursorPosition;
385 mRawYCursorPosition = other->mRawYCursorPosition;
Evan Rosky84f07f02021-04-16 10:42:42 -0700386 mDisplayWidth = other->mDisplayWidth;
387 mDisplayHeight = other->mDisplayHeight;
Jeff Brown5912f952013-07-01 19:10:31 -0700388 mDownTime = other->mDownTime;
389 mPointerProperties = other->mPointerProperties;
390
391 if (keepHistory) {
392 mSampleEventTimes = other->mSampleEventTimes;
393 mSamplePointerCoords = other->mSamplePointerCoords;
394 } else {
395 mSampleEventTimes.clear();
Siarhei Vishniakou46a27742020-09-09 13:57:28 -0500396 mSampleEventTimes.push_back(other->getEventTime());
Jeff Brown5912f952013-07-01 19:10:31 -0700397 mSamplePointerCoords.clear();
398 size_t pointerCount = other->getPointerCount();
399 size_t historySize = other->getHistorySize();
400 mSamplePointerCoords.appendArray(other->mSamplePointerCoords.array()
401 + (historySize * pointerCount), pointerCount);
402 }
403}
404
405void MotionEvent::addSample(
406 int64_t eventTime,
407 const PointerCoords* pointerCoords) {
Siarhei Vishniakou46a27742020-09-09 13:57:28 -0500408 mSampleEventTimes.push_back(eventTime);
Jeff Brown5912f952013-07-01 19:10:31 -0700409 mSamplePointerCoords.appendArray(pointerCoords, getPointerCount());
410}
411
Garfield Tan00f511d2019-06-12 16:55:40 -0700412float MotionEvent::getXCursorPosition() const {
chaviw9eaa22c2020-07-01 16:21:27 -0700413 vec2 vals = mTransform.transform(getRawXCursorPosition(), getRawYCursorPosition());
414 return vals.x;
Garfield Tan00f511d2019-06-12 16:55:40 -0700415}
416
417float MotionEvent::getYCursorPosition() const {
chaviw9eaa22c2020-07-01 16:21:27 -0700418 vec2 vals = mTransform.transform(getRawXCursorPosition(), getRawYCursorPosition());
419 return vals.y;
Garfield Tan00f511d2019-06-12 16:55:40 -0700420}
421
Garfield Tan937bb832019-07-25 17:48:31 -0700422void MotionEvent::setCursorPosition(float x, float y) {
chaviw9eaa22c2020-07-01 16:21:27 -0700423 ui::Transform inverse = mTransform.inverse();
424 vec2 vals = inverse.transform(x, y);
425 mRawXCursorPosition = vals.x;
426 mRawYCursorPosition = vals.y;
Garfield Tan937bb832019-07-25 17:48:31 -0700427}
428
Jeff Brown5912f952013-07-01 19:10:31 -0700429const PointerCoords* MotionEvent::getRawPointerCoords(size_t pointerIndex) const {
430 return &mSamplePointerCoords[getHistorySize() * getPointerCount() + pointerIndex];
431}
432
433float MotionEvent::getRawAxisValue(int32_t axis, size_t pointerIndex) const {
Evan Rosky84f07f02021-04-16 10:42:42 -0700434 return getHistoricalRawAxisValue(axis, pointerIndex, getHistorySize());
Jeff Brown5912f952013-07-01 19:10:31 -0700435}
436
437float MotionEvent::getAxisValue(int32_t axis, size_t pointerIndex) const {
chaviw9eaa22c2020-07-01 16:21:27 -0700438 return getHistoricalAxisValue(axis, pointerIndex, getHistorySize());
Jeff Brown5912f952013-07-01 19:10:31 -0700439}
440
441const PointerCoords* MotionEvent::getHistoricalRawPointerCoords(
442 size_t pointerIndex, size_t historicalIndex) const {
443 return &mSamplePointerCoords[historicalIndex * getPointerCount() + pointerIndex];
444}
445
446float MotionEvent::getHistoricalRawAxisValue(int32_t axis, size_t pointerIndex,
447 size_t historicalIndex) const {
Evan Rosky84f07f02021-04-16 10:42:42 -0700448 if (axis != AMOTION_EVENT_AXIS_X && axis != AMOTION_EVENT_AXIS_Y) {
449 return getHistoricalRawPointerCoords(pointerIndex, historicalIndex)->getAxisValue(axis);
450 }
451 // 0x7 encapsulates all 3 rotations (see ui::Transform::RotationFlags)
452 static const int ALL_ROTATIONS_MASK = 0x7;
453 uint32_t orientation = (mTransform.getOrientation() & ALL_ROTATIONS_MASK);
454 if (orientation == ui::Transform::ROT_0) {
455 return getHistoricalRawPointerCoords(pointerIndex, historicalIndex)->getAxisValue(axis);
456 }
457
458 // For compatibility, convert raw coordinates into "oriented screen space". Once app developers
459 // are educated about getRaw, we can consider removing this.
460 vec2 xy = getHistoricalRawPointerCoords(pointerIndex, historicalIndex)->getXYValue();
461 const float unrotatedX = xy.x;
462 if (orientation == ui::Transform::ROT_90) {
463 xy.x = mDisplayHeight - xy.y;
464 xy.y = unrotatedX;
465 } else if (orientation == ui::Transform::ROT_180) {
466 xy.x = mDisplayWidth - xy.x;
467 xy.y = mDisplayHeight - xy.y;
468 } else if (orientation == ui::Transform::ROT_270) {
469 xy.x = xy.y;
470 xy.y = mDisplayWidth - unrotatedX;
471 }
472 static_assert(AMOTION_EVENT_AXIS_X == 0 && AMOTION_EVENT_AXIS_Y == 1);
473 return xy[axis];
Jeff Brown5912f952013-07-01 19:10:31 -0700474}
475
476float MotionEvent::getHistoricalAxisValue(int32_t axis, size_t pointerIndex,
477 size_t historicalIndex) const {
chaviw9eaa22c2020-07-01 16:21:27 -0700478 if (axis != AMOTION_EVENT_AXIS_X && axis != AMOTION_EVENT_AXIS_Y) {
479 return getHistoricalRawPointerCoords(pointerIndex, historicalIndex)->getAxisValue(axis);
480 }
481
Evan Rosky84f07f02021-04-16 10:42:42 -0700482 vec2 vals = mTransform.transform(
483 getHistoricalRawPointerCoords(pointerIndex, historicalIndex)->getXYValue());
484 static_assert(AMOTION_EVENT_AXIS_X == 0 && AMOTION_EVENT_AXIS_Y == 1);
485 return vals[axis];
Jeff Brown5912f952013-07-01 19:10:31 -0700486}
487
488ssize_t MotionEvent::findPointerIndex(int32_t pointerId) const {
489 size_t pointerCount = mPointerProperties.size();
490 for (size_t i = 0; i < pointerCount; i++) {
491 if (mPointerProperties.itemAt(i).id == pointerId) {
492 return i;
493 }
494 }
495 return -1;
496}
497
498void MotionEvent::offsetLocation(float xOffset, float yOffset) {
chaviw9eaa22c2020-07-01 16:21:27 -0700499 float currXOffset = mTransform.tx();
500 float currYOffset = mTransform.ty();
501 mTransform.set(currXOffset + xOffset, currYOffset + yOffset);
Jeff Brown5912f952013-07-01 19:10:31 -0700502}
503
Robert Carre07e1032018-11-26 12:55:53 -0800504void MotionEvent::scale(float globalScaleFactor) {
chaviw9eaa22c2020-07-01 16:21:27 -0700505 mTransform.set(mTransform.tx() * globalScaleFactor, mTransform.ty() * globalScaleFactor);
Robert Carre07e1032018-11-26 12:55:53 -0800506 mXPrecision *= globalScaleFactor;
507 mYPrecision *= globalScaleFactor;
Jeff Brown5912f952013-07-01 19:10:31 -0700508
509 size_t numSamples = mSamplePointerCoords.size();
510 for (size_t i = 0; i < numSamples; i++) {
chaviw9eaa22c2020-07-01 16:21:27 -0700511 mSamplePointerCoords.editItemAt(i).scale(globalScaleFactor, globalScaleFactor,
512 globalScaleFactor);
Jeff Brown5912f952013-07-01 19:10:31 -0700513 }
514}
515
chaviw9eaa22c2020-07-01 16:21:27 -0700516static vec2 transformPoint(const std::array<float, 9>& matrix, float x, float y) {
Jeff Brown5a2f68e2013-07-15 17:28:19 -0700517 // Apply perspective transform like Skia.
518 float newX = matrix[0] * x + matrix[1] * y + matrix[2];
519 float newY = matrix[3] * x + matrix[4] * y + matrix[5];
520 float newZ = matrix[6] * x + matrix[7] * y + matrix[8];
521 if (newZ) {
522 newZ = 1.0f / newZ;
523 }
chaviw9eaa22c2020-07-01 16:21:27 -0700524 vec2 transformedPoint;
525 transformedPoint.x = newX * newZ;
526 transformedPoint.y = newY * newZ;
527 return transformedPoint;
Jeff Brown5a2f68e2013-07-15 17:28:19 -0700528}
529
chaviw9eaa22c2020-07-01 16:21:27 -0700530static float transformAngle(const std::array<float, 9>& matrix, float angleRadians, float originX,
531 float originY) {
Jeff Brown5912f952013-07-01 19:10:31 -0700532 // Construct and transform a vector oriented at the specified clockwise angle from vertical.
533 // Coordinate system: down is increasing Y, right is increasing X.
Jeff Brown5a2f68e2013-07-15 17:28:19 -0700534 float x = sinf(angleRadians);
535 float y = -cosf(angleRadians);
chaviw9eaa22c2020-07-01 16:21:27 -0700536 vec2 transformedPoint = transformPoint(matrix, x, y);
537
538 transformedPoint.x -= originX;
539 transformedPoint.y -= originY;
Jeff Brown5912f952013-07-01 19:10:31 -0700540
541 // Derive the transformed vector's clockwise angle from vertical.
chaviw9eaa22c2020-07-01 16:21:27 -0700542 float result = atan2f(transformedPoint.x, -transformedPoint.y);
Jeff Brown5912f952013-07-01 19:10:31 -0700543 if (result < - M_PI_2) {
544 result += M_PI;
545 } else if (result > M_PI_2) {
546 result -= M_PI;
547 }
548 return result;
549}
550
chaviw9eaa22c2020-07-01 16:21:27 -0700551void MotionEvent::transform(const std::array<float, 9>& matrix) {
552 // We want to preserve the rawX and rawY so we just update the transform
553 // using the values of the transform passed in
554 ui::Transform newTransform;
555 newTransform.set(matrix);
556 mTransform = newTransform * mTransform;
Jeff Brown5912f952013-07-01 19:10:31 -0700557
Jeff Brown5a2f68e2013-07-15 17:28:19 -0700558 // Determine how the origin is transformed by the matrix so that we
559 // can transform orientation vectors.
chaviw9eaa22c2020-07-01 16:21:27 -0700560 vec2 origin = transformPoint(matrix, 0, 0);
Garfield Tan00f511d2019-06-12 16:55:40 -0700561
Jeff Brown5912f952013-07-01 19:10:31 -0700562 // Apply the transformation to all samples.
563 size_t numSamples = mSamplePointerCoords.size();
564 for (size_t i = 0; i < numSamples; i++) {
565 PointerCoords& c = mSamplePointerCoords.editItemAt(i);
Jeff Brown5912f952013-07-01 19:10:31 -0700566 float orientation = c.getAxisValue(AMOTION_EVENT_AXIS_ORIENTATION);
Jeff Brown5a2f68e2013-07-15 17:28:19 -0700567 c.setAxisValue(AMOTION_EVENT_AXIS_ORIENTATION,
chaviw9eaa22c2020-07-01 16:21:27 -0700568 transformAngle(matrix, orientation, origin.x, origin.y));
Jeff Brown5912f952013-07-01 19:10:31 -0700569 }
570}
571
Evan Roskyd4d4d802021-05-03 20:12:21 -0700572void MotionEvent::applyTransform(const std::array<float, 9>& matrix) {
573 // Determine how the origin is transformed by the matrix so that we
574 // can transform orientation vectors.
575 vec2 origin = transformPoint(matrix, 0, 0);
576
577 // Apply the transformation to all samples.
578 size_t numSamples = mSamplePointerCoords.size();
579 for (size_t i = 0; i < numSamples; i++) {
580 PointerCoords& c = mSamplePointerCoords.editItemAt(i);
581 float orientation = c.getAxisValue(AMOTION_EVENT_AXIS_ORIENTATION);
582 c.setAxisValue(AMOTION_EVENT_AXIS_ORIENTATION,
583 transformAngle(matrix, orientation, origin.x, origin.y));
584 vec2 xy = transformPoint(matrix, c.getX(), c.getY());
585 c.setAxisValue(AMOTION_EVENT_AXIS_X, xy.x);
586 c.setAxisValue(AMOTION_EVENT_AXIS_Y, xy.y);
587 }
588}
589
Brett Chabotfaa986c2020-11-04 17:39:36 -0800590#ifdef __linux__
chaviw9eaa22c2020-07-01 16:21:27 -0700591static status_t readFromParcel(ui::Transform& transform, const Parcel& parcel) {
592 float dsdx, dtdx, tx, dtdy, dsdy, ty;
593 status_t status = parcel.readFloat(&dsdx);
594 status |= parcel.readFloat(&dtdx);
595 status |= parcel.readFloat(&tx);
596 status |= parcel.readFloat(&dtdy);
597 status |= parcel.readFloat(&dsdy);
598 status |= parcel.readFloat(&ty);
599
600 transform.set({dsdx, dtdx, tx, dtdy, dsdy, ty, 0, 0, 1});
601 return status;
602}
603
604static status_t writeToParcel(const ui::Transform& transform, Parcel& parcel) {
605 status_t status = parcel.writeFloat(transform.dsdx());
606 status |= parcel.writeFloat(transform.dtdx());
607 status |= parcel.writeFloat(transform.tx());
608 status |= parcel.writeFloat(transform.dtdy());
609 status |= parcel.writeFloat(transform.dsdy());
610 status |= parcel.writeFloat(transform.ty());
611 return status;
612}
613
Jeff Brown5912f952013-07-01 19:10:31 -0700614status_t MotionEvent::readFromParcel(Parcel* parcel) {
615 size_t pointerCount = parcel->readInt32();
616 size_t sampleCount = parcel->readInt32();
Flanker552a8a52015-09-07 15:28:58 +0800617 if (pointerCount == 0 || pointerCount > MAX_POINTERS ||
618 sampleCount == 0 || sampleCount > MAX_SAMPLES) {
Jeff Brown5912f952013-07-01 19:10:31 -0700619 return BAD_VALUE;
620 }
621
Garfield Tan4cc839f2020-01-24 11:26:14 -0800622 mId = parcel->readInt32();
Jeff Brown5912f952013-07-01 19:10:31 -0700623 mDeviceId = parcel->readInt32();
Siarhei Vishniakou3826d472020-01-27 10:44:40 -0600624 mSource = parcel->readUint32();
Siarhei Vishniakou777a10b2018-01-31 16:45:06 -0800625 mDisplayId = parcel->readInt32();
Siarhei Vishniakou9c858ac2020-01-23 14:20:11 -0600626 std::vector<uint8_t> hmac;
627 status_t result = parcel->readByteVector(&hmac);
628 if (result != OK || hmac.size() != 32) {
629 return BAD_VALUE;
630 }
631 std::move(hmac.begin(), hmac.begin() + hmac.size(), mHmac.begin());
Jeff Brown5912f952013-07-01 19:10:31 -0700632 mAction = parcel->readInt32();
Michael Wright7b159c92015-05-14 14:48:03 +0100633 mActionButton = parcel->readInt32();
Jeff Brown5912f952013-07-01 19:10:31 -0700634 mFlags = parcel->readInt32();
635 mEdgeFlags = parcel->readInt32();
636 mMetaState = parcel->readInt32();
637 mButtonState = parcel->readInt32();
Siarhei Vishniakou49e59222018-12-28 18:17:15 -0800638 mClassification = static_cast<MotionClassification>(parcel->readByte());
chaviw9eaa22c2020-07-01 16:21:27 -0700639
640 result = android::readFromParcel(mTransform, *parcel);
641 if (result != OK) {
642 return result;
643 }
Jeff Brown5912f952013-07-01 19:10:31 -0700644 mXPrecision = parcel->readFloat();
645 mYPrecision = parcel->readFloat();
Garfield Tan937bb832019-07-25 17:48:31 -0700646 mRawXCursorPosition = parcel->readFloat();
647 mRawYCursorPosition = parcel->readFloat();
Evan Rosky84f07f02021-04-16 10:42:42 -0700648 mDisplayWidth = parcel->readInt32();
649 mDisplayHeight = parcel->readInt32();
Jeff Brown5912f952013-07-01 19:10:31 -0700650 mDownTime = parcel->readInt64();
651
652 mPointerProperties.clear();
653 mPointerProperties.setCapacity(pointerCount);
654 mSampleEventTimes.clear();
Siarhei Vishniakou46a27742020-09-09 13:57:28 -0500655 mSampleEventTimes.reserve(sampleCount);
Jeff Brown5912f952013-07-01 19:10:31 -0700656 mSamplePointerCoords.clear();
657 mSamplePointerCoords.setCapacity(sampleCount * pointerCount);
658
659 for (size_t i = 0; i < pointerCount; i++) {
660 mPointerProperties.push();
661 PointerProperties& properties = mPointerProperties.editTop();
662 properties.id = parcel->readInt32();
663 properties.toolType = parcel->readInt32();
664 }
665
Dan Austinc94fc452015-09-22 14:22:41 -0700666 while (sampleCount > 0) {
667 sampleCount--;
Siarhei Vishniakou46a27742020-09-09 13:57:28 -0500668 mSampleEventTimes.push_back(parcel->readInt64());
Jeff Brown5912f952013-07-01 19:10:31 -0700669 for (size_t i = 0; i < pointerCount; i++) {
670 mSamplePointerCoords.push();
671 status_t status = mSamplePointerCoords.editTop().readFromParcel(parcel);
672 if (status) {
673 return status;
674 }
675 }
676 }
677 return OK;
678}
679
680status_t MotionEvent::writeToParcel(Parcel* parcel) const {
681 size_t pointerCount = mPointerProperties.size();
682 size_t sampleCount = mSampleEventTimes.size();
683
684 parcel->writeInt32(pointerCount);
685 parcel->writeInt32(sampleCount);
686
Garfield Tan4cc839f2020-01-24 11:26:14 -0800687 parcel->writeInt32(mId);
Jeff Brown5912f952013-07-01 19:10:31 -0700688 parcel->writeInt32(mDeviceId);
Siarhei Vishniakou3826d472020-01-27 10:44:40 -0600689 parcel->writeUint32(mSource);
Siarhei Vishniakou777a10b2018-01-31 16:45:06 -0800690 parcel->writeInt32(mDisplayId);
Siarhei Vishniakou9c858ac2020-01-23 14:20:11 -0600691 std::vector<uint8_t> hmac(mHmac.begin(), mHmac.end());
692 parcel->writeByteVector(hmac);
Jeff Brown5912f952013-07-01 19:10:31 -0700693 parcel->writeInt32(mAction);
Michael Wright7b159c92015-05-14 14:48:03 +0100694 parcel->writeInt32(mActionButton);
Jeff Brown5912f952013-07-01 19:10:31 -0700695 parcel->writeInt32(mFlags);
696 parcel->writeInt32(mEdgeFlags);
697 parcel->writeInt32(mMetaState);
698 parcel->writeInt32(mButtonState);
Siarhei Vishniakou49e59222018-12-28 18:17:15 -0800699 parcel->writeByte(static_cast<int8_t>(mClassification));
chaviw9eaa22c2020-07-01 16:21:27 -0700700
701 status_t result = android::writeToParcel(mTransform, *parcel);
702 if (result != OK) {
703 return result;
704 }
Jeff Brown5912f952013-07-01 19:10:31 -0700705 parcel->writeFloat(mXPrecision);
706 parcel->writeFloat(mYPrecision);
Garfield Tan937bb832019-07-25 17:48:31 -0700707 parcel->writeFloat(mRawXCursorPosition);
708 parcel->writeFloat(mRawYCursorPosition);
Evan Rosky84f07f02021-04-16 10:42:42 -0700709 parcel->writeInt32(mDisplayWidth);
710 parcel->writeInt32(mDisplayHeight);
Jeff Brown5912f952013-07-01 19:10:31 -0700711 parcel->writeInt64(mDownTime);
712
713 for (size_t i = 0; i < pointerCount; i++) {
714 const PointerProperties& properties = mPointerProperties.itemAt(i);
715 parcel->writeInt32(properties.id);
716 parcel->writeInt32(properties.toolType);
717 }
718
719 const PointerCoords* pc = mSamplePointerCoords.array();
720 for (size_t h = 0; h < sampleCount; h++) {
Siarhei Vishniakou46a27742020-09-09 13:57:28 -0500721 parcel->writeInt64(mSampleEventTimes[h]);
Jeff Brown5912f952013-07-01 19:10:31 -0700722 for (size_t i = 0; i < pointerCount; i++) {
723 status_t status = (pc++)->writeToParcel(parcel);
724 if (status) {
725 return status;
726 }
727 }
728 }
729 return OK;
730}
Brett Chabotfaa986c2020-11-04 17:39:36 -0800731#endif
Jeff Brown5912f952013-07-01 19:10:31 -0700732
Siarhei Vishniakou3826d472020-01-27 10:44:40 -0600733bool MotionEvent::isTouchEvent(uint32_t source, int32_t action) {
Jeff Brown5912f952013-07-01 19:10:31 -0700734 if (source & AINPUT_SOURCE_CLASS_POINTER) {
735 // Specifically excludes HOVER_MOVE and SCROLL.
736 switch (action & AMOTION_EVENT_ACTION_MASK) {
737 case AMOTION_EVENT_ACTION_DOWN:
738 case AMOTION_EVENT_ACTION_MOVE:
739 case AMOTION_EVENT_ACTION_UP:
740 case AMOTION_EVENT_ACTION_POINTER_DOWN:
741 case AMOTION_EVENT_ACTION_POINTER_UP:
742 case AMOTION_EVENT_ACTION_CANCEL:
743 case AMOTION_EVENT_ACTION_OUTSIDE:
744 return true;
745 }
746 }
747 return false;
748}
749
Michael Wright872db4f2014-04-22 15:03:51 -0700750const char* MotionEvent::getLabel(int32_t axis) {
Chris Ye4958d062020-08-20 13:21:10 -0700751 return InputEventLookup::getAxisLabel(axis);
Michael Wright872db4f2014-04-22 15:03:51 -0700752}
753
754int32_t MotionEvent::getAxisFromLabel(const char* label) {
Chris Ye4958d062020-08-20 13:21:10 -0700755 return InputEventLookup::getAxisByLabel(label);
Michael Wright872db4f2014-04-22 15:03:51 -0700756}
757
Siarhei Vishniakouc68fdec2020-10-22 14:58:14 -0500758std::string MotionEvent::actionToString(int32_t action) {
Siarhei Vishniakoud44dddf2020-03-25 16:16:40 -0700759 // Convert MotionEvent action to string
760 switch (action & AMOTION_EVENT_ACTION_MASK) {
761 case AMOTION_EVENT_ACTION_DOWN:
762 return "DOWN";
Siarhei Vishniakoud44dddf2020-03-25 16:16:40 -0700763 case AMOTION_EVENT_ACTION_UP:
764 return "UP";
Siarhei Vishniakouc68fdec2020-10-22 14:58:14 -0500765 case AMOTION_EVENT_ACTION_MOVE:
766 return "MOVE";
Siarhei Vishniakoud44dddf2020-03-25 16:16:40 -0700767 case AMOTION_EVENT_ACTION_CANCEL:
768 return "CANCEL";
Siarhei Vishniakouc68fdec2020-10-22 14:58:14 -0500769 case AMOTION_EVENT_ACTION_OUTSIDE:
770 return "OUTSIDE";
Siarhei Vishniakoud44dddf2020-03-25 16:16:40 -0700771 case AMOTION_EVENT_ACTION_POINTER_DOWN:
772 return "POINTER_DOWN";
773 case AMOTION_EVENT_ACTION_POINTER_UP:
774 return "POINTER_UP";
Siarhei Vishniakouc68fdec2020-10-22 14:58:14 -0500775 case AMOTION_EVENT_ACTION_HOVER_MOVE:
776 return "HOVER_MOVE";
777 case AMOTION_EVENT_ACTION_SCROLL:
778 return "SCROLL";
779 case AMOTION_EVENT_ACTION_HOVER_ENTER:
780 return "HOVER_ENTER";
781 case AMOTION_EVENT_ACTION_HOVER_EXIT:
782 return "HOVER_EXIT";
783 case AMOTION_EVENT_ACTION_BUTTON_PRESS:
784 return "BUTTON_PRESS";
785 case AMOTION_EVENT_ACTION_BUTTON_RELEASE:
786 return "BUTTON_RELEASE";
Siarhei Vishniakoud44dddf2020-03-25 16:16:40 -0700787 }
Siarhei Vishniakouc68fdec2020-10-22 14:58:14 -0500788 return android::base::StringPrintf("%" PRId32, action);
Siarhei Vishniakoud44dddf2020-03-25 16:16:40 -0700789}
790
Siarhei Vishniakou7feb2ea2019-11-25 15:11:23 -0800791// --- FocusEvent ---
792
Garfield Tan4cc839f2020-01-24 11:26:14 -0800793void FocusEvent::initialize(int32_t id, bool hasFocus, bool inTouchMode) {
794 InputEvent::initialize(id, ReservedInputDeviceId::VIRTUAL_KEYBOARD_ID, AINPUT_SOURCE_UNKNOWN,
Siarhei Vishniakou9c858ac2020-01-23 14:20:11 -0600795 ADISPLAY_ID_NONE, INVALID_HMAC);
Siarhei Vishniakou7feb2ea2019-11-25 15:11:23 -0800796 mHasFocus = hasFocus;
797 mInTouchMode = inTouchMode;
798}
799
800void FocusEvent::initialize(const FocusEvent& from) {
801 InputEvent::initialize(from);
802 mHasFocus = from.mHasFocus;
803 mInTouchMode = from.mInTouchMode;
804}
Jeff Brown5912f952013-07-01 19:10:31 -0700805
Prabir Pradhan3f37b7b2020-11-10 16:50:18 -0800806// --- CaptureEvent ---
807
808void CaptureEvent::initialize(int32_t id, bool pointerCaptureEnabled) {
809 InputEvent::initialize(id, ReservedInputDeviceId::VIRTUAL_KEYBOARD_ID, AINPUT_SOURCE_UNKNOWN,
810 ADISPLAY_ID_NONE, INVALID_HMAC);
811 mPointerCaptureEnabled = pointerCaptureEnabled;
812}
813
814void CaptureEvent::initialize(const CaptureEvent& from) {
815 InputEvent::initialize(from);
816 mPointerCaptureEnabled = from.mPointerCaptureEnabled;
817}
818
arthurhung7632c332020-12-30 16:58:01 +0800819// --- DragEvent ---
820
821void DragEvent::initialize(int32_t id, float x, float y, bool isExiting) {
822 InputEvent::initialize(id, ReservedInputDeviceId::VIRTUAL_KEYBOARD_ID, AINPUT_SOURCE_UNKNOWN,
823 ADISPLAY_ID_NONE, INVALID_HMAC);
824 mIsExiting = isExiting;
825 mX = x;
826 mY = y;
827}
828
829void DragEvent::initialize(const DragEvent& from) {
830 InputEvent::initialize(from);
831 mIsExiting = from.mIsExiting;
832 mX = from.mX;
833 mY = from.mY;
834}
835
Jeff Brown5912f952013-07-01 19:10:31 -0700836// --- PooledInputEventFactory ---
837
838PooledInputEventFactory::PooledInputEventFactory(size_t maxPoolSize) :
839 mMaxPoolSize(maxPoolSize) {
840}
841
842PooledInputEventFactory::~PooledInputEventFactory() {
Jeff Brown5912f952013-07-01 19:10:31 -0700843}
844
845KeyEvent* PooledInputEventFactory::createKeyEvent() {
Siarhei Vishniakou727a44e2019-11-23 12:59:16 -0800846 if (mKeyEventPool.empty()) {
847 return new KeyEvent();
Jeff Brown5912f952013-07-01 19:10:31 -0700848 }
Siarhei Vishniakou727a44e2019-11-23 12:59:16 -0800849 KeyEvent* event = mKeyEventPool.front().release();
850 mKeyEventPool.pop();
851 return event;
Jeff Brown5912f952013-07-01 19:10:31 -0700852}
853
854MotionEvent* PooledInputEventFactory::createMotionEvent() {
Siarhei Vishniakou727a44e2019-11-23 12:59:16 -0800855 if (mMotionEventPool.empty()) {
856 return new MotionEvent();
Jeff Brown5912f952013-07-01 19:10:31 -0700857 }
Siarhei Vishniakou727a44e2019-11-23 12:59:16 -0800858 MotionEvent* event = mMotionEventPool.front().release();
859 mMotionEventPool.pop();
860 return event;
Jeff Brown5912f952013-07-01 19:10:31 -0700861}
862
Siarhei Vishniakou7feb2ea2019-11-25 15:11:23 -0800863FocusEvent* PooledInputEventFactory::createFocusEvent() {
864 if (mFocusEventPool.empty()) {
865 return new FocusEvent();
866 }
867 FocusEvent* event = mFocusEventPool.front().release();
868 mFocusEventPool.pop();
869 return event;
870}
871
Prabir Pradhan3f37b7b2020-11-10 16:50:18 -0800872CaptureEvent* PooledInputEventFactory::createCaptureEvent() {
873 if (mCaptureEventPool.empty()) {
874 return new CaptureEvent();
875 }
876 CaptureEvent* event = mCaptureEventPool.front().release();
877 mCaptureEventPool.pop();
878 return event;
879}
880
arthurhung7632c332020-12-30 16:58:01 +0800881DragEvent* PooledInputEventFactory::createDragEvent() {
882 if (mDragEventPool.empty()) {
883 return new DragEvent();
884 }
885 DragEvent* event = mDragEventPool.front().release();
886 mDragEventPool.pop();
887 return event;
888}
889
Jeff Brown5912f952013-07-01 19:10:31 -0700890void PooledInputEventFactory::recycle(InputEvent* event) {
891 switch (event->getType()) {
892 case AINPUT_EVENT_TYPE_KEY:
893 if (mKeyEventPool.size() < mMaxPoolSize) {
Siarhei Vishniakou727a44e2019-11-23 12:59:16 -0800894 mKeyEventPool.push(std::unique_ptr<KeyEvent>(static_cast<KeyEvent*>(event)));
Jeff Brown5912f952013-07-01 19:10:31 -0700895 return;
896 }
897 break;
898 case AINPUT_EVENT_TYPE_MOTION:
899 if (mMotionEventPool.size() < mMaxPoolSize) {
Siarhei Vishniakou727a44e2019-11-23 12:59:16 -0800900 mMotionEventPool.push(std::unique_ptr<MotionEvent>(static_cast<MotionEvent*>(event)));
Jeff Brown5912f952013-07-01 19:10:31 -0700901 return;
902 }
903 break;
Siarhei Vishniakou7feb2ea2019-11-25 15:11:23 -0800904 case AINPUT_EVENT_TYPE_FOCUS:
905 if (mFocusEventPool.size() < mMaxPoolSize) {
906 mFocusEventPool.push(std::unique_ptr<FocusEvent>(static_cast<FocusEvent*>(event)));
907 return;
908 }
909 break;
Prabir Pradhan3f37b7b2020-11-10 16:50:18 -0800910 case AINPUT_EVENT_TYPE_CAPTURE:
911 if (mCaptureEventPool.size() < mMaxPoolSize) {
912 mCaptureEventPool.push(
913 std::unique_ptr<CaptureEvent>(static_cast<CaptureEvent*>(event)));
914 return;
915 }
916 break;
arthurhung7632c332020-12-30 16:58:01 +0800917 case AINPUT_EVENT_TYPE_DRAG:
918 if (mDragEventPool.size() < mMaxPoolSize) {
919 mDragEventPool.push(std::unique_ptr<DragEvent>(static_cast<DragEvent*>(event)));
920 return;
921 }
922 break;
Jeff Brown5912f952013-07-01 19:10:31 -0700923 }
924 delete event;
925}
926
927} // namespace android