exp2l: Work around a NetBSD 10.0/i386 bug.
[gnulib.git] / lib / getugroups.c
blob445e7b2691f776adb68b58fec557658775334f37
1 /* getugroups.c -- return a list of the groups a user is in
3 Copyright (C) 1990-1991, 1998-2000, 2003-2024 Free Software Foundation, Inc.
5 This file is free software: you can redistribute it and/or modify
6 it under the terms of the GNU Lesser General Public License as
7 published by the Free Software Foundation; either version 2.1 of the
8 License, or (at your option) any later version.
10 This file 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 Lesser General Public License for more details.
15 You should have received a copy of the GNU Lesser 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 (_GL_UNUSED int maxcount,
40 _GL_UNUSED gid_t *grouplist,
41 _GL_UNUSED char const *username,
42 _GL_UNUSED gid_t gid)
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 */