summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorIngo Schwarze <schwarze@cvs.openbsd.org>2009-12-19 22:41:40 +0000
committerIngo Schwarze <schwarze@cvs.openbsd.org>2009-12-19 22:41:40 +0000
commit5b2b598d3bd2c4374a944b8b2c66147ec7bdf9a0 (patch)
treed9aaeb952fb48e92726a3e788870cb85e1eb12e1
parent1fe64736e6d768c64774010591775e1b292863f2 (diff)
Implement proper error handling for getgrent(3), getgrnam(3) and getgrgid(3).
1) Distinguish normal end of data (like eof, YPERR_NOMORE, YPERR_KEY) from error conditions. 2) Save errno(2) and restore it in case of success, as suggested by deraadt@ and beck@. While this code is touched, * move the common code for the yp_first and yp_next calls after the respective if/else-block; note that free(data) is not needed in case of failure and that __ypcurrent = NULL does no harm after failing yp_first. * return 0, not NULL: the return value of grscan is int, not a pointer. This is the last part of the work done on getgrent.c during c2k9, heavily relying on the other improvements done at that time. This has been rotting in my tree far too long. "get it in" deraadt@; also lightly tested by simon@ some time ago
-rw-r--r--lib/libc/gen/getgrent.c70
1 files changed, 43 insertions, 27 deletions
diff --git a/lib/libc/gen/getgrent.c b/lib/libc/gen/getgrent.c
index 5b52f960797..0acf4e8183c 100644
--- a/lib/libc/gen/getgrent.c
+++ b/lib/libc/gen/getgrent.c
@@ -1,4 +1,4 @@
-/* $OpenBSD: getgrent.c,v 1.35 2009/11/09 00:18:27 kurt Exp $ */
+/* $OpenBSD: getgrent.c,v 1.36 2009/12/19 22:41:39 schwarze Exp $ */
/*
* Copyright (c) 1989, 1993
* The Regents of the University of California. All rights reserved.
@@ -271,11 +271,13 @@ grscan(int search, gid_t gid, const char *name, struct group *p_gr,
#endif
char **members;
char *line;
+ int saved_errno;
if (gs == NULL)
return 0;
members = gs->members;
line = gs->line;
+ saved_errno = errno;
for (;;) {
#ifdef YP
@@ -285,35 +287,33 @@ grscan(int search, gid_t gid, const char *name, struct group *p_gr,
__ypcurrent, __ypcurrentlen,
&key, &keylen, &data, &datalen);
free(__ypcurrent);
- if (r) {
- __ypcurrent = NULL;
- __ypmode = 0;
- free(data);
- continue;
- }
__ypcurrent = key;
__ypcurrentlen = keylen;
- bcopy(data, line, datalen);
- free(data);
} else {
r = yp_first(__ypdomain, "group.byname",
&__ypcurrent, &__ypcurrentlen,
&data, &datalen);
- if (r) {
- __ypmode = 0;
- free(data);
+ }
+ if (r) {
+ __ypmode = 0;
+ __ypcurrent = NULL;
+ if (r == YPERR_NOMORE)
continue;
- }
- bcopy(data, line, datalen);
- free(data);
+ else
+ return 0;
}
+ bcopy(data, line, datalen);
+ free(data);
line[datalen] = '\0';
bp = line;
goto parse;
}
#endif
- if (!fgets(line, sizeof(gs->line), _gr_fp))
- return(0);
+ if (!fgets(line, sizeof(gs->line), _gr_fp)) {
+ if (feof(_gr_fp) && !ferror(_gr_fp))
+ errno = saved_errno;
+ return 0;
+ }
bp = line;
/* skip lines that are too big */
if (!strchr(line, '\n')) {
@@ -346,7 +346,8 @@ grscan(int search, gid_t gid, const char *name, struct group *p_gr,
case '\n':
if (foundyp) {
*foundyp = 1;
- return (NULL);
+ errno = saved_errno;
+ return 0;
}
if (!search) {
__ypmode = 1;
@@ -362,8 +363,14 @@ grscan(int search, gid_t gid, const char *name, struct group *p_gr,
r = yp_match(__ypdomain, "group.bygid",
buf, strlen(buf), &data, &datalen);
}
- if (r != 0)
+ switch (r) {
+ case 0:
+ break;
+ case YPERR_KEY:
continue;
+ default:
+ return 0;
+ }
bcopy(data, line, datalen);
free(data);
line[datalen] = '\0';
@@ -390,20 +397,28 @@ grscan(int search, gid_t gid, const char *name, struct group *p_gr,
continue;
r = yp_match(__ypdomain, "group.byname",
bp, strlen(bp), &data, &datalen);
- if (r)
+ switch (r) {
+ case 0:
+ break;
+ case YPERR_KEY:
continue;
+ default:
+ return 0;
+ }
bcopy(data, line, datalen);
free(data);
line[datalen] = '\0';
bp = line;
}
} else if (line[0] == '-') {
- if(!__ypexclude_add(&__ypexhead,
- strsep(&line, ":\n") + 1))
- if (foundyp) {
- *foundyp = -1;
- return (NULL);
- }
+ if (__ypexclude_add(&__ypexhead,
+ strsep(&line, ":\n") + 1))
+ return 0;
+ if (foundyp) {
+ *foundyp = -1;
+ errno = saved_errno;
+ return 0;
+ }
continue;
}
parse:
@@ -449,7 +464,8 @@ parse:
cp = bp;
}
*m = NULL;
- return(1);
+ errno = saved_errno;
+ return 1;
}
/* NOTREACHED */
}