diff options
author | Ingo Schwarze <schwarze@cvs.openbsd.org> | 2009-12-19 22:41:40 +0000 |
---|---|---|
committer | Ingo Schwarze <schwarze@cvs.openbsd.org> | 2009-12-19 22:41:40 +0000 |
commit | 5b2b598d3bd2c4374a944b8b2c66147ec7bdf9a0 (patch) | |
tree | d9aaeb952fb48e92726a3e788870cb85e1eb12e1 | |
parent | 1fe64736e6d768c64774010591775e1b292863f2 (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.c | 70 |
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 */ } |