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)
97 AFS_STATCNT(afs_getgroups);
99 gid_t *gp = cr->cr_groups;
100 int n = cr->cr_ngroups;
102 for (i = 0; (i < n) && (*gp != (gid_t) NOGROUP); i++)
108 #if !defined(AFS_LINUX26_ENV)
109 /* Only propogate the PAG to the parent process. Unix's propogate to
110 * all processes sharing the cred.
113 set_pag_in_parent(int pag, int g0, int g1)
116 #ifdef STRUCT_TASK_STRUCT_HAS_PARENT
117 gid_t *gp = current->parent->groups;
118 int ngroups = current->parent->ngroups;
120 gid_t *gp = current->p_pptr->groups;
121 int ngroups = current->p_pptr->ngroups;
124 if ((ngroups < 2) || (afs_get_pag_from_groups(gp[0], gp[1]) == NOPAG)) {
125 /* We will have to shift grouplist to make room for pag */
126 if (ngroups + 2 > NGROUPS) {
129 for (i = ngroups - 1; i >= 0; i--) {
136 if (ngroups < NGROUPS)
137 gp[ngroups] = NOGROUP;
139 #ifdef STRUCT_TASK_STRUCT_HAS_PARENT
140 current->parent->ngroups = ngroups;
142 current->p_pptr->ngroups = ngroups;
149 setpag(cred_t ** cr, afs_uint32 pagvalue, afs_uint32 * newpag,
152 #if defined(AFS_LINUX26_ENV)
153 struct group_info *group_info;
158 group_info = afs_getgroups(*cr);
159 g0 = GROUP_AT(group_info, 0);
160 g1 = GROUP_AT(group_info, 1);
162 if (afs_get_pag_from_groups(g0, g1) == NOPAG) {
163 /* We will have to make sure group_info is big enough for pag */
164 struct group_info *tmp;
167 tmp = groups_alloc(group_info->ngroups + 2);
168 for (i = 0; i < group_info->ngroups; ++i)
169 GROUP_AT(tmp, i + 2) = GROUP_AT(group_info, i);
170 put_group_info(group_info);
174 *newpag = (pagvalue == -1 ? genpag() : pagvalue);
175 afs_get_groups_from_pag(*newpag, &g0, &g1);
176 GROUP_AT(group_info, 0) = g0;
177 GROUP_AT(group_info, 1) = g1;
179 afs_setgroups(cr, group_info, change_parent);
181 put_group_info(group_info);
186 afs_int32 ngroups, code = 0;
191 gidset = (gid_t *) osi_Alloc(NGROUPS * sizeof(gidset[0]));
192 ngroups = afs_getgroups(*cr, gidset);
194 if (afs_get_pag_from_groups(gidset[0], gidset[1]) == NOPAG) {
195 /* We will have to shift grouplist to make room for pag */
196 if (ngroups + 2 > NGROUPS) {
197 osi_Free((char *)gidset, NGROUPS * sizeof(int));
200 for (j = ngroups - 1; j >= 0; j--) {
201 gidset[j + 2] = gidset[j];
205 *newpag = (pagvalue == -1 ? genpag() : pagvalue);
206 afs_get_groups_from_pag(*newpag, &gidset[0], &gidset[1]);
207 code = afs_setgroups(cr, ngroups, gidset, change_parent);
209 /* If change_parent is set, then we should set the pag in the parent as
212 if (change_parent && !code) {
213 code = set_pag_in_parent(*newpag, gidset[0], gidset[1]);
216 osi_Free((char *)gidset, NGROUPS * sizeof(int));
222 /* Intercept the standard system call. */
223 extern long (*sys_setgroupsp) (int gidsetsize, gid_t * grouplist);
225 afs_xsetgroups(int gidsetsize, gid_t * grouplist)
228 cred_t *cr = crref();
233 old_pag = PagInCred(cr);
237 code = (*sys_setgroupsp) (gidsetsize, grouplist);
244 if (old_pag != NOPAG && PagInCred(cr) == NOPAG) {
245 /* re-install old pag if there's room. */
246 code = setpag(&cr, old_pag, &junk, 0);
251 /* Linux syscall ABI returns errno as negative */
255 #if defined(AFS_LINUX24_ENV)
256 /* Intercept the standard uid32 system call. */
257 extern long (*sys_setgroups32p) (int gidsetsize, gid_t * grouplist);
259 afs_xsetgroups32(int gidsetsize, gid_t * grouplist)
262 cred_t *cr = crref();
267 old_pag = PagInCred(cr);
271 code = (*sys_setgroups32p) (gidsetsize, grouplist);
279 if (old_pag != NOPAG && PagInCred(cr) == NOPAG) {
280 /* re-install old pag if there's room. */
281 code = setpag(&cr, old_pag, &junk, 0);
286 /* Linux syscall ABI returns errno as negative */
291 #if defined(AFS_PPC64_LINUX20_ENV)
292 /* Intercept the uid16 system call as used by 32bit programs. */
293 extern long (*sys32_setgroupsp)(int gidsetsize, gid_t *grouplist);
294 asmlinkage long afs32_xsetgroups(int gidsetsize, gid_t *grouplist)
297 cred_t *cr = crref();
302 old_pag = PagInCred(cr);
306 code = (*sys32_setgroupsp)(gidsetsize, grouplist);
313 if (old_pag != NOPAG && PagInCred(cr) == NOPAG) {
314 /* re-install old pag if there's room. */
315 code = setpag(&cr, old_pag, &junk, 0);
320 /* Linux syscall ABI returns errno as negative */
325 #if defined(AFS_SPARC64_LINUX20_ENV) || defined(AFS_AMD64_LINUX20_ENV)
326 /* Intercept the uid16 system call as used by 32bit programs. */
327 extern long (*sys32_setgroupsp) (int gidsetsize, u16 * grouplist);
329 afs32_xsetgroups(int gidsetsize, u16 * grouplist)
332 cred_t *cr = crref();
337 old_pag = PagInCred(cr);
341 code = (*sys32_setgroupsp) (gidsetsize, grouplist);
348 if (old_pag != NOPAG && PagInCred(cr) == NOPAG) {
349 /* re-install old pag if there's room. */
350 code = setpag(&cr, old_pag, &junk, 0);
355 /* Linux syscall ABI returns errno as negative */
359 #ifdef AFS_LINUX24_ENV
360 /* Intercept the uid32 system call as used by 32bit programs. */
361 extern long (*sys32_setgroups32p) (int gidsetsize, gid_t * grouplist);
363 afs32_xsetgroups32(int gidsetsize, gid_t * grouplist)
366 cred_t *cr = crref();
371 old_pag = PagInCred(cr);
375 code = (*sys32_setgroups32p) (gidsetsize, grouplist);
382 if (old_pag != NOPAG && PagInCred(cr) == NOPAG) {
383 /* re-install old pag if there's room. */
384 code = setpag(&cr, old_pag, &junk, 0);
389 /* Linux syscall ABI returns errno as negative */