blob: 9c5411424aa241791a077b405adfd9403a166050 [file] [log] [blame]
Siarhei Vishniakoud0784762019-11-01 15:33:48 -07001/*
2 * Copyright (C) 2019 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#include <benchmark/benchmark.h>
18
19#include <binder/Binder.h>
20#include "../dispatcher/InputDispatcher.h"
21
Siarhei Vishniakouae6229e2019-12-30 16:23:19 -080022using android::os::InputEventInjectionResult;
23using android::os::InputEventInjectionSync;
24
Siarhei Vishniakoud0784762019-11-01 15:33:48 -070025namespace android::inputdispatcher {
26
27// An arbitrary device id.
28static const int32_t DEVICE_ID = 1;
29
30// An arbitrary injector pid / uid pair that has permission to inject events.
31static const int32_t INJECTOR_PID = 999;
32static const int32_t INJECTOR_UID = 1001;
33
Siarhei Vishniakou097c3db2020-05-06 14:18:38 -070034static constexpr std::chrono::duration INJECT_EVENT_TIMEOUT = 5s;
35static constexpr std::chrono::nanoseconds DISPATCHING_TIMEOUT = 100ms;
Siarhei Vishniakoud0784762019-11-01 15:33:48 -070036
37static nsecs_t now() {
38 return systemTime(SYSTEM_TIME_MONOTONIC);
39}
40
41// --- FakeInputDispatcherPolicy ---
42
43class FakeInputDispatcherPolicy : public InputDispatcherPolicyInterface {
44public:
45 FakeInputDispatcherPolicy() {}
46
47protected:
48 virtual ~FakeInputDispatcherPolicy() {}
49
50private:
Siarhei Vishniakouc1ae5562020-06-30 14:22:57 -050051 void notifyConfigurationChanged(nsecs_t) override {}
Siarhei Vishniakoud0784762019-11-01 15:33:48 -070052
Siarhei Vishniakou234129c2020-10-22 22:28:12 -050053 void notifyNoFocusedWindowAnr(
54 const std::shared_ptr<InputApplicationHandle>& applicationHandle) override {
55 ALOGE("There is no focused window for %s", applicationHandle->getName().c_str());
Siarhei Vishniakoud0784762019-11-01 15:33:48 -070056 }
57
Siarhei Vishniakou3c63fa42020-12-15 02:59:54 +000058 void notifyWindowUnresponsive(const sp<IBinder>& connectionToken,
59 const std::string& reason) override {
60 ALOGE("Window is not responding: %s", reason.c_str());
Siarhei Vishniakou234129c2020-10-22 22:28:12 -050061 }
62
Siarhei Vishniakou3c63fa42020-12-15 02:59:54 +000063 void notifyWindowResponsive(const sp<IBinder>& connectionToken) override {}
64
65 void notifyMonitorUnresponsive(int32_t pid, const std::string& reason) override {
66 ALOGE("Monitor is not responding: %s", reason.c_str());
67 }
68
69 void notifyMonitorResponsive(int32_t pid) override {}
Siarhei Vishniakou234129c2020-10-22 22:28:12 -050070
Siarhei Vishniakouc1ae5562020-06-30 14:22:57 -050071 void notifyInputChannelBroken(const sp<IBinder>&) override {}
Siarhei Vishniakoud0784762019-11-01 15:33:48 -070072
Siarhei Vishniakouc1ae5562020-06-30 14:22:57 -050073 void notifyFocusChanged(const sp<IBinder>&, const sp<IBinder>&) override {}
Siarhei Vishniakoud0784762019-11-01 15:33:48 -070074
Chris Yef59a2f42020-10-16 12:55:26 -070075 void notifySensorEvent(int32_t deviceId, InputDeviceSensorType sensorType,
76 InputDeviceSensorAccuracy accuracy, nsecs_t timestamp,
77 const std::vector<float>& values) override {}
78
79 void notifySensorAccuracy(int32_t deviceId, InputDeviceSensorType sensorType,
80 InputDeviceSensorAccuracy accuracy) override {}
81
Chris Yefb552902021-02-03 17:18:37 -080082 void notifyVibratorState(int32_t deviceId, bool isOn) override {}
83
Bernardo Rufino2e1f6512020-10-08 13:42:07 +000084 void notifyUntrustedTouch(const std::string& obscuringPackage) override {}
85
Siarhei Vishniakouc1ae5562020-06-30 14:22:57 -050086 void getDispatcherConfiguration(InputDispatcherConfiguration* outConfig) override {
Siarhei Vishniakoud0784762019-11-01 15:33:48 -070087 *outConfig = mConfig;
88 }
89
Siarhei Vishniakouc1ae5562020-06-30 14:22:57 -050090 bool filterInputEvent(const InputEvent* inputEvent, uint32_t policyFlags) override {
Siarhei Vishniakoud0784762019-11-01 15:33:48 -070091 return true;
92 }
93
Siarhei Vishniakouc1ae5562020-06-30 14:22:57 -050094 void interceptKeyBeforeQueueing(const KeyEvent*, uint32_t&) override {}
Siarhei Vishniakoud0784762019-11-01 15:33:48 -070095
Siarhei Vishniakouc1ae5562020-06-30 14:22:57 -050096 void interceptMotionBeforeQueueing(int32_t, nsecs_t, uint32_t&) override {}
Siarhei Vishniakoud0784762019-11-01 15:33:48 -070097
Siarhei Vishniakouc1ae5562020-06-30 14:22:57 -050098 nsecs_t interceptKeyBeforeDispatching(const sp<IBinder>&, const KeyEvent*, uint32_t) override {
Siarhei Vishniakoud0784762019-11-01 15:33:48 -070099 return 0;
100 }
101
Siarhei Vishniakouc1ae5562020-06-30 14:22:57 -0500102 bool dispatchUnhandledKey(const sp<IBinder>&, const KeyEvent*, uint32_t, KeyEvent*) override {
Siarhei Vishniakoud0784762019-11-01 15:33:48 -0700103 return false;
104 }
105
Siarhei Vishniakouc1ae5562020-06-30 14:22:57 -0500106 void notifySwitch(nsecs_t, uint32_t, uint32_t, uint32_t) override {}
Siarhei Vishniakoud0784762019-11-01 15:33:48 -0700107
Sean Stoutb4e0a592021-02-23 07:34:53 -0800108 void pokeUserActivity(nsecs_t, int32_t, int32_t) override {}
Siarhei Vishniakoud0784762019-11-01 15:33:48 -0700109
Siarhei Vishniakouc1ae5562020-06-30 14:22:57 -0500110 bool checkInjectEventsPermissionNonReentrant(int32_t, int32_t) override { return false; }
Siarhei Vishniakoud0784762019-11-01 15:33:48 -0700111
Siarhei Vishniakouc1ae5562020-06-30 14:22:57 -0500112 void onPointerDownOutsideFocus(const sp<IBinder>& newToken) override {}
Siarhei Vishniakoud0784762019-11-01 15:33:48 -0700113
Prabir Pradhan99987712020-11-10 18:43:05 -0800114 void setPointerCapture(bool enabled) override {}
115
Siarhei Vishniakoud0784762019-11-01 15:33:48 -0700116 InputDispatcherConfiguration mConfig;
117};
118
119class FakeApplicationHandle : public InputApplicationHandle {
120public:
121 FakeApplicationHandle() {}
122 virtual ~FakeApplicationHandle() {}
123
124 virtual bool updateInfo() {
Siarhei Vishniakou70622952020-07-30 11:17:23 -0500125 mInfo.dispatchingTimeoutMillis =
126 std::chrono::duration_cast<std::chrono::milliseconds>(DISPATCHING_TIMEOUT).count();
Siarhei Vishniakoud0784762019-11-01 15:33:48 -0700127 return true;
128 }
129};
130
131class FakeInputReceiver {
132public:
133 void consumeEvent() {
Siarhei Vishniakoud549b252020-08-11 11:25:26 -0500134 uint32_t consumeSeq = 0;
Siarhei Vishniakoud0784762019-11-01 15:33:48 -0700135 InputEvent* event;
136
Siarhei Vishniakouadfd4fa2019-12-20 11:02:58 -0800137 std::chrono::time_point start = std::chrono::steady_clock::now();
Siarhei Vishniakoud0784762019-11-01 15:33:48 -0700138 status_t result = WOULD_BLOCK;
139 while (result == WOULD_BLOCK) {
Siarhei Vishniakouadfd4fa2019-12-20 11:02:58 -0800140 std::chrono::duration elapsed = std::chrono::steady_clock::now() - start;
141 if (elapsed > 10ms) {
142 ALOGE("Waited too long for consumer to produce an event, giving up");
143 break;
144 }
Siarhei Vishniakoud0784762019-11-01 15:33:48 -0700145 result = mConsumer->consume(&mEventFactory, true /*consumeBatches*/, -1, &consumeSeq,
146 &event);
147 }
148 if (result != OK) {
149 ALOGE("Received result = %d from consume()", result);
150 }
151 result = mConsumer->sendFinishedSignal(consumeSeq, true);
152 if (result != OK) {
153 ALOGE("Received result = %d from sendFinishedSignal", result);
154 }
155 }
156
157protected:
158 explicit FakeInputReceiver(const sp<InputDispatcher>& dispatcher, const std::string name)
159 : mDispatcher(dispatcher) {
Garfield Tan15601662020-09-22 15:32:38 -0700160 mClientChannel = *mDispatcher->createInputChannel(name);
Siarhei Vishniakoud0784762019-11-01 15:33:48 -0700161 mConsumer = std::make_unique<InputConsumer>(mClientChannel);
162 }
163
164 virtual ~FakeInputReceiver() {}
165
166 sp<InputDispatcher> mDispatcher;
Garfield Tan15601662020-09-22 15:32:38 -0700167 std::shared_ptr<InputChannel> mClientChannel;
Siarhei Vishniakoud0784762019-11-01 15:33:48 -0700168 std::unique_ptr<InputConsumer> mConsumer;
169 PreallocatedInputEventFactory mEventFactory;
170};
171
172class FakeWindowHandle : public InputWindowHandle, public FakeInputReceiver {
173public:
174 static const int32_t WIDTH = 200;
175 static const int32_t HEIGHT = 200;
176
Chris Yea209fde2020-07-22 13:54:51 -0700177 FakeWindowHandle(const std::shared_ptr<InputApplicationHandle>& inputApplicationHandle,
Siarhei Vishniakoud0784762019-11-01 15:33:48 -0700178 const sp<InputDispatcher>& dispatcher, const std::string name)
179 : FakeInputReceiver(dispatcher, name), mFrame(Rect(0, 0, WIDTH, HEIGHT)) {
Siarhei Vishniakoud0784762019-11-01 15:33:48 -0700180 inputApplicationHandle->updateInfo();
181 mInfo.applicationInfo = *inputApplicationHandle->getInfo();
182 }
183
184 virtual bool updateInfo() override {
Garfield Tan15601662020-09-22 15:32:38 -0700185 mInfo.token = mClientChannel->getConnectionToken();
Siarhei Vishniakoud0784762019-11-01 15:33:48 -0700186 mInfo.name = "FakeWindowHandle";
Michael Wright44753b12020-07-08 13:48:11 +0100187 mInfo.type = InputWindowInfo::Type::APPLICATION;
Siarhei Vishniakouc1ae5562020-06-30 14:22:57 -0500188 mInfo.dispatchingTimeout = DISPATCHING_TIMEOUT;
Siarhei Vishniakoud0784762019-11-01 15:33:48 -0700189 mInfo.frameLeft = mFrame.left;
190 mInfo.frameTop = mFrame.top;
191 mInfo.frameRight = mFrame.right;
192 mInfo.frameBottom = mFrame.bottom;
193 mInfo.globalScaleFactor = 1.0;
194 mInfo.touchableRegion.clear();
195 mInfo.addTouchableRegion(mFrame);
196 mInfo.visible = true;
Vishnu Nair47074b82020-08-14 11:54:47 -0700197 mInfo.focusable = true;
Siarhei Vishniakoud0784762019-11-01 15:33:48 -0700198 mInfo.hasWallpaper = false;
199 mInfo.paused = false;
Siarhei Vishniakoud0784762019-11-01 15:33:48 -0700200 mInfo.ownerPid = INJECTOR_PID;
201 mInfo.ownerUid = INJECTOR_UID;
Siarhei Vishniakoud0784762019-11-01 15:33:48 -0700202 mInfo.displayId = ADISPLAY_ID_DEFAULT;
203
204 return true;
205 }
206
207protected:
208 Rect mFrame;
209};
210
211static MotionEvent generateMotionEvent() {
212 PointerProperties pointerProperties[1];
213 PointerCoords pointerCoords[1];
214
215 pointerProperties[0].clear();
216 pointerProperties[0].id = 0;
217 pointerProperties[0].toolType = AMOTION_EVENT_TOOL_TYPE_FINGER;
218
219 pointerCoords[0].clear();
220 pointerCoords[0].setAxisValue(AMOTION_EVENT_AXIS_X, 100);
221 pointerCoords[0].setAxisValue(AMOTION_EVENT_AXIS_Y, 100);
222
223 const nsecs_t currentTime = now();
224
chaviw9eaa22c2020-07-01 16:21:27 -0700225 ui::Transform identityTransform;
Siarhei Vishniakoud0784762019-11-01 15:33:48 -0700226 MotionEvent event;
Garfield Tanfbe732e2020-01-24 11:26:14 -0800227 event.initialize(InputEvent::nextId(), DEVICE_ID, AINPUT_SOURCE_TOUCHSCREEN,
228 ADISPLAY_ID_DEFAULT, INVALID_HMAC, AMOTION_EVENT_ACTION_DOWN,
229 /* actionButton */ 0, /* flags */ 0,
Siarhei Vishniakoud0784762019-11-01 15:33:48 -0700230 /* edgeFlags */ 0, AMETA_NONE, /* buttonState */ 0, MotionClassification::NONE,
chaviw9eaa22c2020-07-01 16:21:27 -0700231 identityTransform, /* xPrecision */ 0,
Siarhei Vishniakoud0784762019-11-01 15:33:48 -0700232 /* yPrecision */ 0, AMOTION_EVENT_INVALID_CURSOR_POSITION,
233 AMOTION_EVENT_INVALID_CURSOR_POSITION, currentTime, currentTime,
234 /*pointerCount*/ 1, pointerProperties, pointerCoords);
235 return event;
236}
237
238static NotifyMotionArgs generateMotionArgs() {
239 PointerProperties pointerProperties[1];
240 PointerCoords pointerCoords[1];
241
242 pointerProperties[0].clear();
243 pointerProperties[0].id = 0;
244 pointerProperties[0].toolType = AMOTION_EVENT_TOOL_TYPE_FINGER;
245
246 pointerCoords[0].clear();
247 pointerCoords[0].setAxisValue(AMOTION_EVENT_AXIS_X, 100);
248 pointerCoords[0].setAxisValue(AMOTION_EVENT_AXIS_Y, 100);
249
250 const nsecs_t currentTime = now();
251 // Define a valid motion event.
Garfield Tanc51d1ba2020-01-28 13:24:04 -0800252 NotifyMotionArgs args(/* id */ 0, currentTime, DEVICE_ID, AINPUT_SOURCE_TOUCHSCREEN,
Siarhei Vishniakoud0784762019-11-01 15:33:48 -0700253 ADISPLAY_ID_DEFAULT, POLICY_FLAG_PASS_TO_USER, AMOTION_EVENT_ACTION_DOWN,
254 /* actionButton */ 0, /* flags */ 0, AMETA_NONE, /* buttonState */ 0,
255 MotionClassification::NONE, AMOTION_EVENT_EDGE_FLAG_NONE, 1,
256 pointerProperties, pointerCoords,
257 /* xPrecision */ 0, /* yPrecision */ 0,
258 AMOTION_EVENT_INVALID_CURSOR_POSITION,
259 AMOTION_EVENT_INVALID_CURSOR_POSITION, currentTime, /* videoFrames */ {});
260
261 return args;
262}
263
264static void benchmarkNotifyMotion(benchmark::State& state) {
265 // Create dispatcher
266 sp<FakeInputDispatcherPolicy> fakePolicy = new FakeInputDispatcherPolicy();
267 sp<InputDispatcher> dispatcher = new InputDispatcher(fakePolicy);
268 dispatcher->setInputDispatchMode(/*enabled*/ true, /*frozen*/ false);
269 dispatcher->start();
270
271 // Create a window that will receive motion events
Chris Yea209fde2020-07-22 13:54:51 -0700272 std::shared_ptr<FakeApplicationHandle> application = std::make_shared<FakeApplicationHandle>();
Siarhei Vishniakoud0784762019-11-01 15:33:48 -0700273 sp<FakeWindowHandle> window = new FakeWindowHandle(application, dispatcher, "Fake Window");
274
Arthur Hung72d8dc32020-03-28 00:48:39 +0000275 dispatcher->setInputWindows({{ADISPLAY_ID_DEFAULT, {window}}});
Siarhei Vishniakoud0784762019-11-01 15:33:48 -0700276
277 NotifyMotionArgs motionArgs = generateMotionArgs();
278
279 for (auto _ : state) {
280 // Send ACTION_DOWN
281 motionArgs.action = AMOTION_EVENT_ACTION_DOWN;
Garfield Tanc51d1ba2020-01-28 13:24:04 -0800282 motionArgs.id = 0;
Siarhei Vishniakoud0784762019-11-01 15:33:48 -0700283 motionArgs.downTime = now();
284 motionArgs.eventTime = motionArgs.downTime;
285 dispatcher->notifyMotion(&motionArgs);
286
287 // Send ACTION_UP
288 motionArgs.action = AMOTION_EVENT_ACTION_UP;
Garfield Tanc51d1ba2020-01-28 13:24:04 -0800289 motionArgs.id = 1;
Siarhei Vishniakoud0784762019-11-01 15:33:48 -0700290 motionArgs.eventTime = now();
291 dispatcher->notifyMotion(&motionArgs);
292
293 window->consumeEvent();
294 window->consumeEvent();
295 }
296
297 dispatcher->stop();
298}
299
300static void benchmarkInjectMotion(benchmark::State& state) {
301 // Create dispatcher
302 sp<FakeInputDispatcherPolicy> fakePolicy = new FakeInputDispatcherPolicy();
303 sp<InputDispatcher> dispatcher = new InputDispatcher(fakePolicy);
304 dispatcher->setInputDispatchMode(/*enabled*/ true, /*frozen*/ false);
305 dispatcher->start();
306
307 // Create a window that will receive motion events
Chris Yea209fde2020-07-22 13:54:51 -0700308 std::shared_ptr<FakeApplicationHandle> application = std::make_shared<FakeApplicationHandle>();
Siarhei Vishniakoud0784762019-11-01 15:33:48 -0700309 sp<FakeWindowHandle> window = new FakeWindowHandle(application, dispatcher, "Fake Window");
310
Arthur Hung72d8dc32020-03-28 00:48:39 +0000311 dispatcher->setInputWindows({{ADISPLAY_ID_DEFAULT, {window}}});
Siarhei Vishniakoud0784762019-11-01 15:33:48 -0700312
Siarhei Vishniakoud0784762019-11-01 15:33:48 -0700313 for (auto _ : state) {
Siarhei Vishniakouadfd4fa2019-12-20 11:02:58 -0800314 MotionEvent event = generateMotionEvent();
Siarhei Vishniakoud0784762019-11-01 15:33:48 -0700315 // Send ACTION_DOWN
Siarhei Vishniakoud0784762019-11-01 15:33:48 -0700316 dispatcher->injectInputEvent(&event, INJECTOR_PID, INJECTOR_UID,
Siarhei Vishniakouae6229e2019-12-30 16:23:19 -0800317 InputEventInjectionSync::NONE, INJECT_EVENT_TIMEOUT,
Siarhei Vishniakoud0784762019-11-01 15:33:48 -0700318 POLICY_FLAG_FILTERED | POLICY_FLAG_PASS_TO_USER);
319
320 // Send ACTION_UP
321 event.setAction(AMOTION_EVENT_ACTION_UP);
322 dispatcher->injectInputEvent(&event, INJECTOR_PID, INJECTOR_UID,
Siarhei Vishniakouae6229e2019-12-30 16:23:19 -0800323 InputEventInjectionSync::NONE, INJECT_EVENT_TIMEOUT,
Siarhei Vishniakoud0784762019-11-01 15:33:48 -0700324 POLICY_FLAG_FILTERED | POLICY_FLAG_PASS_TO_USER);
325
326 window->consumeEvent();
327 window->consumeEvent();
328 }
329
330 dispatcher->stop();
331}
332
333BENCHMARK(benchmarkNotifyMotion);
334BENCHMARK(benchmarkInjectMotion);
335
336} // namespace android::inputdispatcher
337
338BENCHMARK_MAIN();