unistr/u{8,16,32}-uctomb: Avoid possible trouble with huge strings.
[gnulib.git] / lib / getugroups.c
blobc339161a494f46350e0cdf2df9978c16c847c5d3
1 /* getugroups.c -- return a list of the groups a user is in
3 Copyright (C) 1990-1991, 1998-2000, 2003-2020 Free Software Foundation, Inc.
5 This program is free software: you can redistribute it and/or modify
6 it under the terms of the GNU General Public License as published by
7 the Free Software Foundation; either version 3 of the License, or
8 (at your option) any later version.
10 This program is distributed in the hope that it will be useful,
11 but WITHOUT ANY WARRANTY; without even the implied warranty of
12 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
13 GNU General Public License for more details.
15 You should have received a copy of the GNU General Public License
16 along with this program. If not, see <https://www.gnu.org/licenses/>. */
18 /* Written by David MacKenzie. */
20 #include <config.h>
22 #include "getugroups.h"
24 #include <errno.h>
25 #include <limits.h>
26 #include <stdio.h> /* grp.h on alpha OSF1 V2.0 uses "FILE *". */
27 #include <string.h>
28 #include <unistd.h>
30 #if !HAVE_GRP_H || defined __ANDROID__
32 /* Mingw lacks all things related to group management. The best we
33 can do is fail with ENOSYS.
35 Bionic declares e.g. getgrent() in <grp.h> but it isn't actually
36 defined in the library. */
38 int
39 getugroups (int maxcount _GL_UNUSED,
40 gid_t *grouplist _GL_UNUSED,
41 char const *username _GL_UNUSED,
42 gid_t gid _GL_UNUSED)
44 errno = ENOSYS;
45 return -1;
48 #else /* HAVE_GRP_H */
49 # include <grp.h>
51 # define STREQ(a, b) (strcmp (a, b) == 0)
53 /* Like 'getgroups', but for user USERNAME instead of for the current
54 process. Store at most MAXCOUNT group IDs in the GROUPLIST array.
55 If GID is not -1, store it first (if possible). GID should be the
56 group ID (pw_gid) obtained from getpwuid, in case USERNAME is not
57 listed in /etc/groups. Upon failure, set errno and return -1.
58 Otherwise, return the number of IDs we've written into GROUPLIST. */
60 int
61 getugroups (int maxcount, gid_t *grouplist, char const *username,
62 gid_t gid)
64 int count = 0;
66 if (gid != (gid_t) -1)
68 if (maxcount != 0)
69 grouplist[count] = gid;
70 ++count;
73 setgrent ();
74 while (1)
76 char **cp;
77 struct group *grp;
79 errno = 0;
80 grp = getgrent ();
81 if (grp == NULL)
82 break;
84 for (cp = grp->gr_mem; *cp; ++cp)
86 int n;
88 if ( ! STREQ (username, *cp))
89 continue;
91 /* See if this group number is already on the list. */
92 for (n = 0; n < count; ++n)
93 if (grouplist && grouplist[n] == grp->gr_gid)
94 break;
96 /* If it's a new group number, then try to add it to the list. */
97 if (n == count)
99 if (maxcount != 0)
101 if (count >= maxcount)
102 goto done;
103 grouplist[count] = grp->gr_gid;
105 if (count == INT_MAX)
107 errno = EOVERFLOW;
108 goto done;
110 count++;
115 if (errno != 0)
116 count = -1;
118 done:
120 int saved_errno = errno;
121 endgrent ();
122 errno = saved_errno;
125 return count;
128 #endif /* HAVE_GRP_H */