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 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157
|
/*
* Check decoding of setgroups/setgroups32 syscalls.
*
* Copyright (c) 2016 Dmitry V. Levin <ldv@strace.io>
* Copyright (c) 2016-2024 The strace developers.
* All rights reserved.
*
* SPDX-License-Identifier: GPL-2.0-or-later
*/
#ifdef __NR_setgroups32
# define SYSCALL_NR __NR_setgroups32
# define SYSCALL_NAME "setgroups32"
# define GID_TYPE unsigned int
#else /* __NR_setgroups */
# include "tests.h"
# include "scno.h"
# define SYSCALL_NR __NR_setgroups
# define SYSCALL_NAME "setgroups"
# if defined __NR_setgroups32 && __NR_setgroups != __NR_setgroups32
# define GID_TYPE unsigned short
# else
# define GID_TYPE unsigned int
# endif
#endif
#include <stdio.h>
#include <unistd.h>
static void
printuid(GID_TYPE id)
{
if (id == (GID_TYPE) -1U)
printf("-1");
else
printf("%u", id);
}
int
main(void)
{
const char *errstr;
/* check how the first argument is decoded */
long rc = syscall(SYSCALL_NR, 0, 0);
printf("%s(0, NULL) = %s\n", SYSCALL_NAME, sprintrc(rc));
rc = syscall(SYSCALL_NR, F8ILL_KULONG_MASK, 0);
printf("%s(0, NULL) = %s\n", SYSCALL_NAME, sprintrc(rc));
rc = syscall(SYSCALL_NR, 1, 0);
printf("%s(1, NULL) = %s\n", SYSCALL_NAME, sprintrc(rc));
rc = syscall(SYSCALL_NR, (long) 0xffffffff00000001ULL, 0);
printf("%s(1, NULL) = %s\n", SYSCALL_NAME, sprintrc(rc));
rc = syscall(SYSCALL_NR, -1U, 0);
printf("%s(%d, NULL) = %s\n", SYSCALL_NAME, -1, sprintrc(rc));
rc = syscall(SYSCALL_NR, -1L, 0);
printf("%s(%d, NULL) = %s\n", SYSCALL_NAME, -1, sprintrc(rc));
/* check how the second argument is decoded */
TAIL_ALLOC_OBJECT_CONST_PTR(const GID_TYPE, g1);
TAIL_ALLOC_OBJECT_CONST_ARR(GID_TYPE, g2, 2);
TAIL_ALLOC_OBJECT_CONST_ARR(GID_TYPE, g3, 3);
rc = syscall(SYSCALL_NR, 0, g1 + 1);
printf("%s(0, []) = %s\n", SYSCALL_NAME, sprintrc(rc));
rc = syscall(SYSCALL_NR, 1, g1);
errstr = sprintrc(rc);
printf("%s(1, [", SYSCALL_NAME);
printuid(*g1);
printf("]) = %s\n", errstr);
rc = syscall(SYSCALL_NR, 1, g1 + 1);
printf("%s(1, %p) = %s\n", SYSCALL_NAME, g1 + 1, sprintrc(rc));
rc = syscall(SYSCALL_NR, 1, -1L);
printf("%s(1, %#lx) = %s\n", SYSCALL_NAME, -1L, sprintrc(rc));
rc = syscall(SYSCALL_NR, 2, g1);
errstr = sprintrc(rc);
printf("%s(2, [", SYSCALL_NAME);
printuid(*g1);
printf(", ... /* %p */]) = %s\n", g1 + 1, errstr);
g2[0] = -2;
g2[1] = -3;
rc = syscall(SYSCALL_NR, 2, g2);
errstr = sprintrc(rc);
printf("%s(2, [", SYSCALL_NAME);
printuid(g2[0]);
printf(", ");
printuid(g2[1]);
printf("]) = %s\n", errstr);
rc = syscall(SYSCALL_NR, 3, g2);
errstr = sprintrc(rc);
printf("%s(3, [", SYSCALL_NAME);
printuid(g2[0]);
printf(", ");
printuid(g2[1]);
printf(", ... /* %p */]) = %s\n", g2 + 2, errstr);
g3[0] = 0;
g3[1] = 1;
rc = syscall(SYSCALL_NR, 3, g3);
errstr = sprintrc(rc);
printf("%s(3, [", SYSCALL_NAME);
printuid(g3[0]);
printf(", ");
printuid(g3[1]);
printf(", ...]) = %s\n", errstr);
rc = syscall(SYSCALL_NR, 4, g3);
errstr = sprintrc(rc);
printf("%s(4, [", SYSCALL_NAME);
printuid(g3[0]);
printf(", ");
printuid(g3[1]);
printf(", ...]) = %s\n", errstr);
rc = sysconf(_SC_NGROUPS_MAX);
const unsigned ngroups_max = rc;
if ((unsigned long) rc == ngroups_max && (int) ngroups_max > 0) {
rc = syscall(SYSCALL_NR, ngroups_max, g3);
errstr = sprintrc(rc);
printf("%s(%d, [", SYSCALL_NAME, ngroups_max);
printuid(g3[0]);
printf(", ");
printuid(g3[1]);
printf(", ...]) = %s\n", errstr);
rc = syscall(SYSCALL_NR, F8ILL_KULONG_MASK | ngroups_max, g3);
errstr = sprintrc(rc);
printf("%s(%d, [", SYSCALL_NAME, ngroups_max);
printuid(g3[0]);
printf(", ");
printuid(g3[1]);
printf(", ...]) = %s\n", errstr);
rc = syscall(SYSCALL_NR, ngroups_max + 1, g3);
printf("%s(%d, %p) = %s\n", SYSCALL_NAME,
ngroups_max + 1, g3, sprintrc(rc));
}
puts("+++ exited with 0 +++");
return 0;
}
|