fix handling of d_name in struct dirent
authorRich Felker <dalias@aerifal.cx>
Mon, 6 Jun 2011 22:04:28 +0000 (18:04 -0400)
committerRich Felker <dalias@aerifal.cx>
Mon, 6 Jun 2011 22:04:28 +0000 (18:04 -0400)
basically there are 3 choices for how to implement this variable-size
string member:
1. C99 flexible array member: breaks using dirent.h with pre-C99 compiler.
2. old way: length-1 string: generates array bounds warnings in caller.
3. new way: length-NAME_MAX string. no problems, simplifies all code.

of course the usable part in the pointer returned by readdir might be
shorter than NAME_MAX+1 bytes, but that is allowed by the standard and
doesn't hurt anything.

include/dirent.h
src/dirent/scandir.c
src/regex/glob.c

index ca000bd..7b70abd 100644 (file)
@@ -18,7 +18,7 @@ struct dirent
        off_t d_off;
        unsigned short d_reclen;
        unsigned char d_type;
-       char d_name[1];
+       char d_name[256];
 };
 
 #define d_fileno d_ino
index aad813a..a85cfac 100644 (file)
@@ -12,7 +12,7 @@ int scandir(const char *path, struct dirent ***res,
 {
        DIR *d = opendir(path);
        struct dirent *de, **names=0, **tmp;
-       size_t cnt=0, len=0, size;
+       size_t cnt=0, len=0;
        int old_errno = errno;
 
        if (!d) return -1;
@@ -26,10 +26,9 @@ int scandir(const char *path, struct dirent ***res,
                        if (!tmp) break;
                        names = tmp;
                }
-               size = offsetof(struct dirent,d_name) + strlen(de->d_name) + 1;
-               names[cnt] = malloc(size);
+               names[cnt] = malloc(de->d_reclen);
                if (!names[cnt]) break;
-               memcpy(names[cnt++], de, size);
+               memcpy(names[cnt++], de, de->d_reclen);
        }
 
        closedir(d);
index 67f84bc..550f655 100644 (file)
@@ -53,8 +53,7 @@ static int append(struct match **tail, const char *name, size_t len, int mark)
 static int match_in_dir(const char *d, const char *p, int flags, int (*errfunc)(const char *path, int err), struct match **tail)
 {
        DIR *dir;
-       long long de_buf[(sizeof(struct dirent) + NAME_MAX + sizeof(long long))/sizeof(long long)];
-       struct dirent *de;
+       struct dirent de_buf, *de;
        char pat[strlen(p)+1];
        char *p2;
        size_t l = strlen(d);
@@ -94,7 +93,7 @@ static int match_in_dir(const char *d, const char *p, int flags, int (*errfunc)(
                closedir(dir);
                return error;
        }
-       while (!(error = readdir_r(dir, (void *)de_buf, &de)) && de) {
+       while (!(error = readdir_r(dir, &de_buf, &de)) && de) {
                char namebuf[l+de->d_reclen+2], *name = namebuf;
                if (!literal && fnmatch(p, de->d_name, fnm_flags))
                        continue;