Skip to content
Navigation Menu
Toggle navigation
Sign in
In this repository
All GitHub Enterprise
↵
Jump to
↵
No suggested jump to results
In this repository
All GitHub Enterprise
↵
Jump to
↵
In this organization
All GitHub Enterprise
↵
Jump to
↵
In this repository
All GitHub Enterprise
↵
Jump to
↵
Sign in
Reseting focus
You signed in with another tab or window.
Reload
to refresh your session.
You signed out in another tab or window.
Reload
to refresh your session.
You switched accounts on another tab or window.
Reload
to refresh your session.
Dismiss alert
{{ message }}
git-mirror
/
glibc
Public
Notifications
You must be signed in to change notification settings
Fork
0
Star
0
Code
Pull requests
0
Actions
Projects
0
Security
Insights
Additional navigation options
Code
Pull requests
Actions
Projects
Security
Insights
Files
82e87e0
abilist
aout
argp
assert
bits
c_stubs
catgets
conf
conform
crypt
csu
ctype
debug
dirent
dlfcn
elf
fedora
gmon
gnulib
grp
gshadow
hesiod
nss_hesiod
hesiod-grp.c
hesiod-init.c
hesiod-proto.c
hesiod-pwd.c
hesiod-service.c
nss_hesiod.h
Makefile
README.hesiod
Versions
hesiod.c
hesiod.h
hesiod_p.h
hurd
iconv
iconvdata
include
inet
intl
io
libidn
libio
locale
localedata
login
mach
malloc
manual
math
misc
nis
nptl
nptl_db
nscd
nss
po
posix
pwd
resolv
resource
rt
rtkaio
scripts
setjmp
shadow
signal
socket
soft-fp
stdio-common
stdlib
streams
string
sunrpc
sysdeps
sysvipc
termios
time
timezone
wcsmbs
wctype
.gitignore
BUGS
CANCEL-FCT-WAIVE
CANCEL-FILE-WAIVE
CONFORMANCE
COPYING
COPYING.LIB
ChangeLog
ChangeLog.1
ChangeLog.10
ChangeLog.11
ChangeLog.12
ChangeLog.13
ChangeLog.14
ChangeLog.15
ChangeLog.16
ChangeLog.2
ChangeLog.3
ChangeLog.4
ChangeLog.5
ChangeLog.6
ChangeLog.7
ChangeLog.8
ChangeLog.9
FAQ
FAQ.in
INSTALL
LICENSES
Makeconfig
Makefile
Makefile.in
Makerules
NAMESPACE
NEWS
NOTES
PROJECTS
README
README.libm
Rules
Versions.def
WUR-REPORT
abi-tags
aclocal.m4
config-name.in
config.h.in
config.make.in
configure
configure.in
cppflags-iterator.mk
extra-lib.mk
extra-modules.mk
o-iterator.mk
shlib-versions
test-skeleton.c
tls.make.c
version.h
Breadcrumbs
glibc
/
hesiod
/
nss_hesiod
/
hesiod-grp.c
Blame
Blame
Latest commit
History
History
294 lines (245 loc) · 6.68 KB
Breadcrumbs
glibc
/
hesiod
/
nss_hesiod
/
hesiod-grp.c
Top
File metadata and controls
Code
Blame
294 lines (245 loc) · 6.68 KB
Raw
/* Copyright (C) 1997, 1999, 2000, 2002 Free Software Foundation, Inc. This file is part of the GNU C Library. Contributed by Mark Kettenis <kettenis@phys.uva.nl>, 1997. The GNU C Library is free software; you can redistribute it and/or modify it under the terms of the GNU Lesser General Public License as published by the Free Software Foundation; either version 2.1 of the License, or (at your option) any later version. The GNU C Library is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU Lesser General Public License for more details. You should have received a copy of the GNU Lesser General Public License along with the GNU C Library; if not, write to the Free Software Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA. */ #include <ctype.h> #include <errno.h> #include <grp.h> #include <hesiod.h> #include <nss.h> #include <stdio.h> #include <stdlib.h> #include <string.h> #include <sys/param.h> #include "nss_hesiod.h" /* Get the declaration of the parser function. */ #define ENTNAME grent #define STRUCTURE group #define EXTERN_PARSER #include <nss/nss_files/files-parse.c> enum nss_status _nss_hesiod_setgrent (int stayopen) { return NSS_STATUS_SUCCESS; } enum nss_status _nss_hesiod_endgrent (void) { return NSS_STATUS_SUCCESS; } static enum nss_status lookup (const char *name, const char *type, struct group *grp, char *buffer, size_t buflen, int *errnop) { struct parser_data *data = (void *) buffer; size_t linebuflen; void *context; char **list; int parse_res; size_t len; int olderr = errno; context = _nss_hesiod_init (); if (context == NULL) return NSS_STATUS_UNAVAIL; list = hesiod_resolve (context, name, type); if (list == NULL) { int err = errno; hesiod_end (context); __set_errno (olderr); return err == ENOENT ? NSS_STATUS_NOTFOUND : NSS_STATUS_UNAVAIL; } linebuflen = buffer + buflen - data->linebuffer; len = strlen (*list) + 1; if (linebuflen < len) { hesiod_free_list (context, list); hesiod_end (context); *errnop = ERANGE; return NSS_STATUS_TRYAGAIN; } memcpy (data->linebuffer, *list, len); hesiod_free_list (context, list); hesiod_end (context); parse_res = _nss_files_parse_grent (buffer, grp, data, buflen, errnop); if (parse_res < 1) { __set_errno (olderr); return parse_res == -1 ? NSS_STATUS_TRYAGAIN : NSS_STATUS_NOTFOUND; } return NSS_STATUS_SUCCESS; } enum nss_status _nss_hesiod_getgrnam_r (const char *name, struct group *grp, char *buffer, size_t buflen, int *errnop) { return lookup (name, "group", grp, buffer, buflen, errnop); } enum nss_status _nss_hesiod_getgrgid_r (gid_t gid, struct group *grp, char *buffer, size_t buflen, int *errnop) { char gidstr[21]; /* We will probably never have a gid_t with more than 64 bits. */ snprintf (gidstr, sizeof gidstr, "%d", gid); return lookup (gidstr, "gid", grp, buffer, buflen, errnop); } static int internal_gid_in_list (const gid_t *list, const gid_t g, long int len) { while (len > 0) { if (*list == g) return 1; --len; ++list; } return 0; } static enum nss_status internal_gid_from_group (void *context, const char *groupname, gid_t *group) { char **grp_res; enum nss_status status = NSS_STATUS_NOTFOUND; grp_res = hesiod_resolve (context, groupname, "group"); if (grp_res != NULL && *grp_res != NULL) { char *p = *grp_res; /* Skip to third field. */ while (*p != '\0' && *p != ':') ++p; if (*p != '\0') ++p; while (*p != '\0' && *p != ':') ++p; if (*p != '\0') { char *endp; char *q = ++p; long int val; while (*q != '\0' && *q != ':') ++q; val = strtol (p, &endp, 10); if (sizeof (gid_t) == sizeof (long int) || (gid_t) val == val) { *group = val; if (endp == q && endp != p) status = NSS_STATUS_SUCCESS; } } hesiod_free_list (context, grp_res); } return status; } enum nss_status _nss_hesiod_initgroups_dyn (const char *user, gid_t group, long int *start, long int *size, gid_t **groupsp, long int limit, int *errnop) { enum nss_status status = NSS_STATUS_SUCCESS; char **list = NULL; char *p; void *context; gid_t *groups = *groupsp; int save_errno; context = _nss_hesiod_init (); if (context == NULL) return NSS_STATUS_UNAVAIL; list = hesiod_resolve (context, user, "grplist"); if (list == NULL) { hesiod_end (context); return errno == ENOENT ? NSS_STATUS_NOTFOUND : NSS_STATUS_UNAVAIL; } if (!internal_gid_in_list (groups, group, *start)) { if (__builtin_expect (*start == *size, 0)) { /* Need a bigger buffer. */ gid_t *newgroups; long int newsize; if (limit > 0 && *size == limit) /* We reached the maximum. */ goto done; if (limit <= 0) newsize = 2 * *size; else newsize = MIN (limit, 2 * *size); newgroups = realloc (groups, newsize * sizeof (*groups)); if (newgroups == NULL) goto done; *groupsp = groups = newgroups; *size = newsize; } groups[(*start)++] = group; } save_errno = errno; p = *list; while (*p != '\0') { char *endp; char *q; long int val; status = NSS_STATUS_NOTFOUND; q = p; while (*q != '\0' && *q != ':' && *q != ',') ++q; if (*q != '\0') *q++ = '\0'; __set_errno (0); val = strtol (p, &endp, 10); /* Test whether the number is representable in a variable of type `gid_t'. If not ignore the number. */ if ((sizeof (gid_t) == sizeof (long int) || (gid_t) val == val) && errno == 0) { if (*endp == '\0' && endp != p) { group = val; status = NSS_STATUS_SUCCESS; } else status = internal_gid_from_group (context, p, &group); if (status == NSS_STATUS_SUCCESS && !internal_gid_in_list (groups, group, *start)) { if (__builtin_expect (*start == *size, 0)) { /* Need a bigger buffer. */ gid_t *newgroups; long int newsize; if (limit > 0 && *size == limit) /* We reached the maximum. */ goto done; if (limit <= 0) newsize = 2 * *size; else newsize = MIN (limit, 2 * *size); newgroups = realloc (groups, newsize * sizeof (*groups)); if (newgroups == NULL) goto done; *groupsp = groups = newgroups; *size = newsize; } groups[(*start)++] = group; } } p = q; } __set_errno (save_errno); done: hesiod_free_list (context, list); hesiod_end (context); return NSS_STATUS_SUCCESS; }
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
You can’t perform that action at this time.