在OpenCV中使用地址清理程序

Plu*_*luc 10 c++ opencv sanitizer clang++ address-sanitizer

我正在尝试将Google的Address Sanitizer与CUDA项目结合使用,更准确地说是使用OpenCV cuda功能.但是我在第一次cuda电话上遇到了"内存不足"的错误.

OpenCV Error: Gpu API call (out of memory) in getDevice, file opencv-2.4.11/src/opencv-2.4.11/modules/dynamicuda/include/opencv2/dynamicuda/dynamicuda.hpp, line 664
terminate called after throwing an instance of 'cv::Exception'
  what():  opencv-2.4.11/src/opencv-2.4.11/modules/dynamicuda/include/opencv2/dynamicuda/dynamicuda.hpp:664: error: (-217) out of memory in function getDevice
Run Code Online (Sandbox Code Playgroud)

它可以复制

#include <opencv2/gpu/gpu.hpp>
int main()
{
  cv::gpu::printCudaDeviceInfo(cv::gpu::getDevice());
  return 0;
}
Run Code Online (Sandbox Code Playgroud)

用.编译

clang++ -fsanitize=address -lstdc++ -lopencv_gpu -lopencv_core -o sanitizer sanitizer.cpp && LD_LIBRARY_PATH=/usr/local/lib ./sanitizer
Run Code Online (Sandbox Code Playgroud)

我用gcc得到了同样的结果.我也尝试过将cuda函数列入黑名单而没有结果.

现在使用没有opencv的cuda:

#include <cuda_runtime.h>
int main()
{
  int count = -1;
  cudaGetDevice(&count);
  cout << "Device count: " << count << endl;
  return 0;
}
clang++ -O1 -g -fsanitize=address -fsanitize-blacklist=asan.blacklist -stdlib=libstdc++ -lstdc++ -I/opt/cuda/include -L/opt/cuda/lib64 -lcudart  -o sanitizer sanitizer.cpp && ./sanitizer
Run Code Online (Sandbox Code Playgroud)

消毒剂在内存泄漏时停止:

=================================================================
==25344==ERROR: LeakSanitizer: detected memory leaks

Direct leak of 136 byte(s) in 1 object(s) allocated from:
    #0 0x4bc4a2  (/home/pluc/work/tests/sanitizer+0x4bc4a2)
    #1 0x7f71f0fa69ba  (<unknown module>)

SUMMARY: AddressSanitizer: 136 byte(s) leaked in 1 allocation(s).
Run Code Online (Sandbox Code Playgroud)

我的问题是如何使用地址清洁剂来清理我的软件而不会遇到这种情况?我怎样才能至少正确地将所有与cuda相关的电话列入黑名单?

我没有在着名的网络搜索引擎上找到任何相关内容.这就像人们不使用cuda或asan或两者兼而有之.那些人是否只有cuda完全禁用的构建?

我猜asan在使用cuda内存管理方面遇到了困难,但我正在寻找一种方法,至少可以在我的代码库的其余部分使用这个工具.

小智 0

对于在这里查看的人:我找到的解决方案(基于@BenC提到的相关github问题)是运行export ASAN_OPTIONS=protect_shadow_gap=0.

我还没有成功地找到一种方法来改变编译时的行为。