日韩av黄I国产麻豆传媒I国产91av视频在线观看I日韩一区二区三区在线看I美女国产在线I麻豆视频国产在线观看I成人黄色短片

歡迎訪問 生活随笔!

生活随笔

當(dāng)前位置: 首頁 > 编程语言 > c/c++ >内容正文

c/c++

使用LeakTracer检测android NDK C/C++代码中的memory leak

發(fā)布時間:2024/4/11 c/c++ 38 豆豆
生活随笔 收集整理的這篇文章主要介紹了 使用LeakTracer检测android NDK C/C++代码中的memory leak 小編覺得挺不錯的,現(xiàn)在分享給大家,幫大家做個參考.

Memory issue是C/C++開發(fā)中比較常遇到,經(jīng)常帶給人比較大困擾,debug起來又常常讓人無從下手的一類問題,memory issue主要又分為memory leak,野指針,及其它非法訪問等問題。在android平臺上,使用NDK開發(fā)C/C++ code,由于沒有其它成熟的平臺,如Windows,Linux等上面可用的許多工具,使得memory issue變得更為棘手。

問題存在,那解決辦法總是有的。比較好用又可靠的一個debug android NDK C/C++ code memory issue的辦法就是,把android NDK的C/C++代碼,移植到其它平臺上并運行起來,然后使用那個平臺下的工具,比如Linux desktop Ubuntu等,我們就可以使用諸如valgrind等異常強大的工具了,StackOverflow上有一道題Detect memory leak in android native code,其中一個答案總結(jié)了一些memory leak的檢測工具。特別是android和Linux desktop都使用相同的linux kernel,使得移植這項工作并不是特別復(fù)雜。

當(dāng)然還有另外一種解決問題的方法,那就是把傳統(tǒng)上其它平臺的一些工具給移植到android上來使用,比如valgrind就可以用在android上,但只是不太方便而已。而這里,我們就是將LeakTracer這一linux平臺上常用的memory leak檢測工具給移植到android上來使用。

LeakTracer的下載、編譯

LeakTracer official site。LeakTracer github repo。可以通過git clone將LeakTracer的code download下來,這個項目的結(jié)構(gòu)如下:


helpers目錄下是一些輔助腳本,用來幫助分析產(chǎn)生的trace文件的;libleaktracer目錄下是主要用于trace memory leak的代碼,也是需要我們集成進我們項目的代碼;test目錄下的test 可以參考來對LeakTracer進行集成;README則說明了使用LeakTracer的方法。

可以以3種方式來使用使用LeakTracer:

  • 將自己的程序與libleaktracer.a進行鏈接,也就是將自己的程序一個靜態(tài)鏈接庫libleaktracer.a進行鏈接,我們知道靜態(tài)鏈接是會將庫的代碼揉進我們自己項目的目標(biāo)代碼so中的。
  • 將自己的程序與libleaktracer.so進行鏈接。需要將-lleaktracer選項做為鏈接命令的第一個選項。當(dāng)對程序執(zhí)行"objdump -p"時,應(yīng)該能看到leaktracer.so是Dynamic Section的第一個NEEDED entry才對。

  • 通過LD_PRELOAD環(huán)境變量來使得libleaktracer.so在任何其它動態(tài)鏈接庫之前被加載,然后不需要對程序做任何的更改,還可以通過環(huán)境變量來對LeakTracer的行為進行定制。

這三種方法中的第一種,可以通過將libleaktracer的code復(fù)制到我們的項目中,與我們項目中的其它代碼一起編譯來實現(xiàn)。第二種和第三種都是想要使得leaktracer.so成為程序第一個被加載的library,我們知道,在android上zygote在fork一個應(yīng)用進程時,也會連帶將它之前加載的動態(tài)鏈接庫一并傳給我們的應(yīng)用進程,這項任務(wù)看起來似乎并不是太容易實現(xiàn)。

這里我們就將libleaktracer的代碼復(fù)制進我們的項目中,放在jni/3rd/目錄下。然后修改我們的jni的Android.mk,主要改動內(nèi)容為在適當(dāng)?shù)奈恢迷黾尤缦聝?nèi)容:

LOCAL_C_INCLUDES += $(LOCAL_PATH)/3rd/libleaktracer/include/ LOCAL_SRC_FILES += 3rd/libleaktracer/src/AllocationHandlers.cpp \3rd/libleaktracer/src/MemoryTrace.cpp

同時呢,還要修改jni的Application.mk,增加如下內(nèi)容:

APP_OPTIM := debug

這樣才能在編譯的時候,帶進更多的debug信息進目標(biāo)文件。

進行到這里,進行編譯,一切都o(jì)k。但想要通過Eclipse啟動運行app則遇到了麻煩。Eclipse檢測到libleaktracer下有個LeakTracerC.c文件libleaktracer/src/LeakTracerC.c,這個文件主要用于純C的項目,而我們這里是一個C++的項目,因而并不會用到這個文件。但Eclipse檢測到這個C文件中,卻用了C++的語法,因而會標(biāo)示語法錯誤。我們可以將這個文件直接刪掉或者將它的后綴改為cpp來解決問題。

LeakTracer的集成

要使用LeakTracer的最后一公里,也就是啟動trace,并在結(jié)束trace時,將檢測到的memory leak信息寫入文件。參考LeakTracer/tests/test.cc的code,我們可以在我們自己的library的初始化函數(shù)中加入如下的code來啟動trace:

// starting monitoring allocationsleaktracer::MemoryTrace::GetInstance().startMonitoringAllThreads();

然后在結(jié)束時的銷毀函數(shù)中加入如下的code來將memory leak信息寫入文件:

leaktracer::MemoryTrace::GetInstance().stopAllMonitoring();Poco::Thread::sleep(3000);LOGI("To writeLeaksToFile %s.", "/sdcard/leaks.out");leaktracer::MemoryTrace::GetInstance().writeLeaksToFile("/sdcard/leaks.out");

這個地方為什么要sleep呢?這是為了等待我們library的其它資源的釋放,比如一些線程握有的資源等,以減少LeakTracer的false alarm。

集成結(jié)束,開始運行來檢測memory leak。我們的app剛一運行起來,就發(fā)生了一個SIGSEGV的crash:


看上去是一個空指針,這空指針產(chǎn)生略詭異。我們試圖通過在System.loadLibrary()前加一段sleep 5s的代碼,并用Eclipse的"Debug As"的"Android Native Application"運行程序,以期能獲得更多空指針發(fā)生的位置的信息,但似乎并不能獲得更多這一crash發(fā)生的backtrace的信息。

但不難想到,這個空指針很可能發(fā)生在libleaktracer初始化的代碼里。這個library并沒有太多的code,我們可以將這個library初始化相關(guān)的所有函數(shù)的開始結(jié)束處都加上log,來追查空指針到底發(fā)生在什么地方。主要包括如下的這些函數(shù):

MemoryTrace::init_no_alloc_allowed() MemoryTrace::init_full_from_once() MemoryTrace::init_full() int MemoryTrace::Setup(void) void MemoryTrace::MemoryTraceOnInit(void)void* operator new(size_t size) void* operator new[] (size_t size) void operator delete (void *p) void operator delete[] (void *p) void *malloc(size_t size) void free(void* ptr) void* realloc(void *ptr, size_t size) void* calloc(size_t nmemb, size_t size)

加完了log,再次運行我們的程序,這次能夠看到這樣的一些信息:


可以看到,這個crash發(fā)生在libleaktracer的malloc函數(shù)中,調(diào)用棧為operator new() -> MemoryTrace::Setup(void) -> MemoryTrace::init_full_from_once() -> MemoryTrace::init_full() -> malloc()。

malloc的代碼如下:

void *malloc(size_t size) {void *p;leaktracer::MemoryTrace::Setup();leaktracer::MemoryTrace::GetInstance().InternalMonitoringDisablerThreadUp();p = LT_MALLOC(size);leaktracer::MemoryTrace::GetInstance().InternalMonitoringDisablerThreadDown();leaktracer::MemoryTrace::GetInstance().registerAllocation(p, size, false);return p; }

我們就繼續(xù)加log,在任意兩行之間都加上log。這次運行我們的程序,可以看到這樣的一些log輸出:


由此不難看出,正是如下的這一行訪問了空指針:

p = LT_MALLOC(size);

LT_MALLOC是一個宏(jni/3rd/libleaktracer/include/ObjectsPool.hpp),是一個函數(shù)指針的別名:

