InputManagerService分析一:IMS的啓動與事件傳遞

從這一節裏面,我們開始介紹InputManagerService部分的知識。它用於管理整個系統的輸入部分,包括鍵盤、鼠標、觸摸屏等等。這一章裏面我們主要就要介紹IMS。首先從IMS的啓動來分析:

[java]  view plain  copy
  1. HandlerThread wmHandlerThread = new HandlerThread("WindowManager");  
  2. wmHandlerThread.start();  
  3. Handler wmHandler = new Handler(wmHandlerThread.getLooper());    
  4.   
  5.     inputManager = new InputManagerService(context, wmHandler);             
  6.     wm = WindowManagerService.main(context, power, display, inputManager,  
  7.             wmHandler, factoryTest != SystemServer.FACTORY_TEST_LOW_LEVEL,  
  8.             !firstBoot, onlyCore);  
  9.     ServiceManager.addService(Context.WINDOW_SERVICE, wm);  
  10.     ServiceManager.addService(Context.INPUT_SERVICE, inputManager);  
  11.   
  12.     inputManager.setWindowManagerCallbacks(wm.getInputMonitor());  
  13.     inputManager.start();  


因爲 在Android系統中,按鍵事件是由InputManager來收集並由WindowManagerService服務來分發給各個Activity處理的,所以在介紹WMS的啓動時,我們也一起來介紹一下InputManager的啓動,首先來看InputManagerService的構造函數:

InputManagerService的啓動

[java]  view plain  copy
  1. public InputManagerService(Context context, Handler handler) {  
  2.     this.mContext = context;  
  3.     this.mHandler = new InputManagerHandler(handler.getLooper());  
  4.   
  5.     mUseDevInputEventForAudioJack =  
  6.             context.getResources().getBoolean(R.bool.config_useDevInputEventForAudioJack);  
  7.     mPtr = nativeInit(this, mContext, mHandler.getLooper().getQueue());  
  8. }  
這裏首先構造InputManagerHandler用於處理消息,當然這個handle是綁定在"WindowManager"這個looper上的。然後調用nativeInit去完成初natvie層的初始化:
[java]  view plain  copy
  1. static jint nativeInit(JNIEnv* env, jclass clazz,  
  2.         jobject serviceObj, jobject contextObj, jobject messageQueueObj) {  
  3.     sp<MessageQueue> messageQueue = android_os_MessageQueue_getMessageQueue(env, messageQueueObj);  
  4.   
  5.     NativeInputManager* im = new NativeInputManager(contextObj, serviceObj,  
  6.             messageQueue->getLooper());  
  7.     im->incStrong(0);  
  8.     return reinterpret_cast<jint>(im);  
  9. }  
android_os_MessageQueue_getMessageQueue這個函數我們在介紹Looper的時候已經分析過,因爲Java層的MessageQueue總是對應native層的NativeMessageQueue對象,所以首先先取得native層的messageQueue,並構造NativeInputManager對象:
[java]  view plain  copy
  1. NativeInputManager::NativeInputManager(jobject contextObj,  
  2.         jobject serviceObj, const sp<Looper>& looper) :  
  3.         mLooper(looper) {  
  4.     JNIEnv* env = jniEnv();  
  5.   
  6.     mContextObj = env->NewGlobalRef(contextObj);  
  7.     mServiceObj = env->NewGlobalRef(serviceObj);  
  8.   
  9.     {  
  10.         AutoMutex _l(mLock);  
  11.         mLocked.systemUiVisibility = ASYSTEM_UI_VISIBILITY_STATUS_BAR_VISIBLE;  
  12.         mLocked.pointerSpeed = 0;  
  13.         mLocked.pointerGesturesEnabled = true;  
  14.         mLocked.showTouches = false;  
  15.     }  
  16.   
  17.     sp<EventHub> eventHub = new EventHub();  
  18.     mInputManager = new InputManager(eventHub, thisthis);  
  19. }  
上面的InputManagerService的NativeInputManager的類圖關係如下:



在NativeInputManager的構造函數中,首先對mLocked中的一些變量賦初值。然後構造EventHub對象,並通過它最終構造InputManager對象。首先來看EventHub的構造函數:
[java]  view plain  copy
  1. EventHub::EventHub(void) :  
  2.         mBuiltInKeyboardId(NO_BUILT_IN_KEYBOARD), mNextDeviceId(1), mControllerNumbers(),  
  3.         mOpeningDevices(0), mClosingDevices(0),  
  4.         mNeedToSendFinishedDeviceScan(false),  
  5.         mNeedToReopenDevices(false), mNeedToScanDevices(true),  
  6.         mPendingEventCount(0), mPendingEventIndex(0), mPendingINotify(false) {  
  7.     acquire_wake_lock(PARTIAL_WAKE_LOCK, WAKE_LOCK_ID);  
  8.   
  9.     mEpollFd = epoll_create(EPOLL_SIZE_HINT);  
  10.     LOG_ALWAYS_FATAL_IF(mEpollFd < 0"Could not create epoll instance.  errno=%d", errno);  
  11.   
  12.     mINotifyFd = inotify_init();  
  13.     int result = inotify_add_watch(mINotifyFd, DEVICE_PATH, IN_DELETE | IN_CREATE);  
  14.     LOG_ALWAYS_FATAL_IF(result < 0"Could not register INotify for %s.  errno=%d",  
  15.             DEVICE_PATH, errno);  
  16.   
  17.     struct epoll_event eventItem;  
  18.     memset(&eventItem, 0, sizeof(eventItem));  
  19.     eventItem.events = EPOLLIN;  
  20.     eventItem.data.u32 = EPOLL_ID_INOTIFY;  
  21.     result = epoll_ctl(mEpollFd, EPOLL_CTL_ADD, mINotifyFd, &eventItem);  
  22.     LOG_ALWAYS_FATAL_IF(result != 0"Could not add INotify to epoll instance.  errno=%d", errno);  
  23.   
  24.     int wakeFds[2];  
  25.     result = pipe(wakeFds);  
  26.     LOG_ALWAYS_FATAL_IF(result != 0"Could not create wake pipe.  errno=%d", errno);  
  27.   
  28.     mWakeReadPipeFd = wakeFds[0];  
  29.     mWakeWritePipeFd = wakeFds[1];  
  30.   
  31.     result = fcntl(mWakeReadPipeFd, F_SETFL, O_NONBLOCK);  
  32.     LOG_ALWAYS_FATAL_IF(result != 0"Could not make wake read pipe non-blocking.  errno=%d",  
  33.             errno);  
  34.   
  35.     result = fcntl(mWakeWritePipeFd, F_SETFL, O_NONBLOCK);  
  36.     LOG_ALWAYS_FATAL_IF(result != 0"Could not make wake write pipe non-blocking.  errno=%d",  
  37.             errno);  
  38.   
  39.     eventItem.data.u32 = EPOLL_ID_WAKE;  
  40.     result = epoll_ctl(mEpollFd, EPOLL_CTL_ADD, mWakeReadPipeFd, &eventItem);  
  41.     LOG_ALWAYS_FATAL_IF(result != 0"Could not add wake read pipe to epoll instance.  errno=%d",  
  42.             errno);  
  43. }  

首先是對一些成員變量做初始化,這些變量等到我們用到時再來一一介紹。然後調用inotify機制對/dev/input文件目錄下的增和刪操作做監聽,/dev/input目錄就是所有的input設備文件。並調用epoll機制EPOLL_CTL_ADD命令對上面的inotify的句柄和mWakeReadPipeFd句柄加入到epoll的句柄mEpollFd中。我們可以看到當有/dev/input下面的增刪操作或者在mWakeWritePipeFd中寫數據時,epoll_wait將會返回。再來看InputManager的構造函數:

[java]  view plain  copy
  1. InputManager::InputManager(  
  2.         const sp<EventHubInterface>& eventHub,  
  3.         const sp<InputReaderPolicyInterface>& readerPolicy,  
  4.         const sp<InputDispatcherPolicyInterface>& dispatcherPolicy) {  
  5.     mDispatcher = new InputDispatcher(dispatcherPolicy);  
  6.     mReader = new InputReader(eventHub, readerPolicy, mDispatcher);  
  7.     initialize();  
  8. }  
  9.   
  10. void InputManager::initialize() {  
  11.     mReaderThread = new InputReaderThread(mReader);  
  12.     mDispatcherThread = new InputDispatcherThread(mDispatcher);  
  13. }  

