- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
所有,我不知道我达到了什么限制,也不知道这是否是 valgrind
、libc
或 me
的问题,但我需要知道这是否可以重现,如果可以,问题出在哪里。我已将问题归结为可在我的 2 个 AMD 机器上生成的 MCVE。基本上,我动态分配 struct dirent * 指针,然后为每个成功的 readdir 分配一个 struct dirent 。 valgrind
对 1017
没有任何提示,但在数字 1018
上,我收到一个 invalid read
错误(没有重新分配)涉及),例如
==9881== Invalid read of size 8
==9881== at 0x4C2F316: memcpy@@GLIBC_2.14 (in /usr/lib/valgrind/vgpreload_memcheck-amd64-linux.so)
==9881== by 0x40098E: main (readdir_mcve.c:35)
==9881== Address 0x51df070 is 0 bytes after a block of size 32,816 alloc'd
==9881== at 0x4C2ABD0: malloc (in /usr/lib/valgrind/vgpreload_memcheck-amd64-linux.so)
==9881== by 0x4EE93E3: __alloc_dir (in /usr/lib/libc-2.23.so)
==9881== by 0x4EE94D2: opendir_tail (in /usr/lib/libc-2.23.so)
==9881== by 0x400802: main (readdir_mcve.c:9)
(大小为 32,816 的 block
看起来很奇怪,但我没有找到任何帮助来分解它)
该代码将要打开的目录名称作为第一个参数,然后将要读取的文件限制作为第二个参数(默认为 1000
):
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <sys/types.h>
#include <dirent.h>
int main (int argc, char **argv) {
DIR *dp = opendir (argc > 1 ? argv[1] : "."); /* open directory (. default) */
struct dirent *de = NULL, **dlist = NULL; /* ptr and ptr2ptr to dirent */
size_t nptrs = argc > 2 ? (size_t)strtoul (argv[2], NULL, 10) : 1000,
i = 0, idx = 0; /* index, allocation counter */
if (!dp) {
fprintf (stderr, "error: opendir failed.\n");
return 1;
}
/* allocate nptrs dirent pointers */
if (!(dlist = calloc (nptrs, sizeof *dlist))) {
fprintf (stderr, "error: virtual memory exhausted - dlist\n");
return 1;
}
while ((de = readdir (dp))) {
/* skip dot files */
if (!strcmp (de->d_name, ".") || !strcmp (de->d_name, ".."))
continue;
if (!(dlist[idx] = calloc (1, sizeof **dlist))) { /* alloc dirent */
fprintf (stderr, "error: dlist memory allocation failed\n");
return 1;
}
memcpy (dlist[idx++], de, sizeof *de); /* copy de to dlist[idx] */
if (idx == nptrs) /* post-check/realloc, insures sentinel NULL */
break;
}
closedir (dp);
for (i = 0; i < idx; i++) {
printf (" file[%3zu] : %s\n", i, dlist[i]->d_name);
free (dlist[i]);
}
free (dlist);
return 0;
}
您可以使用以下命令创建一个简单的测试目录:
$ mkdir readdir_tst
$ for i in {1..1024}; do
printf -v fname "file%04d" "$i"
touch "readdir_tst/$fname"
done
然后一切都很好,读取 1014
文件名(1017
分配):
$ valgrind ./bin/readdir_mcve readdir_tst 1014 > /dev/null
==9880== Memcheck, a memory error detector
==9880== Copyright (C) 2002-2015, and GNU GPL'd, by Julian Seward et al.
==9880== Using Valgrind-3.11.0 and LibVEX; rerun with -h for copyright info
==9880== Command: ./bin/readdir_mcve readdir_tst 1014
==9880==
==9880==
==9880== HEAP SUMMARY:
==9880== in use at exit: 0 bytes in 0 blocks
==9880== total heap usage: 1,017 allocs, 1,017 frees, 328,944 bytes allocated
==9880==
==9880== All heap blocks were freed -- no leaks are possible
==9880==
==9880== For counts of detected and suppressed errors, rerun with: -v
==9880== ERROR SUMMARY: 0 errors from 0 contexts (suppressed: 0 from 0)
但是在文件 1015
(分配 1018
)上,我遇到了 __alloc_dir
问题,并且 valgrind 抛出 Invalid read of size 8 。 .. 在分配了大小为 32,816 的 block 之后是 0 字节
:
$ valgrind ./bin/readdir_mcve readdir_tst 1015 > /dev/null
==9881== Memcheck, a memory error detector
==9881== Copyright (C) 2002-2015, and GNU GPL'd, by Julian Seward et al.
==9881== Using Valgrind-3.11.0 and LibVEX; rerun with -h for copyright info
==9881== Command: ./bin/readdir_mcve readdir_tst 1015
==9881==
==9881== Invalid read of size 8
==9881== at 0x4C2F316: memcpy@@GLIBC_2.14 (in /usr/lib/valgrind/vgpreload_memcheck-amd64-linux.so)
==9881== by 0x40098E: main (readdir_mcve.c:35)
==9881== Address 0x51df070 is 0 bytes after a block of size 32,816 alloc'd
==9881== at 0x4C2ABD0: malloc (in /usr/lib/valgrind/vgpreload_memcheck-amd64-linux.so)
==9881== by 0x4EE93E3: __alloc_dir (in /usr/lib/libc-2.23.so)
==9881== by 0x4EE94D2: opendir_tail (in /usr/lib/libc-2.23.so)
==9881== by 0x400802: main (readdir_mcve.c:9)
==9881==
==9881==
==9881== HEAP SUMMARY:
==9881== in use at exit: 0 bytes in 0 blocks
==9881== total heap usage: 1,018 allocs, 1,018 frees, 329,232 bytes allocated
==9881==
==9881== All heap blocks were freed -- no leaks are possible
==9881==
==9881== For counts of detected and suppressed errors, rerun with: -v
==9881== ERROR SUMMARY: 1 errors from 1 contexts (suppressed: 0 from 0)
代码继续正常读取和打印所有目录条目,但令我困惑的是 valgrind
错误。我让它重新分配,而不是在达到分配限制时调用 break
,并且它处理 /usr/bin
中的 4000+
文件,没有任何其他问题比 valgrind
错误。 (因与 MCVE 无关而被删除)。我在 SO 上找到的最接近的东西是 Valgrind malloc leaks ,但这不适用于此处。其他人可以重现这个吗?如果可以,这是 valgrind
、libc
还是 me
?
注意:我使用libc-2.18.so
得到了相同的结果。
GNU libc dirent.h 提供了更多信息
在答案指向正确的方向并继续搜索之后,libc 似乎可以通过多种方式确定 d_name
的长度。它将取决于编译器可用的各种定义。在 dirent.h解释:
46 /* This file defines `struct dirent'.
47
48 It defines the macro `_DIRENT_HAVE_D_NAMLEN' iff there is a `d_namlen'
49 member that gives the length of `d_name'.
...
59 */
...
67 /* These macros extract size information from a `struct dirent *'.
68 They may evaluate their argument multiple times, so it must not
69 have side effects. Each of these may involve a relatively costly
70 call to `strlen' on some systems, so these values should be cached.
71
72 _D_EXACT_NAMLEN (DP) returns the length of DP->d_name, not including
73 its terminating null character.
74
75 _D_ALLOC_NAMLEN (DP) returns a size at least (_D_EXACT_NAMLEN (DP) + 1);
76 that is, the allocation size needed to hold the DP->d_name string.
77 Use this macro when you don't need the exact length, just an upper bound.
78 This macro is less likely to require calling `strlen' than _D_EXACT_NAMLEN.
79 */
80
81 #ifdef _DIRENT_HAVE_D_NAMLEN
82 # define _D_EXACT_NAMLEN(d) ((d)->d_namlen)
83 # define _D_ALLOC_NAMLEN(d) (_D_EXACT_NAMLEN (d) + 1)
84 #else
85 # define _D_EXACT_NAMLEN(d) (strlen ((d)->d_name))
86 # ifdef _DIRENT_HAVE_D_RECLEN
87 # define _D_ALLOC_NAMLEN(d) (((char *) (d) + (d)->d_reclen) - &(d)->d_name[0])
88 # else
89 # define _D_ALLOC_NAMLEN(d) (sizeof (d)->d_name > 1 ? sizeof (d)->d_name : \
90 _D_EXACT_NAMLEN (d) + 1)
91 # endif
92 #endif
...
虽然根据不同的定义可以采用多种不同的编译路径,但如果设置了 __USE_XOPEN2K8
,则会出现最具可读性的编译路径:
221 #ifdef __USE_XOPEN2K8
222 ...
230 # ifdef __USE_MISC
231 # ifndef MAXNAMLEN
232 /* Get the definitions of the POSIX.1 limits. */
233 # include <bits/posix1_lim.h>
234
235 /* `MAXNAMLEN' is the BSD name for what POSIX calls `NAME_MAX'. */
236 # ifdef NAME_MAX
237 # define MAXNAMLEN NAME_MAX
238 # else
239 # define MAXNAMLEN 255
240 # endif
241 # endif
242 # endif
因此,在这种情况下,d_name
是 NAME_MAX
或 255
,具体取决于 NAME_MAX
定义(因此通过 _D_ALLOC_NAMLEN (DP)
宏设置为 256
)。感谢放松为我指明了正确的方向。我不知道我们是否能知道为什么 1017
struct dirent
分配没有问题发生以及为什么 valgrind
开始提示的确切答案编号 1018
,但至少我们现在了解问题的根源在哪里,以及为什么使用 memcpy
复制 struct dirent
可能会出现问题。
最佳答案
你似乎不能像这样复制struct dirent
the manual page并且代码不同步。
struct dirent
{
#ifndef __USE_FILE_OFFSET64
__ino_t d_ino; /* File serial number. */
#else
__ino64_t d_ino;
#endif
unsigned short int d_reclen; /* Length of the whole `struct dirent'. */
unsigned char d_type; /* File type, possibly unknown. */
unsigned char d_namlen; /* Length of the file name. */
/* Only this member is in the POSIX standard. */
char d_name[1]; /* File name (actually longer). */
};
显然,由于 d_name
被声明为 [1]
,因此您将无法使用 sizeof
获得正确的大小。您需要进行更巧妙的存储,即 strdup()
名称或其他内容(如果您只对名称感兴趣)。
我不是 100% 确定为什么这会导致损坏,但我打赌您会看到某种 UB(请注意,如果您到达 printf()
,您将在阅读过程中死去在您复制的会触发 UB 的字符串上)。
关于c - Valgrind 1017 calloc/memcopy of struct dirent OK,第 1018 次 -- 无效读取,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/38117417/
我不明白为什么我的 C 编译器会抛出一个严重的错误。我根据 man page 初始化了所有东西,或者至少我认为我做了。 .然而我仍然被抛出一个严重的错误。它一直说 expected int (*)(c
我正在尝试编写一个程序来读取有关程序所在文件夹的信息。我已经仔细检查了其他在线资源,看起来我正在使用与其他正确解决方案相同的模式(见下文)。 #include #include #include
最终更新 - 答案位于已接受答案的评论中。 首先我意识到这个问题还有很多其他答案。我已经经历了其中的大部分,并且这段代码是经历了许多其他答案的组合。我想要做的就是获取目录中每个文件的完整路径。 #in
平台:Windows XP Service Pack 3 编译器:Code::Blocks 版本 12.11 我目前正在编写一个程序,该程序将使用 POSIX 目录函数递归删除给定目录。但我对 rea
你能告诉我,如何在下面的代码中只将文件名(没有目录)放入我的 vector 中吗?: int getDir (string dir, vector &files) { DIR *dp;
我已经编写了用于我在 Windows 上的应用程序的虚拟系统 (VFS)。现在,我将我的应用程序移至 iOS,但我遇到了 dirrent 问题。 Windows 端口添加了关于当前文件夹的信息,文件所
我正在 Ubunutu 16.04 linux 上开发一个 C++ 程序 它是从 shell 读取目录路径和组宽度。然后它应该在目录中导航并跟踪文件,如果它找到一个目录进入它并跟踪那些文件。并在末尾打
public:vector> cont; void memContent(string path, int f){ DIR *dir; struct dirent *ent; int
如果给定一个 std::string 形式的目录路径,我如何获得指向该目录的 dirent 结构? 我可以先使用 opendir() 获取所需目录的 DIR*。然后我可以在我的 DIR* 上使用 re
我尝试对文件夹中的文件进行计数,但 readdir 函数会跳过包含 unicode 字符的文件。我在 c 中使用 dirent。 int filecount(char* path) { int
我写了一个小函数,它接收一个文件夹的路径,并且应该返回其中的文件数。 出于某种原因,我的答案不正确:打印时我看到它打印了 . 和 ... 谁能告诉我为什么?减少 2 会解决问题吗? 代码如下:
我搜索了 dirent 以找到 C 例程 find / -iregex ".*/dirent.h$" 2>/dev/null 哪些返回相同的系列 /Library/Developer/CommandL
我有一个目录(我们称之为目录 A),其中包含约 15000 个目录(B1、B2、..、B15000),B 的每个目录都包含一个我想要的文件(我们称之为“原始”)读。所以我想阅读:A/B1/原始A/B2
我正在学习使用 dirent.h。虽然这个过程引人入胜且有趣,但我在使用 d_name 时遇到了问题。 我想使用 d_name 做两件事。 递归搜索子目录。对此,当我遇到DT_DIR类型的文件时,我会
有一些问题试图想出一个好方法来拥有一个只选择 wav 文件并专门返回这些文件的字符串的目录遍历器,例如只是wav文件。我可以停止 dirent.h 在子目录中查找,我很确定我只需要一个 if 语句,它
我正在构建一个需要 dirent.h 的 C++ 代码我发现这个文件已经被include了SYSROOT = $(NDK)/platforms/android-/arch-arm/ $SYSROOT/
我需要以单独的颜色打印 .c 文件、可执行文件和目录。请给我一些想法。提前致谢 int main(void) { DIR *d; int iNum = 0;
这个问题已经有答案了: Checking if a dir. entry returned by readdir is a directory, link or file. dent->d_type
我以前从未使用过dirent.h。我正在使用 istringstream 来读取文本文件(单数),但需要尝试修改程序以读取目录中的多个文本文件。这是我尝试实现 dirent 的地方,但它不起作用。 也
我正在尝试使用 dirent.h 中提供的功能递归打开文件 我的问题是:我无法跳过无法打开的目录。我希望它打开它可以打开的目录并跳过它不能打开的目录并移动到下一个目录而不是失败退出。我应该怎么做才能解
我是一名优秀的程序员,十分优秀!