2 * Copyright 2000, International Business Machines Corporation and others.
5 * This software has been released under the terms of the IBM Public
6 * License. For details, see the LICENSE file in the top-level source
7 * directory or online at http://www.openafs.org/dl/license10.html
16 #include <afsconfig.h>
17 #include "afs/param.h"
22 #include "afs/sysincludes.h"
23 #include "afsincludes.h"
24 #include "afs/afs_stats.h" /* statistics */
25 #ifdef AFS_LINUX22_ENV
26 #include "h/smp_lock.h"
29 #if defined(AFS_LINUX26_ENV)
31 afs_setgroups(cred_t **cr, struct group_info *group_info, int change_parent)
33 struct group_info *old_info;
35 AFS_STATCNT(afs_setgroups);
37 old_info = (*cr)->cr_group_info;
38 get_group_info(group_info);
39 (*cr)->cr_group_info = group_info;
40 put_group_info(old_info);
45 old_info = current->parent->group_info;
46 get_group_info(group_info);
47 current->parent->group_info = group_info;
48 put_group_info(old_info);
55 afs_setgroups(cred_t **cr, int ngroups, gid_t * gidset, int change_parent)
61 AFS_STATCNT(afs_setgroups);
63 if (ngroups > NGROUPS)
66 gp = (*cr)->cr_groups;
67 if (ngroups < NGROUPS)
68 gp[ngroups] = (gid_t) NOGROUP;
70 for (i = ngroups; i > 0; i--) {
74 (*cr)->cr_ngroups = ngroups;
80 #if defined(AFS_LINUX26_ENV)
81 static struct group_info *
82 afs_getgroups(cred_t * cr)
84 AFS_STATCNT(afs_getgroups);
86 get_group_info(cr->cr_group_info);
87 return cr->cr_group_info;
90 /* Returns number of groups. And we trust groups to be large enough to
91 * hold all the groups.
94 afs_getgroups(cred_t *cr, gid_t *groups)
100 AFS_STATCNT(afs_getgroups);
105 for (i = 0; (i < n) && (*gp != (gid_t) NOGROUP); i++)
111 #if !defined(AFS_LINUX26_ENV)
112 /* Only propogate the PAG to the parent process. Unix's propogate to
113 * all processes sharing the cred.
116 set_pag_in_parent(int pag, int g0, int g1)
119 #ifdef STRUCT_TASK_STRUCT_HAS_PARENT
120 gid_t *gp = current->parent->groups;
121 int ngroups = current->parent->ngroups;
123 gid_t *gp = current->p_pptr->groups;
124 int ngroups = current->p_pptr->ngroups;
127 if ((ngroups < 2) || (afs_get_pag_from_groups(gp[0], gp[1]) == NOPAG)) {
128 /* We will have to shift grouplist to make room for pag */
129 if (ngroups + 2 > NGROUPS) {
132 for (i = ngroups - 1; i >= 0; i--) {
139 if (ngroups < NGROUPS)
140 gp[ngroups] = NOGROUP;
142 #ifdef STRUCT_TASK_STRUCT_HAS_PARENT
143 current->parent->ngroups = ngroups;
145 current->p_pptr->ngroups = ngroups;
152 setpag(cred_t ** cr, afs_uint32 pagvalue, afs_uint32 * newpag,
155 #if defined(AFS_LINUX26_ENV)
156 struct group_info *group_info;
158 struct group_info *tmp;
164 group_info = afs_getgroups(*cr);
165 if (group_info->ngroups < 2
166 || afs_get_pag_from_groups(GROUP_AT(group_info, 0),
167 GROUP_AT(group_info, 1)) == NOPAG)
168 /* We will have to make sure group_info is big enough for pag */
171 tmp = groups_alloc(group_info->ngroups + need_space);
173 for (i = 0; i < group_info->ngroups; ++i)
174 GROUP_AT(tmp, i + need_space) = GROUP_AT(group_info, i);
175 put_group_info(group_info);
178 *newpag = (pagvalue == -1 ? genpag() : pagvalue);
179 afs_get_groups_from_pag(*newpag, &g0, &g1);
180 GROUP_AT(group_info, 0) = g0;
181 GROUP_AT(group_info, 1) = g1;
183 afs_setgroups(cr, group_info, change_parent);
185 put_group_info(group_info);
190 afs_int32 ngroups, code = 0;
195 gidset = (gid_t *) osi_Alloc(NGROUPS * sizeof(gidset[0]));
196 ngroups = afs_getgroups(*cr, gidset);
198 if (afs_get_pag_from_groups(gidset[0], gidset[1]) == NOPAG) {
199 /* We will have to shift grouplist to make room for pag */
200 if (ngroups + 2 > NGROUPS) {
201 osi_Free((char *)gidset, NGROUPS * sizeof(int));
204 for (j = ngroups - 1; j >= 0; j--) {
205 gidset[j + 2] = gidset[j];
209 *newpag = (pagvalue == -1 ? genpag() : pagvalue);
210 afs_get_groups_from_pag(*newpag, &gidset[0], &gidset[1]);
211 code = afs_setgroups(cr, ngroups, gidset, change_parent);
213 /* If change_parent is set, then we should set the pag in the parent as
216 if (change_parent && !code) {
217 code = set_pag_in_parent(*newpag, gidset[0], gidset[1]);
220 osi_Free((char *)gidset, NGROUPS * sizeof(int));
226 /* Intercept the standard system call. */
227 extern asmlinkage long (*sys_setgroupsp) (int gidsetsize, gid_t * grouplist);
229 afs_xsetgroups(int gidsetsize, gid_t * grouplist)
232 cred_t *cr = crref();
237 old_pag = PagInCred(cr);
241 code = (*sys_setgroupsp) (gidsetsize, grouplist);
248 if (old_pag != NOPAG && PagInCred(cr) == NOPAG) {
249 /* re-install old pag if there's room. */
250 code = setpag(&cr, old_pag, &junk, 0);
255 /* Linux syscall ABI returns errno as negative */
259 #if defined(AFS_LINUX24_ENV)
260 /* Intercept the standard uid32 system call. */
261 extern asmlinkage long (*sys_setgroups32p) (int gidsetsize, gid_t * grouplist);
263 afs_xsetgroups32(int gidsetsize, gid_t * grouplist)
266 cred_t *cr = crref();
271 old_pag = PagInCred(cr);
275 code = (*sys_setgroups32p) (gidsetsize, grouplist);
283 if (old_pag != NOPAG && PagInCred(cr) == NOPAG) {
284 /* re-install old pag if there's room. */
285 code = setpag(&cr, old_pag, &junk, 0);
290 /* Linux syscall ABI returns errno as negative */
295 #if defined(AFS_PPC64_LINUX20_ENV)
296 /* Intercept the uid16 system call as used by 32bit programs. */
297 extern long (*sys32_setgroupsp)(int gidsetsize, gid_t *grouplist);
298 asmlinkage long afs32_xsetgroups(int gidsetsize, gid_t *grouplist)
301 cred_t *cr = crref();
306 old_pag = PagInCred(cr);
310 code = (*sys32_setgroupsp)(gidsetsize, grouplist);
317 if (old_pag != NOPAG && PagInCred(cr) == NOPAG) {
318 /* re-install old pag if there's room. */
319 code = setpag(&cr, old_pag, &junk, 0);
324 /* Linux syscall ABI returns errno as negative */
329 #if defined(AFS_SPARC64_LINUX20_ENV) || defined(AFS_AMD64_LINUX20_ENV)
330 /* Intercept the uid16 system call as used by 32bit programs. */
331 extern long (*sys32_setgroupsp) (int gidsetsize, u16 * grouplist);
333 afs32_xsetgroups(int gidsetsize, u16 * grouplist)
336 cred_t *cr = crref();
341 old_pag = PagInCred(cr);
345 code = (*sys32_setgroupsp) (gidsetsize, grouplist);
352 if (old_pag != NOPAG && PagInCred(cr) == NOPAG) {
353 /* re-install old pag if there's room. */
354 code = setpag(&cr, old_pag, &junk, 0);
359 /* Linux syscall ABI returns errno as negative */
363 #ifdef AFS_LINUX24_ENV
364 /* Intercept the uid32 system call as used by 32bit programs. */
365 extern long (*sys32_setgroups32p) (int gidsetsize, gid_t * grouplist);
367 afs32_xsetgroups32(int gidsetsize, gid_t * grouplist)
370 cred_t *cr = crref();
375 old_pag = PagInCred(cr);
379 code = (*sys32_setgroups32p) (gidsetsize, grouplist);
386 if (old_pag != NOPAG && PagInCred(cr) == NOPAG) {
387 /* re-install old pag if there's room. */
388 code = setpag(&cr, old_pag, &junk, 0);
393 /* Linux syscall ABI returns errno as negative */