這InputManager的構造函數中首先構造InputDispatcher和InputReader兩個對象,然後在initialize構造InputReaderThread和InputDispatcherThread分別用於收按鍵輸入和分發按鍵消息。InputDispatcher是分發消息的對象,InputReader是從驅動獲取按鍵消息,首先來看一下它們的構造函數:

[java]  view plain  copy
  1. InputDispatcher::InputDispatcher(const sp<InputDispatcherPolicyInterface>& policy) :  
  2.     mPolicy(policy),  
  3.     mPendingEvent(NULL), mAppSwitchSawKeyDown(false), mAppSwitchDueTime(LONG_LONG_MAX),  
  4.     mNextUnblockedEvent(NULL),  
  5.     mDispatchEnabled(false), mDispatchFrozen(false), mInputFilterEnabled(false),  
  6.     mInputTargetWaitCause(INPUT_TARGET_WAIT_CAUSE_NONE) {  
  7.     mLooper = new Looper(false);  
  8.   
  9.     mKeyRepeatState.lastKeyEntry = NULL;  
  10.   
  11.     policy->getDispatcherConfiguration(&mConfig);  
  12. }  
InputDispatcher的構造函數中首先初始化一些成員變量,然後構造一個Native層的Looper,並通過IMS獲取VirtualKeyQuietime和ExcludedDevice系統配置保存到InputDispatcherConfiguration選項。接着來看InputReader的構造函數:
[java]  view plain  copy
  1. InputReader::InputReader(const sp<EventHubInterface>& eventHub,  
  2.         const sp<InputReaderPolicyInterface>& policy,  
  3.         const sp<InputListenerInterface>& listener) :  
  4.         mContext(this), mEventHub(eventHub), mPolicy(policy),  
  5.         mGlobalMetaState(0), mGeneration(1),  
  6.         mDisableVirtualKeysTimeout(LLONG_MIN), mNextTimeout(LLONG_MAX),  
  7.         mConfigurationChangesToRefresh(0) {  
  8.     mQueuedListener = new QueuedInputListener(listener);  
  9.   
  10.     { // acquire lock  
  11.         AutoMutex _l(mLock);  
  12.   
  13.         refreshConfigurationLocked(0);  
  14.         updateGlobalMetaStateLocked();  
  15.     } // release lock  
  16. }  

這裏首先由傳入的InputDispatcher構造一個QueuedInputListener並保存在mQueuedListener中。回到InputManager的構造函數中,接着調用initialize構造分別構造InputReaderThread和InputDispatcherThread兩個thread。

InputReader和InputDispatcher兩個類都比較龐大,我們先來看一下它們的類圖:




再來看一下上面IMS構造過程的流程圖:




到這裏InputManagerService的構造函數就介紹完了,回到SystemService中接着調用WMS的main方法,並把前面創建的InputManagerService作爲參數傳入:
[java]  view plain  copy
  1. private WindowManagerService(Context context, PowerManagerService pm,  
  2.         DisplayManagerService displayManager, InputManagerService inputManager,  
  3.         boolean haveInputMethods, boolean showBootMsgs, boolean onlyCore) {  
  4.           
  5.     ......  
  6.       
  7.     mInputManager = inputManager; // Must be before createDisplayContentLocked.  
  8.   
  9.     mPointerEventDispatcher = new PointerEventDispatcher(mInputManager.monitorInput(TAG));  
  10.       
  11.     .....  
  12. }  

在WMS的構造函數中,首先在mInputManager中保存InputManagerService對象,然後構造PointerEventDispatcher對用於分析點擊事件。首先來看InputManagerService的monitorInput方法:

[java]  view plain  copy
  1. public InputChannel monitorInput(String inputChannelName) {  
  2.     InputChannel[] inputChannels = InputChannel.openInputChannelPair(inputChannelName);  
  3.     nativeRegisterInputChannel(mPtr, inputChannels[0], nulltrue);  
  4.     inputChannels[0].dispose(); // don't need to retain the Java object reference  
  5.     return inputChannels[1];  
  6. }  

