diff options
Diffstat (limited to 'libpwdgrp/initgroups.c')
-rw-r--r-- | libpwdgrp/initgroups.c | 115 |
1 files changed, 115 insertions, 0 deletions
diff --git a/libpwdgrp/initgroups.c b/libpwdgrp/initgroups.c new file mode 100644 index 000000000..9b5dcbcb2 --- /dev/null +++ b/libpwdgrp/initgroups.c | |||
@@ -0,0 +1,115 @@ | |||
1 | /* | ||
2 | * initgroups.c - This file is part of the libc-8086/grp package for ELKS, | ||
3 | * Copyright (C) 1995, 1996 Nat Friedman <ndf@linux.mit.edu>. | ||
4 | * | ||
5 | * This library is free software; you can redistribute it and/or | ||
6 | * modify it under the terms of the GNU Library General Public | ||
7 | * License as published by the Free Software Foundation; either | ||
8 | * version 2 of the License, or (at your option) any later version. | ||
9 | * | ||
10 | * This library 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 GNU | ||
13 | * Library General Public License for more details. | ||
14 | * | ||
15 | * You should have received a copy of the GNU Library General Public | ||
16 | * License along with this library; if not, write to the Free | ||
17 | * Software Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA. | ||
18 | * | ||
19 | */ | ||
20 | |||
21 | #include "busybox.h" | ||
22 | |||
23 | #include <unistd.h> | ||
24 | #include <string.h> | ||
25 | #include <fcntl.h> | ||
26 | #include "grp.h" | ||
27 | |||
28 | /* | ||
29 | * Define GR_SCALE_DYNAMIC if you want grp to dynamically scale its read buffer | ||
30 | * so that lines of any length can be used. On very very small systems, | ||
31 | * you may want to leave this undefined becasue it will make the grp functions | ||
32 | * somewhat larger (because of the inclusion of malloc and the code necessary). | ||
33 | * On larger systems, you will want to define this, because grp will _not_ | ||
34 | * deal with long lines gracefully (they will be skipped). | ||
35 | */ | ||
36 | #undef GR_SCALE_DYNAMIC | ||
37 | |||
38 | #ifndef GR_SCALE_DYNAMIC | ||
39 | /* | ||
40 | * If scaling is not dynamic, the buffers will be statically allocated, and | ||
41 | * maximums must be chosen. GR_MAX_LINE_LEN is the maximum number of | ||
42 | * characters per line in the group file. GR_MAX_MEMBERS is the maximum | ||
43 | * number of members of any given group. | ||
44 | */ | ||
45 | #define GR_MAX_LINE_LEN 128 | ||
46 | /* GR_MAX_MEMBERS = (GR_MAX_LINE_LEN-(24+3+6))/9 */ | ||
47 | #define GR_MAX_MEMBERS 11 | ||
48 | |||
49 | #endif /* !GR_SCALE_DYNAMIC */ | ||
50 | |||
51 | |||
52 | /* | ||
53 | * Define GR_DYNAMIC_GROUP_LIST to make initgroups() dynamically allocate | ||
54 | * space for it's GID array before calling setgroups(). This is probably | ||
55 | * unnecessary scalage, so it's undefined by default. | ||
56 | */ | ||
57 | #undef GR_DYNAMIC_GROUP_LIST | ||
58 | |||
59 | #ifndef GR_DYNAMIC_GROUP_LIST | ||
60 | /* | ||
61 | * GR_MAX_GROUPS is the size of the static array initgroups() uses for | ||
62 | * its static GID array if GR_DYNAMIC_GROUP_LIST isn't defined. | ||
63 | */ | ||
64 | #define GR_MAX_GROUPS 64 | ||
65 | |||
66 | #endif /* !GR_DYNAMIC_GROUP_LIST */ | ||
67 | |||
68 | int initgroups(__const char *user, gid_t gid) | ||
69 | { | ||
70 | register struct group *group; | ||
71 | |||
72 | #ifndef GR_DYNAMIC_GROUP_LIST | ||
73 | gid_t group_list[GR_MAX_GROUPS]; | ||
74 | #else | ||
75 | gid_t *group_list = NULL; | ||
76 | #endif | ||
77 | register char **tmp_mem; | ||
78 | int num_groups; | ||
79 | int grp_fd; | ||
80 | |||
81 | |||
82 | if ((grp_fd = open("/etc/group", O_RDONLY)) < 0) | ||
83 | return -1; | ||
84 | |||
85 | num_groups = 0; | ||
86 | #ifdef GR_DYNAMIC_GROUP_LIST | ||
87 | group_list = (gid_t *) realloc(group_list, 1); | ||
88 | #endif | ||
89 | group_list[num_groups] = gid; | ||
90 | #ifndef GR_DYNAMIC_GROUP_LIST | ||
91 | while (num_groups < GR_MAX_GROUPS && | ||
92 | (group = __getgrent(grp_fd)) != NULL) | ||
93 | #else | ||
94 | while ((group = __getgrent(grp_fd)) != NULL) | ||
95 | #endif | ||
96 | { | ||
97 | if (group->gr_gid != gid); | ||
98 | { | ||
99 | tmp_mem = group->gr_mem; | ||
100 | while (*tmp_mem != NULL) { | ||
101 | if (!strcmp(*tmp_mem, user)) { | ||
102 | num_groups++; | ||
103 | #ifdef GR_DYNAMIC_GROUP_LIST | ||
104 | group_list = (gid_t *) realloc(group_list, num_groups * | ||
105 | sizeof(gid_t *)); | ||
106 | #endif | ||
107 | group_list[num_groups] = group->gr_gid; | ||
108 | } | ||
109 | tmp_mem++; | ||
110 | } | ||
111 | } | ||
112 | } | ||
113 | close(grp_fd); | ||
114 | return setgroups(num_groups, group_list); | ||
115 | } | ||