[Input] socket连接的创建

前一篇文章,我们讲了Input ANR是怎么产生的[ANR]Input ANR是怎样产生的,着重讲的Input ANR的触发和判定原理。主要分析了system_server进程中的InputDispatcher线程的运行流程。这个线程主要负责事件的分发,通过socket将事件发送给App端进行处理。

system_server进程的InputDispatcher线程,与App端的主线程进行通信,需要先建立socket连接。这篇文章,我们讲讲socket连接的建立过程。

socketpair使用

socketpair方法,主要用于创建一对无名的、相互连接的套接字。

#include <sys/types.h> 
#include <sys/socket.h>

int socketpair(int domain, int type, int protocol, int sv[2]);

参数:

  • domain:协议家族
    • AF_LOCAL
    • AF_UNIX
  • type:套接字类型
    • SOCKET_STREAM:基于TCP
    • SOCKET_DGRAM:基于UDP
    • SOCKET_SEQPACKET:序列化包,提供一个序列化的、可靠的、双向的基本连接的数据传输通道,数据长度定长。
  • protocol:协议类型,只能是0
  • sv:返回的套接字对

返回值:

  • 0:成功
  • 1:失败

read方法

ssize_t read(int fd, void * buf, size_t count);

作用:将fd所指的文件传送count个字节到buf指针所指的内存中
返回值:

  • 返回实际读到的字节数
  • 0:表示读到文件尾,无刻度数据
  • 小于0:读取出错,需要看具体的errno

write方法

ssize_t write (int fd, const void * buf, size_t count); 

作用:将buf所指的内存写入count个字节到参数fd所指的文件中。
返回值:

  • 返回实际写入的字节数
  • -1:发生错误,需要看具体的errno。

这里先简单介绍一下read和write方法,之后再出文章详细介绍UNIX域套接字。

App跨进程调用WMS

主线程创建Activity的时候,会调用setContentView,最后会调用到ViewRootImplsetView方法。

public void setView(View view, WindowManager.LayoutParams attrs, View panelParentView) {
    synchronized (this) {
      ...
      if ((mWindowAttributes.inputFeatures
          & WindowManager.LayoutParams.INPUT_FEATURE_NO_INPUT_CHANNEL) == 0) {
          mInputChannel = new InputChannel(); //创建InputChannel对象
      }
      //通过Binder调用,进入system进程的Session
      res = mWindowSession.addToDisplay(mWindow, mSeq, mWindowAttributes,
                  getHostVisibility(), mDisplay.getDisplayId(),
                  mAttachInfo.mContentInsets, mAttachInfo.mStableInsets,
                  mAttachInfo.mOutsets, mInputChannel);
      ...
      if (mInputChannel != null) {
          if (mInputQueueCallback != null) {
              mInputQueue = new InputQueue();
              mInputQueueCallback.onInputQueueCreated(mInputQueue);
          }
          //创建WindowInputEventReceiver对象,并且传入刚刚创建好的mInputChannel
          mInputEventReceiver = new WindowInputEventReceiver(mInputChannel,  Looper.myLooper());
      }
    }
}

这个方法,主要是一个跨进程的Binder调用,最后调用的WMS中的addWinow方法。

WMS生成两个InputChannel

public int addWindow(Session session, IWindow client, int seq, WindowManager.LayoutParams attrs, int viewVisibility, int displayId, Rect outContentInsets, Rect outStableInsets, Rect outOutsets, InputChannel outInputChannel) {
        //创建一对InputChannel
        InputChannel[] inputChannels = InputChannel.openInputChannelPair(name);
        //将socket服务端保存到WindowState的mInputChannel
        win.setInputChannel(inputChannels[0]);
        //socket客户端传递给outInputChannel,最后会作为跨进程调用的返回值,传递给App端
        inputChannels[1].transferTo(outInputChannel);
        //利用socket服务端作为参数,注册到system_server的IMS中
        mInputManager.registerInputChannel(win.mInputChannel, win.mInputWindowHandle);
    }
    //设置当前聚焦窗口
    mInputMonitor.updateInputWindowsLw(false /*force*/);
}

这个方法,主要调用了InputChannelopenInputChannePair,生成一对InputChannel

  • InputChannel[0] 保存到WindowStatemInputChannel
  • InputChannel[1]传递给客户端,即ViewRootImpl中的mInputChannel,最后会和WindowInputEventReceiver关联。InputChannel可以跨进程通信。
status_t InputChannel::openInputChannelPair(const String8& name,
        sp<InputChannel>& outServerChannel, sp<InputChannel>& outClientChannel) {
    int sockets[2];
    //真正创建socket对的地方【核心】
    if (socketpair(AF_UNIX, SOCK_SEQPACKET, 0, sockets)) {
        ...
        return result;
    }

    int bufferSize = SOCKET_BUFFER_SIZE; //32k
    setsockopt(sockets[0], SOL_SOCKET, SO_SNDBUF, &bufferSize, sizeof(bufferSize));
    setsockopt(sockets[0], SOL_SOCKET, SO_RCVBUF, &bufferSize, sizeof(bufferSize));
    setsockopt(sockets[1], SOL_SOCKET, SO_SNDBUF, &bufferSize, sizeof(bufferSize));
    setsockopt(sockets[1], SOL_SOCKET, SO_RCVBUF, &bufferSize, sizeof(bufferSize));

    String8 serverChannelName = name;
    serverChannelName.append(" (server)");
    //创建InputChannel对象
    outServerChannel = new InputChannel(serverChannelName, sockets[0]);

    String8 clientChannelName = name;
    clientChannelName.append(" (client)");
    //创建InputChannel对象
    outClientChannel = new InputChannel(clientChannelName, sockets[1]);
    return OK;
}

在这个方法里,主要做了以下事情:

  • 创建了一个socket pair,生成一对无名的,相互连接的套接字
  • 设两个套接口的缓冲区大小为32kb
  • 分别创建服务端和客户端的InputChannel对象
    • socket[0]对应的是server
    • socket[1]对应的是client

到这里,socket的创建就完成了。

总结

Android输入系统,system_server和app之间socket的创建流程:

  • 首先App端在初始化view的时候,会通过跨进程Binder调用WMS的addWindow方法
  • WMS在addWindow中会创建socket连接,生成两个inputChannel对象,一个设置给IMS,一个通过Binder传回给App。

拓展阅读

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,752评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,100评论 3 387
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,244评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,099评论 1 286
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,210评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,307评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,346评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,133评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,546评论 1 306
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,849评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,019评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,702评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,331评论 3 319
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,030评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,260评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,871评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,898评论 2 351

推荐阅读更多精彩内容

  • 原创内容,转载请注明出处,多谢配合。 上节讲到InputDispatcher通过publishKeyEvent把i...
    Stan_Z阅读 10,105评论 5 23
  • https://www.jianshu.com/p/2bff4ecd86c9本篇博客主要是过一下Android I...
    wbo4958阅读 7,791评论 4 19
  • 最近在准备android面试,整理了下相关的面试题,分为如下三个部分:android部分、Java部分、算法面试题...
    JasmineBen阅读 7,075评论 10 137
  • Android Input架构 Linux Input子系统简介 Android 是基于Linux 内核,Linu...
    Nothing_655f阅读 2,010评论 0 1
  • 一、引言 本文作为Android系统架构的开篇,起到提纲挈领的作用,从系统整体架构角度概要讲解Android系统的...
    迷途小码农h阅读 2,043评论 2 25