先來看一下InputChannel的類圖結構:




InputChannel用於從InputDispatcher中後去所有的input消息,所以這裏構造一對InputChannel,一個註冊到底層的InputDispatcher中,一個用於PointerEventDispatcher給activity分發消息。先來看InputChannel的openInputChannelPair函數:
[java]  view plain  copy
  1.     public static InputChannel[] openInputChannelPair(String name) {  
  2.         return nativeOpenInputChannelPair(name);  
  3.     }  
  4.   
  5. static jobjectArray android_view_InputChannel_nativeOpenInputChannelPair(JNIEnv* env,  
  6.         jclass clazz, jstring nameObj) {  
  7.     const char* nameChars = env->GetStringUTFChars(nameObj, NULL);  
  8.     String8 name(nameChars);  
  9.     env->ReleaseStringUTFChars(nameObj, nameChars);  
  10.   
  11.     sp<InputChannel> serverChannel;  
  12.     sp<InputChannel> clientChannel;  
  13.     status_t result = InputChannel::openInputChannelPair(name, serverChannel, clientChannel);  
  14.   
  15.     if (result) {  
  16.         String8 message;  
  17.         message.appendFormat("Could not open input channel pair.  status=%d", result);  
  18.         jniThrowRuntimeException(env, message.string());  
  19.         return NULL;  
  20.     }  
  21.   
  22.     jobjectArray channelPair = env->NewObjectArray(2, gInputChannelClassInfo.clazz, NULL);  
  23.     if (env->ExceptionCheck()) {  
  24.         return NULL;  
  25.     }  
  26.   
  27.     jobject serverChannelObj = android_view_InputChannel_createInputChannel(env,  
  28.             new NativeInputChannel(serverChannel));  
  29.     if (env->ExceptionCheck()) {  
  30.         return NULL;  
  31.     }  
  32.   
  33.     jobject clientChannelObj = android_view_InputChannel_createInputChannel(env,  
  34.             new NativeInputChannel(clientChannel));  
  35.     if (env->ExceptionCheck()) {  
  36.         return NULL;  
  37.     }  
  38.   
  39.     env->SetObjectArrayElement(channelPair, 0, serverChannelObj);  
  40.     env->SetObjectArrayElement(channelPair, 1, clientChannelObj);  
  41.     return channelPair;  
  42. }  

android_view_InputChannel_nativeOpenInputChannelPair函數首先調用Native層的InputChannel的openInputChannelPair函數創建一對socket,先來看openInputChannelPair的實現:
[cpp]  view plain  copy
  1. status_t InputChannel::openInputChannelPair(const String8& name,  
  2.         sp<InputChannel>& outServerChannel, sp<InputChannel>& outClientChannel) {  
  3.     int sockets[2];  
  4.     if (socketpair(AF_UNIX, SOCK_SEQPACKET, 0, sockets)) {  
  5.         status_t result = -errno;  
  6.         ALOGE("channel '%s' ~ Could not create socket pair.  errno=%d",  
  7.                 name.string(), errno);  
  8.         outServerChannel.clear();  
  9.         outClientChannel.clear();  
  10.         return result;  
  11.     }  
  12.   
  13.     int bufferSize = SOCKET_BUFFER_SIZE;  
  14.     setsockopt(sockets[0], SOL_SOCKET, SO_SNDBUF, &bufferSize, sizeof(bufferSize));  
  15.     setsockopt(sockets[0], SOL_SOCKET, SO_RCVBUF, &bufferSize, sizeof(bufferSize));  
  16.     setsockopt(sockets[1], SOL_SOCKET, SO_SNDBUF, &bufferSize, sizeof(bufferSize));  
  17.     setsockopt(sockets[1], SOL_SOCKET, SO_RCVBUF, &bufferSize, sizeof(bufferSize));  
  18.   
  19.     String8 serverChannelName = name;  
  20.     serverChannelName.append(" (server)");  
  21.     outServerChannel = new InputChannel(serverChannelName, sockets[0]);  
  22.   
  23.     String8 clientChannelName = name;  
  24.     clientChannelName.append(" (client)");  
  25.     outClientChannel = new InputChannel(clientChannelName, sockets[1]);  
  26.     return OK;  
  27. }  
