VSYNC on Android N

VSYNC on Android N

1.      什么是VSYNC

当屏幕从缓冲区扫描完一帧到屏幕上之后,开始扫描下一帧之前,发出的一个同步信号,该信号用来切换前缓冲区和后缓冲区。

2.      HWComposer、SurfaceFlinger和VSYNC

HWComposer封装了显示设备,而SurfaceFlinger用于合成显示图层,再将其刷入到HWComposer封装的显示设备中。

当显示驱动支持时,VSYNC由硬件定时发送到HWComposer中;否则HWComposer会新起一个线程模拟发送VSYNC信号,其原理也十分简单,就是定期唤起线程发送信号。

HWComposer每次收到VSYNC信号,都会触发SurfaceFlinger中的回调onVSyncReceived()

VSYNC信号的初始化和发送如下图所示:


a. HAL层的VSYNCenable后,会不断发送VSYNC信号,并触发HWC的回调函数hook_vsync(),继而调用到SFonVSyncReceived()

b. systrace里有两个HW_VSYNC,一个叫HW_VSYNC_ON_0,其实就是记录的0 display的硬件VSYNC打开或关闭事件。记录点在HWComposer::eventControl()中,每当设置的enable与之前的enable状态不一致时(即enable->disable,disable->enable)记录。

另外一个叫HW_VSYNC_0,记录的是0 display的硬件VSYNC事件,记录点在HWComposer::vsync()中,即每当HALVSYNC事件发生,并回调到HWC的回调函数vsync()时记录。

c. 可以看出SFmEventControlThread线程用于控制HWCVSYNC信号的开关:通过调用EventControlThread::setVsyncEnabled()打开或关闭HWCVSYNC信号;SFonVSyncReceived()代表接收到了HWC层的VSYNC信号。

3.      SurfaceFlingerDispSync

SF的成员变量mPrimaryDispSync,引用一个DispSync对象。

DispSync可以理解为SF层的sync信号。APPSF自身会监听该信号,用于控制App绘制UISF合成显示图层的节奏。

DispSync中有一个成员变量DispSyncThread,该线程的threadLoop()为一个无限循环:

a. 首先判断mPeriod是否等于0

b. 不为0的话计算下个唤醒时间,继续3, 4, 5,否则挂在mCond上;

c. 在唤醒时间点,判断是否有注册的mEventListeners

d. 有的话则依次调用监听者的回调函数onDispSyncEvent()

e. 继续循环进入步骤1

系统只有一个DispSync,有两个EventListerner注册在它的mThread中,分别为appsfDispSyncSource对应的EventListener

SurfaceFlinger::init()中会层层调用到DispSync::setPeriod(),继而触发DispSyncmThread开始定期发送sync信号,并且调用监听者的回调。

4.      VSYNC-sf

VSYNC-sf用于控制SF合成Layer的节奏,可以理解为sf对应的DispSyncSource发出的sync信号,再触发EventThread的回调并转换成事件写入写端,最后触发读端的回调MessageQueue::cb_eventReceiver()SF开始合成显示图层。

DispSyncSource其实就是DispSync的监听者,通过DispSync再转发出自己所需的sync信号。

其流程图如图所示:


a. EventThread::threadLoop()发现有Connection注册了才会调用成员函数enableVSyncLocked():注册自己为CallbackDispSyncSource中,并且调用DispSyncSource::setVSyncEnabled()DispSyncSource注册为监听者到DispSync中;

b. DispSyncThread::threadLoop()只要DispSyncThread::mPeriod被设置,则开始定期发送信号。信号流为DispSyncSource::onDispSyncEvent()-->EventThread::onVSyncEvent()EventThread::threadLoop()收到事件后,再通过EventThread::Connection::postEvent()在写端写数据;对于sfEventThread来说,读端读到数据,则触发回调MessageQueue::cb_eventReceiver(),继而调用SFdispatchInvalidate(),开始合成显示图层。

5.      VSYNC-app

VSYNC-app用于控制APP绘制UI的节奏,类似4,其实就是SFapp-DispSyncSource根据自己需要,转发DispSyncsync信号。

4不同的是,SFinit()函数就已经创建了Connection,并注册在sf-EventThread中,而APP是后续有应用有需要了才会创建Connection并注册到SFapp-EventThread中。

APP中和VSYNC-app打交道的类为ChoreographerChoreographer的成员属性mDisplayEventReceiver:FrameDisplayEventReceiver用于接收SF端的VSYNC-app信号,其原理和4类似,最后就是通过native层的DisplayEventReceiver,创建并注册一个ConnectionSFsf-EventThread中,native层的DisplayEventDispatcher再将Connection的读端和自己通过addFd注册到mLooper中,后续VSYNC-app信号会触发DisplayEventDispatcherhandleEvent(),继而调用到FrameDisplayEventReceiveronVsync()

6.      总体类图

见下图,绿色的类代表有一个独立的线程,对于硬件支持VSYNC的情况下,一共有四个线程。

 

   参考文档:

a.http://windrunnerlihuan.com/2017/05/25/Android-SurfaceFlinger-%E5%AD%A6%E4%B9%A0%E4%B9%8B%E8%B7%AF-%E4%BA%94-VSync-%E5%B7%A5%E4%BD%9C%E5%8E%9F%E7%90%86/

b.https://blog.youkuaiyun.com/jinzhuojun/article/details/17293325

