summaryrefslogtreecommitdiff
path: root/src/libposixplz/setgroups.c
blob: d064ed2b162732b9dcd5894dac15204439031613 (plain)
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
/* ISC license. */

#include <skalibs/sysdeps.h>

#ifdef SKALIBS_HASSETGROUPS

#include <skalibs/nonposix.h>
#include <string.h>
#include <unistd.h>
#include <grp.h>
#include <skalibs/setgroups.h>

int setgroups_and_gid (gid_t g, size_t n, gid_t const *tab)
{
  size_t i = 1 ;
  if (!n) return setgroups(1, &g) ;
  if (tab[0] == g) return setgroups(n, tab) ;
  for (; i < n ; i++) if (tab[i] == g) break ;
  if (i < n)
  {
    gid_t newtab[n] ;
    newtab[0] = g ;
    memcpy(newtab + 1, tab, i * sizeof(gid_t)) ;
    memcpy(newtab + i + 1, tab + i + 1, (n - i - 1) * sizeof(gid_t)) ;
    return setgroups(n, newtab) ;
  }
  else
  {
    gid_t newtab[n+1] ;
    newtab[0] = g ;
    memcpy(newtab + 1, tab, n * sizeof(gid_t)) ;
    return setgroups(n+1, newtab) ;
  }
}

int setgroups_with_egid (size_t n, gid_t const *tab)
{
  return setgroups_and_gid(getegid(), n, tab) ;
}

int skalibs_setgroups (size_t n, gid_t const *tab)
{
#ifdef SKALIBS_BSD_SUCKS
  return setgroups_with_egid(n, tab) ;
#else
  return setgroups(n, tab) ;
#endif
}

#endif