summaryrefslogtreecommitdiff
path: root/libc/sysdeps/linux/common/getgroups.c
blob: 29ceaf33bd684f077d66d24b481be4bf65db0f8d (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
51
52
53
54
55
56
57
/*
 * getgroups() for uClibc
 *
 * Copyright (C) 2000-2006 Erik Andersen <andersen@uclibc.org>
 *
 * Licensed under the LGPL v2.1, see the file COPYING.LIB in this tarball.
 */

#include <sys/syscall.h>
#include <unistd.h>

#if defined(__NR_getgroups32)
# undef __NR_getgroups
# define __NR_getgroups __NR_getgroups32
_syscall2(int, getgroups, int, size, gid_t *, list)

#elif __WORDSIZE == 64
_syscall2(int, getgroups, int, size, gid_t *, list)

#else
# include <errno.h>
# include <stdlib.h>
# include <sys/types.h>
# include <sys/param.h>

# define __NR___syscall_getgroups __NR_getgroups
static __always_inline
_syscall2(int, __syscall_getgroups, int, size, __kernel_gid_t *, list)

int getgroups(int size, gid_t groups[])
{
	if (unlikely(size < 0)) {
ret_error:
		__set_errno(EINVAL);
		return -1;
	} else {
		int i, ngids;
		__kernel_gid_t *kernel_groups;

		size = MIN(size, sysconf(_SC_NGROUPS_MAX));
		kernel_groups = (__kernel_gid_t *)malloc(sizeof(*kernel_groups) * size);
		if (size && kernel_groups == NULL)
			goto ret_error;

		ngids = __syscall_getgroups(size, kernel_groups);
		if (size != 0 && ngids > 0) {
			for (i = 0; i < ngids; i++) {
				groups[i] = kernel_groups[i];
			}
		}

		free(kernel_groups);
		return ngids;
	}
}
#endif
libc_hidden_def(getgroups)