使用 VLD 記憶體漏失檢測工具輔助開發時整理的學習筆記。本篇主要介紹 VLD 庫提供的 22 個外部介面。同系列文章目錄可見 《記憶體漏失檢測工具》目錄
VLD 2.5.1 安裝 完成後,安裝目錄的 include
資料夾下有兩個標頭檔案:vld.h
和 vld_def.h
,其中 vld.h
檔案會 #include "vld_def.h"
,因此在實際使用時,專案中要同時新增這兩個標頭檔案(或將這兩個檔案放在編譯器的搜尋路徑中),但只需包含 vld.h
檔案。
這個檔案裡主要以宏的形式定義了 15 個 VLD 設定項的掩碼,這 15 個設定項與 vld.ini
組態檔中的 14 個設定項不是完全對應的,將這些設定項掩碼與 vld.h
檔案中的介面結合起來用,可以實現在執行過程中對 VLD 的設定進行動態修改。其中 9 個設定項可作為介面 VLDSetOptions
的輸入,另外 4 個設定項可作為介面 VLDSetReportOptions
的輸入,剩餘的 2 個設定項分別是 VLD_OPT_SELF_TEST
與 VLD_OPT_VLDOFF
,其中 VLD_OPT_SELF_TEST
只能通過修改 vld.ini
檔案中的 SelfTest
進行設定(詳見 設定項 SelfTest),VLD_OPT_VLDOFF
只能通過修改 vld.ini
檔案中的 VLD
進行設定(詳見 設定項 VLD),當 VLD_OPT_VLDOFF
被設定後,所有介面也都會變得不可用。
#define VLD_OPT_AGGREGATE_DUPLICATES 0x0001 // If set, aggregate duplicate leaks in the leak report.
#define VLD_OPT_MODULE_LIST_INCLUDE 0x0002 // If set, modules in the module list are included, all others are excluded.
#define VLD_OPT_REPORT_TO_DEBUGGER 0x0004 // If set, the memory leak report is sent to the debugger.
#define VLD_OPT_REPORT_TO_FILE 0x0008 // If set, the memory leak report is sent to a file.
#define VLD_OPT_SAFE_STACK_WALK 0x0010 // If set, the stack is walked using the "safe" method (StackWalk64).
#define VLD_OPT_SELF_TEST 0x0020 // If set, perform a self-test to verify memory leak self-checking.
#define VLD_OPT_SLOW_DEBUGGER_DUMP 0x0040 // If set, inserts a slight delay between sending output to the debugger.
#define VLD_OPT_START_DISABLED 0x0080 // If set, memory leak detection will initially disabled.
#define VLD_OPT_TRACE_INTERNAL_FRAMES 0x0100 // If set, include useless frames (e.g. internal to VLD) in call stacks.
#define VLD_OPT_UNICODE_REPORT 0x0200 // If set, the leak report will be encoded UTF-16 instead of ASCII.
#define VLD_OPT_VLDOFF 0x0400 // If set, VLD will be completely deactivated. It will not attach to any modules.
#define VLD_OPT_REPORT_TO_STDOUT 0x0800 // If set, the memory leak report is sent to stdout.
#define VLD_OPT_SKIP_HEAPFREE_LEAKS 0x1000 // If set, VLD skip HeapFree memory leaks.
#define VLD_OPT_VALIDATE_HEAPFREE 0x2000 // If set, VLD verifies and reports heap consistency for HeapFree calls.
#define VLD_OPT_SKIP_CRTSTARTUP_LEAKS 0x4000 // If set, VLD skip crt srtartup memory leaks.
這個檔案裡主要宣告了 VLD 的 22 個外部介面。檔案前面的編譯條件 defined _DEBUG || defined VLD_FORCE_ENABLE
表明 VLD 通常只能在 DEBUG
模式下執行,若要在 RELEASE
模式下執行,可以在包含標頭檔案 vld.h
前預先定義 VLD_FORCE_ENABLE
宏。一個有趣的現象是,這個 _DEBUG
宏有時候在 RELEASE
模式下也會被定義,參考文章 神祕的 _DEBUG 宏從何處來?。介面中的一些型別別名定義如下:
typedef int VLD_BOOL;
typedef unsigned int VLD_UINT;
typedef size_t VLD_SIZET;
typedef void* VLD_HMODULE;
該函數用於禁用當前執行緒的記憶體漏失檢測功能。
// VLDDisable - Disables Visual Leak Detector's memory leak detection at
// runtime. If memory leak detection is already disabled, then calling this
// function has no effect.
//
// Note: In multithreaded programs, this function operates on a per-thread
// basis. In other words, if you call this function from one thread, then
// memory leak detection is only disabled for that thread. If memory leak
// detection is enabled for other threads, then it will remain enabled for
// those other threads. It was designed to work this way to insulate you,
// the programmer, from having to ensure thread synchronization when calling
// VLDEnable() and VLDDisable(). Without this, calling these two functions
// unsynchronized could result in unpredictable and unintended behavior.
// But this also means that if you want to disable memory leak detection
// process-wide, then you need to call this function from every thread in
// the process.
//
// Return Value:
//
// None.
//
__declspec(dllimport) void VLDDisable ();
該函數用於啟用當前執行緒的記憶體漏失檢測功能。
// VLDEnable - Enables Visual Leak Detector's memory leak detection at runtime.
// If memory leak detection is already enabled, which it is by default, then
// calling this function has no effect.
//
// Note: In multithreaded programs, this function operates on a per-thread
// basis. In other words, if you call this function from one thread, then
// memory leak detection is only enabled for that thread. If memory leak
// detection is disabled for other threads, then it will remain disabled for
// those other threads. It was designed to work this way to insulate you,
// the programmer, from having to ensure thread synchronization when calling
// VLDEnable() and VLDDisable(). Without this, calling these two functions
// unsynchronized could result in unpredictable and unintended behavior.
// But this also means that if you want to enable memory leak detection
// process-wide, then you need to call this function from every thread in
// the process.
//
// Return Value:
//
// None.
//
__declspec(dllimport) void VLDEnable ();
該函數用於還原當前執行緒記憶體漏失檢測功能的開關狀態。
// VLDRestore - Restore Visual Leak Detector's previous state.
//
// Return Value:
//
// None.
//
__declspec(dllimport) void VLDRestore ();
為了便於理解,下面貼出這一介面對應的原始碼(詳見 vld.cpp 第 2548~2561
行):
void VisualLeakDetector::RestoreLeakDetectionState ()
{
tls_t *tls;
if (m_options & VLD_OPT_VLDOFF) {
// VLD has been turned off.
return;
}
// Restore state memory leak detection for the current thread.
tls = getTls();
tls->flags &= ~(VLD_TLS_DISABLED | VLD_TLS_ENABLED);
tls->flags |= tls->oldFlags & (VLD_TLS_DISABLED | VLD_TLS_ENABLED);
}
該函數用於全域性禁用 VLD 功能。
// VLDGlobalDisable - Disables Visual Leak Detector's memory leak detection at
// runtime in all threads. If memory leak detection is already disabled,
// then calling this function has no effect.
//
// Return Value:
//
// None.
//
__declspec(dllimport) void VLDGlobalDisable ();
該函數用於全域性啟用 VLD 功能。
// VLDGlobalEnable - Enables Visual Leak Detector's memory leak detection
// at runtime in all threads. If memory leak detection is already enabled,
// which it is by default, then calling this function has no effect.
//
// Return Value:
//
// None.
//
__declspec(dllimport) void VLDGlobalEnable ();
該函數用於列印整個程式當前執行此函數前的記憶體漏失報告。由於在程式關閉後 VLD 會自動輸出報告,因此若不需要在程式執行過程中輸出報告,就不需使用此函數。
// VLDReportLeaks - Report leaks up to the execution point.
//
// Return Value:
//
// None.
//
__declspec(dllimport) VLD_UINT VLDReportLeaks ();
該函數用於列印指定執行緒在執行此函數前的記憶體漏失報告。
// VLDReportThreadLeaks - Report thread leaks up to the execution point.
//
// threadId: thread Id.
//
// Return Value:
//
// None.
//
__declspec(dllimport) VLD_UINT VLDReportThreadLeaks (VLD_UINT threadId);
該函數用於獲取整個程式當前的記憶體漏失數量。
// VLDGetLeaksCount - Return memory leaks count to the execution point.
//
// Return Value:
//
// None.
//
__declspec(dllimport) VLD_UINT VLDGetLeaksCount ();
該函數用於獲取指定執行緒當前的記憶體漏失數量。
// VLDGetThreadLeaksCount - Return thread memory leaks count to the execution point.
//
// threadId: thread Id.
//
// Return Value:
//
// None.
//
__declspec(dllimport) VLD_UINT VLDGetThreadLeaksCount (VLD_UINT threadId);
該函數用於標記當前的洩漏為已經報告過,後續不再報告。
// VLDMarkAllLeaksAsReported - Mark all leaks as reported.
//
// Return Value:
//
// None.
//
__declspec(dllimport) void VLDMarkAllLeaksAsReported ();
該函數用於標記指定執行緒當前的洩漏為已經報告過,後續不再報告。
// VLDMarkThreadLeaksAsReported - Mark thread leaks as reported.
//
// threadId: thread Id.
//
// Return Value:
//
// None.
//
__declspec(dllimport) void VLDMarkThreadLeaksAsReported (VLD_UINT threadId);
該函數用於重新整理載入的模組列表,以便針對動態載入的模組進行記憶體漏失檢查。
// VLDRefreshModules - Look for recently loaded DLLs and patch them if necessary.
//
// Return Value:
//
// None.
//
__declspec(dllimport) void VLDRefreshModules();
該函數用於對指定模組啟用記憶體漏失檢測。輸入引數為指定模組(dll
或者 exe
)的控制程式碼,可以通過呼叫 Win32
的 API
來獲得(常用的有 GetModuleHandleW 函數、LoadLibraryA 函數)。
// VLDEnableModule - Enable Memory leak checking on the specified module.
//
// module: module handle.
//
// Return Value:
//
// None.
//
__declspec(dllimport) void VLDEnableModule(VLD_HMODULE module);
例如,若想檢測一個已載入當前程序的動態連結庫 test.dll
是否存在記憶體漏失,可以按以下方式使用該介面(需包含標頭檔案 Windows.h
,若為動態載入的模組,還需額外使用 VLDRefreshModules()
重新整理內部的模組列表):
// 呼叫 Win32 API 獲得模組控制程式碼
HMODULE h_dll = GetModuleHandleW(L"test.dll");
// 對該模組啟用 VLD 功能
VLDEnableModule(h_dll);
// 呼叫模組中的函數
...
該函數用於對指定模組禁用記憶體漏失檢測。與 VLDEnableModule
相對應,輸入引數為指定模組(dll
或者 exe
)的控制程式碼,可以通過呼叫 Win32
的API
來獲得。
// VLDDisableModule - Disable Memory leak checking on the specified module.
//
// module: module handle.
//
// Return Value:
//
// None.
//
__declspec(dllimport) void VLDDisableModule(VLD_HMODULE module);
該函數用於獲取當前的設定掩碼值(與 VLDSetOptions
相對應),將掩碼值結合 VLDSetOptions
的 9 個設定掩碼宏可以人工推斷出 VLD 當前的設定。
// VLDGetOptions - Return all current options.
//
// Return Value:
//
// Mask of current options.
//
__declspec(dllimport) VLD_UINT VLDGetOptions();
例如,若先前使用 VLDSetOptions
介面對 VLD 做了以下設定:
VLDSetOptions(VLD_OPT_AGGREGATE_DUPLICATES | VLD_OPT_SKIP_HEAPFREE_LEAKS, 256, 64);
則 VLDGetOptions()
的返回值為 4097
,計算方式如下:
VLD_OPT_AGGREGATE_DUPLICATES | VLD_OPT_SKIP_HEAPFREE_LEAKS == 0x1001 == 4097
若呼叫此函數前,未使用 VLDSetOptions
介面更改設定,且未修改 vld.ini
組態檔,則該函數的返回值為預設的 16386
,即 0x4002
。
該函數用於獲取生成的洩漏報告檔案路徑。注意:用於儲存檔案路徑的 wchar_t
陣列 filename
需預分配足夠大的記憶體,以防出現意外的情況,MAX_PATH
為 windows
系統下檔案路徑的最大長度(詳見 MAX_PATH,值為 260
)。
// VLDGetReportFilename - Return current report filename.
//
// filename: current report filename (max characters - MAX_PATH).
//
// Return Value:
//
// None.
//
__declspec(dllimport) void VLDGetReportFilename(wchar_t *filename);
該函數用於設定設定掩碼值(由相應的設定掩碼宏通過邏輯運算得到)、設定為每個洩漏塊資料轉儲的最大位元組數(參考 設定項 MaxDataDump)、設定對每個洩漏塊進行堆疊跟蹤的最大幀數(參考 設定項 MaxTraceFrames)。
// VLDSetOptions - Update the report options via function call rather than INI file.
//
// option_mask: Only the following flags are checked
// VLD_OPT_AGGREGATE_DUPLICATES
// VLD_OPT_MODULE_LIST_INCLUDE
// VLD_OPT_SAFE_STACK_WALK
// VLD_OPT_SLOW_DEBUGGER_DUMP
// VLD_OPT_TRACE_INTERNAL_FRAMES
// VLD_OPT_START_DISABLED
// VLD_OPT_SKIP_HEAPFREE_LEAKS
// VLD_OPT_VALIDATE_HEAPFREE
//
// maxDataDump: maximum number of user-data bytes to dump for each leaked block.
//
// maxTraceFrames: maximum number of frames per stack trace for each leaked block.
//
// Return Value:
//
// None.
//
__declspec(dllimport) void VLDSetOptions(VLD_UINT option_mask, VLD_SIZET maxDataDump, VLD_UINT maxTraceFrames);
檢視該介面的 原始碼 可知,第一個引數的合法輸入為以下 9 個設定掩碼宏之一(上面的介面說明並不全面)、或者它們的邏輯或(|
)組合:
// option_mask 的合法輸入:以下 9 個設定宏之一
#define VLD_OPT_AGGREGATE_DUPLICATES 0x0001 // If set, aggregate duplicate leaks in the leak report.
#define VLD_OPT_MODULE_LIST_INCLUDE 0x0002 // If set, modules in the module list are included, all others are excluded.
#define VLD_OPT_SAFE_STACK_WALK 0x0010 // If set, the stack is walked using the "safe" method (StackWalk64).
#define VLD_OPT_SLOW_DEBUGGER_DUMP 0x0040 // If set, inserts a slight delay between sending output to the debugger.
#define VLD_OPT_START_DISABLED 0x0080 // If set, memory leak detection will initially disabled.
#define VLD_OPT_TRACE_INTERNAL_FRAMES 0x0100 // If set, include useless frames (e.g. internal to VLD) in call stacks.
#define VLD_OPT_SKIP_HEAPFREE_LEAKS 0x1000 // If set, VLD skip HeapFree memory leaks.
#define VLD_OPT_VALIDATE_HEAPFREE 0x2000 // If set, VLD verifies and reports heap consistency for HeapFree calls.
#define VLD_OPT_SKIP_CRTSTARTUP_LEAKS 0x4000 // If set, VLD skip crt srtartup memory leaks.
例如,我想同時開啟 VLD_OPT_AGGREGATE_DUPLICATES
與 VLD_OPT_SKIP_HEAPFREE_LEAKS
,並設定 maxDataDump=256
,maxTraceFrames=64
,可以用以下幾種方式傳參,它們的效果是一樣的:
VLDSetOptions(VLD_OPT_AGGREGATE_DUPLICATES | VLD_OPT_SKIP_HEAPFREE_LEAKS, 256, 64);
VLDSetOptions(0x1001, 256, 64);
VLDSetOptions(4097, 256, 64);
需要注意的是,每次使用此函數時,內部都會先將這 9 個設定置零,然後使用新輸入的設定,因此,若輸入不合法,就會丟失對應 bit
處上一次的設定狀態,具體細節可檢視原始碼。
該函數用於設定要包含或者排除洩漏檢測的模組列表。閱讀 原始碼 可知,第一個引數 modules
的最大有效長度為 512
。
// VLDSetModulesList - Set list of modules included/excluded in leak detection
// depending on parameter "includeModules".
//
// modules: list of modules to be forcefully included/excluded in leak detection.
//
// includeModules: include or exclude that modules.
//
// Return Value:
//
// None.
//
__declspec(dllimport) void VLDSetModulesList(const wchar_t *modules, VLD_BOOL includeModules);
原始碼(詳見 vld.cpp 第 867~882 行)中判斷某個模組 modulename
是否在所給列表 m_forcedModuleList
中的核心程式碼如下,由於使用的是 wcsstr
函數 進行字串比較,因此模組名之間可用任意字元或字串進行分隔,且不區分大小寫,這與組態檔 vld.ini
中的 ForceIncludeModules
設定項一樣(詳見 設定項 ForceIncludeModules)。另外,從原始碼中也可以看出,當第二個引數為 false
時,將執行後面的判斷語句(條件為 wcsstr(m_forcedModuleList, modulename) != NULL
),意味著不在列表中的模組都會被開啟記憶體檢測,這可能會導致很多誤判,被大量的偽洩漏刷屏,因此需慎傳 false
。
// This module does not import VLD. This means that none of the module's
// sources #included vld.h.
if ((m_options & VLD_OPT_MODULE_LIST_INCLUDE) != 0)
{
if (wcsstr(m_forcedModuleList, modulename) == NULL) {
// Exclude this module from leak detection.
moduleFlags |= VLD_MODULE_EXCLUDED;
}
}
else
{
if (wcsstr(m_forcedModuleList, modulename) != NULL) {
// Exclude this module from leak detection.
moduleFlags |= VLD_MODULE_EXCLUDED;
}
}
在 QT 中實際使用時(測試環境:QT 5.9.2,MSVC 2015 32bit,Debug 模式,VLD 版本為 2.5.1),發現這個函數好像只能修改內部的 m_forcedModuleList
列表,並沒有按照第二個引數 includeModules
立馬更新指定模組的檢測狀態,即使接著使用 VLDRefreshModules()
函數進行重新整理,也沒有實現按新列表進行檢測的效果。因此,要實現對某個 DLL
庫的動態檢測控制,最好還是使用 VLDEnableModule
、VLDDisableModule
、VLDRefreshModules
這三個介面。若沒有動態檢測控制的需求,則可以修改組態檔 vld.ini
中的 ForceIncludeModules
設定項(詳見 設定項 ForceIncludeModules),修改組態檔是沒有這個問題的。
該函數用於獲取檢測中的模組列表。獲取的模組列表是由 VLDSetModulesList
設定的,也可能是由 vld.ini
檔案中的 ForceIncludeModules
設定的,用於儲存模組列表的 wchar_t
陣列 modules
需預分配足夠大的記憶體(通常為 512
),以防出現意外的情況。第二個引數 size
為想要獲取的字串長度,一般設定為第一個引數 modules
的長度(通常為 512
)。
// VLDGetModulesList - Return current list of included/excluded modules
// depending on flag VLD_OPT_TRACE_INTERNAL_FRAMES.
//
// modules: destination string for list of included/excluded modules (maximum length 512 characters).
//
// size: maximum string size.
//
// Return Value:
//
// VLD_BOOL: TRUE if include modules, otherwise FALSE.
//
__declspec(dllimport) VLD_BOOL VLDGetModulesList(wchar_t *modules, VLD_UINT size);
該函數用於設定洩漏報告輸出方式。
// VLDSetReportOptions - Update the report options via function call rather than INI file.
//
// Only the following flags are checked
// VLD_OPT_REPORT_TO_DEBUGGER
// VLD_OPT_REPORT_TO_FILE
// VLD_OPT_REPORT_TO_STDOUT
// VLD_OPT_UNICODE_REPORT
//
// filename is optional and can be NULL.
//
// Return Value:
//
// None.
//
__declspec(dllimport) void VLDSetReportOptions(VLD_UINT option_mask, const wchar_t *filename);
第一個引數的合法輸入為以下 4 個設定掩碼宏之一、或者它們的邏輯或(|
)組合,第二個引數可以為 NULL
但不能省略。
// option_mask 的合法輸入:以下 4 個設定宏之一
#define VLD_OPT_REPORT_TO_DEBUGGER 0x0004 // If set, the memory leak report is sent to the debugger.
#define VLD_OPT_REPORT_TO_FILE 0x0008 // If set, the memory leak report is sent to a file.
#define VLD_OPT_REPORT_TO_STDOUT 0x0800 // If set, the memory leak report is sent to stdout.
#define VLD_OPT_UNICODE_REPORT 0x0200 // If set, the leak report will be encoded UTF-16 instead of ASCII.
需要注意的是,如果設定了 VLD_OPT_UNICODE_REPORT
,即使沒設定 VLD_OPT_REPORT_TO_FILE
,洩漏報告也會輸出到檔案,這與 vld.ini
檔案中的 設定項 ReportEncoding 效果一樣。閱讀 原始碼 可知,若設定了 VLD_OPT_REPORT_TO_STDOUT
,程式會呼叫 fputs(messagea, stdout)
輸出洩漏報告至標準輸出窗,若設定了 VLD_OPT_REPORT_TO_DEBUGGER
,程式會呼叫 OutputDebugStringW(messagew)
輸出洩露報告至偵錯窗(詳見 OutputDebugStringW 函數)。
該函數用於自定義記憶體漏失報告回撥函數,與 CrtSetReportHook 函數 很相似。
// VLDSetReportHook - Installs or uninstalls a client-defined reporting function by hooking it
// into the C run-time debug reporting process (debug version only).
//
// mode: The action to take: VLD_RPTHOOK_INSTALL or VLD_RPTHOOK_REMOVE.
//
// pfnNewHook: Report hook to install or remove.
//
// Return Value:
//
// int: 0 if success.
//
__declspec(dllimport) int VLDSetReportHook(int mode, VLD_REPORT_HOOK pfnNewHook);
其中的 VLD_REPORT_HOOK
在 vld_def.h
中有給出別名宣告,自定義的報告函數必須是以下型別的函數。
typedef int (__cdecl * VLD_REPORT_HOOK)(int reportType, wchar_t *message, int *returnValue);
其中的 VLD_RPTHOOK_INSTALL
與 VLD_RPTHOOK_REMOVE
也在 vld_def.h
中有給出宏定義,安裝自定義報告函數時傳 VLD_RPTHOOK_INSTALL
,解除安裝時傳 VLD_RPTHOOK_INSTALL
,若設定成功,VLDSetReportHook
返回 0
,否則返回 -1
。
#define VLD_RPTHOOK_INSTALL 0
#define VLD_RPTHOOK_REMOVE 1
檢視 API 原始碼(詳見 utility.cpp 第 674~774 行),可以得到以下幾點資訊:
true
(非零)時,其後不會呼叫預設的報告輸出函數,返回值為 false
(零)時,其後仍會呼叫預設的報告輸出函數。reportType
值恆為 0
,可以不使用這個值;OutputDebugStringW
或 fputs
進行列印,其他輸出函數可能無法正常使用,因為 VLD 原始碼未包含對應標頭檔案);*returnValue = 1
時,會觸發 __debugbreak()
函數(詳見 __debugbreak 函數),將在程式碼中引起斷點,並在其中提示使用者執行偵錯程式,當 *returnValue != 1
時,無任何影響。例如,我定義了以下自定義報告函數:
int MyReportHook(int, wchar_t* message, int* returnValue)
{
// 不讓VLD觸發__debugbreak()函數
*returnValue = 0;
// 使用預設的 Block 資訊
if (wcsstr(message, L"Block") != NULL) {
return false;
}
// 使用自定義的退出資訊
if (wcsstr(message, L"Visual Leak Detector is now exiting") != NULL) {
wchar_t wcs[512]{};
wcscpy (wcs, L"This is a custom output: ");
wcscat (wcs, message);
OutputDebugStringW(wcs);
return true;
}
// 忽略其他資訊
return true;
}
然後在 main
主函數的開頭新增上:
VLDSetReportHook(VLD_RPTHOOK_INSTALL, MyReportHook);
DEBUG
模式下執行程式,程式結束後得到以下結果:
Visual Leak Detector read settings from: D:\Program Files (x86)\Visual Leak Detector\vld.ini
Visual Leak Detector Version 2.5.1 installed.
---------- Block 2 at 0x015D2418: 40 bytes ----------
---------- Block 1 at 0x015D27E0: 40 bytes ----------
This is a custom output: Visual Leak Detector is now exiting.
未使用自定義報告函數時(或者緊接著使用 VLD_RPTHOOK_REMOVE
解除安裝 MyReportHook
)的輸出為:
Visual Leak Detector read settings from: D:\Program Files (x86)\Visual Leak Detector\vld.ini
Visual Leak Detector Version 2.5.1 installed.
WARNING: Visual Leak Detector detected memory leaks!
---------- Block 2 at 0x00964198: 40 bytes ----------
Leak Hash: 0x01EF1389, Count: 1, Total 40 bytes
Call Stack (TID 10072):
ucrtbased.dll!malloc()
f:\dd\vctools\crt\vcstartup\src\heap\new_array.cpp (15): testVLD.exe!operator new[]() + 0x9 bytes
e:\cworkspace\qt 5.9\qtdemo\testvld\main.cpp (60): testVLD.exe!main() + 0x7 bytes
f:\dd\vctools\crt\vcstartup\src\startup\exe_common.inl (74): testVLD.exe!invoke_main() + 0x1B bytes
f:\dd\vctools\crt\vcstartup\src\startup\exe_common.inl (264): testVLD.exe!__scrt_common_main_seh() + 0x5 bytes
f:\dd\vctools\crt\vcstartup\src\startup\exe_common.inl (309): testVLD.exe!__scrt_common_main()
f:\dd\vctools\crt\vcstartup\src\startup\exe_main.cpp (17): testVLD.exe!mainCRTStartup()
KERNEL32.DLL!BaseThreadInitThunk() + 0x19 bytes
ntdll.dll!RtlGetAppContainerNamedObjectPath() + 0x11E bytes
ntdll.dll!RtlGetAppContainerNamedObjectPath() + 0xEE bytes
Data:
CD CD CD CD CD CD CD CD CD CD CD CD CD CD CD CD ........ ........
CD CD CD CD CD CD CD CD CD CD CD CD CD CD CD CD ........ ........
CD CD CD CD CD CD CD CD ........ ........
---------- Block 1 at 0x00964248: 40 bytes ----------
Leak Hash: 0x5D0E4327, Count: 1, Total 40 bytes
Call Stack (TID 10072):
ucrtbased.dll!malloc()
f:\dd\vctools\crt\vcstartup\src\heap\new_array.cpp (15): testVLD.exe!operator new[]() + 0x9 bytes
e:\cworkspace\qt 5.9\qtdemo\testvld\main.cpp (59): testVLD.exe!main() + 0x7 bytes
f:\dd\vctools\crt\vcstartup\src\startup\exe_common.inl (74): testVLD.exe!invoke_main() + 0x1B bytes
f:\dd\vctools\crt\vcstartup\src\startup\exe_common.inl (264): testVLD.exe!__scrt_common_main_seh() + 0x5 bytes
f:\dd\vctools\crt\vcstartup\src\startup\exe_common.inl (309): testVLD.exe!__scrt_common_main()
f:\dd\vctools\crt\vcstartup\src\startup\exe_main.cpp (17): testVLD.exe!mainCRTStartup()
KERNEL32.DLL!BaseThreadInitThunk() + 0x19 bytes
ntdll.dll!RtlGetAppContainerNamedObjectPath() + 0x11E bytes
ntdll.dll!RtlGetAppContainerNamedObjectPath() + 0xEE bytes
Data:
CD CD CD CD CD CD CD CD CD CD CD CD CD CD CD CD ........ ........
CD CD CD CD CD CD CD CD CD CD CD CD CD CD CD CD ........ ........
CD CD CD CD CD CD CD CD ........ ........
Visual Leak Detector detected 2 memory leaks (152 bytes).
Largest number used: 152 bytes.
Total allocations: 152 bytes.
Visual Leak Detector is now exiting.
該函數用於對所有已跟蹤的呼叫堆疊進行符號解析,當呼叫堆疊中存在已解除安裝的模組函數時,可以呼叫此函數重新整理呼叫堆疊中的符號資訊(函數名、行號、指令偏移量等資訊)。
// VLDResolveCallstacks - Performs symbol resolution for all saved extent CallStack's that have
// been tracked by Visual Leak Detector. This function is necessary for applications that
// dynamically load and unload modules, and through which memory leaks might be included.
// If this is NOT called, stack traces may have stack frames with no symbol information. This
// happens because the symbol API's cannot look up symbols for a binary / module that has been unloaded
// from the process.
//
// Return Value:
//
// int: 0 if successfully resolved all callstacks.
//
__declspec(dllexport) int VLDResolveCallstacks();
按編碼規範來說,這個介面前面應該是 __declspec(dllimport)
,但它實際卻是 __declspec(dllexport)
,這種用法不知是庫作者筆誤,還是別有用途,有興趣的可以深究下去。
VLDDisable
、VLDEnable
、VLDRestore
可以做到只檢測指定執行緒的記憶體漏失,或者排除指定執行緒的記憶體漏失檢測。VLDGlobalDisable
、VLDGlobalEnable
可以做到只檢測特定時間階段的記憶體漏失,比如只檢測程式完成某項任務期間的記憶體漏失,而不檢測其他時間段的記憶體漏失。VLDEnableModule
、VLDDisableModule
、VLDSetModulesList
、VLDGetModulesList
、VLDRefreshModules
、VLDResolveCallstacks
可以實現對指定模組進行記憶體檢測的動態控制。VLDReportLeaks
、VLDReportThreadLeaks
、VLDMarkAllLeaksAsReported
、VLDMarkThreadLeaksAsReported
、VLDSetReportOptions
、VLDSetReportHook
、VLDGetReportFilename
可以實現對洩漏報告的動態客製化。VLDGetLeaksCount
、VLDGetThreadLeaksCount
可以實現對洩漏資訊的實時獲取。VLDGetOptions
、VLDSetOptions
可以實現執行過程中動態修改 VLD 的設定。本文作者:木三百川
本文連結:https://www.cnblogs.com/young520/p/17324641.html
版權宣告:本文系博主原創文章,著作權歸作者所有。商業轉載請聯絡作者獲得授權,非商業轉載請附上出處連結。遵循 署名-非商業性使用-相同方式共用 4.0 國際版 (CC BY-NC-SA 4.0) 版權協定。