c.https://blog.youkuaiyun.com/sosesoA/article/details/51087639

d.https://blog.youkuaiyun.com/a740169405/article/details/70548443#t6

 

 

 

 

 

 


----- pid 23676 at 2025-08-18 17:27:01.435529533+0530 ----- Cmd line: com.amazon.avod.thirdpartyclient Build fingerprint: 'realme/RMX3999IN/RE5C94L1:16/BP2A.250605.015/U.R4T2.17996cc_790005_78c773:user/release-keys' ABI: 'arm64' Build type: optimized Debug Store: 3,16,1199864::ID:8,C:E,T:1038521||ID:8,C:S,T:1038521,N:SvcStart,D:stId=2;flg=0;act=null;comp=ComponentInfo{com.amazon.avod.thirdpartyclient/com.amazon.avod.app.termination.UserForceCloseApplicationDetectorService};pkg=null||ID:7,C:E,T:1038521||ID:7,C:S,T:1038520,N:SvcBind,D:rebind=false;act=null;cmp=ComponentInfo{com.amazon.avod.thirdpartyclient/com.amazon.avod.mdso.MdsoSignInBackgroundService};pkg=null||ID:6,C:E,T:1038519||ID:6,C:S,T:1038498,N:SvcCreate,D:name=com.amazon.avod.mdso.MdsoSignInBackgroundService;pkg=com.amazon.avod.thirdpartyclient||ID:5,C:E,T:948054||ID:5,C:S,T:948054,N:SvcStart,D:stId=1;flg=0;act=null;comp=ComponentInfo{com.amazon.avod.thirdpartyclient/com.amazon.avod.app.termination.UserForceCloseApplicationDetectorService};pkg=null||ID:4,C:E,T:948054||ID:4,C:S,T:948053,N:SvcCreate,D:name=com.amazon.avod.app.termination.UserForceCloseApplicationDetectorService;pkg=com.amazon.avod.thirdpartyclient||ID:0,C:P,T:948023,N:Finish,D:tname=main;tid=2;prid=4b4d2ac||ID:3,C:E,T:948023||ID:3,C:S,T:948023,N:BcRcvReg,D:tname=main;tid=2;act=android.net.conn.CONNECTIVITY_CHANGE;cmp=null;pkg=null;prid=4b4d2ac||ID:1,C:E,T:947742||ID:2,C:E,T:947643||ID:2,C:S,T:947643,N:SchRcvReg,D:tname=binder:23676_3;tid=1325;; suspend all histogram: Sum: 4.600ms 99% C.I. 3us-3449.600us Avg: 170.370us Max: 3957us DALVIK THREADS (73): "main" prio=5 tid=1 Native | group="main" sCount=1 ucsCount=0 flags=1 obj=0x73b7cfa8 self=0xb400007662cd3000 | sysTid=23676 nice=-10 cgrp=top-app sched=0/0 handle=0x771a5fe098 | state=S schedstat=( 36553517643 1171375374 26823 ) utm=2036 stm=1618 core=7 HZ=100 | stack=0x7ffb988000-0x7ffb98a000 stackSize=8188KB | held mutexes= native: #00 pc 000aeb1c /apex/com.android.runtime/lib64/bionic/libc.so (syscall+28) (BuildId: 154bc2ed884eb3475ddfa6bd20c12cb5) native: #01 pc 00201230 /apex/com.android.art/lib64/libart.so (art::ConditionVariable::WaitHoldingLocks+136) (BuildId: 80d2ab18f9d259d8e546c1e6bae752b1) native: #02 pc 00621508 /apex/com.android.art/lib64/libart.so (art::JNI<false>::CallObjectMethodV+1452) (BuildId: 80d2ab18f9d259d8e546c1e6bae752b1) native: #03 pc 00101e28 /system/lib64/libandroid_runtime.so (_JNIEnv::CallObjectMethod+120) (BuildId: 09a3ecef1e87b94c1ce97d0d8cacfa29) native: #04 pc 001a6af0 /system/lib64/libandroid_runtime.so (android::NativeDisplayEventReceiver::dispatchVsync+64) (BuildId: 09a3ecef1e87b94c1ce97d0d8cacfa29) native: #05 pc 000f3e2c /system/lib64/libgui.so (android::DisplayEventDispatcher::handleEvent+652) (BuildId: e7efdd9061014c212a06730846094438) native: #06 pc 00016d94 /system/lib64/libutils.so (android::Looper::pollOnce+1188) (BuildId: c0a38ad307b147e3cfcf1f790dd28610) native: #07 pc 001e270c /system/lib64/libandroid_runtime.so (android::android_os_MessageQueue_nativePollOnce+44) (BuildId: 09a3ecef1e87b94c1ce97d0d8cacfa29) at android.os.MessageQueue.nativePollOnce(Native method) at android.os.MessageQueue.nextLegacy(MessageQueue.java:937) at android.os.MessageQueue.next(MessageQueue.java:1049) at android.os.Looper.loopOnce(Looper.java:221) at android.os.Looper.loop(Looper.java:408) at android.app.ActivityThread.main(ActivityThread.java:9952) at java.lang.reflect.Method.invoke(Native method) at com.android.internal.os.RuntimeInit$MethodAndArgsCaller.run(RuntimeInit.java:613) at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:1074) DumpLatencyMs: 1.23292详细解析一下这个ANR问题并给出原因
最新发布
08-22
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值