From 51d45a710eff6da1749ae66ce9cab218144c388b Mon Sep 17 00:00:00 2001 From: Jeff Brown Date: Thu, 17 Jun 2010 20:52:56 -0700 Subject: More native input event dispatching. Added ANRs handling. Added event injection. Fixed a NPE ActivityManagerServer writing ANRs to the drop box. Fixed HOME key interception. Fixed trackball reporting. Fixed pointer rotation in landscape mode. Change-Id: I50340f559f22899ab924e220a78119ffc79469b7 --- include/ui/Input.h | 4 +- include/ui/InputDispatcher.h | 133 +++++++++++++++++++++++++++++++++++-------- include/ui/InputManager.h | 12 ++++ 3 files changed, 124 insertions(+), 25 deletions(-) (limited to 'include/ui') diff --git a/include/ui/Input.h b/include/ui/Input.h index 92ff872..979d6e8 100644 --- a/include/ui/Input.h +++ b/include/ui/Input.h @@ -58,8 +58,6 @@ struct RawEvent { /* * Flags that flow alongside events in the input dispatch system to help with certain * policy decisions such as waking from device sleep. - * - * TODO This enumeration should probably be split up or relabeled for clarity. */ enum { /* These flags originate in RawEvents and are generally set in the key map. */ @@ -73,6 +71,8 @@ enum { POLICY_FLAG_MENU = 0x00000040, POLICY_FLAG_LAUNCHER = 0x00000080, + POLICY_FLAG_RAW_MASK = 0x0000ffff, + /* These flags are set by the input reader policy as it intercepts each event. */ // Indicates that the screen was off when the event was received and the event diff --git a/include/ui/InputDispatcher.h b/include/ui/InputDispatcher.h index 80a20c9..511ad20 100644 --- a/include/ui/InputDispatcher.h +++ b/include/ui/InputDispatcher.h @@ -35,6 +35,28 @@ namespace android { /* + * Constants used to report the outcome of input event injection. + */ +enum { + /* (INTERNAL USE ONLY) Specifies that injection is pending and its outcome is unknown. */ + INPUT_EVENT_INJECTION_PENDING = -1, + + /* Injection succeeded. */ + INPUT_EVENT_INJECTION_SUCCEEDED = 0, + + /* Injection failed because the injector did not have permission to inject + * into the application with input focus. */ + INPUT_EVENT_INJECTION_PERMISSION_DENIED = 1, + + /* Injection failed because there were no available input targets. */ + INPUT_EVENT_INJECTION_FAILED = 2, + + /* Injection failed due to a timeout. */ + INPUT_EVENT_INJECTION_TIMED_OUT = 3 +}; + + +/* * An input target specifies how an input event is to be dispatched to a particular window * including the window's input channel, control flags, a timeout, and an X / Y offset to * be added to input event coordinates to compensate for the absolute position of the @@ -70,6 +92,7 @@ struct InputTarget { float xOffset, yOffset; }; + /* * Input dispatcher policy interface. * @@ -91,8 +114,11 @@ public: /* Notifies the system that an input channel is unrecoverably broken. */ virtual void notifyInputChannelBroken(const sp& inputChannel) = 0; - /* Notifies the system that an input channel is not responding. */ - virtual void notifyInputChannelANR(const sp& inputChannel) = 0; + /* Notifies the system that an input channel is not responding. + * Returns true and a new timeout value if the dispatcher should keep waiting. + * Otherwise returns false. */ + virtual bool notifyInputChannelANR(const sp& inputChannel, + nsecs_t& outNewTimeout) = 0; /* Notifies the system that an input channel recovered from ANR. */ virtual void notifyInputChannelRecoveredFromANR(const sp& inputChannel) = 0; @@ -100,12 +126,22 @@ public: /* Gets the key repeat timeout or -1 if automatic key repeating is disabled. */ virtual nsecs_t getKeyRepeatTimeout() = 0; - /* Gets the input targets for a key event. */ - virtual void getKeyEventTargets(KeyEvent* keyEvent, uint32_t policyFlags, + /* Gets the input targets for a key event. + * If the event is being injected, injectorPid and injectorUid should specify the + * process id and used id of the injecting application, otherwise they should both + * be -1. + * Returns one of the INPUT_EVENT_INJECTION_XXX constants. */ + virtual int32_t getKeyEventTargets(KeyEvent* keyEvent, uint32_t policyFlags, + int32_t injectorPid, int32_t injectorUid, Vector& outTargets) = 0; - /* Gets the input targets for a motion event. */ - virtual void getMotionEventTargets(MotionEvent* motionEvent, uint32_t policyFlags, + /* Gets the input targets for a motion event. + * If the event is being injected, injectorPid and injectorUid should specify the + * process id and used id of the injecting application, otherwise they should both + * be -1. + * Returns one of the INPUT_EVENT_INJECTION_XXX constants. */ + virtual int32_t getMotionEventTargets(MotionEvent* motionEvent, uint32_t policyFlags, + int32_t injectorPid, int32_t injectorUid, Vector& outTargets) = 0; }; @@ -139,6 +175,17 @@ public: uint32_t pointerCount, const int32_t* pointerIds, const PointerCoords* pointerCoords, float xPrecision, float yPrecision, nsecs_t downTime) = 0; + /* Injects an input event and optionally waits for sync. + * This method may block even if sync is false because it must wait for previous events + * to be dispatched before it can determine whether input event injection will be + * permitted based on the current input focus. + * Returns one of the INPUT_EVENT_INJECTION_XXX constants. + * + * This method may be called on any thread (usually by the input manager). + */ + virtual int32_t injectInputEvent(const InputEvent* event, + int32_t injectorPid, int32_t injectorUid, bool sync, int32_t timeoutMillis) = 0; + /* Registers or unregister input channels that may be used as targets for input events. * * These methods may be called on any thread (usually by the input manager). @@ -183,6 +230,9 @@ public: uint32_t pointerCount, const int32_t* pointerIds, const PointerCoords* pointerCoords, float xPrecision, float yPrecision, nsecs_t downTime); + virtual int32_t injectInputEvent(const InputEvent* event, + int32_t injectorPid, int32_t injectorUid, bool sync, int32_t timeoutMillis); + virtual status_t registerInputChannel(const sp& inputChannel); virtual status_t unregisterInputChannel(const sp& inputChannel); @@ -205,7 +255,13 @@ private: int32_t type; nsecs_t eventTime; + int32_t injectionResult; // initially INPUT_EVENT_INJECTION_PENDING + int32_t injectorPid; // -1 if not injected + int32_t injectorUid; // -1 if not injected + bool dispatchInProgress; // initially false, set to true while dispatching + + inline bool isInjected() { return injectorPid >= 0; } }; struct ConfigurationChangedEntry : EventEntry { @@ -293,6 +349,7 @@ private: struct CommandEntry; typedef void (InputDispatcher::*Command)(CommandEntry* commandEntry); + class Connection; struct CommandEntry : Link { CommandEntry(); ~CommandEntry(); @@ -300,7 +357,7 @@ private: Command command; // parameters for the command (usage varies by command) - sp inputChannel; + sp connection; }; // Generic queue implementation. @@ -353,9 +410,16 @@ private: public: Allocator(); - ConfigurationChangedEntry* obtainConfigurationChangedEntry(); - KeyEntry* obtainKeyEntry(); - MotionEntry* obtainMotionEntry(); + ConfigurationChangedEntry* obtainConfigurationChangedEntry(nsecs_t eventTime); + KeyEntry* obtainKeyEntry(nsecs_t eventTime, + int32_t deviceId, int32_t nature, uint32_t policyFlags, int32_t action, + int32_t flags, int32_t keyCode, int32_t scanCode, int32_t metaState, + int32_t repeatCount, nsecs_t downTime); + MotionEntry* obtainMotionEntry(nsecs_t eventTime, + int32_t deviceId, int32_t nature, uint32_t policyFlags, int32_t action, + int32_t metaState, int32_t edgeFlags, float xPrecision, float yPrecision, + nsecs_t downTime, uint32_t pointerCount, + const int32_t* pointerIds, const PointerCoords* pointerCoords); DispatchEntry* obtainDispatchEntry(EventEntry* eventEntry); CommandEntry* obtainCommandEntry(Command command); @@ -367,7 +431,7 @@ private: void releaseCommandEntry(CommandEntry* entry); void appendMotionSample(MotionEntry* motionEntry, - nsecs_t eventTime, int32_t pointerCount, const PointerCoords* pointerCoords); + nsecs_t eventTime, const PointerCoords* pointerCoords); private: Pool mConfigurationChangeEntryPool; @@ -376,6 +440,8 @@ private: Pool mMotionSamplePool; Pool mDispatchEntryPool; Pool mCommandEntryPool; + + void initializeEventEntry(EventEntry* entry, int32_t type, nsecs_t eventTime); }; /* Manages the dispatch state associated with a single input channel. */ @@ -439,6 +505,8 @@ private: } status_t initialize(); + + void setNextTimeoutTime(nsecs_t currentTime, nsecs_t timeout); }; sp mPolicy; @@ -455,8 +523,17 @@ private: KeyedVector > mConnectionsByReceiveFd; // Active connections are connections that have a non-empty outbound queue. + // We don't use a ref-counted pointer here because we explicitly abort connections + // during unregistration which causes the connection's outbound queue to be cleared + // and the connection itself to be deactivated. Vector mActiveConnections; + // List of connections that have timed out. Only used by dispatchOnce() + // We don't use a ref-counted pointer here because it is not possible for a connection + // to be unregistered while processing timed out connections since we hold the lock for + // the duration. + Vector mTimedOutConnections; + // Preallocated key and motion event objects used only to ask the input dispatcher policy // for the targets of an event that is to be dispatched. KeyEvent mReusableKeyEvent; @@ -468,6 +545,13 @@ private: Vector mCurrentInputTargets; bool mCurrentInputTargetsValid; // false while targets are being recomputed + // Event injection and synchronization. + Condition mInjectionResultAvailableCondition; + Condition mFullySynchronizedCondition; + bool isFullySynchronizedLocked(); + EventEntry* createEntryFromInputEventLocked(const InputEvent* event); + void setInjectionResultLocked(EventEntry* entry, int32_t injectionResult); + // Key repeat tracking. // XXX Move this up to the input reader instead. struct KeyRepeatState { @@ -500,13 +584,18 @@ private: nsecs_t currentTime, EventEntry* entry, bool resumeWithAppendedMotionSample); // Manage the dispatch cycle for a single connection. - void prepareDispatchCycleLocked(nsecs_t currentTime, Connection* connection, + // These methods are deliberately not Interruptible because doing all of the work + // with the mutex held makes it easier to ensure that connection invariants are maintained. + // If needed, the methods post commands to run later once the critical bits are done. + void prepareDispatchCycleLocked(nsecs_t currentTime, const sp& connection, EventEntry* eventEntry, const InputTarget* inputTarget, bool resumeWithAppendedMotionSample); - void startDispatchCycleLocked(nsecs_t currentTime, Connection* connection); - void finishDispatchCycleLocked(nsecs_t currentTime, Connection* connection); - bool timeoutDispatchCycleLocked(nsecs_t currentTime, Connection* connection); - bool abortDispatchCycleLocked(nsecs_t currentTime, Connection* connection, + void startDispatchCycleLocked(nsecs_t currentTime, const sp& connection); + void finishDispatchCycleLocked(nsecs_t currentTime, const sp& connection); + void timeoutDispatchCycleLocked(nsecs_t currentTime, const sp& connection); + void resumeAfterTimeoutDispatchCycleLocked(nsecs_t currentTime, + const sp& connection, nsecs_t newTimeout); + void abortDispatchCycleLocked(nsecs_t currentTime, const sp& connection, bool broken); static bool handleReceiveCallback(int receiveFd, int events, void* data); @@ -514,19 +603,17 @@ private: void activateConnectionLocked(Connection* connection); void deactivateConnectionLocked(Connection* connection); - // Outbound policy interactions. - void doNotifyConfigurationChangedLockedInterruptible(CommandEntry* commandEntry); - // Interesting events that we might like to log or tell the framework about. void onDispatchCycleStartedLocked( - nsecs_t currentTime, Connection* connection); + nsecs_t currentTime, const sp& connection); void onDispatchCycleFinishedLocked( - nsecs_t currentTime, Connection* connection, bool recoveredFromANR); + nsecs_t currentTime, const sp& connection, bool recoveredFromANR); void onDispatchCycleANRLocked( - nsecs_t currentTime, Connection* connection); + nsecs_t currentTime, const sp& connection); void onDispatchCycleBrokenLocked( - nsecs_t currentTime, Connection* connection); + nsecs_t currentTime, const sp& connection); + // Outbound policy interactions. void doNotifyInputChannelBrokenLockedInterruptible(CommandEntry* commandEntry); void doNotifyInputChannelANRLockedInterruptible(CommandEntry* commandEntry); void doNotifyInputChannelRecoveredFromANRLockedInterruptible(CommandEntry* commandEntry); diff --git a/include/ui/InputManager.h b/include/ui/InputManager.h index 3872c26..7509dd2 100644 --- a/include/ui/InputManager.h +++ b/include/ui/InputManager.h @@ -78,6 +78,15 @@ public: /* Unregisters an input channel. */ virtual status_t unregisterInputChannel(const sp& inputChannel) = 0; + /* Injects an input event and optionally waits for sync. + * This method may block even if sync is false because it must wait for previous events + * to be dispatched before it can determine whether input event injection will be + * permitted based on the current input focus. + * Returns one of the INPUT_EVENT_INJECTION_XXX constants. + */ + virtual int32_t injectInputEvent(const InputEvent* event, + int32_t injectorPid, int32_t injectorUid, bool sync, int32_t timeoutMillis) = 0; + /* Gets input device configuration. */ virtual void getInputConfiguration(InputConfiguration* outConfiguration) const = 0; @@ -118,6 +127,9 @@ public: virtual status_t registerInputChannel(const sp& inputChannel); virtual status_t unregisterInputChannel(const sp& inputChannel); + virtual int32_t injectInputEvent(const InputEvent* event, + int32_t injectorPid, int32_t injectorUid, bool sync, int32_t timeoutMillis); + virtual void getInputConfiguration(InputConfiguration* outConfiguration) const; virtual int32_t getScanCodeState(int32_t deviceId, int32_t deviceClasses, int32_t scanCode) const; -- cgit v1.1