• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /*
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 package android.view;
18 
19 import android.os.Build;
20 import android.util.Log;
21 
22 /**
23  * Checks whether a sequence of input events is self-consistent.
24  * Logs a description of each problem detected.
25  * <p>
26  * When a problem is detected, the event is tainted.  This mechanism prevents the same
27  * error from being reported multiple times.
28  * </p>
29  *
30  * @hide
31  */
32 public final class InputEventConsistencyVerifier {
33     private static final boolean IS_ENG_BUILD = "eng".equals(Build.TYPE);
34 
35     private static final String EVENT_TYPE_KEY = "KeyEvent";
36     private static final String EVENT_TYPE_TRACKBALL = "TrackballEvent";
37     private static final String EVENT_TYPE_TOUCH = "TouchEvent";
38     private static final String EVENT_TYPE_GENERIC_MOTION = "GenericMotionEvent";
39 
40     // The number of recent events to log when a problem is detected.
41     // Can be set to 0 to disable logging recent events but the runtime overhead of
42     // this feature is negligible on current hardware.
43     private static final int RECENT_EVENTS_TO_LOG = 5;
44 
45     // The object to which the verifier is attached.
46     private final Object mCaller;
47 
48     // Consistency verifier flags.
49     private final int mFlags;
50 
51     // Tag for logging which a client can set to help distinguish the output
52     // from different verifiers since several can be active at the same time.
53     // If not provided defaults to the simple class name.
54     private final String mLogTag;
55 
56     // The most recently checked event and the nesting level at which it was checked.
57     // This is only set when the verifier is called from a nesting level greater than 0
58     // so that the verifier can detect when it has been asked to verify the same event twice.
59     // It does not make sense to examine the contents of the last event since it may have
60     // been recycled.
61     private int mLastEventSeq;
62     private String mLastEventType;
63     private int mLastNestingLevel;
64 
65     // Copy of the most recent events.
66     private InputEvent[] mRecentEvents;
67     private boolean[] mRecentEventsUnhandled;
68     private int mMostRecentEventIndex;
69 
70     // Current event and its type.
71     private InputEvent mCurrentEvent;
72     private String mCurrentEventType;
73 
74     // Linked list of key state objects.
75     private KeyState mKeyStateList;
76 
77     // Current state of the trackball.
78     private boolean mTrackballDown;
79     private boolean mTrackballUnhandled;
80 
81     // Bitfield of pointer ids that are currently down.
82     // Assumes that the largest possible pointer id is 31, which is potentially subject to change.
83     // (See MAX_POINTER_ID in frameworks/base/include/ui/Input.h)
84     private int mTouchEventStreamPointers;
85 
86     // The device id and source of the current stream of touch events.
87     private int mTouchEventStreamDeviceId = -1;
88     private int mTouchEventStreamSource;
89 
90     // Set to true when we discover that the touch event stream is inconsistent.
91     // Reset on down or cancel.
92     private boolean mTouchEventStreamIsTainted;
93 
94     // Set to true if the touch event stream is partially unhandled.
95     private boolean mTouchEventStreamUnhandled;
96 
97     // Set to true if we received hover enter.
98     private boolean mHoverEntered;
99 
100     // The current violation message.
101     private StringBuilder mViolationMessage;
102 
103     /**
104      * Indicates that the verifier is intended to act on raw device input event streams.
105      * Disables certain checks for invariants that are established by the input dispatcher
106      * itself as it delivers input events, such as key repeating behavior.
107      */
108     public static final int FLAG_RAW_DEVICE_INPUT = 1 << 0;
109 
110     /**
111      * Creates an input consistency verifier.
112      * @param caller The object to which the verifier is attached.
113      * @param flags Flags to the verifier, or 0 if none.
114      */
InputEventConsistencyVerifier(Object caller, int flags)115     public InputEventConsistencyVerifier(Object caller, int flags) {
116         this(caller, flags, InputEventConsistencyVerifier.class.getSimpleName());
117     }
118 
119     /**
120      * Creates an input consistency verifier.
121      * @param caller The object to which the verifier is attached.
122      * @param flags Flags to the verifier, or 0 if none.
123      * @param logTag Tag for logging. If null defaults to the short class name.
124      */
InputEventConsistencyVerifier(Object caller, int flags, String logTag)125     public InputEventConsistencyVerifier(Object caller, int flags, String logTag) {
126         this.mCaller = caller;
127         this.mFlags = flags;
128         this.mLogTag = (logTag != null) ? logTag : "InputEventConsistencyVerifier";
129     }
130 
131     /**
132      * Determines whether the instrumentation should be enabled.
133      * @return True if it should be enabled.
134      */
isInstrumentationEnabled()135     public static boolean isInstrumentationEnabled() {
136         return IS_ENG_BUILD;
137     }
138 
139     /**
140      * Resets the state of the input event consistency verifier.
141      */
reset()142     public void reset() {
143         mLastEventSeq = -1;
144         mLastNestingLevel = 0;
145         mTrackballDown = false;
146         mTrackballUnhandled = false;
147         mTouchEventStreamPointers = 0;
148         mTouchEventStreamIsTainted = false;
149         mTouchEventStreamUnhandled = false;
150         mHoverEntered = false;
151 
152         while (mKeyStateList != null) {
153             final KeyState state = mKeyStateList;
154             mKeyStateList = state.next;
155             state.recycle();
156         }
157     }
158 
159     /**
160      * Checks an arbitrary input event.
161      * @param event The event.
162      * @param nestingLevel The nesting level: 0 if called from the base class,
163      * or 1 from a subclass.  If the event was already checked by this consistency verifier
164      * at a higher nesting level, it will not be checked again.  Used to handle the situation
165      * where a subclass dispatching method delegates to its superclass's dispatching method
166      * and both dispatching methods call into the consistency verifier.
167      */
onInputEvent(InputEvent event, int nestingLevel)168     public void onInputEvent(InputEvent event, int nestingLevel) {
169         if (event instanceof KeyEvent) {
170             final KeyEvent keyEvent = (KeyEvent)event;
171             onKeyEvent(keyEvent, nestingLevel);
172         } else {
173             final MotionEvent motionEvent = (MotionEvent)event;
174             if (motionEvent.isTouchEvent()) {
175                 onTouchEvent(motionEvent, nestingLevel);
176             } else if ((motionEvent.getSource() & InputDevice.SOURCE_CLASS_TRACKBALL) != 0) {
177                 onTrackballEvent(motionEvent, nestingLevel);
178             } else {
179                 onGenericMotionEvent(motionEvent, nestingLevel);
180             }
181         }
182     }
183 
184     /**
185      * Checks a key event.
186      * @param event The event.
187      * @param nestingLevel The nesting level: 0 if called from the base class,
188      * or 1 from a subclass.  If the event was already checked by this consistency verifier
189      * at a higher nesting level, it will not be checked again.  Used to handle the situation
190      * where a subclass dispatching method delegates to its superclass's dispatching method
191      * and both dispatching methods call into the consistency verifier.
192      */
onKeyEvent(KeyEvent event, int nestingLevel)193     public void onKeyEvent(KeyEvent event, int nestingLevel) {
194         if (!startEvent(event, nestingLevel, EVENT_TYPE_KEY)) {
195             return;
196         }
197 
198         try {
199             ensureMetaStateIsNormalized(event.getMetaState());
200 
201             final int action = event.getAction();
202             final int deviceId = event.getDeviceId();
203             final int source = event.getSource();
204             final int keyCode = event.getKeyCode();
205             switch (action) {
206                 case KeyEvent.ACTION_DOWN: {
207                     KeyState state = findKeyState(deviceId, source, keyCode, /*remove*/ false);
208                     if (state != null) {
209                         // If the key is already down, ensure it is a repeat.
210                         // We don't perform this check when processing raw device input
211                         // because the input dispatcher itself is responsible for setting
212                         // the key repeat count before it delivers input events.
213                         if (state.unhandled) {
214                             state.unhandled = false;
215                         } else if ((mFlags & FLAG_RAW_DEVICE_INPUT) == 0
216                                 && event.getRepeatCount() == 0) {
217                             problem("ACTION_DOWN but key is already down and this event "
218                                     + "is not a key repeat.");
219                         }
220                     } else {
221                         addKeyState(deviceId, source, keyCode);
222                     }
223                     break;
224                 }
225                 case KeyEvent.ACTION_UP: {
226                     KeyState state = findKeyState(deviceId, source, keyCode, /*remove*/ true);
227                     if (state == null) {
228                         problem("ACTION_UP but key was not down.");
229                     } else {
230                         state.recycle();
231                     }
232                     break;
233                 }
234                 case KeyEvent.ACTION_MULTIPLE:
235                     break;
236                 default:
237                     problem("Invalid action " + KeyEvent.actionToString(action)
238                             + " for key event.");
239                     break;
240             }
241         } finally {
242             finishEvent();
243         }
244     }
245 
246     /**
247      * Checks a trackball event.
248      * @param event The event.
249      * @param nestingLevel The nesting level: 0 if called from the base class,
250      * or 1 from a subclass.  If the event was already checked by this consistency verifier
251      * at a higher nesting level, it will not be checked again.  Used to handle the situation
252      * where a subclass dispatching method delegates to its superclass's dispatching method
253      * and both dispatching methods call into the consistency verifier.
254      */
onTrackballEvent(MotionEvent event, int nestingLevel)255     public void onTrackballEvent(MotionEvent event, int nestingLevel) {
256         if (!startEvent(event, nestingLevel, EVENT_TYPE_TRACKBALL)) {
257             return;
258         }
259 
260         try {
261             ensureMetaStateIsNormalized(event.getMetaState());
262 
263             final int action = event.getAction();
264             final int source = event.getSource();
265             if ((source & InputDevice.SOURCE_CLASS_TRACKBALL) != 0) {
266                 switch (action) {
267                     case MotionEvent.ACTION_DOWN:
268                         if (mTrackballDown && !mTrackballUnhandled) {
269                             problem("ACTION_DOWN but trackball is already down.");
270                         } else {
271                             mTrackballDown = true;
272                             mTrackballUnhandled = false;
273                         }
274                         ensureHistorySizeIsZeroForThisAction(event);
275                         ensurePointerCountIsOneForThisAction(event);
276                         break;
277                     case MotionEvent.ACTION_UP:
278                         if (!mTrackballDown) {
279                             problem("ACTION_UP but trackball is not down.");
280                         } else {
281                             mTrackballDown = false;
282                             mTrackballUnhandled = false;
283                         }
284                         ensureHistorySizeIsZeroForThisAction(event);
285                         ensurePointerCountIsOneForThisAction(event);
286                         break;
287                     case MotionEvent.ACTION_MOVE:
288                         ensurePointerCountIsOneForThisAction(event);
289                         break;
290                     default:
291                         problem("Invalid action " + MotionEvent.actionToString(action)
292                                 + " for trackball event.");
293                         break;
294                 }
295 
296                 if (mTrackballDown && event.getPressure() <= 0) {
297                     problem("Trackball is down but pressure is not greater than 0.");
298                 } else if (!mTrackballDown && event.getPressure() != 0) {
299                     problem("Trackball is up but pressure is not equal to 0.");
300                 }
301             } else {
302                 problem("Source was not SOURCE_CLASS_TRACKBALL.");
303             }
304         } finally {
305             finishEvent();
306         }
307     }
308 
309     /**
310      * Checks a touch event.
311      * @param event The event.
312      * @param nestingLevel The nesting level: 0 if called from the base class,
313      * or 1 from a subclass.  If the event was already checked by this consistency verifier
314      * at a higher nesting level, it will not be checked again.  Used to handle the situation
315      * where a subclass dispatching method delegates to its superclass's dispatching method
316      * and both dispatching methods call into the consistency verifier.
317      */
onTouchEvent(MotionEvent event, int nestingLevel)318     public void onTouchEvent(MotionEvent event, int nestingLevel) {
319         if (!startEvent(event, nestingLevel, EVENT_TYPE_TOUCH)) {
320             return;
321         }
322 
323         final int action = event.getAction();
324         final boolean newStream = action == MotionEvent.ACTION_DOWN
325                 || action == MotionEvent.ACTION_CANCEL;
326         if (newStream && (mTouchEventStreamIsTainted || mTouchEventStreamUnhandled)) {
327             mTouchEventStreamIsTainted = false;
328             mTouchEventStreamUnhandled = false;
329             mTouchEventStreamPointers = 0;
330         }
331         if (mTouchEventStreamIsTainted) {
332             event.setTainted(true);
333         }
334 
335         try {
336             ensureMetaStateIsNormalized(event.getMetaState());
337 
338             final int deviceId = event.getDeviceId();
339             final int source = event.getSource();
340 
341             if (!newStream && mTouchEventStreamDeviceId != -1
342                     && (mTouchEventStreamDeviceId != deviceId
343                             || mTouchEventStreamSource != source)) {
344                 problem("Touch event stream contains events from multiple sources: "
345                         + "previous device id " + mTouchEventStreamDeviceId
346                         + ", previous source " + Integer.toHexString(mTouchEventStreamSource)
347                         + ", new device id " + deviceId
348                         + ", new source " + Integer.toHexString(source));
349             }
350             mTouchEventStreamDeviceId = deviceId;
351             mTouchEventStreamSource = source;
352 
353             final int pointerCount = event.getPointerCount();
354             if ((source & InputDevice.SOURCE_CLASS_POINTER) != 0) {
355                 switch (action) {
356                     case MotionEvent.ACTION_DOWN:
357                         if (mTouchEventStreamPointers != 0) {
358                             problem("ACTION_DOWN but pointers are already down.  "
359                                     + "Probably missing ACTION_UP from previous gesture.");
360                         }
361                         ensureHistorySizeIsZeroForThisAction(event);
362                         ensurePointerCountIsOneForThisAction(event);
363                         mTouchEventStreamPointers = 1 << event.getPointerId(0);
364                         break;
365                     case MotionEvent.ACTION_UP:
366                         ensureHistorySizeIsZeroForThisAction(event);
367                         ensurePointerCountIsOneForThisAction(event);
368                         mTouchEventStreamPointers = 0;
369                         mTouchEventStreamIsTainted = false;
370                         break;
371                     case MotionEvent.ACTION_MOVE: {
372                         final int expectedPointerCount =
373                                 Integer.bitCount(mTouchEventStreamPointers);
374                         if (pointerCount != expectedPointerCount) {
375                             problem("ACTION_MOVE contained " + pointerCount
376                                     + " pointers but there are currently "
377                                     + expectedPointerCount + " pointers down.");
378                             mTouchEventStreamIsTainted = true;
379                         }
380                         break;
381                     }
382                     case MotionEvent.ACTION_CANCEL:
383                         mTouchEventStreamPointers = 0;
384                         mTouchEventStreamIsTainted = false;
385                         break;
386                     case MotionEvent.ACTION_OUTSIDE:
387                         if (mTouchEventStreamPointers != 0) {
388                             problem("ACTION_OUTSIDE but pointers are still down.");
389                         }
390                         ensureHistorySizeIsZeroForThisAction(event);
391                         ensurePointerCountIsOneForThisAction(event);
392                         mTouchEventStreamIsTainted = false;
393                         break;
394                     default: {
395                         final int actionMasked = event.getActionMasked();
396                         final int actionIndex = event.getActionIndex();
397                         if (actionMasked == MotionEvent.ACTION_POINTER_DOWN) {
398                             if (mTouchEventStreamPointers == 0) {
399                                 problem("ACTION_POINTER_DOWN but no other pointers were down.");
400                                 mTouchEventStreamIsTainted = true;
401                             }
402                             if (actionIndex < 0 || actionIndex >= pointerCount) {
403                                 problem("ACTION_POINTER_DOWN index is " + actionIndex
404                                         + " but the pointer count is " + pointerCount + ".");
405                                 mTouchEventStreamIsTainted = true;
406                             } else {
407                                 final int id = event.getPointerId(actionIndex);
408                                 final int idBit = 1 << id;
409                                 if ((mTouchEventStreamPointers & idBit) != 0) {
410                                     problem("ACTION_POINTER_DOWN specified pointer id " + id
411                                             + " which is already down.");
412                                     mTouchEventStreamIsTainted = true;
413                                 } else {
414                                     mTouchEventStreamPointers |= idBit;
415                                 }
416                             }
417                             ensureHistorySizeIsZeroForThisAction(event);
418                         } else if (actionMasked == MotionEvent.ACTION_POINTER_UP) {
419                             if (actionIndex < 0 || actionIndex >= pointerCount) {
420                                 problem("ACTION_POINTER_UP index is " + actionIndex
421                                         + " but the pointer count is " + pointerCount + ".");
422                                 mTouchEventStreamIsTainted = true;
423                             } else {
424                                 final int id = event.getPointerId(actionIndex);
425                                 final int idBit = 1 << id;
426                                 if ((mTouchEventStreamPointers & idBit) == 0) {
427                                     problem("ACTION_POINTER_UP specified pointer id " + id
428                                             + " which is not currently down.");
429                                     mTouchEventStreamIsTainted = true;
430                                 } else {
431                                     mTouchEventStreamPointers &= ~idBit;
432                                 }
433                             }
434                             ensureHistorySizeIsZeroForThisAction(event);
435                         } else {
436                             problem("Invalid action " + MotionEvent.actionToString(action)
437                                     + " for touch event.");
438                         }
439                         break;
440                     }
441                 }
442             } else {
443                 problem("Source was not SOURCE_CLASS_POINTER.");
444             }
445         } finally {
446             finishEvent();
447         }
448     }
449 
450     /**
451      * Checks a generic motion event.
452      * @param event The event.
453      * @param nestingLevel The nesting level: 0 if called from the base class,
454      * or 1 from a subclass.  If the event was already checked by this consistency verifier
455      * at a higher nesting level, it will not be checked again.  Used to handle the situation
456      * where a subclass dispatching method delegates to its superclass's dispatching method
457      * and both dispatching methods call into the consistency verifier.
458      */
onGenericMotionEvent(MotionEvent event, int nestingLevel)459     public void onGenericMotionEvent(MotionEvent event, int nestingLevel) {
460         if (!startEvent(event, nestingLevel, EVENT_TYPE_GENERIC_MOTION)) {
461             return;
462         }
463 
464         try {
465             ensureMetaStateIsNormalized(event.getMetaState());
466 
467             final int action = event.getAction();
468             final int source = event.getSource();
469             if ((source & InputDevice.SOURCE_CLASS_POINTER) != 0) {
470                 switch (action) {
471                     case MotionEvent.ACTION_HOVER_ENTER:
472                         ensurePointerCountIsOneForThisAction(event);
473                         mHoverEntered = true;
474                         break;
475                     case MotionEvent.ACTION_HOVER_MOVE:
476                         ensurePointerCountIsOneForThisAction(event);
477                         break;
478                     case MotionEvent.ACTION_HOVER_EXIT:
479                         ensurePointerCountIsOneForThisAction(event);
480                         if (!mHoverEntered) {
481                             problem("ACTION_HOVER_EXIT without prior ACTION_HOVER_ENTER");
482                         }
483                         mHoverEntered = false;
484                         break;
485                     case MotionEvent.ACTION_SCROLL:
486                         ensureHistorySizeIsZeroForThisAction(event);
487                         ensurePointerCountIsOneForThisAction(event);
488                         break;
489                     default:
490                         problem("Invalid action for generic pointer event.");
491                         break;
492                 }
493             } else if ((source & InputDevice.SOURCE_CLASS_JOYSTICK) != 0) {
494                 switch (action) {
495                     case MotionEvent.ACTION_MOVE:
496                         ensurePointerCountIsOneForThisAction(event);
497                         break;
498                     default:
499                         problem("Invalid action for generic joystick event.");
500                         break;
501                 }
502             }
503         } finally {
504             finishEvent();
505         }
506     }
507 
508     /**
509      * Notifies the verifier that a given event was unhandled and the rest of the
510      * trace for the event should be ignored.
511      * This method should only be called if the event was previously checked by
512      * the consistency verifier using {@link #onInputEvent} and other methods.
513      * @param event The event.
514      * @param nestingLevel The nesting level: 0 if called from the base class,
515      * or 1 from a subclass.  If the event was already checked by this consistency verifier
516      * at a higher nesting level, it will not be checked again.  Used to handle the situation
517      * where a subclass dispatching method delegates to its superclass's dispatching method
518      * and both dispatching methods call into the consistency verifier.
519      */
onUnhandledEvent(InputEvent event, int nestingLevel)520     public void onUnhandledEvent(InputEvent event, int nestingLevel) {
521         if (nestingLevel != mLastNestingLevel) {
522             return;
523         }
524 
525         if (mRecentEventsUnhandled != null) {
526             mRecentEventsUnhandled[mMostRecentEventIndex] = true;
527         }
528 
529         if (event instanceof KeyEvent) {
530             final KeyEvent keyEvent = (KeyEvent)event;
531             final int deviceId = keyEvent.getDeviceId();
532             final int source = keyEvent.getSource();
533             final int keyCode = keyEvent.getKeyCode();
534             final KeyState state = findKeyState(deviceId, source, keyCode, /*remove*/ false);
535             if (state != null) {
536                 state.unhandled = true;
537             }
538         } else {
539             final MotionEvent motionEvent = (MotionEvent)event;
540             if (motionEvent.isTouchEvent()) {
541                 mTouchEventStreamUnhandled = true;
542             } else if ((motionEvent.getSource() & InputDevice.SOURCE_CLASS_TRACKBALL) != 0) {
543                 if (mTrackballDown) {
544                     mTrackballUnhandled = true;
545                 }
546             }
547         }
548     }
549 
ensureMetaStateIsNormalized(int metaState)550     private void ensureMetaStateIsNormalized(int metaState) {
551         final int normalizedMetaState = KeyEvent.normalizeMetaState(metaState);
552         if (normalizedMetaState != metaState) {
553             problem(String.format("Metastate not normalized.  Was 0x%08x but expected 0x%08x.",
554                     metaState, normalizedMetaState));
555         }
556     }
557 
ensurePointerCountIsOneForThisAction(MotionEvent event)558     private void ensurePointerCountIsOneForThisAction(MotionEvent event) {
559         final int pointerCount = event.getPointerCount();
560         if (pointerCount != 1) {
561             problem("Pointer count is " + pointerCount + " but it should always be 1 for "
562                     + MotionEvent.actionToString(event.getAction()));
563         }
564     }
565 
ensureHistorySizeIsZeroForThisAction(MotionEvent event)566     private void ensureHistorySizeIsZeroForThisAction(MotionEvent event) {
567         final int historySize = event.getHistorySize();
568         if (historySize != 0) {
569             problem("History size is " + historySize + " but it should always be 0 for "
570                     + MotionEvent.actionToString(event.getAction()));
571         }
572     }
573 
startEvent(InputEvent event, int nestingLevel, String eventType)574     private boolean startEvent(InputEvent event, int nestingLevel, String eventType) {
575         // Ignore the event if we already checked it at a higher nesting level.
576         final int seq = event.getSequenceNumber();
577         if (seq == mLastEventSeq && nestingLevel < mLastNestingLevel
578                 && eventType == mLastEventType) {
579             return false;
580         }
581 
582         if (nestingLevel > 0) {
583             mLastEventSeq = seq;
584             mLastEventType = eventType;
585             mLastNestingLevel = nestingLevel;
586         } else {
587             mLastEventSeq = -1;
588             mLastEventType = null;
589             mLastNestingLevel = 0;
590         }
591 
592         mCurrentEvent = event;
593         mCurrentEventType = eventType;
594         return true;
595     }
596 
finishEvent()597     private void finishEvent() {
598         if (mViolationMessage != null && mViolationMessage.length() != 0) {
599             if (!mCurrentEvent.isTainted()) {
600                 // Write a log message only if the event was not already tainted.
601                 mViolationMessage.append("\n  in ").append(mCaller);
602                 mViolationMessage.append("\n  ");
603                 appendEvent(mViolationMessage, 0, mCurrentEvent, false);
604 
605                 if (RECENT_EVENTS_TO_LOG != 0 && mRecentEvents != null) {
606                     mViolationMessage.append("\n  -- recent events --");
607                     for (int i = 0; i < RECENT_EVENTS_TO_LOG; i++) {
608                         final int index = (mMostRecentEventIndex + RECENT_EVENTS_TO_LOG - i)
609                                 % RECENT_EVENTS_TO_LOG;
610                         final InputEvent event = mRecentEvents[index];
611                         if (event == null) {
612                             break;
613                         }
614                         mViolationMessage.append("\n  ");
615                         appendEvent(mViolationMessage, i + 1, event, mRecentEventsUnhandled[index]);
616                     }
617                 }
618 
619                 Log.d(mLogTag, mViolationMessage.toString());
620 
621                 // Taint the event so that we do not generate additional violations from it
622                 // further downstream.
623                 mCurrentEvent.setTainted(true);
624             }
625             mViolationMessage.setLength(0);
626         }
627 
628         if (RECENT_EVENTS_TO_LOG != 0) {
629             if (mRecentEvents == null) {
630                 mRecentEvents = new InputEvent[RECENT_EVENTS_TO_LOG];
631                 mRecentEventsUnhandled = new boolean[RECENT_EVENTS_TO_LOG];
632             }
633             final int index = (mMostRecentEventIndex + 1) % RECENT_EVENTS_TO_LOG;
634             mMostRecentEventIndex = index;
635             if (mRecentEvents[index] != null) {
636                 mRecentEvents[index].recycle();
637             }
638             mRecentEvents[index] = mCurrentEvent.copy();
639             mRecentEventsUnhandled[index] = false;
640         }
641 
642         mCurrentEvent = null;
643         mCurrentEventType = null;
644     }
645 
appendEvent(StringBuilder message, int index, InputEvent event, boolean unhandled)646     private static void appendEvent(StringBuilder message, int index,
647             InputEvent event, boolean unhandled) {
648         message.append(index).append(": sent at ").append(event.getEventTimeNano());
649         message.append(", ");
650         if (unhandled) {
651             message.append("(unhandled) ");
652         }
653         message.append(event);
654     }
655 
problem(String message)656     private void problem(String message) {
657         if (mViolationMessage == null) {
658             mViolationMessage = new StringBuilder();
659         }
660         if (mViolationMessage.length() == 0) {
661             mViolationMessage.append(mCurrentEventType).append(": ");
662         } else {
663             mViolationMessage.append("\n  ");
664         }
665         mViolationMessage.append(message);
666     }
667 
findKeyState(int deviceId, int source, int keyCode, boolean remove)668     private KeyState findKeyState(int deviceId, int source, int keyCode, boolean remove) {
669         KeyState last = null;
670         KeyState state = mKeyStateList;
671         while (state != null) {
672             if (state.deviceId == deviceId && state.source == source
673                     && state.keyCode == keyCode) {
674                 if (remove) {
675                     if (last != null) {
676                         last.next = state.next;
677                     } else {
678                         mKeyStateList = state.next;
679                     }
680                     state.next = null;
681                 }
682                 return state;
683             }
684             last = state;
685             state = state.next;
686         }
687         return null;
688     }
689 
addKeyState(int deviceId, int source, int keyCode)690     private void addKeyState(int deviceId, int source, int keyCode) {
691         KeyState state = KeyState.obtain(deviceId, source, keyCode);
692         state.next = mKeyStateList;
693         mKeyStateList = state;
694     }
695 
696     private static final class KeyState {
697         private static Object mRecycledListLock = new Object();
698         private static KeyState mRecycledList;
699 
700         public KeyState next;
701         public int deviceId;
702         public int source;
703         public int keyCode;
704         public boolean unhandled;
705 
KeyState()706         private KeyState() {
707         }
708 
obtain(int deviceId, int source, int keyCode)709         public static KeyState obtain(int deviceId, int source, int keyCode) {
710             KeyState state;
711             synchronized (mRecycledListLock) {
712                 state = mRecycledList;
713                 if (state != null) {
714                     mRecycledList = state.next;
715                 } else {
716                     state = new KeyState();
717                 }
718             }
719             state.deviceId = deviceId;
720             state.source = source;
721             state.keyCode = keyCode;
722             state.unhandled = false;
723             return state;
724         }
725 
recycle()726         public void recycle() {
727             synchronized (mRecycledListLock) {
728                 next = mRecycledList;
729                 mRecycledList = next;
730             }
731         }
732     }
733 }
734