zoukankan      html  css  js  c++  java
  • Android SurfaceFlinger

    Android 系统启动过程
    Activity 创建过程
    Activity 与 Window 与 View 之间的关系

    Android 系统从按下开机键到桌面,从桌面点击 App 图标到 Activity 显示的过程。但是 Activity 是怎么显示在屏幕上的呢?下面我们就来讨论下这一过程。

    SurfaceFlinger 启动过程

    SurfaceFlinger 启动过程:

    SurfaceFlinger 进程是由 init 进程创建的,运行在独立的 SurfaceFlinger 进程中。init 进程读取 init.rc 文件启动 SurfaceFlinger。 

    1  service surfaceflinger /system/bin/surfaceflinger
    2    class core
    3    user system
    4    group graphics drmrpc
    5    onrestart restart zygote
    6    writepid /dev/cpuset/system-background/tasks
    7

    SurfaceFlinger 的创建会执行 main() 方法:
    main_surfaceflinger.cpp

     1 int main(int, char**) {
     2    ProcessState::self()->setThreadPoolMaxThreadCount(4);
     3
     4    sp<ProcessState> ps(ProcessState::self());
     5    ps->startThreadPool();
     6
     7    //实例化 surfaceflinger
     8    sp<SurfaceFlinger> flinger =  new SurfaceFlinger();
     9
    10    setpriority(PRIO_PROCESS, 0, PRIORITY_URGENT_DISPLAY);
    11    set_sched_policy(0, SP_FOREGROUND);
    12
    13    //初始化
    14    flinger->init();
    15
    16    //发布 surface flinger,注册到 ServiceManager
    17    sp<IServiceManager> sm(defaultServiceManager());
    18    sm->addService(String16(SurfaceFlinger::getServiceName()), flinger, false);
    19
    20    // 运行在当前线程
    21    flinger->run();
    22
    23    return 0;
    24}

    SurfaceFlinger 的实例化会执行到:onFirstRef()

    1 void SurfaceFlinger::onFirstRef() {
    2    mEventQueue.init(this);
    3}

     onFirstRef() 中会创建 Handler 并初始化。
    MessageQueue.cpp:

    1void MessageQueue::init(const sp<SurfaceFlinger>& flinger) {
    2    mFlinger = flinger;
    3    mLooper = new Looper(true);
    4    mHandler = new Handler(*this);
    5}

    然后会执行到 SurfaceFlinger::init():

    1void SurfaceFlinger::init() {
     2    Mutex::Autolock _l(mStateLock);
     3
     4    //初始化 EGL,作为默认的显示
     5    mEGLDisplay = eglGetDisplay(EGL_DEFAULT_DISPLAY);
     6    eglInitialize(mEGLDisplay, NULL, NULL);
     7
     8    // 初始化硬件 composer 对象
     9    mHwc = new HWComposer(this, *static_cast<HWComposer::EventHandler *>(this));
    10
    11    //获取 RenderEngine 引擎
    12    mRenderEngine = RenderEngine::create(mEGLDisplay, mHwc->getVisualID());
    13
    14    //检索创建的 EGL 上下文
    15    mEGLContext = mRenderEngine->getEGLContext();
    16
    17    //初始化非虚拟显示屏
    18    for (size_t i=0 ; i<DisplayDevice::NUM_BUILTIN_DISPLAY_TYPES ; i++) {
    19        DisplayDevice::DisplayType type((DisplayDevice::DisplayType)i);
    20        //建立已连接的显示设备
    21        if (mHwc->isConnected(i) || type==DisplayDevice::DISPLAY_PRIMARY) {
    22            bool isSecure = true;
    23            createBuiltinDisplayLocked(type);
    24            wp<IBinder> token = mBuiltinDisplays[i];
    25
    26            sp<IGraphicBufferProducer> producer;
    27            sp<IGraphicBufferConsumer> consumer;
    28            //创建 BufferQueue 的生产者和消费者
    29            BufferQueue::createBufferQueue(&producer, &consumer,
    30                    new GraphicBufferAlloc());
    31
    32            sp<FramebufferSurface> fbs = new FramebufferSurface(*mHwc, i, consumer);
    33            int32_t hwcId = allocateHwcDisplayId(type);
    34            //创建显示设备
    35            sp<DisplayDevice> hw = new DisplayDevice(this,
    36                    type, hwcId, mHwc->getFormat(hwcId), isSecure, token,
    37                    fbs, producer,
    38                    mRenderEngine->getEGLConfig());
    39            if (i > DisplayDevice::DISPLAY_PRIMARY) {
    40                hw->setPowerMode(HWC_POWER_MODE_NORMAL);
    41            }
    42            mDisplays.add(token, hw);
    43        }
    44    }
    45
    46    getDefaultDisplayDevice()->makeCurrent(mEGLDisplay, mEGLContext);
    47
    48    //当应用和 sf 的 vsync 偏移量一致时,则只创建一个 EventThread 线程
    49    if (vsyncPhaseOffsetNs != sfVsyncPhaseOffsetNs) {
    50        sp<VSyncSource> vsyncSrc = new DispSyncSource(&mPrimaryDispSync,
    51                vsyncPhaseOffsetNs, true, "app");
    52        mEventThread = new EventThread(vsyncSrc);
    53        sp<VSyncSource> sfVsyncSrc = new DispSyncSource(&mPrimaryDispSync,
    54                sfVsyncPhaseOffsetNs, true, "sf");
    55        mSFEventThread = new EventThread(sfVsyncSrc);
    56        mEventQueue.setEventThread(mSFEventThread);
    57    } else {
    58        //创建 DispSyncSource 对象
    59        sp<VSyncSource> vsyncSrc = new DispSyncSource(&mPrimaryDispSync,
    60                vsyncPhaseOffsetNs, true, "sf-app");
    61        //创建线程 EventThread
    62        mEventThread = new EventThread(vsyncSrc);
    63        //设置 EventThread
    64        mEventQueue.setEventThread(mEventThread);
    65    }
    66
    67    //创建 EventControl
    68    mEventControlThread = new EventControlThread(this);
    69    mEventControlThread->run("EventControl", PRIORITY_URGENT_DISPLAY);
    70
    71    //当不存在 HWComposer 时,则设置软件 vsync
    72    if (mHwc->initCheck() != NO_ERROR) {
    73        mPrimaryDispSync.setPeriod(16666667);
    74    }
    75
    76    //初始化绘图状态
    77    mDrawingState = mCurrentState;
    78
    79    //初始化显示设备
    80    initializeDisplays();
    81
    82    //启动开机动画
    83    startBootAnim();
    84}

    该方法主要功能是:

    1. 初始化 EGL

    2. 创建 HWComposer

    3. 初始化非虚拟显示屏

    4. 启动 EventThread 线程

    5. 启动开机动画

    创建 HWComposer:

    1HWComposer::HWComposer(const sp<SurfaceFlinger>& flinger, EventHandler& handler):mFlinger(flinger), mFbDev(0), mHwc(0), mNumDisplays(1), mCBContext(new cb_context), mEventHandler(handler), mDebugForceFakeVSync(false) {
     2    ...
     3    bool needVSyncThread = true;
     4    int fberr = loadFbHalModule(); //加载 framebuffer 的 HAL 层模块
     5    loadHwcModule(); //加载 HWComposer 模块
     6
     7    //标记已分配的 display ID
     8    for (size_t i=0 ; i<NUM_BUILTIN_DISPLAYS ; i++) {
     9        mAllocatedDisplayIDs.markBit(i);
    10    }
    11
    12    if (mHwc) {
    13        if (mHwc->registerProcs) {
    14            mCBContext->hwc = this;
    15            mCBContext->procs.invalidate = &hook_invalidate;
    16            //VSYNC 信号的回调方法
    17            mCBContext->procs.vsync = &hook_vsync;
    18            if (hwcHasApiVersion(mHwc, HWC_DEVICE_API_VERSION_1_1))
    19                mCBContext->procs.hotplug = &hook_hotplug;
    20            else
    21                mCBContext->procs.hotplug = NULL;
    22            memset(mCBContext->procs.zero, 0, sizeof(mCBContext->procs.zero));
    23            //注册回调函数
    24            mHwc->registerProcs(mHwc, &mCBContext->procs);
    25        }
    26
    27        //进入此处,说明已成功打开硬件 composer 设备,则不再需要 vsync 线程
    28        needVSyncThread = false;
    29        eventControl(HWC_DISPLAY_PRIMARY, HWC_EVENT_VSYNC, 0);
    30        ...
    31    }
    32    ...
    33    if (needVSyncThread) {
    34        //不支持硬件的 VSYNC,则会创建线程来模拟定时 VSYNC 信号
    35        mVSyncThread = new VSyncThread(*this);
    36    }
    37}

    HWComposer 代表着硬件显示设备,注册了 VSYNC 信号的回调。VSYNC 信号本身是由显示驱动产生的,在不支持硬件的 VSYNC,则会创建“VSyncThread”线程来模拟定时 VSYNC 信号。
    当硬件产生VSYNC信号时,则会发送消息,handler 收到消息进行处理。当 SurfaceFlinger 进程收到 VSync 信号后经层层调用,最终调用到该对象的 handleMessageRefresh() 方法。
    SurfaceFlinger.cpp:

    1void SurfaceFlinger::handleMessageRefresh() {
    2    ATRACE_CALL();
    3    preComposition();//处理显示设备与 layers 的改变,更新光标
    4    rebuildLayerStacks();//重建所有可见 Layer 列表,根据Z轴排序
    5    setUpHWComposer();//更新 HWComposer 图层
    6    doDebugFlashRegions(); 
    7    doComposition();//生成 OpenGL 纹理图像
    8    postComposition();//将图像传递到物理屏幕
    9}

    Surface 创建过程

    Surface 创建过程:

    Surface 创建的过程就是 Activity 显示的过程,在 ActivityThread.handleResumeActivity() 中调用了 Activity.makeVisible(),我们接着看下 Activity 是怎么显示出来的。

     final void handleResumeActivity(IBinder token,
                boolean clearHide, boolean isForward, boolean reallyResume, int seq, String reason) {
    
            //把activity数据记录更新到ActivityClientRecord
            ActivityClientRecord r = mActivities.get(token);
    
            r = performResumeActivity(token, clearHide, reason);
    
            if (r != null) {
    
                if (r.window == null && !a.mFinished && willBeVisible) {
                    r.window = r.activity.getWindow();
                    View decor = r.window.getDecorView();
                    decor.setVisibility(View.INVISIBLE);//不可见
                    ViewManager wm = a.getWindowManager();
                    WindowManager.LayoutParams l = r.window.getAttributes();
                    a.mDecor = decor;
                    l.type = WindowManager.LayoutParams.TYPE_BASE_APPLICATION;
    
                //...
                    if (a.mVisibleFromClient && !a.mWindowAdded) {
                        a.mWindowAdded = true;
                        wm.addView(decor, l);//把decor添加到窗口上(划重点)
                    }
    
                } 
                    //屏幕参数发生了改变
                    performConfigurationChanged(r.activity, r.tmpConfig);
    
                    WindowManager.LayoutParams l = r.window.getAttributes();
    
                        if (r.activity.mVisibleFromClient) {
                            ViewManager wm = a.getWindowManager();
                            View decor = r.window.getDecorView();
                            wm.updateViewLayout(decor, l);//更新窗口状态
                        }
    
    
                    ``````
                    if (r.activity.mVisibleFromClient) {
                        //已经成功添加到窗口上了(绘制和事件接收),设置为可见
                        r.activity.makeVisible();
                    }
    
    
                //通知ActivityManagerService,Activity完成Resumed
                 ActivityManagerNative.getDefault().activityResumed(token);
            } 
        }

    Activity.makeVisible:

    1 void makeVisible() {
    2    if (!mWindowAdded) {
    3        ViewManager wm = getWindowManager();//此处 getWindowManager 获取的是 WindowManagerImpl 对象
    4        wm.addView(mDecor, getWindow().getAttributes());
    5        mWindowAdded = true;
    6    }
    7    mDecor.setVisibility(View.VISIBLE);
    8}

    WindowManagerImpl.java:

    1 public void addView(@NonNull View view, @NonNull ViewGroup.LayoutParams params) {
    2    applyDefaultToken(params);
    3    mGlobal.addView(view, params, mDisplay, mParentWindow);
    4}

    WindowManagerGlobal.java:

     1 public void addView(View view, ViewGroup.LayoutParams params, Display display, Window parentWindow) {
     2    ...
     3    final WindowManager.LayoutParams wparams = (WindowManager.LayoutParams) params;
     4    //创建 ViewRootImpl
     5    ViewRootImpl root = new ViewRootImpl(view.getContext(), display);
     6    view.setLayoutParams(wparams);
     7    mViews.add(view);
     8    mRoots.add(root);
     9    mParams.add(wparams);
    10    //设置 View
    11    root.setView(view, wparams, panelParentView);
    12    ...
    13}

    创建 ViewRootImpl:

     1 public final class ViewRootImpl implements ViewParent,
     2        View.AttachInfo.Callbacks, ThreadedRenderer.DrawCallbacks {
     3    ...
     4    final Surface mSurface = new Surface(); //创建 Surface,此时 Surface 创建完什么都没有,详见下面分析
     5    ...
     6    public ViewRootImpl(Context context, Display display) {
     7        mContext = context;
     8        //获取 IWindowSession 的代理类
     9        mWindowSession = WindowManagerGlobal.getWindowSession();
    10        mDisplay = display;
    11        mThread = Thread.currentThread(); //主线程
    12        mWindow = new W(this);
    13        mChoreographer = Choreographer.getInstance();
    14        ...
    15    }
    16}

    WindowManagerGlobal.java:

     1 public static IWindowSession getWindowSession() {
     2    synchronized (WindowManagerGlobal.class) {
     3        if (sWindowSession == null) {
     4            try {
     5                //获取 IMS 的代理类
     6                InputMethodManager imm = InputMethodManager.getInstance();
     7                //获取 WMS 的代理类
     8                IWindowManager windowManager = getWindowManagerService();
     9                //经过 Binder 调用,最终调用 WMS
    10                sWindowSession = windowManager.openSession(
    11                        new IWindowSessionCallback.Stub() {...},
    12                        imm.getClient(), imm.getInputContext());
    13            } catch (RemoteException e) {
    14                ...
    15            }
    16        }
    17        return sWindowSession
    18    }
    19}

    WMS的openSession函数创建应用程序与WMS之间的连接通道,即获取IWindowSession代理对象,并将该代理对象保存到ViewRootImpl的静态成员变量sWindowSession中,因此在应用程序进程中有且只有一个IWindowSession代理对象。

    WindowManagerService.openSession:

    1 public IWindowSession openSession(IWindowSessionCallback callback, IInputMethodClient client, IInputContext inputContext) {
    2    //创建 Session 对象
    3    Session session = new Session(this, callback, client, inputContext);
    4    return session;
    5}

    在WMS服务端构造了一个Session实例对象。ViewRootImpl 是一很重要的类,类似 ActivityThread 负责跟AmS通信一样,ViewRootImpl 的一个重要职责就是跟 WmS 通信,它通静态变量 sWindowSession(IWindowSession实例)与 WmS 进行通信。每个应用进程,仅有一个 sWindowSession 对象,它对应了 WmS 中的 Session 子类,WmS 为每一个应用进程分配一个 Session 对象。WindowState 类有一个 IWindow mClient 参数,是在构造方法中赋值的,是由 Session 调用 addWindow 传递过来了,对应了 ViewRootImpl 中的 W 类的实例。

    再次经过 Binder 将数据写回 app 进程,则获取的便是 Session 的代理对象 IWindowSession。

    创建完 ViewRootImpl 对象后,接下来调用该对象的 setView() 方法。 


    ViewRootImpl:

     1 public void setView(View view, WindowManager.LayoutParams attrs, View panelParentView) {
     2  synchronized (this) {
     3
     4    requestLayout(); //详见下面分析
     5    ...
     6    //通过 Binder调用,进入 system 进程的 Session
     7    res = mWindowSession.addToDisplay(mWindow, mSeq, mWindowAttributes,
     8          getHostVisibility(), mDisplay.getDisplayId(),
     9          mAttachInfo.mContentInsets, mAttachInfo.mStableInsets,
    10          mAttachInfo.mOutsets, mInputChannel);
    11    ...
    12  }
    13}
    1final class Session extends IWindowSession.Stub implements IBinder.DeathRecipient {
    2
    3    public int addToDisplay(IWindow window, int seq, WindowManager.LayoutParams attrs, int viewVisibility, int displayId, Rect outContentInsets, Rect outStableInsets, Rect outOutsets, InputChannel outInputChannel) {
    4        //调用 WMS.addWindow
    5        return mService.addWindow(this, window, seq, attrs, viewVisibility, displayId,
    6                outContentInsets, outStableInsets, outOutsets, outInputChannel);
    7    }
    8}

    WindowManagerService.java:

     1public int addWindow(Session session, IWindow client, int seq, WindowManager.LayoutParams attrs, int viewVisibility, int displayId, Rect outContentInsets, Rect outStableInsets, Rect outOutsets, InputChannel outInputChannel) {
     2    ...
     3    WindowToken token = mTokenMap.get(attrs.token);
     4    //创建 WindowState
     5    WindowState win = new WindowState(this, session, client, token,
     6                attachedWindow, appOp[0], seq, attrs, viewVisibility, displayContent);
     7    ...
     8    //调整 WindowManager 的 LayoutParams 参数
     9    mPolicy.adjustWindowParamsLw(win.mAttrs);
    10    res = mPolicy.prepareAddWindowLw(win, attrs);
    11    addWindowToListInOrderLocked(win, true);
    12    // 设置 input
    13    mInputManager.registerInputChannel(win.mInputChannel, win.mInputWindowHandle);
    14    //详见下面分析
    15    win.attach();
    16    mWindowMap.put(client.asBinder(), win);
    17
    18    if (win.canReceiveKeys()) {
    19        //当该窗口能接收按键事件,则更新聚焦窗口
    20        focusChanged = updateFocusedWindowLocked(UPDATE_FOCUS_WILL_ASSIGN_LAYERS,
    21                false /*updateInputWindows*/);
    22    }
    23    assignLayersLocked(displayContent.getWindowList());
    24    ...
    25}
    26
    27//WindowState.java
    28void attach() {
    29    mSession.windowAddedLocked();
    30}

    创建 SurfaceSession 对象,并将当前 Session 添加到 WMS.mSessions 成员变量。
    Session.java:

     1void windowAddedLocked() {
     2    if (mSurfaceSession == null) {
     3        mSurfaceSession = new SurfaceSession();
     4        mService.mSessions.add(this);
     5        if (mLastReportedAnimatorScale != mService.getCurrentAnimatorScale()) {
     6            mService.dispatchNewAnimatorScaleLocked(this);
     7        }
     8    }
     9    mNumWindow++;
    10}

    SurfaceSession 的创建会调用 JNI,在 JNI 调用 nativeCreate()。
    android_view_SurfaceSession.cpp:

    1 static jlong nativeCreate(JNIEnv* env, jclass clazz) {
    2    SurfaceComposerClient* client = new SurfaceComposerClient();
    3    client->incStrong((void*)nativeCreate);
    4    return reinterpret_cast<jlong>(client);
    5}

    创建 SurfaceComposerClient 对象, 作为跟 SurfaceFlinger 通信的代理对象。

     1 SurfaceComposerClient::SurfaceComposerClient() {
     2    //getComposerService() 将返回 SF 的 Binder 代理端的 BpSurfaceFlinger 对象
     3    sp<ISurfaceComposer> sm(getComposerService());
     4
     5    //先调用 SF 的 createConnection(),再调用_init
     6    _init(sm, sm->createConnection());
     7    if(mClient != 0) {
     8       Mutex::Autolock _l(gLock);
     9
    10       //gActiveConnections 是全局变量,把刚才创建的 client 保存到这个 map 中去
    11       gActiveConnections.add(mClient->asBinder(), this);
    12    }
    13}

    SurfaceFlinger.cpp:

     1 sp<ISurfaceFlingerClient>SurfaceFlinger::createConnection() {
     2    Mutex::Autolock _l(mStateLock);
     3    uint32_t token = mTokens.acquire();
     4
     5    //先创建一个Client
     6    sp<Client> client = new Client(token, this);
     7
     8    //把这个Client对象保存到mClientsMap中,token是它的标识。
     9    status_t err = mClientsMap.add(token, client);
    10
    11    /*
    12    创建一个用于 Binder 通信的 BClient,BClient 派生于 ISurfaceFlingerClient,
    13    它的作用是接受客户端的请求,然后把处理提交给 SF,注意,并不是提交给 Client。
    14    Client 会创建一块共享内存,该内存由 getControlBlockMemory 函数返回。
    15    */
    16    sp<BClient> bclient = new BClient(this, token,client->getControlBlockMemory());
    17    return bclient;
    18}
    19
    20
    21 Client::Client(ClientID clientID, constsp<SurfaceFlinger>& flinger):ctrlblk(0), cid(clientID), mPid(0), mBitmap(0), mFlinger(flinger) {
    22 const int pgsize = getpagesize();
    23    //下面这个操作会使 cblksize 为页的大小,目前是4096字节
    24    constint cblksize = ((sizeof(SharedClient)+(pgsize-1))&~(pgsize-1));
    25    mCblkHeap = new MemoryHeapBase(cblksize, 0, "SurfaceFlinger Clientcontrol-block");
    26
    27    ctrlblk = static_cast<SharedClient *>(mCblkHeap->getBase());
    28    if(ctrlblk) {
    29       new(ctrlblk) SharedClient;//原来 Surface 的 CB 对象就是在共享内存中创建的这个 SharedClient 对象
    30    }
    31}

    SharedClient:

    1class SharedClient {
     2
     3 public:
     4   SharedClient();
     5   ~SharedClient();
     6   status_t validate(size_t token) const;
     7   uint32_t getIdentity(size_t token) const;
     8
     9private:
    10    Mutexlock;
    11    Condition cv; //支持跨进程的同步对象
    12
    13    //NUM_LAYERS_MAX 为 31,SharedBufferStack 是什么?
    14    SharedBufferStack surfaces[ NUM_LAYERS_MAX ];
    15
    16};
    17
    18//SharedClient的构造函数,没什么新意,不如Audio的CB对象复杂
    19SharedClient::SharedClient():lock(Mutex::SHARED), cv(Condition::SHARED) {
    20}

    一个 Client 最多支持 31 个显示层。每一个显示层的生产/消费步调都由会对应的 SharedBufferStack 来控制。而它内部就用了几个成员变量来控制读写位置。

    SharedBufferStack.h:

     1class  SharedBufferStack{
     2     ......
     3    //Buffer 是按块使用的,每个 Buffer 都有自己的编号,其实就是数组中的索引号。
     4    volatile int32_t head;     //FrontBuffer 的编号
     5    volatile int32_t available; //空闲 Buffer 的个数
     6    volatile int32_t queued;  //脏 Buffer 的个数,脏 Buffer 表示有新数据的 Buffer
     7    volatile int32_t inUse; //SF 当前正在使用的 Buffer 的编号   
     8    volatilestatus_t status; //状态码
     9     ......
    10  }

    SF 的一个 Client 分配一个跨进程共享的 SharedClient 对象。这个对象有31个 SharedBufferStack 元素,每一个 SharedBufferStack 对应于一个显示层。
    一个显示层将创建两个 Buffer,后续的 PageFlipping 就是基于这两个 Buffer 展开的。
    接着看 SurfaceComposerClient 中这个_init函数:

    1void SurfaceComposerClient::_init(
     2       const sp<ISurfaceComposer>& sm, constsp<ISurfaceFlingerClient>& conn) {
     3    mPrebuiltLayerState = 0;
     4    mTransactionOpen = 0;
     5    mStatus = NO_ERROR;
     6    mControl = 0;
     7
     8    mClient = conn;// mClient 就是 BClient 的客户端
     9    mControlMemory =mClient->getControlBlock();
    10    mSignalServer = sm;// mSignalServer 就是 BpSurfaceFlinger
    11    //mControl 就是那个创建于共享内存之中的 SharedClient
    12    mControl = static_cast<SharedClient*>(mControlMemory->getBase());
    13}

    创建完 ViewRootImpl 对象后,接下来调用该对象的 setView() 方法。在 setView() 中调用了 requestLayout() 方法我们来看下这个方法:

    1public void requestLayout() {
     2   checkThread();
     3   mLayoutRequested = true;
     4   scheduleTraversals();
     5}
     6
     7public void scheduleTraversals() {
     8    if(!mTraversalScheduled) {
     9       mTraversalScheduled = true;
    10       sendEmptyMessage(DO_TRAVERSAL); //发送 DO_TRAVERSAL 消息
    11    }
    12}
    13
    14public void handleMessage(Message msg) {
    15   switch (msg.what) {
    16    ......
    17    case DO_TRAVERSAL:
    18        ......
    19        performTraversals();//调用 performTraversals()
    20        ......
    21        break;
    22    ......
    23    }
    24}
    25
    26private void performTraversals() {
    27    finalView host = mView;//还记得这mView吗?它就是 DecorView
    28    booleaninitialized = false;
    29    booleancontentInsetsChanged = false;
    30    booleanvisibleInsetsChanged;
    31
    32    try {
    33        relayoutResult= // 1. 关键函数relayoutWindow
    34        relayoutWindow(params, viewVisibility,insetsPending);
    35    }
    36    ......
    37    draw(fullRedrawNeeded);// 2. 开始绘制
    38    ......
    39}
    40
    41private int relayoutWindow(WindowManager.LayoutParams params, int viewVisibility, boolean insetsPending)throws RemoteException {
    42       //原来是调用 IWindowSession 的 relayout(),暂且记住这个调用
    43       int relayoutResult = sWindowSession.relayout(mWindow, params, (int) (mView.mMeasuredWidth * appScale + 0.5f),  (int) (mView.mMeasuredHeight * appScale + 0.5f), viewVisibility, insetsPending, mWinFrame, mPendingContentInsets, mPendingVisibleInsets, mPendingConfiguration, mSurface); //mSurface 做为参数传进去了。
    44       }
    45   ......
    46}
    47
    48private void draw(boolean fullRedrawNeeded) {
    49    Surface surface = mSurface;//mSurface 是 ViewRoot 的成员变量
    50    ......
    51    Canvascanvas;
    52
    53    try {
    54       int left = dirty.left;
    55       int top = dirty.top;
    56       int right = dirty.right;
    57       int bottom = dirty.bottom;
    58
    59       //从 mSurface 中 lock 一块 Canvas
    60       canvas = surface.lockCanvas(dirty);
    61       ......
    62       mView.draw(canvas);//调用 DecorView 的 draw 函数,canvas 就是画布
    63       ......
    64       //unlock 画布,屏幕上马上就能看到 View 的样子了
    65       surface.unlockCanvasAndPost(canvas);
    66    }
    67    ......
    68}

    在 ViewRoot 构造时,会创建一个 Surface,它使用无参构造函数,代码如下所示:

    final Surface mSurface = new Surface();

    此时创建完的 Surface 是空的,什么都没有。接着继续分析 relayoutWindow(),在 relayoutWindow() 中会调用 IWindowSession 的 relayout(),这是一个跨进程方法会调用到 WMS 中的 Session.relayout(),最后调用到 WindowManagerService.relayoutWindow()。

    1public int relayoutWindow(Session session,IWindow client,
     2           WindowManager.LayoutParams attrs, int requestedWidth,
     3           int requestedHeight, int viewVisibility, boolean insetsPending,
     4           Rect outFrame, Rect outContentInsets, Rect outVisibleInsets,
     5            Configuration outConfig, SurfaceoutSurface){
     6        .....
     7
     8    try {
     9         //win 就是 WinState,这里将创建一个本地的 Surface 对象
    10        Surfacesurface = win.createSurfaceLocked();
    11        if(surface != null) {
    12            //先创建一个本地 surface,然后在 outSurface 的对象上调用 copyFrom
    13            //将本地 Surface 的信息拷贝到 outSurface 中,为什么要这么麻烦呢?
    14            outSurface.copyFrom(surface);
    15        ......
    16}

    WindowManagerService.java::WindowState:

     1Surface createSurfaceLocked() {
     2    ......
     3    try {
     4        //mSurfaceSession 就是在 Session 上创建的 SurfaceSession 对象
     5        //这里,以它为参数,构造一个新的 Surface 对象
     6        mSurface = new Surface(mSession.mSurfaceSession, mSession.mPid, mAttrs.getTitle().toString(), 0, w, h, mAttrs.format, flags);
     7    }
     8    Surface.openTransaction();//打开一个事务处理
     9    ......
    10    Surface.closeTransaction();//关闭一个事务处理
    11    ......
    12}

    构造 Surface 对象:

    1 public Surface(SurfaceSession s,//传入一个SurfaceSession对象
    2    int pid, String name, int display, int w, int h, int format, int flags) throws OutOfResourcesException {
    3        ......
    4        mCanvas = new CompatibleCanvas();
    5        //又一个 native 函数
    6        init(s,pid,name,display,w,h,format,flags);
    7        mName = name;
    8    }
     1static void Surface_init(JNIEnv*env, jobject clazz, jobject session, jint pid, jstring jname, jint dpy, jint w, jint h, jint format, jintflags) {
     2
     3    //从 SurfaceSession 对象中取出之前创建的那个 SurfaceComposerClient 对象
     4    SurfaceComposerClient* client = (SurfaceComposerClient*)env->GetIntField(session, sso.client);
     5    sp<SurfaceControl> surface;//注意它的类型是 SurfaceControl
     6    if (jname == NULL) {
     7        //调用 SurfaceComposerClient 的 createSurface 函数,返回的 surface 是一个 SurfaceControl 类型
     8        surface = client->createSurface(pid, dpy, w, h, format, flags);
     9    } else{
    10        ......
    11    }
    12
    13   //把这个 surfaceControl 对象设置到 Java 层的 Surface 对象中
    14   setSurfaceControl(env, clazz, surface);
    15}

    在 createSurface 内部会使用 Binder 通信将请求发给 SurfaceFlinger:

    1sp<ISurface>SurfaceFlinger::createSurface(ClientID clientId, int pid, const String8& name, ISurfaceFlingerClient::surface_data_t* params, DisplayID d, uint32_t w, uint32_t h, PixelFormat format, uint32_t flags) {
     2    sp<LayerBaseClient> layer;//LayerBaseClient 是 Layer 家族的基类
     3    //这里又冒出一个 LayerBaseClient 的内部类,它也叫Surface
     4    sp<LayerBaseClient::Surface> surfaceHandle;
     5    Mutex::Autolock _l(mStateLock);
     6
     7    //根据 clientId 找到 createConnection 时加入的那个 Client 对象
     8    sp<Client> client = mClientsMap.valueFor(clientId);
     9    ......
    10    //注意这个 id,它的值表示 Client 创建的是第几个显示层
    11    //同时也表示将使用 SharedBufferStatck 数组的第 id 个元素
    12    int32_t id = client->generateId(pid);
    13
    14    //一个 Client 不能创建多于 NUM_LAYERS_MAX 个的Layer
    15    if(uint32_t(id) >= NUM_LAYERS_MAX) {
    16       return surfaceHandle;
    17    }
    18
    19    //根据 flags 参数来创建不同类型的显示层
    20    switch(flags & eFXSurfaceMask) {
    21        case eFXSurfaceNormal:
    22           if (UNLIKELY(flags & ePushBuffers)) {
    23             //创建 PushBuffer 类型的显示层
    24            layer = createPushBuffersSurfaceLocked(client, d, id, w, h, flags);
    25            } else {
    26               //创建 Normal 类型的显示层
    27               layer = createNormalSurfaceLocked(client, d, id, w, h, flags, format);
    28           }
    29           break;
    30        case eFXSurfaceBlur:
    31            //创建 Blur 类型的显示层
    32           layer = createBlurSurfaceLocked(client, d, id, w, h, flags);
    33           break;
    34        case eFXSurfaceDim:
    35            //创建 Dim 类型的显示层
    36           layer = createDimSurfaceLocked(client, d, id, w, h, flags);
    37           break;
    38    }
    39
    40    if(layer != 0) {
    41        layer->setName(name);
    42        setTransactionFlags(eTransactionNeeded);
    43        //从显示层对象中取出一个 ISurface 对象赋值给 SurfaceHandle
    44        surfaceHandle = layer->getSurface();
    45        if(surfaceHandle != 0) {
    46           params->token = surfaceHandle->getToken();
    47           params->identity = surfaceHandle->getIdentity();
    48           params->width = w;
    49           params->height = h;
    50           params->format = format;
    51        }
    52    }
    53    return surfaceHandle;//ISurface 的 Bn 端就是这个对象
    54}
    1sp<LayerBaseClient>SurfaceFlinger::createNormalSurfaceLocked(const sp<Client>& client, DisplayID display, int32_t id, uint32_t w, uint32_t h, uint32_t flags, PixelFormat& format) {
     2    switch(format) { //一些图像方面的参数设置,可以不去管它
     3    case PIXEL_FORMAT_TRANSPARENT:
     4    case PIXEL_FORMAT_TRANSLUCENT:
     5       format = PIXEL_FORMAT_RGBA_8888;
     6       break;
     7    case PIXEL_FORMAT_OPAQUE:
     8       format = PIXEL_FORMAT_RGB_565;
     9       break;
    10    }
    11
    12    //创建一个 Layer 类型的对象
    13    sp<Layer> layer = new Layer(this, display,client, id);
    14
    15    //设置 Buffer
    16    status_t err = layer->setBuffers(w, h, format, flags);
    17    if (LIKELY(err == NO_ERROR)) {
    18        //初始化这个新 layer 的一些状态
    19        layer->initStates(w, h, flags);
    20        //下面这个函数把这个 layer 加入到 Z 轴集合中
    21        addLayer_l(layer);
    22    }
    23......
    24    return layer;
    25}

    createNormalSurfaceLocked 函数有三个关键点,它们是:

    • 构造一个Layer对象。

    • 调用Layer对象的setBuffers函数。

    • 调用SF的addLayer_l函数。

    当跨进程的 createSurface() 执行完返回一个 ISurface 对象,接下来会创建 SurfaceControl 对象:

    1SurfaceControl::SurfaceControl(
     2       const sp<SurfaceComposerClient>& client,
     3       const sp<ISurface>& surface,
     4       const ISurfaceFlingerClient::surface_data_t& data,
     5       uint32_t w, uint32_t h, PixelFormat format, uint32_t flags)
     6    //mClient 为 SurfaceComposerClient,而 mSurface 指向跨进程 createSurface() 调用返回的 ISurface 对象
     7    :mClient(client), mSurface(surface),
     8     mToken(data.token), mIdentity(data.identity),
     9     mWidth(data.width), mHeight(data.height), mFormat(data.format),
    10     mFlags(flags){
    11     ......
    12}

    SurfaceControl 类可以看作是一个 wrapper 类,它封装了一些函数,通过这些函数可以方便地调用 mClient 或 ISurface 提供的函数。

    最后会执行 copyFrom() 返回给 App 客户端:

     1static void Surface_copyFrom(JNIEnv* env,jobject clazz, jobject other) {
     2    //根据JNI函数的规则,clazz 是 copyFrom 的调用对象,而 other 是 copyFrom 的参数。
     3    //目标对象此时还没有设置 SurfaceControl,而源对象在前面已经创建了 SurfaceControl
     4    constsp<SurfaceControl>& surface = getSurfaceControl(env, clazz);
     5    constsp<SurfaceControl>& rhs = getSurfaceControl(env, other);
     6    if (!SurfaceControl::isSameSurface(surface, rhs)) {
     7        //把源 SurfaceControl 对象设置到目标 Surface 中
     8        setSurfaceControl(env, clazz, rhs);
     9    }
    10}

    copyFrom 期间一共有三个关键对象,它们分别是:

    • SurfaceComposerClient

    • SurfaceControl

    • Surface,这个 Surface 对象属于 Native 层,和 Java 层的 Surface 相对应

    其中转移到 ViewRoot 成员变量 mSurface 中的,就是最后这个 Surface 对象了。

    在 SurfaceFlinger 进程中,Client 的一个 Layer 将使用 SharedBufferStack 数组中的一个成员,并通过 SharedBufferServer 结构来控制这个成员,我们知道 SurfaceFlinger 是消费者,所以可由 SharedBufferServer 来控制数据的读取。

    与之相对应,客户端的进程也会有一个对象来使用这个 SharedBufferStack,可它是通过另外一个叫 SharedBufferClient 的结构来控制的。客户端为 SurfaceFlinger 提供数据,所以可由 SharedBufferClient 控制数据的写入。

    Surface 显示过程

    如图所示,在 App 进程中创建 PhoneWindow 后会创建 ViewRoot。ViewRoot 的创建会创建一个 Surface,这个 Surface 其实是空的,通过与 WindowManagerService 通信 copyFrom() 一个 NativeSurface。在与 SurfaceFlinger 通信时,会创建 SharedClient 一段共享内存,里面存放的是 SharedBufferStack 对应 SurfaceFlinger 中的 SurfaceLayer 每个 Layer 其实是一个 FrameBuffer,每个 FrameBuffer 中有两个 GraphicBuffer 记作 FrontBuffer 和 BackBuffer。

    在 SurfaceFlinger 中 SharedBufferServer 来管理 FrameBuffer。同时在 App 端 copyFrom() 出来 NativeSurface 时会创建一个 SharedBufferClient 与 SharedClient 这块共享内存关联。当客户端 addView() 或者需要更新 View 时,会通过 SharedBufferClient 写入数据到 ShareClient 中,SurfaceFlinger 中的 SharedBufferServer 接收到通知会将 FrameBuffer 中的数据传输到屏幕上。

    HWComposer 是基于硬件来产生 VSync 信号的,来通知 SurfaceFlinger 重绘控制显示的帧率。

    
    
  • 相关阅读:
    Python第三方库之openpyxl(3)
    python的openpyxl的使用笔记
    openpyxl模块(excel操作)
    openpyxl
    OpenPyXl的使用
    python操作Excel模块openpyxl
    re,xpath,BeautifulSoup三种方法爬取古诗词网上诗歌
    python爬虫demo01
    ffmpeg录制报错
    解决ffmpeg执行报错“ffmpeg: error while loading shared libraries: libavdevice.so.58: cannot open shared object file: No such file or directory”的问题
  • 原文地址:https://www.cnblogs.com/mingfeng002/p/10458152.html
Copyright © 2011-2022 走看看