Merge pull request #13899 from poettering/in-gid-tweak

user-util: tweak to in_gid()
This commit is contained in:
Yu Watanabe 2019-11-03 22:54:32 +09:00 committed by GitHub
commit f4dad55b8b
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
2 changed files with 44 additions and 15 deletions

View File

@ -409,8 +409,10 @@ char* gid_to_name(gid_t gid) {
}
int in_gid(gid_t gid) {
gid_t *gids;
int ngroups, r, i;
_cleanup_free_ gid_t *allocated = NULL;
gid_t local[16], *p = local;
int ngroups = ELEMENTSOF(local);
unsigned attempt = 0;
if (getgid() == gid)
return 1;
@ -421,23 +423,39 @@ int in_gid(gid_t gid) {
if (!gid_is_valid(gid))
return -EINVAL;
ngroups = getgroups(0, NULL);
if (ngroups < 0)
return -errno;
if (ngroups == 0)
return 0;
for (;;) {
ngroups = getgroups(ngroups, p);
if (ngroups >= 0)
break;
if (errno != EINVAL)
return -errno;
gids = newa(gid_t, ngroups);
/* Give up eventually */
if (attempt++ > 10)
return -EINVAL;
r = getgroups(ngroups, gids);
if (r < 0)
return -errno;
/* Get actual size needed, and size the array explicitly. Note that this is potentially racy
* to use (in multi-threaded programs), hence let's call this in a loop. */
ngroups = getgroups(0, NULL);
if (ngroups < 0)
return -errno;
if (ngroups == 0)
return false;
for (i = 0; i < r; i++)
if (gids[i] == gid)
return 1;
free(allocated);
return 0;
allocated = new(gid_t, ngroups);
if (!allocated)
return -ENOMEM;
p = allocated;
}
for (int i = 0; i < ngroups; i++)
if (p[i] == gid)
return true;
return false;
}
int in_group(const char *name) {

View File

@ -286,6 +286,15 @@ static void test_make_salt(void) {
assert(!streq(s, t));
}
static void test_in_gid(void) {
assert(in_gid(getgid()) >= 0);
assert(in_gid(getegid()) >= 0);
assert(in_gid(GID_INVALID) < 0);
assert(in_gid(TTY_GID) == 0); /* The TTY gid is for owning ttys, it would be really really weird if we were in it. */
}
int main(int argc, char *argv[]) {
test_uid_to_name_one(0, "root");
test_uid_to_name_one(UID_NOBODY, NOBODY_USER_NAME);
@ -320,5 +329,7 @@ int main(int argc, char *argv[]) {
test_make_salt();
test_in_gid();
return 0;
}