#define LT_MALLOC (*lt_malloc) #define LT_FREE (*lt_free) #define LT_REALLOC (*lt_realloc) #define LT_CALLOC (*lt_calloc)

函數(shù)指針則定義在jni/3rd/libleaktracer/src/AllocationHandlers.cpp中:

void* (*lt_malloc)(size_t size); void (*lt_free)(void* ptr); void* (*lt_realloc)(void *ptr, size_t size); void* (*lt_calloc)(size_t nmemb, size_t size);

函數(shù)指針則定義在jni/3rd/libleaktracer/src/AllocationHandlers.cpp中:

void* (*lt_malloc)(size_t size); void (*lt_free)(void* ptr); void* (*lt_realloc)(void *ptr, size_t size); void* (*lt_calloc)(size_t nmemb, size_t size);

這些個函數(shù)指針都通過結(jié)構(gòu)體數(shù)組static libc_alloc_func_t libc_alloc_funcs,在MemoryTrace::init_no_alloc_allowed()中進行初始化( jni/3rd/libleaktracer/src/MemoryTrace.cpp ):

typedef struct {const char *symbname;void *libcsymbol;void **localredirect; } libc_alloc_func_t;static libc_alloc_func_t libc_alloc_funcs[] = {{ "calloc", (void*)__libc_calloc, (void**)(&lt_calloc) },{ "malloc", (void*)__libc_malloc, (void**)(&lt_malloc) },{ "realloc", (void*)__libc_realloc, (void**)(&lt_realloc) },{ "free", (void*)__libc_free, (void**)(&lt_free) } };void MemoryTrace::init_no_alloc_allowed() {libc_alloc_func_t *curfunc;unsigned i;for (i=0; i<(sizeof(libc_alloc_funcs)/sizeof(libc_alloc_funcs[0])); ++i) {curfunc = &libc_alloc_funcs[i];if (!*curfunc->localredirect) {if (curfunc->libcsymbol) {*curfunc->localredirect = curfunc->libcsymbol;} else {*curfunc->localredirect = dlsym(RTLD_NEXT, curfunc->symbname);}}}

MemoryTrace::init_no_alloc_allowed()中進行初始化的這段代碼,主要是找到系統(tǒng)本來的那組分配/釋放內(nèi)存的函數(shù)的地址,并保存在lt_malloc這一組函數(shù)指針中。

android的標(biāo)準(zhǔn)C庫中,并沒有__libc_malloc這一組符號,因而lt_malloc的值應(yīng)該來自于dlsym(RTLD_NEXT, curfunc->symbname)。

我們知道dlsym()函數(shù)可以與dlopen()配合,動態(tài)加載一個動態(tài)鏈接庫,并獲取里面的函數(shù)指針。dlsym()函數(shù)接收兩個參數(shù),一個是dlopen()打開的動態(tài)鏈接庫的handle,另一個符號名。動態(tài)鏈接庫的handle也可以不來自于dlopen(),而是RTLD_DEFAULT和RTLD_NEXT這兩個特殊的值,其中前者表示從當(dāng)前應(yīng)用加載的第一個動態(tài)鏈接庫開始查找符號地址,而后者則表示從當(dāng)前so的下一個so開始查找符號地址。

看到這里,也就不難理解為何需要先加載libleaktracer了。總結(jié)一下,有兩個原因,一是在動態(tài)鏈接時,程序?qū)τ诜峙?釋放內(nèi)存的函數(shù)的調(diào)用能鏈接到libleaktracer的實現(xiàn),二是這里能夠找到系統(tǒng)的那組分配/釋放內(nèi)存的函數(shù)。

但在android平臺上,我們是沒有辦法讓libleaktracer的code早于其它所有的動態(tài)鏈接庫加載的,因而我們需要將這里的**RTLD_NEXT給替換成RTLD_DEFAULT**,以便于能夠找到系統(tǒng)的那組內(nèi)存分配/釋放函數(shù)的地址。

至此libleaktracer的初始化過程終于可以正常的執(zhí)行了。內(nèi)存的分配/釋放函數(shù)調(diào)用的頻率實在是太高了,因而去掉剛剛加的那些log,準(zhǔn)備迎接下一步的挑戰(zhàn)。

但運行起來之后,又出現(xiàn)了crash了。


這次倒是可以通過Eclipse的"Debug As"的"Android Native Application"抓到發(fā)生crash的整個backtrace:


MemoryTrace::storeAllocationStack()的code如下:

inline void MemoryTrace::storeAllocationStack(void* arr[ALLOCATION_STACK_DEPTH]) {unsigned int iIndex = 0; #ifdef USE_BACKTRACEvoid* arrtmp[ALLOCATION_STACK_DEPTH+1];iIndex = backtrace(arrtmp, ALLOCATION_STACK_DEPTH + 1) - 1;memcpy(arr, &arrtmp[1], iIndex*sizeof(void*)); #elsevoid *pFrame;// NOTE: we can't use "for" loop, __builtin_* functions// require the number to be known at compile timearr[iIndex++] = ( (pFrame = __builtin_frame_address(0)) != NULL) ? __builtin_return_address(0) : NULL; if (iIndex == ALLOCATION_STACK_DEPTH) return;arr[iIndex++] = (pFrame != NULL && (pFrame = __builtin_frame_address(1)) != NULL) ? __builtin_return_address(1) : NULL; if (iIndex == ALLOCATION_STACK_DEPTH) return;arr[iIndex++] = (pFrame != NULL && (pFrame = __builtin_frame_address(2)) != NULL) ? __builtin_return_address(2) : NULL; if (iIndex == ALLOCATION_STACK_DEPTH) return;arr[iIndex++] = (pFrame != NULL && (pFrame = __builtin_frame_address(3)) != NULL) ? __builtin_return_address(3) : NULL; if (iIndex == ALLOCATION_STACK_DEPTH) return;arr[iIndex++] = (pFrame != NULL && (pFrame = __builtin_frame_address(4)) != NULL) ? __builtin_return_address(4) : NULL; if (iIndex == ALLOCATION_STACK_DEPTH) return;arr[iIndex++] = (pFrame != NULL && (pFrame = __builtin_frame_address(5)) != NULL) ? __builtin_return_address(5) : NULL; if (iIndex == ALLOCATION_STACK_DEPTH) return;arr[iIndex++] = (pFrame != NULL && (pFrame = __builtin_frame_address(6)) != NULL) ? __builtin_return_address(6) : NULL; if (iIndex == ALLOCATION_STACK_DEPTH) return;arr[iIndex++] = (pFrame != NULL && (pFrame = __builtin_frame_address(7)) != NULL) ? __builtin_return_address(7) : NULL; if (iIndex == ALLOCATION_STACK_DEPTH) return;arr[iIndex++] = (pFrame != NULL && (pFrame = __builtin_frame_address(8)) != NULL) ? __builtin_return_address(8) : NULL; if (iIndex == ALLOCATION_STACK_DEPTH) return;arr[iIndex++] = (pFrame != NULL && (pFrame = __builtin_frame_address(9)) != NULL) ? __builtin_return_address(9) : NULL; #endif// fill remaining spacesfor (; iIndex < ALLOCATION_STACK_DEPTH; iIndex++)arr[iIndex] = NULL; }

可以看到,這個函數(shù)主要是利用gcc的內(nèi)置函數(shù)builtin_frame_address()和builtin_return_address()獲取一次內(nèi)存分配發(fā)生的整個的callstack。由于對內(nèi)存訪問的權(quán)限限制的原因,而會發(fā)生SIGSEGV錯誤。但又不是在第一次調(diào)用這些內(nèi)置函數(shù)時發(fā)生的crash。那就先把保存的callstack的深度調(diào)淺一點好了,比如把ALLOCATION_STACK_DEPTH的值改為3。這倒是不crash了,但每次獲得的backtrace都只有一層,而且每個backtrace都一樣,這些信息真是毫無用處,它們都指向libleaktracer的malloc。

看來通過gcc的內(nèi)置函數(shù)builtin_frame_address()和builtin_return_address()獲取backtrace這條路是行不同了。那android平臺上native層到底有沒有其它可以獲取backtrace的方法呢?答案當(dāng)然是,有~,而且那個接口比gcc的內(nèi)置函數(shù)還有好用許多。這個好用的接口就是_Unwind_Backtrace**()**。include標(biāo)準(zhǔn)庫頭文件#include <unwind.h>就可以使用_Unwind_Backtrace()了,這個函數(shù)的原型如下:

typedef struct _Unwind_Context _Unwind_Context;/* @@@ Use unwind data to perform a stack backtrace. The trace callbackis called for every stack frame in the call chain, but no cleanupactions are performed. */typedef _Unwind_Reason_Code (*_Unwind_Trace_Fn) (_Unwind_Context *, void *);_Unwind_Reason_Code _Unwind_Backtrace(_Unwind_Trace_Fn,void*);

_Unwind_Backtrace()接收兩個參數(shù),一個是_Unwind_Reason_Code () (_Unwind_Context , void *)類型的函數(shù)指針_Unwind_Trace_Fn,另外一個是userdata,會被作為每次調(diào)用 _Unwind_Trace_Fn的第二個參數(shù)傳入。 _Unwind_Backtrace()的執(zhí)行,會針對調(diào)用鏈中的每一級stack frame調(diào)用trace callback _Unwind_Trace_Fn,在這個callback的實現(xiàn)中,我們可以保存stack frame的信息。

我們重寫MemoryTrace::storeAllocationStack()函數(shù):

struct TraceHandle {void **backtrace;int pos; };_Unwind_Reason_Code Unwind_Trace_Fn(_Unwind_Context *context, void *hnd) {struct TraceHandle *traceHanle = (struct TraceHandle *) hnd;_Unwind_Word ip = _Unwind_GetIP(context);if (traceHanle->pos != ALLOCATION_STACK_DEPTH) {traceHanle->backtrace[traceHanle->pos] = (void *) ip;++traceHanle->pos;return _URC_NO_REASON;}return _URC_END_OF_STACK; }// stores allocation stack, up to ALLOCATION_STACK_DEPTH // frames void MemoryTrace::storeAllocationStack(void* arr[ALLOCATION_STACK_DEPTH]) {unsigned int iIndex = 0;TraceHandle traceHandle;traceHandle.backtrace = arr;traceHandle.pos = 0;_Unwind_Backtrace(Unwind_Trace_Fn, &traceHandle);// fill remaining spacesfor (iIndex = traceHandle.pos; iIndex < ALLOCATION_STACK_DEPTH; iIndex++)arr[iIndex] = NULL; }

在這里我們通過_Unwind_Backtrace**()來獲得調(diào)用棧。系統(tǒng)還提供了_Unwind_GetIP**()用來幫助我們獲取每一個stack frame的指令指針I(yè)P。

在我們的library銷毀時保存trace信息到文件中,從設(shè)備上將該文件/sdcard/leaks.out pull出來,可以看到如下的這樣一些內(nèi)容:

# LeakTracer report diff_utc_mono=1448719075.832670 leak, time=9077.889212, stack=0xa3241568 0xa323fac0 0xa323fdd4 0xa321407c 0xa3213fe8, size=12, data=..i.X.i...j. leak, time=9077.889945, stack=0xa3241568 0xa323fac0 0xa323fdd4 0xa32b8a94 0xa32ba1b8, size=27, data=.mi.8A......42.62.105.193.j leak, time=9077.889151, stack=0xa3241568 0xa323fac0 0xa323fdd4 0xa32b8a94 0xa32b96e4, size=84, data=G...G.......{"size":1885076,"spd":1.23937e+06,"tim leak, time=9074.898368, stack=0xa3241568 0xa323fac0 0xa323fdd4 0xa32039ec 0xa3209cec, size=88, data= .0.......i.d.r.......................i........... leak, time=9084.530140, stack=0xa3241568 0xa323fac0 0xa323fdd4 0xa32b8a94 0xa32ba1b8, size=19, data=............ipPriv. leak, time=9088.118207, stack=0xa3241568 0xa323fac0 0xa323fdd4 0xa32b8a94 0xa32b8c68, size=13, data=............. leak, time=9084.530354, stack=0xa3241568 0xa323fac0 0xa323fdd4 0xa31a8e80 0xa31a8bac, size=28, data=....\$j..%j..%j.\"j......$j. leak, time=9084.530232, stack=0xa3241568 0xa323fac0 0xa323fdd4 0xa31ab740 0xa31ab7f4, size=24, data=0A.......#j.H&j..&j..... leak, time=9084.530537, stack=0xa3241568 0xa323fac0 0xa323fdd4 0xa31ab114 0xa3212be4, size=4, data=.._. leak, time=9084.530628, stack=0xa3241568 0xa323fac0 0xa323fdd4 0xa32b8a94 0xa32ba1b8, size=18, data=............ipPub. leak, time=9084.530720, stack=0xa3241568 0xa323fac0 0xa323fdd4 0xa31a8e80 0xa31a8bac, size=28, data=.....%j......&j..$j......$j.

既然有了 沒被正確釋放的內(nèi)存分配時候的backtrace,那就將它們轉(zhuǎn)換到代碼文件的行數(shù)吧。LeakTracer/helpers下的leak-analyze-addr2line工具可以幫我們完成這些。leak-analyze-addr2line的用法如下:

Usage: /usr/bin/leak-analyze-addr2line <PROGRAM> <LEAKFILE>

于是我們輸入如下的命令:

leak-analyze-addr2line /media/data/CorpProjects/git_repos/P2PClient/peerTester/obj/local/armeabi-v7a/libmoretvp2p.so leaks.out

注意這里的so文件的路徑,不是libs/armeabi-v7a下面的那個,那個so中是沒有debug信息的,而是obj/local/armeabi-v7a/下面的。

但執(zhí)行上面的命令,我們卻只得到了這樣的一些信息:

Processing "leaks.out" log for "/media/data/CorpProjects/git_repos/P2PClient/peerTester/obj/local/armeabi-v7a/libmoretvp2p.so" Matching addresses to "/media/data/CorpProjects/git_repos/P2PClient/peerTester/obj/local/armeabi-v7a/libmoretvp2p.so" found 166 leak(s) 252 bytes lost in 9 blocks (one of them allocated at 9084.533162), from following call stack:??:0??:0??:0??:0??:0 4 bytes lost in 1 blocks (one of them allocated at 9084.532643), from following call stack:??:0??:0??:0??:0??:0 317 bytes lost in 15 blocks (one of them allocated at 9075.580071), from following call stack:??:0??:0??:0??:0??:0 4 bytes lost in 1 blocks (one of them allocated at 9084.532307), from following call stack:??:0??:0??:0??:0??:0

貌似完全無法得到backtrace對應(yīng)的代碼行數(shù)呢。問題出在哪里了呢?

回頭再看/sdcard/leaks.out中的backtrace,可以看到內(nèi)存地址都是進程地址空間的絕對地址,動態(tài)鏈接庫在每次加載是都可能被映射在進程內(nèi)存地址空間的不同位置,因而addr2line無法根據(jù)符號的地址空間絕對地址轉(zhuǎn)換到代碼行數(shù)也容易理解。難道我們要先通過/proc/[pid]/maps找到我們的動態(tài)鏈接庫映射的內(nèi)存基地址,然后手動算出backtrace每個地址對應(yīng)的動態(tài)鏈接庫內(nèi)部的偏移地址,再通過addr2line來將內(nèi)存地址轉(zhuǎn)換到代碼文件的行號?

這是比較難辦到的:一來許多production的設(shè)備,根本不允許我們訪問進程的內(nèi)存映射/proc/[pid]/maps;二是backtrace的內(nèi)存地址太多,手動轉(zhuǎn)要猴年馬月才能賺得完。

看起來只要我們能夠獲取我們的library映射到的內(nèi)存的基地址,一切問題也就迎刃而解了,但android平臺是否存在這樣的一種方法呢?答案當(dāng)然是肯定的,這個函數(shù)也就是int dladdr(const void addr, Dl_info info)。dladdr()與dlsym()一樣,同在libdl中。

于是我們定義一個靜態(tài)的Dl_info結(jié)構(gòu)對象s_P2pSODlInfo,并在MemoryTrace::init_no_alloc_allowed()中,初始化lt_calloc等函數(shù)指針的那段code下面加一行對dladdr()的調(diào)用:

dladdr((const void*)init_no_alloc_allowed, &s_P2pSODlInfo);LOGI("s_P2pSODlInfo, dli_fbase = %p, dli_fname = %s, dli_saddr = %p, dli_sname = %s",s_P2pSODlInfo.dli_fbase, s_P2pSODlInfo.dli_fname, s_P2pSODlInfo.dli_saddr, s_P2pSODlInfo.dli_sname);

這里我們就通過函數(shù)init_no_alloc_allowed的地址來獲得整個library內(nèi)存映射的基地址。同時修改_Unwind_Backtrace的Unwind_Trace_Fn為如下這樣:

_Unwind_Reason_Code Unwind_Trace_Fn(_Unwind_Context *context, void *hnd) {struct TraceHandle *traceHanle = (struct TraceHandle *) hnd;_Unwind_Word ip = _Unwind_GetIP(context);if (traceHanle->pos != ALLOCATION_STACK_DEPTH) {traceHanle->backtrace[traceHanle->pos] = (void *) (ip - (_Unwind_Word) s_P2pSODlInfo.dli_fbase);++traceHanle->pos;return _URC_NO_REASON;}return _URC_END_OF_STACK; }

將我們通過_Unwind_GetIP()獲得的IP值都減去library 內(nèi)存映射的基地址。

再次運行我們的程序,產(chǎn)生memory leak的trace文件并pull下來,并用leak-analyze-addr2line工具進行分析,終于可以獲得leak的內(nèi)存分配時的callstack了,如下面這樣:

Processing "leaks.out" log for "/media/data/CorpProjects/git_repos/P2PClient/peerTester/obj/local/armeabi-v7a/libmoretvp2p.so" Matching addresses to "/media/data/CorpProjects/git_repos/P2PClient/peerTester/obj/local/armeabi-v7a/libmoretvp2p.so" found 165 leak(s) 24 bytes lost in 1 blocks (one of them allocated at 10597.290280), from following call stack:/media/data/CorpProjects/git_repos/P2PClient/peerTester/jni/3rd/libleaktracer/src/MemoryTrace.cpp:316/media/data/CorpProjects/git_repos/P2PClient/peerTester/jni/3rd/libleaktracer/include/MemoryTrace.hpp:368/media/data/CorpProjects/git_repos/P2PClient/peerTester/jni/3rd/libleaktracer/src/AllocationHandlers.cpp:29/media/data/CorpProjects/git_repos/P2PClient/peerTester/jni/3rd/json/src/Value.cpp:361/media/data/CorpProjects/git_repos/P2PClient/peerTester/jni/3rd/json/src/Value.cpp:368 (discriminator 1) 69 bytes lost in 3 blocks (one of them allocated at 10585.557730), from following call stack:/media/data/CorpProjects/git_repos/P2PClient/peerTester/jni/3rd/libleaktracer/src/MemoryTrace.cpp:316/media/data/CorpProjects/git_repos/P2PClient/peerTester/jni/3rd/libleaktracer/include/MemoryTrace.hpp:368/media/data/CorpProjects/git_repos/P2PClient/peerTester/jni/3rd/libleaktracer/src/AllocationHandlers.cpp:29libgcc2.c:?libgcc2.c:? 12 bytes lost in 1 blocks (one of them allocated at 10587.229148), from following call stack:/media/data/CorpProjects/git_repos/P2PClient/peerTester/jni/3rd/libleaktracer/src/MemoryTrace.cpp:316/media/data/CorpProjects/git_repos/P2PClient/peerTester/jni/3rd/libleaktracer/include/MemoryTrace.hpp:368/media/data/CorpProjects/git_repos/P2PClient/peerTester/jni/3rd/libleaktracer/src/AllocationHandlers.cpp:29/media/data/CorpProjects/git_repos/P2PClient/peerTester/jni/src/core/PlaySession.cpp:38 (discriminator 7)/media/data/CorpProjects/git_repos/P2PClient/peerTester/jni/src/core/PlaySessionInitializer.cpp:241 (discriminator 3)

可能由于內(nèi)存釋放的時延等原因,LeakTracer報出來的問題不一定是真正的memory leak,也可能只是false alarm,具體問題還需要根據(jù)LeakTracer產(chǎn)生的報告再來做分析。

LeakTracer的設(shè)計與實現(xiàn)

這里我們再來分析一下LeakTracer的設(shè)計與實現(xiàn)。

LeakTracer主要的設(shè)計思路為:

  • 實現(xiàn)一組內(nèi)存的分配/釋放函數(shù),這組函數(shù)的函數(shù)原型與系統(tǒng)的那一組完全一樣,讓被trace的library對于內(nèi)存的分配/釋放函數(shù)的調(diào)用都鏈接到自己實現(xiàn)的這一組函數(shù)中以override掉系統(tǒng)的那組內(nèi)存/分配釋放函數(shù);

  • 自己實現(xiàn)的這組函數(shù)中的內(nèi)存分配函數(shù)記錄分配相關(guān)的信息,包括分配的內(nèi)存的大小,callstack等,并調(diào)用系統(tǒng)本來的內(nèi)存分配函數(shù)去分配內(nèi)存;

  • 自己實現(xiàn)的這組函數(shù)中的內(nèi)存釋放函數(shù)則銷毀內(nèi)存分配的相關(guān)記錄,并使用系統(tǒng)的內(nèi)存釋放函數(shù)真正的釋放內(nèi)存;

  • 在trace結(jié)束時,遍歷所有保存的內(nèi)存分配記錄的信息,并把這些信息保存進文件以供進一步的分析。

  • LeakTracer實現(xiàn)的用于override系統(tǒng)內(nèi)存分配/釋放函數(shù)的那組函數(shù)在jni/3rd/libleaktracer/src/AllocationHandlers.cpp中定義:

    void* (*lt_malloc)(size_t size); void (*lt_free)(void* ptr); void* (*lt_realloc)(void *ptr, size_t size); void* (*lt_calloc)(size_t nmemb, size_t size);void* operator new(size_t size) {void *p;leaktracer::MemoryTrace::Setup();p = LT_MALLOC(size);leaktracer::MemoryTrace::GetInstance().registerAllocation(p, size, false);return p; }void* operator new[] (size_t size) {void *p;leaktracer::MemoryTrace::Setup();p = LT_MALLOC(size);leaktracer::MemoryTrace::GetInstance().registerAllocation(p, size, true);return p; }void operator delete (void *p) {leaktracer::MemoryTrace::Setup();leaktracer::MemoryTrace::GetInstance().registerRelease(p, false);LT_FREE(p); }void operator delete[] (void *p) {leaktracer::MemoryTrace::Setup();leaktracer::MemoryTrace::GetInstance().registerRelease(p, true);LT_FREE(p); }/** -- libc memory operators -- **//* malloc* in some malloc implementation, there is a recursive call to malloc* (for instance, in uClibc 0.9.29 malloc-standard )* we use a InternalMonitoringDisablerThreadUp that use a tls variable to prevent several registration* during the same malloc*/ void *malloc(size_t size) {void *p;leaktracer::MemoryTrace::Setup();leaktracer::MemoryTrace::GetInstance().InternalMonitoringDisablerThreadUp();p = LT_MALLOC(size);leaktracer::MemoryTrace::GetInstance().InternalMonitoringDisablerThreadDown();leaktracer::MemoryTrace::GetInstance().registerAllocation(p, size, false);return p; }void free(void* ptr) {leaktracer::MemoryTrace::Setup();leaktracer::MemoryTrace::GetInstance().registerRelease(ptr, false);LT_FREE(ptr); }void* realloc(void *ptr, size_t size) {void *p;leaktracer::MemoryTrace::Setup();leaktracer::MemoryTrace::GetInstance().InternalMonitoringDisablerThreadUp();p = LT_REALLOC(ptr, size);leaktracer::MemoryTrace::GetInstance().InternalMonitoringDisablerThreadDown();if (p != ptr){if (ptr)leaktracer::MemoryTrace::GetInstance().registerRelease(ptr, false);leaktracer::MemoryTrace::GetInstance().registerAllocation(p, size, false);}else{leaktracer::MemoryTrace::GetInstance().registerReallocation(p, size, false);}return p; }void* calloc(size_t nmemb, size_t size) {void *p;leaktracer::MemoryTrace::Setup();leaktracer::MemoryTrace::GetInstance().InternalMonitoringDisablerThreadUp();p = LT_CALLOC(nmemb, size);leaktracer::MemoryTrace::GetInstance().InternalMonitoringDisablerThreadDown();leaktracer::MemoryTrace::GetInstance().registerAllocation(p, nmemb*size, false);return p; }

    系統(tǒng)的那組內(nèi)存分配/釋放函數(shù)的函數(shù)指針也在這個文件中定義。如我們前面看到的,系統(tǒng)的那組內(nèi)存分配/釋放函數(shù)的函數(shù)指針在MemoryTrace::init_no_alloc_allowed()中通過dlsym(RTLD_DEFAULT, curfunc->symbname)進行初始化。

    LeakTracer通過leaktracer::MemoryTrace::GetInstance().registerAllocation(p, size, false)記錄每一次內(nèi)存分配的相關(guān)信息:

    // adds all relevant info regarding current allocation to map inline void MemoryTrace::registerAllocation(void *p, size_t size, bool is_array) {allocation_info_t *info = NULL;if (!AllMonitoringIsDisabled() && (__monitoringAllThreads || getThreadOptions().monitoringAllocations) && p != NULL) {MutexLock lock(__allocations_mutex);info = __allocations.insert(p);if (info != NULL) {info->size = size;info->isArray = is_array;storeTimestamp(info->timestamp);}}// we store the stack without locking __allocations_mutex// it should be safe enough// prevent a deadlock between backtrave function who are now using advanced dl_iterate_phdr function// and dl_* function which uses malloc functionsif (info != NULL) {storeAllocationStack(info->allocStack);}if (p == NULL) {InternalMonitoringDisablerThreadUp();// WARNINGInternalMonitoringDisablerThreadDown();} }

    并通過leaktracer::MemoryTrace::GetInstance().registerRelease(ptr, false)銷毀一個內(nèi)存分配記錄:

    // removes allocation's info from the map inline void MemoryTrace::registerRelease(void *p, bool is_array) {if (!AllMonitoringIsDisabled() && __monitoringReleases && p != NULL) {MutexLock lock(__allocations_mutex);allocation_info_t *info = __allocations.find(p);if (info != NULL) {if (info->isArray != is_array) {InternalMonitoringDisablerThreadUp();// WARNINGInternalMonitoringDisablerThreadDown();}__allocations.release(p);}} }

    整體來看LeakTracer的設(shè)計與實現(xiàn)都并不復(fù)雜,因而能夠trace的memory issue也就有限。比如,LeakTracer就無法trace多次釋放等問題。但它也足以作為我們編寫更強大的memory issue trace工具的基礎(chǔ)了。

    Done。

    參考文檔:
    Android下打印調(diào)試堆棧方法
    dladdr - 獲取某個地址的符號信息
    LeakTracer for Android
    http://androidxref.com/5.0.0_r2/xref/hardware/ti/omap4-aah/stacktrace.c
    http://www.newsmth.net/nForum/#!article/KernelTech/413

    總結(jié)

    以上是生活随笔為你收集整理的使用LeakTracer检测android NDK C/C++代码中的memory leak的全部內(nèi)容,希望文章能夠幫你解決所遇到的問題。

    如果覺得生活随笔網(wǎng)站內(nèi)容還不錯,歡迎將生活随笔推薦給好友。

    日韩一级电影在线观看 | 日韩精品中文字幕在线播放 | 手机在线黄色网址 | 亚洲一区视频在线播放 | 五月婷婷丁香网 | 在线免费看片 | 日本不卡一区二区三区在线观看 | 99热99| 成人av在线亚洲 | 久久免费大片 | 国产精品国产三级国产aⅴ9色 | www.久久色| 91九色在线视频观看 | 九九涩涩av台湾日本热热 | 午夜精品久久久久久久99水蜜桃 | 97免费视频在线播放 | 一区二区视频播放 | 国产精品破处视频 | 人人爽人人澡人人添人人人人 | 日韩在线观看一区 | 久久久国产毛片 | 久久婷婷国产 | 66av99精品福利视频在线 | 91日韩精品 | 狠狠色丁香久久综合网 | 午夜国产成人 | 91最新国产 | 91中文字幕在线 | 中文字幕久久久精品 | 美女视频黄在线观看 | 国产综合精品一区二区三区 | 亚洲精品看片 | 国产高清视频网 | 久久久高清免费视频 | av在线中文 | 亚洲精品午夜久久久久久久久久久 | 国产美女在线免费观看 | 欧美男同网站 | 人人看人人做人人澡 | 久久婷婷一区二区三区 | 黄色av免费 | 91在线永久 | 亚洲一级免费电影 | 色噜噜在线观看视频 | 四虎影视精品成人 | 色狠狠久久av五月综合 | 国产精品久久久久久久久久免费 | 国产97在线观看 | 天天爱天天操 | 最新日韩在线 | 在线观看国产高清视频 | 香蕉视频网站在线观看 | 久精品在线 | 中文在线a在线 | 免费av网站在线 | 中文字幕在线专区 | 中文字幕中文中文字幕 | 欧美另类网站 | 激情丁香综合 | 日韩高清观看 | 成人一级黄色片 | 亚洲一区二区三区精品在线观看 | 久久无码av一区二区三区电影网 | 国产日韩欧美在线播放 | 国产一区国产二区在线观看 | 偷拍精品一区二区三区 | 97精品超碰一区二区三区 | 国产精品久久二区 | 国产原创在线视频 | 91av超碰 | 中文字幕亚洲在线观看 | 中文字幕av专区 | 亚洲激情视频在线 | 99re中文字幕| 日本电影黄色 | 欧美日韩免费一区二区三区 | 日本中文字幕在线电影 | 伊人成人久久 | 999在线精品| 国产v亚洲v | 日韩中文在线电影 | 欧美日韩3p| 91亚洲精品国偷拍自产在线观看 | 国产在线视频导航 | 久9在线 | 探花视频在线观看+在线播放 | 久久福利小视频 | 久久国产精品99精国产 | 91精品国产99久久久久久久 | 国产成人精品一区一区一区 | 免费视频一二三区 | 久久久国产99久久国产一 | 免费av大片 | 香蕉精品视频在线观看 | 九九视频在线播放 | 欧美成人播放 | 久久免费视频在线观看6 | 国内精品二区 | 午夜视频欧美 | 91麻豆精品一区二区三区 | 欧美日韩高清 | 青青射 | 国产免费黄视频在线观看 | 中文字幕在线看视频国产中文版 | 久久男人视频 | 91人人爽人人爽人人精88v | 亚洲国产精品电影 | 久久精品国亚洲 | 日韩一区二区三区在线观看 | 一区二区三区日韩在线 | 国产一区高清在线 | 天天摸天天干天天操天天射 | 免费看的黄色的网站 | 国产老熟| 中文字幕在线观看一区二区 | 国产精品视频专区 | 日韩和的一区二在线 | 国产精品福利一区 | 97人人模人人爽人人喊中文字 | 久久国产精品99久久久久久老狼 | 国产精品免费久久久久 | 青春草视频在线播放 | 97超碰中文 | 夜夜夜夜夜夜操 | 国产精品日韩在线观看 | 日韩精品免费在线 | 国产视频一区二区在线播放 | 亚洲综合激情小说 | 中文字幕在线看视频国产中文版 | 国产精品99在线播放 | 91av成人 | 日韩黄色网络 | 亚洲精品电影在线 | 91豆花在线 | 国产午夜精品av一区二区 | 免费网站看v片在线a | 国产午夜影院 | 黄色毛片观看 | a视频免费| 色视频网站在线观看一=区 a视频免费在线观看 | 亚洲禁18久人片 | 日韩高清在线一区二区三区 | 天天干婷婷 | 国内成人精品视频 | 精品视频在线视频 | 亚洲码国产日韩欧美高潮在线播放 | 国产精品一区二区免费看 | 成人影片在线免费观看 | 激情网婷婷 | 亚洲日韩精品欧美一区二区 | 免费看黄在线观看 | 一级黄色免费网站 | 久久久久久久久久伊人 | 欧美成人xxxx| 国产精品第一页在线观看 | 国产乱对白刺激视频不卡 | 日韩欧美一区二区三区免费观看 | 欧美analxxxx| 久久久国产一区二区三区四区小说 | 成人av电影免费观看 | 国产一区二区日本 | 狠狠色丁香九九婷婷综合五月 | 成人在线视频免费 | 久久综合给合久久狠狠色 | 天天躁天天狠天天透 | 91精品国产99久久久久久红楼 | 欧洲精品视频一区二区 | 超碰99在线| 色婷婷视频在线观看 | 日日躁你夜夜躁你av蜜 | 99国产视频| 91看片黄色| 字幕网资源站中文字幕 | 色视频在线免费 | 手机在线看片日韩 | 不卡日韩av | 国产精品午夜免费福利视频 | 欧美一区二区三区在线播放 | 久草爱 | 色综合久久88色综合天天免费 | 国产精品久久久久婷婷二区次 | 久久久久伊人 | 最近2019好看的中文字幕免费 | 国产成人黄色网址 | 日韩三区在线 | 欧美激情视频三区 | 俺要去色综合狠狠 | 波多野结衣视频一区 | 久久国产一二区 | 日韩欧美电影在线 | 日日爱影视 | 91免费国产在线观看 | 亚洲日本va午夜在线影院 | 91在线免费公开视频 | 黄色网www| av在线最新 | 91 中文字幕 | 中文字幕精 | 色射色 | 久久er99热精品一区二区三区 | 日韩精品大片 | 国产高清绿奴videos | 国产精品久久久影视 | 在线视频 区 | 黄色大片免费网站 | av丁香花| 欧美日韩xxxxx | 日韩欧美视频二区 | 国产黄视频在线观看 | 亚洲三级网站 | 手机在线中文字幕 | 久久99热精品 | 成人av av在线 | 99久久激情| 精品爱爱 | 国产视频1 | 亚洲干视频在线观看 | 国产麻豆精品免费视频 | 久久久精品国产免费观看同学 | 欧美国产日韩激情 | 欧美日本一区 | 热久久精品在线 | 91传媒在线观看 | 黄色大片日本 | 麻豆国产网站 | 国产麻豆传媒 | 国产裸体永久免费视频网站 | 国内成人精品视频 | 免费91在线 | 成人精品国产 | 91麻豆精品国产91久久久无需广告 | 9999在线 | 91视频麻豆 | 五月天激情综合网 | 美腿丝袜av | 日日爽夜夜爽 | 超黄视频网站 | 国产色资源 | 国产成人精品在线 | 免费亚洲精品视频 | 国产精品黑丝在线观看 | 免费三级av | 99re8这里有精品热视频免费 | 最新日韩视频在线观看 | 91精品免费在线视频 | 婷婷色网视频在线播放 | 91污污视频在线观看 | 97精品超碰一区二区三区 | 午夜久久久久久久久久久 | 国产色中涩 | 久久久91精品国产 | 精品国产乱码久久久久久浪潮 | 日本激情动作片免费看 | 亚洲成色777777在线观看影院 | 国产视频亚洲视频 | 国产成人黄色 | 亚洲另类视频在线 | 九色琪琪久久综合网天天 | 久久久久久综合 | 日韩精品视| 国产精品不卡 | 日韩天天综合 | 亚洲婷婷免费 | 婷婷九月丁香 | 又黄又爽的免费高潮视频 | 色偷偷人人澡久久超碰69 | 国产精品剧情 | 夜夜摸夜夜爽 | www.色婷婷.com| 毛片网站免费 | 国产伦精品一区二区三区… | 亚洲经典视频在线观看 | 亚洲国产欧美一区二区三区丁香婷 | 欧美黄在线| 一区二区理论片 | 500部大龄熟乱视频使用方法 | 久久1区| 国产亚洲久一区二区 | www.狠狠操.com| 日韩欧美高清免费 | 美女av免费| 成人啊 v | 99精品国产视频 | 久久er99热精品一区二区三区 | 激情五月在线视频 | av动态图片 | 精品在线观看一区二区三区 | 日韩欧美一区二区三区免费观看 | 欧美在线一 | 亚洲最新毛片 | 四虎国产精品永久在线国在线 | 日日天天av| 成人在线视频免费观看 | 中文字幕黄色 | 久久久久久久久久久久久久电影 | 久久久高清一区二区三区 | 干av在线| 中文字幕免费在线看 | 丝袜护士aⅴ在线白丝护士 天天综合精品 | 免费色av| 久久久午夜视频 | 成人观看视频 | 日韩久久精品一区二区 | 免费在线播放视频 | 欧美成人猛片 | 视频国产 | 国产视频在线播放 | 免费91麻豆精品国产自产在线观看 | 99热最新地址 | 亚洲国产中文字幕在线视频综合 | 久久久久麻豆v国产 | 91精品国产欧美一区二区成人 | 婷婷在线网| 精品福利在线 | 97免费| 亚洲国产操 | 黄色avwww| 久久免费的精品国产v∧ | 国产精选视频 | 亚洲色图激情文学 | 国内精品视频在线播放 | 日韩在线观看视频中文字幕 | 国产区第一页 | 午夜精品一区二区三区视频免费看 | 日本在线观看中文字幕无线观看 | 免费看国产黄色 | .国产精品成人自产拍在线观看6 | 国产精品毛片一区视频播 | 在线观看免费福利 | 成人一级免费电影 | 国产免费不卡av | 最近的中文字幕大全免费版 | 27xxoo无遮挡动态视频 | 国产成人福利片 | 久草在线最新 | 天天干,夜夜操 | 丁香五香天综合情 | 奇米影视四色8888 | 国产精品亚洲成人 | 狠狠插狠狠干 | 91丨九色丨高潮丰满 | 三级毛片视频 | 欧美激情精品久久久久 | 亚洲少妇久久 | 在线之家免费在线观看电影 | 日本韩国精品一区二区在线观看 | 亚洲精品456在线播放第一页 | 久久久www免费电影网 | 精品96久久久久久中文字幕无 | 激情久久小说 | 在线观看亚洲电影 | 亚洲一级片免费观看 | 日韩大片免费在线观看 | 天堂网中文在线 | 成人在线网站观看 | 热re99久久精品国产99热 | 久久精品毛片基地 | 天天插天天 | 四虎国产精品成人免费4hu | 热久久这里只有精品 | 国产成人黄色网址 | 日韩免费在线观看视频 | 黄色www免费 | 色七七亚洲影院 | 国产69精品久久久久久久久久 | 久久草草影视免费网 | 色综合天天 | 亚洲精品乱码久久 | 亚洲综合小说电影qvod | 国产成人不卡 | 亚洲精品久久视频 | 成人免费一区二区三区在线观看 | 日本中文在线 | 青春草视频在线播放 | 国产美女精彩久久 | 激情av网址 | 亚洲男人天堂2018 | 国产精品一区二区无线 | 天天操网址 | 国产精品手机看片 | 人人网人人爽 | 亚洲三区在线 | 超级碰碰碰免费视频 | 国产精品永久在线观看 | 性日韩欧美在线视频 | ,午夜性刺激免费看视频 | 一区二区三区在线免费播放 | 欧美精品在线视频观看 | 91av原创 | 四虎免费av | 成人av在线一区二区 | 夜夜骑日日操 | 国产 日韩 欧美 自拍 | 国产大片黄色 | 91网在线观看 | 久久久人 | 国产午夜精品一区 | 国产精品永久久久久久久www | 中文字幕在线免费97 | 欧美日韩一区二区三区在线观看视频 | 欧美性生活小视频 | 亚洲综合色视频 | 在线亚洲日本 | 日韩av伦理片 | 91激情视频在线 | 国产精品自在欧美一区 | 天天插日日射 | 亚洲精品黄色 | 久草精品视频在线播放 | 国产精品免费视频一区二区 | 狠狠干成人 | 国产在线1区 | 91精品久久久久久综合乱菊 | 亚洲视频精品在线 | 永久免费视频国产 | 欧美日韩二三区 | 国产视频每日更新 | 亚洲伦理中文字幕 | 久久狠狠一本精品综合网 | 免费看的黄色网 | 黄色三几片 | 99久久日韩精品免费热麻豆美女 | 精品视频不卡 | 日韩精品中字 | 精品视频999 | 免费日韩 精品中文字幕视频在线 | 91九色在线 | 国产精品久久久久久久久久不蜜月 | 成人精品一区二区三区中文字幕 | 久久久精品电影 | 国产精品原创av片国产免费 | 免费男女羞羞的视频网站中文字幕 | 欧美尹人| 蜜臀久久99精品久久久酒店新书 | 国产a精品 | 久久99电影 | 粉嫩av一区二区三区入口 | 色综合久久久久久中文网 | 成年人国产精品 | 国产精品久久久久亚洲影视 | 99精品在线播放 | 成人天堂网| 久久久国产精品麻豆 | 在线观看国产高清视频 | 中文字幕免费久久 | 黄色a视频免费 | 激情中文在线 | 在线黄频 | 欧美国产一区在线 | 免费黄av | 国产精品久久艹 | 91av资源网| 天天操天天射天天爽 | 一区二区三区四区精品 | 奇米影视777四色米奇影院 | 在线观看爱爱视频 | 国产 色| 成人18视频 | 中文字幕第一页在线vr | 天天操人人要 | 日韩精品久久一区二区三区 | 久青草国产在线 | 在线精品亚洲一区二区 | 成人久久18免费网站麻豆 | 天海冀一区二区三区 | 97超碰在| 欧美在线视频一区二区三区 | 日韩毛片在线免费观看 | 久久久久久在线观看 | 天天射天天做 | 欧美精品中文 | 人人干狠狠干 | av丝袜在线| 久久黄色免费视频 | 中文字幕a∨在线乱码免费看 | 日日爽视频 | 99视频国产在线 | 欧美精彩视频在线观看 | 丁香花在线视频观看免费 | 91亚洲精品久久久蜜桃网站 | 国产99久久 | 国内精品久久久久久久久久清纯 | 99精品成人 | 欧美亚洲三级 | 日本女人的性生活视频 | 久色小说 | 91大神dom调教在线观看 | 久久免费av电影 | 狠狠操狠狠操 | 国产精品女主播一区二区三区 | 久久99精品久久久久久久久久久久 | 999成人 | 99久精品视频| 久久理论视频 | 成人午夜毛片 | 免费看一及片 | 午夜少妇 | 天天摸天天操天天舔 | 国产精品刺激对白麻豆99 | 波多野结衣在线播放视频 | 最近2019中文免费高清视频观看www99 | 91久久精品一区 | 色婷婷在线观看视频 | 高清av不卡 | 日韩免费一区二区三区 | 日韩免费一二三区 | 国产精品久久久久久久久岛 | 国内精品久久久久影院一蜜桃 | 精品视频中文字幕 | 99热九九这里只有精品10 | 97超碰人人澡人人 | 天天插天天干天天操 | 国产日韩欧美在线一区 | 成人久久综合 | 午夜视频久久久 | 91大神在线观看视频 | 99久久精品国产亚洲 | 日韩免费网址 | 天天综合视频在线观看 | 日本性生活一级片 | 国产精品久久久久久久久岛 | 久久久久久久久久久久久国产精品 | 久久精品欧美视频 | 精一区二区| 91 在线视频播放 | 国产中文字幕视频在线观看 | 亚洲v精品 | 精品久久网 | 一区在线免费观看 | 天天天色综合a | 麻豆视频国产精品 | 久久成人久久 | 99热最新在线 | 国产亚洲精品久久久久久 | 高清在线一区 | 亚洲精品午夜国产va久久成人 | 国产a级片免费观看 | a视频在线观看免费 | 欧美精品久久久久久久久久 | 国产视频日韩视频欧美视频 | 日韩成人欧美 | 久久久久久久久免费视频 | 99re8这里有精品热视频免费 | 国产在线无 | 中文一区二区三区在线观看 | 黄av免费 | 国产免费又粗又猛又爽 | 亚在线播放中文视频 | 免费色视频网址 | 成人久久18免费网站图片 | 91色亚洲 | 99热99热 | 国产成人精品一区二三区 | 久久久久国产一区二区 | 亚洲视频 中文字幕 | 国产精品高潮在线观看 | 免费观看一区二区三区视频 | 欧美五月婷婷 | 日韩欧美高清免费 | 久久久国产精品网站 | 四虎影视成人永久免费观看视频 | 婷婷五情天综123 | 国产美女免费观看 | av免费观看在线 | 久九视频| 天天操天天射天天添 | 97人人看| 国产一线二线三线在线观看 | 毛片黄色一级 | 欧美精品在线观看 | 2023国产精品自产拍在线观看 | 免费高清男女打扑克视频 | 午夜性色 | 麻豆视频免费在线播放 | 午夜精品福利在线 | 99亚洲精品 | 欧美性视频网站 | 欧美巨乳波霸 | 97香蕉久久国产在线观看 | 国产午夜一区 | 亚洲综合激情小说 | 国产精品久久精品 | 911久久香蕉国产线看观看 | 欧美日韩在线视频免费 | 中文字幕一区二 | 黄色www | 欧美一级日韩免费不卡 | 2021国产在线视频 | 夜夜操狠狠操 | 欧美一区二区三区不卡 | 在线观看免费版高清版 | 韩国av三级| aaaaaa毛片| 日韩在线不卡av | 欧美一区免费观看 | 午夜在线免费观看视频 | 国产黄色成人 | 色婷婷狠| av三级在线免费观看 | 久久免费精彩视频 | 人人干人人草 | 99久久这里有精品 | 精品国产伦一区二区三区观看说明 | 在线视频区 | 手机看片午夜 | 在线视频在线观看 | 一级黄色片在线 | 97av视频 | 国产精品激情 | 在线观看中文字幕 | 天天做天天爱夜夜爽 | 欧美成天堂网地址 | 久久免费观看少妇a级毛片 久久久久成人免费 | 日韩精品在线免费播放 | 美女视频黄频大全免费 | 国产视频精选在线 | 蜜臀精品久久久久久蜜臀 | 精品国产一区二区三区久久久久久 | 中文有码在线视频 | 91精品国产福利在线观看 | 亚洲精品女人 | 国产成人精品av在线观 | 天天干天天综合 | 日韩在线视频播放 | 久久影院亚洲 | 日韩乱码中文字幕 | 97视频精品 | 精品免费| 国产又粗又猛又爽又黄的视频免费 | 337p日本欧洲亚洲大胆裸体艺术 | 国产成人综合图片 | 日韩激情中文字幕 | 国产精品欧美激情在线观看 | 丁香六月欧美 | 97超碰福利久久精品 | 欧美亚洲成人xxx | 亚在线播放中文视频 | 免费网站黄 | 91精品视频网站 | 嫩小bbbb摸bbb摸bbb | 97色狠狠 | 国产手机在线观看 | 丁香综合网 | 中文字幕4 | 天天操偷偷干 | 一级片视频在线 | 在线国产小视频 | 色亚洲网 | 99久久精品国产毛片 | av三级av| 精品视频在线观看 | 免费人做人爱www的视 | 国产裸体视频网站 | 国产一级片久久 | 成人av在线网址 | 五月天激情综合网 | 亚洲 欧美 变态 国产 另类 | 97在线观看 | 日韩免费不卡视频 | 日韩av一区二区三区 | 久久99网站 | 欧美va天堂va视频va在线 | 91视视频在线直接观看在线看网页在线看 | 日色在线视频 | 欧美午夜精品久久久久久浪潮 | 久久国产精品第一页 | 2022中文字幕在线观看 | 玖草影院 | 中文字幕精品三区 | 亚洲最新在线视频 | 丁香婷婷自拍 | 九九久久影视 | av一二三区 | 日本黄色免费在线观看 | 久久黄色精品视频 | 久久婷婷一区二区三区 | 久草在线欧美 | 天天玩天天干天天操 | 91精品国产三级a在线观看 | 久久综合五月 | 日韩av在线影视 | 高清不卡毛片 | 91精品国产乱码久久桃 | 天天操天天摸天天爽 | 丝袜制服天堂 | 91av在线不卡 | 亚洲国产精品电影在线观看 | 日韩一区精品 | 日韩中文字幕91 | 最近的中文字幕大全免费版 | 精品一区二区三区四区在线 | 亚州精品国产 | 亚洲mv大片欧洲mv大片免费 | 九九热视频在线播放 | 国产一区二区久久 | 国产精品99久久99久久久二8 | 在线观看www视频 | 99精品美女 | 超级av在线 | 九九精品无码 | 精品久久毛片 | 黄色在线视频网址 | 九九日九九操 | 99精品视频在线播放观看 | 国产亚洲精品成人av久久影院 | 中文字幕 欧美性 | 亚洲国产天堂av | 亚洲精品小视频 | 在线观看免费一区 | 日韩精品在线视频免费观看 | 国产免费看 | 精品国产激情 | 国产高清av免费在线观看 | 精品国产精品久久 | 欧美日本在线视频 | av日韩中文| 久久伊99综合婷婷久久伊 | 人人干人人做 | 天天操比 | 欧美激精品 | 一区二区不卡在线观看 | 精品一区二区日韩 | 欧美性久久久 | www.在线看片.com | 亚洲视频精品在线 | 久久亚洲福利 | 深爱综合网 | 亚洲禁18久人片 | 亚洲视频高清 | 久久99精品视频 | 国产精品日韩在线观看 | 久久精品视频国产 | 色婷婷一区 | av在线免费观看网站 | 爱爱一区| 黄色国产高清 | 亚洲综合导航 | 天天色综合1 | 激情图片qvod| 操处女逼 | 成人av一区二区三区 | www.色就是色 | 日韩资源在线 | 久久久久久电影 | 99热超碰在线 | 三级毛片视频 | a精品视频 | 国产中文字幕亚洲 | 亚洲国产日本 | 国产99久久久久 | 亚洲狠狠| 精品一区av | 色综合久久久久综合体桃花网 | 成人精品一区二区三区电影免费 | 成人国产精品电影 | 草久在线观看视频 | 国产偷v国产偷∨精品视频 在线草 | 亚洲视频456| 国外成人在线视频网站 | 亚洲乱码精品久久久久 | 一区二区欧美在线观看 | 这里只有精品视频在线 | av在线小说 | 日韩精品视频在线观看免费 | a在线观看视频 | 成人欧美日韩国产 | 国产成人精品不卡 | 国产在线91精品 | 国产精品福利无圣光在线一区 | 亚洲国产字幕 | 天天舔天天搞 | 日韩电影一区二区在线观看 | 日韩色在线观看 | 色网站国产精品 | 色综合久久88色综合天天6 | 久久综合九色综合久久久精品综合 | 日韩久久午夜一级啪啪 | 国产九色91 | 男女全黄一级一级高潮免费看 | 日本系列中文字幕 | 成年人视频在线免费播放 | 国产男女爽爽爽免费视频 | 麻豆国产精品视频 | 天天天天天天操 | 伊人狠狠色丁香婷婷综合 | 久久国产精品系列 | 久色小说 | www.久久精品视频 | 亚洲精品中文在线资源 | 国产一区二区午夜 | 免费国产在线精品 | 亚洲国产美女精品久久久久∴ | 日韩免费二区 | 国产高清视频在线观看 | 五月婷婷香蕉 | av久久久久久 | 日韩美av在线 | 日本精品久久 | 国产成人久久av977小说 | 91干干干 | 亚洲综合视频网 | 天天躁日日躁狠狠躁av中文 | www.色国产 | 在线观看国产区 | 天天干,天天射,天天操,天天摸 | 性色大片在线观看 | 国产精品久久久久久久久久三级 | 久久久午夜精品理论片中文字幕 | 国产精品福利午夜在线观看 | 亚州av网站大全 | 成人在线播放免费观看 | 色偷偷88欧美精品久久久 | 午夜精品久久久久久久99无限制 | 在线播放一区二区三区 | 久久免费国产精品 | 亚洲第一区在线播放 | 中文字幕有码在线 | 免费无遮挡动漫网站 | 九九热在线观看 | 四虎影院在线观看av | 波多在线视频 | 99精品欧美一区二区蜜桃免费 | 夜色.com | 免费a级大片 | 一区二区三区日韩视频在线观看 | 福利网址在线观看 | 国产精品美女久久久久久久 | 亚洲国产日韩av | 久久免费99 | 超级av在线 | 亚洲理论在线观看电影 | 综合色爱 | 六月丁香伊人 | 成人精品一区二区三区电影免费 | 亚洲高清视频一区二区三区 | 日韩高清黄色 | 国产日韩欧美在线观看视频 | 欧美另类交在线观看 | 欧美综合在线观看 | 亚洲美女免费视频 | 波多野结衣精品视频 | 在线免费黄 | 国产一区免费在线观看 | 91麻豆精品一区二区三区 | 国产美女久久久 | 免费h视频 | 2023亚洲精品国偷拍自产在线 | 黄色高清视频在线观看 | 久久久久国产精品一区二区 | 四虎在线观看视频 | 免费在线观看黄色网 | av手机版 | 亚洲成人av在线播放 | 人人爽人人澡人人添人人人人 | 婷婷综合伊人 | 国产精品久久久久久影院 | 国产精久久| 九热在线| 久久国产精品免费一区二区三区 | 激情丁香在线 | 深夜免费福利网站 | 免费视频18 | 一级片视频在线 | 在线天堂中文在线资源网 | 国内久久精品 | 天天玩天天干天天操 | 六月色丁香 | 日韩欧美一区二区在线 | 国产精品一区二区你懂的 | 日韩理论 | 成人h电影在线观看 | 四虎永久免费在线观看 | 久草在线久 | 国产一区二区三区网站 | 在线精品观看国产 | 国产96视频| 欧美国产日韩一区二区三区 | 激情综合电影网 | 成人免费xxxxxx视频 | 中文字幕婷婷 | 在线免费视 | 免费福利在线 | 天天鲁天天干天天射 | 亚洲区另类春色综合小说 | 五月综合久久 | 欧美日韩69 | 香蕉久草 | 亚洲黄色免费网站 | 特级片免费看 | 99视频精品 | 樱空桃av | 亚洲做受高潮欧美裸体 | 久久久国际精品 | 天堂网一区二区三区 | 久久网址 | 天天爽夜夜爽人人爽一区二区 | 午夜资源站 | 最新日本中文字幕 | 亚洲精品视| 国内一区二区视频 | 亚洲最新av网站 | 日韩在线观看不卡 | 亚洲国产精品成人va在线观看 | 888av| 麻豆久久久久久久 | 一级性视频 | 2023av| 欧美日韩高清一区二区 国产亚洲免费看 | 日韩精品久久久久久久电影99爱 | 黄色小网站免费看 | 久久久久久久久久久久久久免费看 | 国内精品久久久久久久影视简单 | 久久久久99精品国产片 | 日韩免费在线视频 | 久久免费的精品国产v∧ | 黄色毛片在线观看 | 99久久久国产精品免费观看 | 欧美aaa视频 | 欧美国产日韩激情 | 粉嫩av一区二区三区四区五区 | 色播五月激情综合网 | 久久免费视频2 | 黄色av免费电影 | 91九色精品女同系列 | 欧美一区二区三区在线观看 | 人人爱在线视频 | 久久电影中文字幕视频 | 亚洲人在线 | 伊人色播 | 国产成人一区二 | 99色国产| 免费av网站在线 | av中文字幕网站 | 久精品在线观看 | av大片免费| 日韩av黄 | 91桃色国产在线播放 | 久久免费看毛片 | 国产特级毛片aaaaaa毛片 | 激情在线免费视频 | 91在线超碰 | 亚洲成人免费在线观看 | 天天干天天做天天爱 | 99精品久久99久久久久 | 在线看国产 | 成人国产精品免费观看 | 国产日韩视频在线播放 | 国产成人精品一区二区在线观看 | 在线观看亚洲精品 | 西西4444www大胆视频 | 国产免码va在线观看免费 | 免费看一及片 | 久久精品这里热有精品 | 999成人 | 国产亚洲精品综合一区91 | 亚洲精品国产精品国自产观看浪潮 | 欧美精品亚洲精品日韩精品 | 中文字幕在线一区观看 | 99久热在线精品视频 | 成人免费91 | 成人黄色小说视频 | 在线 你懂 | 天天搞夜夜骑 | 国产黄影院色大全免费 | 精品久久久999 | 国产精品九九视频 | 69久久夜色精品国产69 | 天天骚夜夜操 | 日韩电影在线观看一区二区 | 97视频人人 | av解说在线 | 日韩视频一区二区三区 | 国产高清一区二区 | 久久爱资源网 | 在线观看www91 | 久久av黄色| 国产五十路毛片 | 日韩爱爱网站 | 国产精品美女久久久久久2018 | 亚洲视频99 | 97超碰人人澡人人爱 | 国产高清成人 | 麻豆91精品91久久久 | 国产一区二区电影在线观看 | 黄污污网站| 99久e精品热线免费 99国产精品久久久久久久久久 | 碰超在线| 色婷婷www | 欧美巨乳波霸 | 成人a在线观看 | 亚在线播放中文视频 | 中文字幕色站 | 日韩精品久久一区二区 | 久久1区| 国产四虎在线 | av在线com| 久久视频在线看 | 国产精品中文 |