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 if (group_info->ngroups < 2
160 || afs_get_pag_from_groups(GROUP_AT(group_info, 0),
161 GROUP_AT(group_info, 1)) == NOPAG) {
162 /* We will have to make sure group_info is big enough for pag */
163 struct group_info *tmp;
166 tmp = groups_alloc(group_info->ngroups + 2);
167 for (i = 0; i < group_info->ngroups; ++i)
168 GROUP_AT(tmp, i + 2) = GROUP_AT(group_info, i);
169 put_group_info(group_info);
173 *newpag = (pagvalue == -1 ? genpag() : pagvalue);
174 afs_get_groups_from_pag(*newpag, &g0, &g1);
175 GROUP_AT(group_info, 0) = g0;
176 GROUP_AT(group_info, 1) = g1;
178 afs_setgroups(cr, group_info, change_parent);
180 put_group_info(group_info);
185 afs_int32 ngroups, code = 0;
190 gidset = (gid_t *) osi_Alloc(NGROUPS * sizeof(gidset[0]));
191 ngroups = afs_getgroups(*cr, gidset);
193 if (afs_get_pag_from_groups(gidset[0], gidset[1]) == NOPAG) {
194 /* We will have to shift grouplist to make room for pag */
195 if (ngroups + 2 > NGROUPS) {
196 osi_Free((char *)gidset, NGROUPS * sizeof(int));
199 for (j = ngroups - 1; j >= 0; j--) {
200 gidset[j + 2] = gidset[j];
204 *newpag = (pagvalue == -1 ? genpag() : pagvalue);
205 afs_get_groups_from_pag(*newpag, &gidset[0], &gidset[1]);
206 code = afs_setgroups(cr, ngroups, gidset, change_parent);
208 /* If change_parent is set, then we should set the pag in the parent as
211 if (change_parent && !code) {
212 code = set_pag_in_parent(*newpag, gidset[0], gidset[1]);
215 osi_Free((char *)gidset, NGROUPS * sizeof(int));
221 /* Intercept the standard system call. */
222 extern long (*sys_setgroupsp) (int gidsetsize, gid_t * grouplist);
224 afs_xsetgroups(int gidsetsize, gid_t * grouplist)
227 cred_t *cr = crref();
232 old_pag = PagInCred(cr);
236 code = (*sys_setgroupsp) (gidsetsize, grouplist);
243 if (old_pag != NOPAG && PagInCred(cr) == NOPAG) {
244 /* re-install old pag if there's room. */
245 code = setpag(&cr, old_pag, &junk, 0);
250 /* Linux syscall ABI returns errno as negative */
254 #if defined(AFS_LINUX24_ENV)
255 /* Intercept the standard uid32 system call. */
256 extern long (*sys_setgroups32p) (int gidsetsize, gid_t * grouplist);
258 afs_xsetgroups32(int gidsetsize, gid_t * grouplist)
261 cred_t *cr = crref();
266 old_pag = PagInCred(cr);
270 code = (*sys_setgroups32p) (gidsetsize, grouplist);
278 if (old_pag != NOPAG && PagInCred(cr) == NOPAG) {
279 /* re-install old pag if there's room. */
280 code = setpag(&cr, old_pag, &junk, 0);
285 /* Linux syscall ABI returns errno as negative */
290 #if defined(AFS_PPC64_LINUX20_ENV)
291 /* Intercept the uid16 system call as used by 32bit programs. */
292 extern long (*sys32_setgroupsp)(int gidsetsize, gid_t *grouplist);
293 asmlinkage long afs32_xsetgroups(int gidsetsize, gid_t *grouplist)
296 cred_t *cr = crref();
301 old_pag = PagInCred(cr);
305 code = (*sys32_setgroupsp)(gidsetsize, grouplist);
312 if (old_pag != NOPAG && PagInCred(cr) == NOPAG) {
313 /* re-install old pag if there's room. */
314 code = setpag(&cr, old_pag, &junk, 0);
319 /* Linux syscall ABI returns errno as negative */
324 #if defined(AFS_SPARC64_LINUX20_ENV) || defined(AFS_AMD64_LINUX20_ENV)
325 /* Intercept the uid16 system call as used by 32bit programs. */
326 extern long (*sys32_setgroupsp) (int gidsetsize, u16 * grouplist);
328 afs32_xsetgroups(int gidsetsize, u16 * grouplist)
331 cred_t *cr = crref();
336 old_pag = PagInCred(cr);
340 code = (*sys32_setgroupsp) (gidsetsize, grouplist);
347 if (old_pag != NOPAG && PagInCred(cr) == NOPAG) {
348 /* re-install old pag if there's room. */
349 code = setpag(&cr, old_pag, &junk, 0);
354 /* Linux syscall ABI returns errno as negative */
358 #ifdef AFS_LINUX24_ENV
359 /* Intercept the uid32 system call as used by 32bit programs. */
360 extern long (*sys32_setgroups32p) (int gidsetsize, gid_t * grouplist);
362 afs32_xsetgroups32(int gidsetsize, gid_t * grouplist)
365 cred_t *cr = crref();
370 old_pag = PagInCred(cr);
374 code = (*sys32_setgroups32p) (gidsetsize, grouplist);
381 if (old_pag != NOPAG && PagInCred(cr) == NOPAG) {
382 /* re-install old pag if there's room. */
383 code = setpag(&cr, old_pag, &junk, 0);
388 /* Linux syscall ABI returns errno as negative */