上面的代碼比較簡單,首先創建一對未命名、相互連接的UNIX域套接字,然後分別通過Fd分別構造兩個Native層的InputChannel代表server和client端。在android_view_InputChannel_nativeOpenInputChannelPair函數中把Native層的InputChannel和Java層的InputChannel通過NativeInputChannel相互綁定起來。

回到InputManagerService的monitorInput方法中,InputChannel的openInputChannelPair返回一對InputChannel對象,其中index爲0的代表server端,index爲1的代表client端。接着調用nativeRegisterInputChannel把InputChannel[0]到InputDispatcher,用於從InputDispatcher獲取觸摸事件:

[cpp]  view plain  copy
  1. static void nativeRegisterInputChannel(JNIEnv* env, jclass clazz,  
  2.         jint ptr, jobject inputChannelObj, jobject inputWindowHandleObj, jboolean monitor) {  
  3.     NativeInputManager* im = reinterpret_cast<NativeInputManager*>(ptr);  
  4.   
  5.     sp<InputChannel> inputChannel = android_view_InputChannel_getInputChannel(env,  
  6.             inputChannelObj);  
  7.     if (inputChannel == NULL) {  
  8.         throwInputChannelNotInitialized(env);  
  9.         return;  
  10.     }  
  11.   
  12.     sp<InputWindowHandle> inputWindowHandle =  
  13.             android_server_InputWindowHandle_getHandle(env, inputWindowHandleObj);  
  14.   
  15.     status_t status = im->registerInputChannel(  
  16.             env, inputChannel, inputWindowHandle, monitor);  
  17.     if (status) {  
  18.     }  
  19. }  
  20.   
  21. status_t NativeInputManager::registerInputChannel(JNIEnv* env,  
  22.         const sp<InputChannel>& inputChannel,  
  23.         const sp<InputWindowHandle>& inputWindowHandle, bool monitor) {  
  24.     return mInputManager->getDispatcher()->registerInputChannel(  
  25.             inputChannel, inputWindowHandle, monitor);  
  26. }  
nativeRegisterInputChannel首先通過Java層的InputChannel對象獲取到Native層的InputChannel對象,然後調用NativeInputManager的registerInputChannel方法,因爲當前沒有設置inputWindowhandler,所以這裏的InputWindowHandler爲NULL:

[cpp]  view plain  copy
  1. status_t InputDispatcher::registerInputChannel(const sp<InputChannel>& inputChannel,  
  2.         const sp<InputWindowHandle>& inputWindowHandle, bool monitor) {  
  3.     { // acquire lock  
  4.         AutoMutex _l(mLock);  
  5.   
  6.         if (getConnectionIndexLocked(inputChannel) >= 0) {  
  7.             ALOGW("Attempted to register already registered input channel '%s'",  
  8.                     inputChannel->getName().string());  
  9.             return BAD_VALUE;  
  10.         }  
  11.   
  12.         sp<Connection> connection = new Connection(inputChannel, inputWindowHandle, monitor);  
  13.   
  14.         int fd = inputChannel->getFd();  
  15.         mConnectionsByFd.add(fd, connection);  
  16.   
  17.         if (monitor) {  
  18.             mMonitoringChannels.push(inputChannel);  
  19.         }  
  20.   
  21.         mLooper->addFd(fd, 0, ALOOPER_EVENT_INPUT, handleReceiveCallback, this);  
  22.     } // release lock  
  23.   
  24.     // Wake the looper because some connections have changed.  
  25.             return BAD_VALUE;  
  26.         }  
  27.   
  28.         sp<Connection> connection = new Connection(inputChannel, inputWindowHandle, monitor);  
  29.   
  30.         int fd = inputChannel->getFd();  
  31.         mConnectionsByFd.add(fd, connection);  
  32.   
  33.         if (monitor) {  
  34.             mMonitoringChannels.push(inputChannel);  
  35.         }  
  36.   
  37.         mLooper->addFd(fd, 0, ALOOPER_EVENT_INPUT, handleReceiveCallback, this);  
  38.     } // release lock  
  39.   
  40.     // Wake the looper because some connections have changed.  
  41.     mLooper->wake();  
  42.     return OK;  
相關文章
相關標籤/搜索