gpt4 book ai didi

c++ - 我遇到了 fopen_s 的内存泄漏,但没有 'new' ed 或 'malloc' ed

转载 作者:塔克拉玛干 更新时间:2023-11-03 07:18:06 27 4
gpt4 key购买 nike

我正在使用 MS VS2010 和一个名为 Deleaker 的插件来发现我可能错过的任何内存泄漏。它告诉我 fopen_s 行有 2 处内存泄漏,但我没有在该行的任何内容上使用 new 或 malloc。每次它发现泄漏时,位置都在现场,所以我不认为它看错了线。有什么建议吗?

注意:图像加载正常,我使用的是 LibPNG,OPAL 是我自己的 DLL,图像和图像->数据在使用此 DLL 的应用程序中释放。

我希望我已经提供了足够的信息(但不要太多)

OPAL_API void LoadPNGImage(const char* filename, OPAL::GUI::ImageStruct *&image)
{
int bit_depth, color_type, interlace_type;
png_structp png_ptr;
png_infop info_ptr;
png_uint_32 width, height;
unsigned char* line;
unsigned int sig_read = 0;
unsigned int x, y;
FILE *fp;
image = (OPAL::GUI::ImageStruct*)malloc(sizeof(OPAL::GUI::ImageStruct));
memset(image, 0, sizeof(OPAL::GUI::ImageStruct));
if(fopen_s(&fp, filename, "rb")) // 2 MEMORY LEAKS DETECTED HERE (False Positive??)
{
image = NULL;
return;
}
png_ptr = png_create_read_struct(PNG_LIBPNG_VER_STRING, NULL, NULL, NULL);
if (png_ptr == NULL)
{
free(image);
image = NULL;
fclose(fp);
return;
}
png_set_error_fn(png_ptr, (png_voidp) NULL, (png_error_ptr) NULL, user_warning_fn);
info_ptr = png_create_info_struct(png_ptr);
if (info_ptr == NULL)
{
free(image);
image = NULL;
fclose(fp);
png_destroy_read_struct(&png_ptr, png_infopp_NULL, png_infopp_NULL);
return;
}
png_init_io(png_ptr, fp);
png_set_sig_bytes(png_ptr, sig_read);
png_read_info(png_ptr, info_ptr);
png_get_IHDR(png_ptr, info_ptr, &width, &height, &bit_depth, &color_type, &interlace_type, int_p_NULL, int_p_NULL);
png_set_packing(png_ptr);
image->width = width;
image->height = height;
image->data = (unsigned char*)malloc(sizeof(unsigned char) * 4 * image->width * image->height);
memset(image->data, 0, sizeof(unsigned char) * 4 * image->width * image->height);
if(!image->data)
{
free(image->data);
free(image);
image = NULL;
fclose(fp);
png_destroy_read_struct(&png_ptr, png_infopp_NULL, png_infopp_NULL);
return;
}
line = (unsigned char*)malloc(width * 4);
if(!line)
{
image = NULL;
fclose(fp);
png_destroy_read_struct(&png_ptr, png_infopp_NULL, png_infopp_NULL);
return;
}
for(y = 0; y < height; y++)
{
png_read_row(png_ptr, (unsigned char*)line, png_bytep_NULL);
for(x = 0; x < width; x++)
{
switch(color_type)
{
case PNG_COLOR_TYPE_GRAY:
// Don't wanna support this mode until I need it
break;
case PNG_COLOR_TYPE_PALETTE:
// Don't wanna support this mode until I need it
break;
case PNG_COLOR_TYPE_RGB:
image->data[4 * ((y * width) + x) + 0] = (unsigned char)line[(3 * x) + 0];
image->data[4 * ((y * width) + x) + 1] = (unsigned char)line[(3 * x) + 1];
image->data[4 * ((y * width) + x) + 2] = (unsigned char)line[(3 * x) + 2];
image->data[4 * ((y * width) + x) + 3] = (unsigned char)255;
break;
case PNG_COLOR_TYPE_RGBA:
image->data[(4 * ((y * width) + x)) + 0] = (unsigned char)line[(4 * x) + 0];
image->data[(4 * ((y * width) + x)) + 1] = (unsigned char)line[(4 * x) + 1];
image->data[(4 * ((y * width) + x)) + 2] = (unsigned char)line[(4 * x) + 2];
image->data[(4 * ((y * width) + x)) + 3] = (unsigned char)line[(4 * x) + 3];
break;
}
}
}
free(line);
png_read_end(png_ptr, info_ptr);
png_destroy_read_struct(&png_ptr, &info_ptr, png_infopp_NULL);
fclose(fp);
}

原来是文件名。当它作为 const char* 直接传递时,我认为我不需要清除它

这是泄漏检测器错误地标记我还是我真的需要以某种方式释放它?

函数是这样调用的:LoadPNGImage("images\\tiles00.png", tiles);

最佳答案

当像 fopen 这样的函数进行一次性分配时,就会发生这种情况。如果您随后再次调用 fopen,您将不会得到更多泄漏。

Deleaker 试图隐藏此类“已知泄漏”,但有时它们仍会显示。

调试这种情况我发现“泄漏”来自 CRT 中的这段代码:

int __cdecl _mtinitlocknum (
int locknum
)
{
...
if ( (pcs = _malloc_crt(sizeof(CRITICAL_SECTION))) == NULL ) { <-- HERE!
errno = ENOMEM;
return FALSE;
}
...
_locktable[locknum].lock = pcs;

这个分配应该在 _mtdeletelocks 中释放:

void __cdecl _mtdeletelocks(
void
)
{
...
for ( locknum = 0 ; locknum < _TOTAL_LOCKS ; locknum++ ) {
if ( _locktable[locknum].lock != NULL &&
_locktable[locknum].kind != lkPrealloc )
...
_free_crt(pcs);

_mtterm(似乎只有 _mtterm!)调用 _mtdeletelocks。但正如我所见,_mtterm 根本没有被调用。如果 CRT 被 EXE 使用,那么 CRT 可能不是一个大的泄漏。

同时,如果 DLL 使用 CRT,则在 DLL_PROCESS_DETACH 处理程序中调用 _mtterm - 对于 DLL,此类泄漏似乎很重要(DLL 可以加载和卸载多次)!

关于c++ - 我遇到了 fopen_s 的内存泄漏,但没有 'new' ed 或 'malloc' ed,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/30512626/

27 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com