全志V853芯片 Tina Linux下网络ADB内存泄露如何修复?

时间:2021-11-16 00:42:26

1.主题

Tina Linux 网络ADB内存泄露修复

2.问题背景

硬件:V853 软件:Tina4.0 Linux-4.9 背景:使用网络adb时,反复connect disconnect,会发生内存泄露的问题。

3.问题描述

3.1复现步骤

1、首先使能网络ADB功能。

killall adbd
export ADB_TRANSPORT_PORT=5555
/bin/adbd -D > /dev/null &

2、连接无线网络,保持小机与测试电脑处于同一个局域网中。 由于每款产品连接无线网络的命令不大一致,因此此处就不对网络连接进行赘述了。

3、编写bat脚本,进行压测。

:loop
adb disconnect
adb connect 192.168.0.101
adb disconnect
timeout /t 3
goto :loop
pause

4、使用内存泄露工具查看内存是否泄露。

3.2具体表现

使用内部一款内存泄露工具观察adbd工具内存占用情况:

行 2499: 632        504        74         /bin/adbd
行 2539: 632        504        74         /bin/adbd
行 2581: 632        504        74         /bin/adbd
行 2629: 632        504        74         /bin/adbd
行 2672: 632        504        74         /bin/adbd
行 2711: 632        504        74         /bin/adbd
行 2753: 632        504        74         /bin/adbd
行 2796: 632        504        74         /bin/adbd
行 2841: 632        504        74         /bin/adbd
行 2881: 632        504        74         /bin/adbd
行 2921: 632        504        74         /bin/adbd
行 2961: 632        504        74         /bin/adbd
行 3008: 632        504        74         /bin/adbd
行 3047: 632        504        74         /bin/adbd
行 3091: 632        504        74         /bin/adbd
行 3131: 632        504        74         /bin/adbd
行 3181: 632        504        74         /bin/adbd
行 3221: 632        504        74         /bin/adbd
行 3265: 632        504        74         /bin/adbd
行 3309: 632        504        74         /bin/adbd
行 3349: 632        504        74         /bin/adbd
行 3399: 632        504        74         /bin/adbd
行 3439: 632        504        74         /bin/adbd
行 3483: 632        504        74         /bin/adbd
行 3523: 632        504        74         /bin/adbd
行 3574: 632        504        74         /bin/adbd
行 3614: 632        504        74         /bin/adbd
行 3654: 632        504        74         /bin/adbd
行 3694: 632        636        206        /bin/adbd
行 3740: 632        636        206        /bin/adbd
行 3780: 632        636        206        /bin/adbd
行 3820: 632        636        206        /bin/adbd
行 3860: 632        636        206        /bin/adbd
行 3900: 632        636        206        /bin/adbd
行 3946: 632        636        206        /bin/adbd
行 3986: 632        636        206        /bin/adbd
行 4026: 632        636        206        /bin/adbd
行 4066: 632        636        206        /bin/adbd
行 4117: 632        636        202        /bin/adbd
行 4159: 632        636        206        /bin/adbd
行 4203: 632        636        202        /bin/adbd
行 4245: 632        636        204        /bin/adbd
行 4292: 632        636        206        /bin/adbd
行 4333: 632        668        234        /bin/adbd

发现确实存在内存泄露的情况。

4.问题分析

使能网络adb时,将命令改成:

killall adbd
export ADB_TRANSPORT_PORT=5555
/bin/adbd -D &

将adbd的log信息输出到控制台上,观察adbd的运行情况。

根据log信息以及对比源码能够发现,在调用transport.c的transport_registration_func()函数时,当读到的action等于0,即退出adb时,就会free掉一些之前malloc的指针。

全志V853芯片 Tina Linux下网络ADB内存泄露如何修复?

5.根本原因

通过在函数transport_registration_func()的free动作前后添加打印能够发现,在adb disconnect时,并没有调用到free的动作。

再在register_socket_transport()中calloc结构体atransport前后添加打印发现,每次adb connect时,都会重新调用register_socket_transport()去注册传输所需要的资源,并且会重新申请一篇内存。

那么就根据前后,就能知道在disconnect时有一篇内存没有释放,而后重新connect时又新申请内存,导致了内存的泄露。

对比有线adb,在连接时会申请结构体atransport的内存,在拔出usb线时也会free掉对应的内存。

再看回来函数transport_registration_func(),

全志V853芯片 Tina Linux下网络ADB内存泄露如何修复?

在初始化时,这里注册了一个异步事情回调函数,当接收到事件的时候,就会调用transport_registration_func()去处理事件。对于拔出USB线来说,就会对gadget进行disconnect,然后composite gadget就会通过uevent通知应用层已经断开连接了,于是adbd就能够获取到事件从而去断开连接,释放资源。

而对于网络adb来说,暂无这种机制。所以就需要依靠在output_thread中,在通信失败后主动触发断连的操作。

在部分SDK中,存在这么一个补丁:

diff --git a/utils/adb/src/transport.c b/utils/adb/src/transport.c
index 9fd6cc2..97a438b 100755
--- a/utils/adb/src/transport.c
+++ b/utils/adb/src/transport.c
@@ -285,7 +285,6 @@ static void *output_thread(void *_t)
         } else {
             D("%s: remote read failed for transport\n", t->serial);
             put_apacket(p);
-            break;
         }
     }

补丁的作用时,当通信失败时,就会断连、释放资源。这个补丁引入是为了修复USB ADB在通信不佳的情况下,触发了通信失败然后导致了USB重新枚举。 但是引入该补丁后,在使用网络ADB时,通信失败也不会被断连了,因此也造成了内存泄露的问题。

6.解决办法

基于上述情况,在output_thread中加入限制的条件。在USB ADB时,取消break,让其一直在循环当中。对于网络ADB来说,则是执行break,当发生断连时,能够及时将资源释放掉。

修复的补丁如附件所示。

0001-adbd-only-transport_local-can-disconnect-in-outputth.patch