summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--usr.bin/diff/diffdir.c37
1 files changed, 22 insertions, 15 deletions
diff --git a/usr.bin/diff/diffdir.c b/usr.bin/diff/diffdir.c
index 1a870701533..771052243fb 100644
--- a/usr.bin/diff/diffdir.c
+++ b/usr.bin/diff/diffdir.c
@@ -1,4 +1,4 @@
-/* $OpenBSD: diffdir.c,v 1.28 2004/10/02 18:13:24 millert Exp $ */
+/* $OpenBSD: diffdir.c,v 1.29 2004/11/26 20:09:56 otto Exp $ */
/*
* Copyright (c) 2003 Todd C. Miller <Todd.Miller@courtesan.com>
@@ -21,7 +21,7 @@
*/
#ifndef lint
-static const char rcsid[] = "$OpenBSD: diffdir.c,v 1.28 2004/10/02 18:13:24 millert Exp $";
+static const char rcsid[] = "$OpenBSD: diffdir.c,v 1.29 2004/11/26 20:09:56 otto Exp $";
#endif /* not lint */
#include <sys/param.h>
@@ -165,7 +165,7 @@ static struct dirent **
slurpdir(char *path, char **bufp, int enoentok)
{
char *buf, *ebuf, *cp;
- size_t bufsize;
+ size_t bufsize, have, need;
long base;
int fd, nbytes, entries;
struct stat sb;
@@ -181,28 +181,35 @@ slurpdir(char *path, char **bufp, int enoentok)
}
return (&dummy);
}
- fstat(fd, &sb);
+ if (fstat(fd, &sb) == -1) {
+ warn("%s", path);
+ close(fd);
+ return (NULL);
+ }
+
+ need = roundup(sb.st_blksize, sizeof(struct dirent));
+ have = bufsize = roundup(MAX(sb.st_size, sb.st_blksize),
+ sizeof(struct dirent)) + need;
+ ebuf = buf = emalloc(bufsize);
- bufsize = 0;
- ebuf = buf = NULL;
do {
- bufsize += roundup(MAX(sb.st_size, sb.st_blksize),
- sizeof(struct dirent));
- if (buf == NULL)
- buf = ebuf = emalloc(bufsize);
- else {
+ if (have < need) {
+ bufsize += need;
+ have += need;
cp = erealloc(buf, bufsize);
ebuf = cp + (ebuf - buf);
buf = cp;
}
- nbytes = getdirentries(fd, ebuf, bufsize, &base);
+ nbytes = getdirentries(fd, ebuf, have, &base);
if (nbytes == -1) {
- free(buf);
warn("%s", path);
+ free(buf);
+ close(fd);
return (NULL);
}
ebuf += nbytes;
- } while (nbytes == bufsize);
+ have -= nbytes;
+ } while (nbytes != 0);
close(fd);
/*
@@ -232,7 +239,7 @@ slurpdir(char *path, char **bufp, int enoentok)
}
dirlist[entries] = NULL;
- qsort(dirlist, entries, sizeof(struct dir *), dircompare);
+ qsort(dirlist, entries, sizeof(struct dirent *), dircompare);
*bufp = buf;
return (dirlist);