mirror of
https://gitlab.freedesktop.org/pulseaudio/pulseaudio.git
synced 2025-10-31 22:25:33 -04:00
core: Add thread-safe group info functions with dynamic buffers
Provides getgrgid, getgrnam, getpwuid & getpwnam replacements that are thread safe (a la getgrgid_r() and friends) that internally handle allocating big-enough buffers to avoid ERANGE errors on large users or groups.
This commit is contained in:
parent
9d1cc133f3
commit
15eb03a5b3
6 changed files with 643 additions and 115 deletions
|
|
@ -280,7 +280,8 @@ TESTS = \
|
||||||
proplist-test \
|
proplist-test \
|
||||||
lock-autospawn-test \
|
lock-autospawn-test \
|
||||||
prioq-test \
|
prioq-test \
|
||||||
sigbus-test
|
sigbus-test \
|
||||||
|
usergroup-test
|
||||||
|
|
||||||
TESTS_BINARIES = \
|
TESTS_BINARIES = \
|
||||||
mainloop-test \
|
mainloop-test \
|
||||||
|
|
@ -318,7 +319,8 @@ TESTS_BINARIES = \
|
||||||
stripnul \
|
stripnul \
|
||||||
lock-autospawn-test \
|
lock-autospawn-test \
|
||||||
prioq-test \
|
prioq-test \
|
||||||
sigbus-test
|
sigbus-test \
|
||||||
|
usergroup-test
|
||||||
|
|
||||||
if HAVE_SIGXCPU
|
if HAVE_SIGXCPU
|
||||||
#TESTS += \
|
#TESTS += \
|
||||||
|
|
@ -557,6 +559,11 @@ alsa_time_test_LDADD = $(AM_LDADD)
|
||||||
alsa_time_test_CFLAGS = $(AM_CFLAGS) $(ASOUNDLIB_CFLAGS)
|
alsa_time_test_CFLAGS = $(AM_CFLAGS) $(ASOUNDLIB_CFLAGS)
|
||||||
alsa_time_test_LDFLAGS = $(AM_LDFLAGS) $(BINLDFLAGS) $(ASOUNDLIB_LIBS)
|
alsa_time_test_LDFLAGS = $(AM_LDFLAGS) $(BINLDFLAGS) $(ASOUNDLIB_LIBS)
|
||||||
|
|
||||||
|
usergroup_test_SOURCES = tests/usergroup-test.c
|
||||||
|
usergroup_test_LDADD = $(AM_LDADD) libpulsecore-@PA_MAJORMINORMICRO@.la
|
||||||
|
usergroup_test_CFLAGS = $(AM_CFLAGS)
|
||||||
|
usergroup_test_LDFLAGS = $(AM_LDFLAGS) $(BINLDFLAGS)
|
||||||
|
|
||||||
###################################
|
###################################
|
||||||
# Common library #
|
# Common library #
|
||||||
###################################
|
###################################
|
||||||
|
|
@ -621,6 +628,7 @@ libpulsecommon_@PA_MAJORMINORMICRO@_la_SOURCES = \
|
||||||
pulsecore/tagstruct.c pulsecore/tagstruct.h \
|
pulsecore/tagstruct.c pulsecore/tagstruct.h \
|
||||||
pulsecore/time-smoother.c pulsecore/time-smoother.h \
|
pulsecore/time-smoother.c pulsecore/time-smoother.h \
|
||||||
pulsecore/tokenizer.c pulsecore/tokenizer.h \
|
pulsecore/tokenizer.c pulsecore/tokenizer.h \
|
||||||
|
pulsecore/usergroup.c pulsecore/usergroup.h \
|
||||||
pulsecore/sndfile-util.c pulsecore/sndfile-util.h \
|
pulsecore/sndfile-util.c pulsecore/sndfile-util.h \
|
||||||
pulsecore/winsock.h
|
pulsecore/winsock.h
|
||||||
|
|
||||||
|
|
|
||||||
|
|
@ -61,38 +61,40 @@
|
||||||
#include <pulsecore/log.h>
|
#include <pulsecore/log.h>
|
||||||
#include <pulsecore/core-util.h>
|
#include <pulsecore/core-util.h>
|
||||||
#include <pulsecore/macro.h>
|
#include <pulsecore/macro.h>
|
||||||
|
#include <pulsecore/usergroup.h>
|
||||||
|
|
||||||
#include "util.h"
|
#include "util.h"
|
||||||
|
|
||||||
char *pa_get_user_name(char *s, size_t l) {
|
char *pa_get_user_name(char *s, size_t l) {
|
||||||
const char *p;
|
const char *p;
|
||||||
|
char *name = NULL;
|
||||||
|
#ifdef OS_IS_WIN32
|
||||||
char buf[1024];
|
char buf[1024];
|
||||||
|
#endif
|
||||||
|
|
||||||
#ifdef HAVE_PWD_H
|
#ifdef HAVE_PWD_H
|
||||||
struct passwd pw, *r;
|
struct passwd *r;
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
pa_assert(s);
|
pa_assert(s);
|
||||||
pa_assert(l > 0);
|
pa_assert(l > 0);
|
||||||
|
|
||||||
if (!(p = (getuid() == 0 ? "root" : NULL)) &&
|
if ((p = (getuid() == 0 ? "root" : NULL)) ||
|
||||||
!(p = getenv("USER")) &&
|
(p = getenv("USER")) ||
|
||||||
!(p = getenv("LOGNAME")) &&
|
(p = getenv("LOGNAME")) ||
|
||||||
!(p = getenv("USERNAME"))) {
|
(p = getenv("USERNAME")))
|
||||||
|
{
|
||||||
|
name = pa_strlcpy(s, p, l);
|
||||||
|
} else {
|
||||||
#ifdef HAVE_PWD_H
|
#ifdef HAVE_PWD_H
|
||||||
|
|
||||||
#ifdef HAVE_GETPWUID_R
|
if ((r = pa_getpwuid_malloc(getuid())) == NULL) {
|
||||||
if (getpwuid_r(getuid(), &pw, buf, sizeof(buf), &r) != 0 || !r) {
|
|
||||||
#else
|
|
||||||
/* XXX Not thread-safe, but needed on OSes (e.g. FreeBSD 4.X)
|
|
||||||
* that do not support getpwuid_r. */
|
|
||||||
if ((r = getpwuid(getuid())) == NULL) {
|
|
||||||
#endif
|
|
||||||
pa_snprintf(s, l, "%lu", (unsigned long) getuid());
|
pa_snprintf(s, l, "%lu", (unsigned long) getuid());
|
||||||
return s;
|
return s;
|
||||||
}
|
}
|
||||||
|
|
||||||
p = r->pw_name;
|
name = pa_strlcpy(s, r->pw_name, l);
|
||||||
|
pa_getpwuid_free(r);
|
||||||
|
|
||||||
#elif defined(OS_IS_WIN32) /* HAVE_PWD_H */
|
#elif defined(OS_IS_WIN32) /* HAVE_PWD_H */
|
||||||
DWORD size = sizeof(buf);
|
DWORD size = sizeof(buf);
|
||||||
|
|
@ -102,7 +104,7 @@ char *pa_get_user_name(char *s, size_t l) {
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
p = buf;
|
name = pa_strlcpy(s, buf, l);
|
||||||
|
|
||||||
#else /* HAVE_PWD_H */
|
#else /* HAVE_PWD_H */
|
||||||
|
|
||||||
|
|
@ -110,7 +112,7 @@ char *pa_get_user_name(char *s, size_t l) {
|
||||||
#endif /* HAVE_PWD_H */
|
#endif /* HAVE_PWD_H */
|
||||||
}
|
}
|
||||||
|
|
||||||
return pa_strlcpy(s, p, l);
|
return name;
|
||||||
}
|
}
|
||||||
|
|
||||||
char *pa_get_host_name(char *s, size_t l) {
|
char *pa_get_host_name(char *s, size_t l) {
|
||||||
|
|
@ -126,11 +128,10 @@ char *pa_get_host_name(char *s, size_t l) {
|
||||||
}
|
}
|
||||||
|
|
||||||
char *pa_get_home_dir(char *s, size_t l) {
|
char *pa_get_home_dir(char *s, size_t l) {
|
||||||
char *e;
|
char *e, *dir;
|
||||||
|
|
||||||
#ifdef HAVE_PWD_H
|
#ifdef HAVE_PWD_H
|
||||||
char buf[1024];
|
struct passwd *r;
|
||||||
struct passwd pw, *r;
|
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
pa_assert(s);
|
pa_assert(s);
|
||||||
|
|
@ -143,22 +144,19 @@ char *pa_get_home_dir(char *s, size_t l) {
|
||||||
return pa_strlcpy(s, e, l);
|
return pa_strlcpy(s, e, l);
|
||||||
|
|
||||||
#ifdef HAVE_PWD_H
|
#ifdef HAVE_PWD_H
|
||||||
|
|
||||||
errno = 0;
|
errno = 0;
|
||||||
#ifdef HAVE_GETPWUID_R
|
if ((r = pa_getpwuid_malloc(getuid())) == NULL) {
|
||||||
if (getpwuid_r(getuid(), &pw, buf, sizeof(buf), &r) != 0 || !r) {
|
|
||||||
#else
|
|
||||||
/* XXX Not thread-safe, but needed on OSes (e.g. FreeBSD 4.X)
|
|
||||||
* that do not support getpwuid_r. */
|
|
||||||
if ((r = getpwuid(getuid())) == NULL) {
|
|
||||||
#endif
|
|
||||||
if (!errno)
|
if (!errno)
|
||||||
errno = ENOENT;
|
errno = ENOENT;
|
||||||
|
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
return pa_strlcpy(s, r->pw_dir, l);
|
dir = pa_strlcpy(s, r->pw_dir, l);
|
||||||
|
|
||||||
|
pa_getpwuid_free(r);
|
||||||
|
|
||||||
|
return dir;
|
||||||
#else /* HAVE_PWD_H */
|
#else /* HAVE_PWD_H */
|
||||||
|
|
||||||
errno = ENOENT;
|
errno = ENOENT;
|
||||||
|
|
|
||||||
|
|
@ -115,6 +115,7 @@
|
||||||
#include <pulsecore/macro.h>
|
#include <pulsecore/macro.h>
|
||||||
#include <pulsecore/thread.h>
|
#include <pulsecore/thread.h>
|
||||||
#include <pulsecore/strbuf.h>
|
#include <pulsecore/strbuf.h>
|
||||||
|
#include <pulsecore/usergroup.h>
|
||||||
|
|
||||||
#include "core-util.h"
|
#include "core-util.h"
|
||||||
|
|
||||||
|
|
@ -969,42 +970,24 @@ fail:
|
||||||
|
|
||||||
/* Check whether the specified GID and the group name match */
|
/* Check whether the specified GID and the group name match */
|
||||||
static int is_group(gid_t gid, const char *name) {
|
static int is_group(gid_t gid, const char *name) {
|
||||||
struct group group, *result = NULL;
|
struct group *group = NULL;
|
||||||
long n;
|
|
||||||
void *data;
|
|
||||||
int r = -1;
|
int r = -1;
|
||||||
|
|
||||||
#ifdef HAVE_GETGRGID_R
|
|
||||||
|
|
||||||
#ifdef _SC_GETGR_R_SIZE_MAX
|
|
||||||
n = sysconf(_SC_GETGR_R_SIZE_MAX);
|
|
||||||
#else
|
|
||||||
n = -1;
|
|
||||||
#endif
|
|
||||||
if (n <= 0)
|
|
||||||
n = 512;
|
|
||||||
|
|
||||||
data = pa_xmalloc((size_t) n);
|
|
||||||
|
|
||||||
if ((errno = getgrgid_r(gid, &group, data, (size_t) n, &result)) || !result)
|
|
||||||
#else
|
|
||||||
errno = 0;
|
errno = 0;
|
||||||
if (!(result = getgrgid(gid)))
|
if (!(group = pa_getgrgid_malloc(gid)))
|
||||||
#endif
|
|
||||||
{
|
{
|
||||||
if (!errno)
|
if (!errno)
|
||||||
errno = ENOENT;
|
errno = ENOENT;
|
||||||
|
|
||||||
pa_log("getgrgid(%u): %s", gid, pa_cstrerror(errno));
|
pa_log("pa_getgrgid_malloc(%u): %s", gid, pa_cstrerror(errno));
|
||||||
|
|
||||||
goto finish;
|
goto finish;
|
||||||
}
|
}
|
||||||
|
|
||||||
r = strcmp(name, result->gr_name) == 0;
|
r = strcmp(name, group->gr_name) == 0;
|
||||||
|
|
||||||
finish:
|
finish:
|
||||||
|
pa_getgrgid_free(group);
|
||||||
pa_xfree(data);
|
|
||||||
|
|
||||||
return r;
|
return r;
|
||||||
}
|
}
|
||||||
|
|
@ -1053,69 +1036,37 @@ finish:
|
||||||
|
|
||||||
/* Check whether the specifc user id is a member of the specified group */
|
/* Check whether the specifc user id is a member of the specified group */
|
||||||
int pa_uid_in_group(uid_t uid, const char *name) {
|
int pa_uid_in_group(uid_t uid, const char *name) {
|
||||||
char *g_buf = NULL, *p_buf = NULL;
|
struct group *group = NULL;
|
||||||
long g_n, p_n;
|
|
||||||
struct group grbuf, *gr = NULL;
|
|
||||||
char **i;
|
char **i;
|
||||||
int r = -1;
|
int r = -1;
|
||||||
|
|
||||||
#ifdef HAVE_GETGRNAM_R
|
|
||||||
|
|
||||||
#ifdef _SC_GETGR_R_SIZE_MAX
|
|
||||||
g_n = sysconf(_SC_GETGR_R_SIZE_MAX);
|
|
||||||
#else
|
|
||||||
g_n = -1;
|
|
||||||
#endif
|
|
||||||
if (g_n <= 0)
|
|
||||||
g_n = 512;
|
|
||||||
|
|
||||||
g_buf = pa_xmalloc((size_t) g_n);
|
|
||||||
|
|
||||||
if ((errno = getgrnam_r(name, &grbuf, g_buf, (size_t) g_n, &gr)) != 0 || !gr)
|
|
||||||
#else
|
|
||||||
errno = 0;
|
errno = 0;
|
||||||
if (!(gr = getgrnam(name)))
|
if (!(group = pa_getgrnam_malloc(name)))
|
||||||
#endif
|
|
||||||
{
|
{
|
||||||
if (!errno)
|
if (!errno)
|
||||||
errno = ENOENT;
|
errno = ENOENT;
|
||||||
goto finish;
|
goto finish;
|
||||||
}
|
}
|
||||||
|
|
||||||
#ifdef HAVE_GETPWNAM_R
|
|
||||||
|
|
||||||
#ifdef _SC_GETPW_R_SIZE_MAX
|
|
||||||
p_n = sysconf(_SC_GETPW_R_SIZE_MAX);
|
|
||||||
#else
|
|
||||||
p_n = -1;
|
|
||||||
#endif
|
|
||||||
if (p_n <= 0)
|
|
||||||
p_n = 512;
|
|
||||||
|
|
||||||
p_buf = pa_xmalloc((size_t) p_n);
|
|
||||||
#endif
|
|
||||||
|
|
||||||
r = 0;
|
r = 0;
|
||||||
for (i = gr->gr_mem; *i; i++) {
|
for (i = group->gr_mem; *i; i++) {
|
||||||
struct passwd pwbuf, *pw = NULL;
|
struct passwd *pw = NULL;
|
||||||
|
|
||||||
#ifdef HAVE_GETPWNAM_R
|
|
||||||
if ((errno = getpwnam_r(*i, &pwbuf, p_buf, (size_t) p_n, &pw)) != 0 || !pw)
|
|
||||||
#else
|
|
||||||
errno = 0;
|
errno = 0;
|
||||||
if (!(pw = getpwnam(*i)))
|
if (!(pw = pa_getpwnam_malloc(*i)))
|
||||||
#endif
|
|
||||||
continue;
|
continue;
|
||||||
|
|
||||||
if (pw->pw_uid == uid) {
|
if (pw->pw_uid == uid)
|
||||||
r = 1;
|
r = 1;
|
||||||
|
|
||||||
|
pa_getpwnam_free(pw);
|
||||||
|
|
||||||
|
if (r == 1)
|
||||||
break;
|
break;
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
finish:
|
finish:
|
||||||
pa_xfree(g_buf);
|
pa_getgrnam_free(group);
|
||||||
pa_xfree(p_buf);
|
|
||||||
|
|
||||||
return r;
|
return r;
|
||||||
}
|
}
|
||||||
|
|
@ -1123,27 +1074,10 @@ finish:
|
||||||
/* Get the GID of a gfiven group, return (gid_t) -1 on failure. */
|
/* Get the GID of a gfiven group, return (gid_t) -1 on failure. */
|
||||||
gid_t pa_get_gid_of_group(const char *name) {
|
gid_t pa_get_gid_of_group(const char *name) {
|
||||||
gid_t ret = (gid_t) -1;
|
gid_t ret = (gid_t) -1;
|
||||||
char *g_buf = NULL;
|
struct group *gr = NULL;
|
||||||
long g_n;
|
|
||||||
struct group grbuf, *gr = NULL;
|
|
||||||
|
|
||||||
#ifdef HAVE_GETGRNAM_R
|
|
||||||
|
|
||||||
#ifdef _SC_GETGR_R_SIZE_MAX
|
|
||||||
g_n = sysconf(_SC_GETGR_R_SIZE_MAX);
|
|
||||||
#else
|
|
||||||
g_n = -1;
|
|
||||||
#endif
|
|
||||||
if (g_n <= 0)
|
|
||||||
g_n = 512;
|
|
||||||
|
|
||||||
g_buf = pa_xmalloc((size_t) g_n);
|
|
||||||
|
|
||||||
if ((errno = getgrnam_r(name, &grbuf, g_buf, (size_t) g_n, &gr)) != 0 || !gr)
|
|
||||||
#else
|
|
||||||
errno = 0;
|
errno = 0;
|
||||||
if (!(gr = getgrnam(name)))
|
if (!(gr = pa_getgrnam_malloc(name)))
|
||||||
#endif
|
|
||||||
{
|
{
|
||||||
if (!errno)
|
if (!errno)
|
||||||
errno = ENOENT;
|
errno = ENOENT;
|
||||||
|
|
@ -1153,7 +1087,7 @@ gid_t pa_get_gid_of_group(const char *name) {
|
||||||
ret = gr->gr_gid;
|
ret = gr->gr_gid;
|
||||||
|
|
||||||
finish:
|
finish:
|
||||||
pa_xfree(g_buf);
|
pa_getgrnam_free(gr);
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
||||||
376
src/pulsecore/usergroup.c
Normal file
376
src/pulsecore/usergroup.c
Normal file
|
|
@ -0,0 +1,376 @@
|
||||||
|
/***
|
||||||
|
This file is part of PulseAudio.
|
||||||
|
|
||||||
|
Copyright 2009 Ted Percival
|
||||||
|
|
||||||
|
PulseAudio is free software; you can redistribute it and/or modify
|
||||||
|
it under the terms of the GNU Lesser General Public License as
|
||||||
|
published by the Free Software Foundation; either version 2.1 of the
|
||||||
|
License, or (at your option) any later version.
|
||||||
|
|
||||||
|
PulseAudio is distributed in the hope that it will be useful, but
|
||||||
|
WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||||||
|
Lesser General Public License for more details.
|
||||||
|
|
||||||
|
You should have received a copy of the GNU Lesser General Public
|
||||||
|
License along with PulseAudio; if not, write to the Free Software
|
||||||
|
Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307
|
||||||
|
USA.
|
||||||
|
***/
|
||||||
|
|
||||||
|
#ifdef HAVE_CONFIG_H
|
||||||
|
#include <config.h>
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#include <sys/types.h>
|
||||||
|
#include <errno.h>
|
||||||
|
|
||||||
|
#ifdef HAVE_PWD_H
|
||||||
|
#include <pwd.h>
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#ifdef HAVE_GRP_H
|
||||||
|
#include <grp.h>
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#include <pulse/xmalloc.h>
|
||||||
|
#include <pulsecore/macro.h>
|
||||||
|
|
||||||
|
#include "usergroup.h"
|
||||||
|
|
||||||
|
#ifdef HAVE_GRP_H
|
||||||
|
|
||||||
|
/* Returns a suitable starting size for a getgrnam_r() or getgrgid_r() buffer,
|
||||||
|
plus the size of a struct group.
|
||||||
|
*/
|
||||||
|
static size_t starting_getgr_buflen(void) {
|
||||||
|
size_t full_size;
|
||||||
|
long n;
|
||||||
|
#ifdef _SC_GETGR_R_SIZE_MAX
|
||||||
|
n = sysconf(_SC_GETGR_R_SIZE_MAX);
|
||||||
|
#else
|
||||||
|
n = -1;
|
||||||
|
#endif
|
||||||
|
if (n <= 0)
|
||||||
|
n = 512;
|
||||||
|
|
||||||
|
full_size = (size_t) n + sizeof(struct group);
|
||||||
|
|
||||||
|
if (full_size < (size_t) n) /* check for integer overflow */
|
||||||
|
return (size_t) n;
|
||||||
|
|
||||||
|
return full_size;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Returns a suitable starting size for a getpwnam_r() or getpwuid_r() buffer,
|
||||||
|
plus the size of a struct passwd.
|
||||||
|
*/
|
||||||
|
static size_t starting_getpw_buflen(void) {
|
||||||
|
long n;
|
||||||
|
size_t full_size;
|
||||||
|
|
||||||
|
#ifdef _SC_GETPW_R_SIZE_MAX
|
||||||
|
n = sysconf(_SC_GETPW_R_SIZE_MAX);
|
||||||
|
#else
|
||||||
|
n = -1;
|
||||||
|
#endif
|
||||||
|
if (n <= 0)
|
||||||
|
n = 512;
|
||||||
|
|
||||||
|
full_size = (size_t) n + sizeof(struct passwd);
|
||||||
|
|
||||||
|
if (full_size < (size_t) n) /* check for integer overflow */
|
||||||
|
return (size_t) n;
|
||||||
|
|
||||||
|
return full_size;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Given a memory allocation (*bufptr) and its length (*buflenptr),
|
||||||
|
double the size of the allocation, updating the given buffer and length
|
||||||
|
arguments. This function should be used in conjunction with the pa_*alloc
|
||||||
|
and pa_xfree functions.
|
||||||
|
|
||||||
|
Unlike realloc(), this function does *not* retain the original buffer's
|
||||||
|
contents.
|
||||||
|
|
||||||
|
Returns 0 on success, nonzero on error. The error cause is indicated by
|
||||||
|
errno.
|
||||||
|
*/
|
||||||
|
static int expand_buffer_trashcontents(void **bufptr, size_t *buflenptr) {
|
||||||
|
size_t newlen;
|
||||||
|
|
||||||
|
if (!bufptr || !*bufptr || !buflenptr) {
|
||||||
|
errno = EINVAL;
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
newlen = *buflenptr * 2;
|
||||||
|
|
||||||
|
if (newlen < *buflenptr) {
|
||||||
|
errno = EOVERFLOW;
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Don't bother retaining memory contents; free & alloc anew */
|
||||||
|
pa_xfree(*bufptr);
|
||||||
|
|
||||||
|
*bufptr = pa_xmalloc(newlen);
|
||||||
|
*buflenptr = newlen;
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
#ifdef HAVE_GETGRGID_R
|
||||||
|
/* Thread-safe getgrgid() replacement.
|
||||||
|
Returned value should be freed using pa_getgrgid_free() when the caller is
|
||||||
|
finished with the returned group data.
|
||||||
|
|
||||||
|
API is the same as getgrgid(), errors are indicated by a NULL return;
|
||||||
|
consult errno for the error cause (zero it before calling).
|
||||||
|
The returned value must be freed using pa_xfree().
|
||||||
|
*/
|
||||||
|
struct group *pa_getgrgid_malloc(gid_t gid) {
|
||||||
|
size_t buflen, getgr_buflen;
|
||||||
|
int err;
|
||||||
|
void *buf;
|
||||||
|
void *getgr_buf;
|
||||||
|
struct group *result = NULL;
|
||||||
|
|
||||||
|
buflen = starting_getgr_buflen();
|
||||||
|
buf = pa_xmalloc(buflen);
|
||||||
|
|
||||||
|
getgr_buflen = buflen - sizeof(struct group);
|
||||||
|
getgr_buf = (char *)buf + sizeof(struct group);
|
||||||
|
|
||||||
|
while ((err = getgrgid_r(gid, (struct group *)buf, getgr_buf,
|
||||||
|
getgr_buflen, &result)) == ERANGE)
|
||||||
|
{
|
||||||
|
if (expand_buffer_trashcontents(&buf, &buflen))
|
||||||
|
break;
|
||||||
|
|
||||||
|
getgr_buflen = buflen - sizeof(struct group);
|
||||||
|
getgr_buf = (char *)buf + sizeof(struct group);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (err || !result) {
|
||||||
|
result = NULL;
|
||||||
|
if (buf) {
|
||||||
|
pa_xfree(buf);
|
||||||
|
buf = NULL;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pa_assert(result == buf || result == NULL);
|
||||||
|
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
void pa_getgrgid_free(struct group *grp) {
|
||||||
|
pa_xfree(grp);
|
||||||
|
}
|
||||||
|
|
||||||
|
#else /* !HAVE_GETGRGID_R */
|
||||||
|
|
||||||
|
struct group *pa_getgrgid_malloc(gid_t gid) {
|
||||||
|
return getgrgid(gid);
|
||||||
|
}
|
||||||
|
|
||||||
|
void pa_getgrgid_free(struct group *grp) {
|
||||||
|
/* nothing */
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
#endif /* !HAVE_GETGRGID_R */
|
||||||
|
|
||||||
|
#ifdef HAVE_GETGRNAM_R
|
||||||
|
/* Thread-safe getgrnam() function.
|
||||||
|
Returned value should be freed using pa_getgrnam_free() when the caller is
|
||||||
|
finished with the returned group data.
|
||||||
|
|
||||||
|
API is the same as getgrnam(), errors are indicated by a NULL return;
|
||||||
|
consult errno for the error cause (zero it before calling).
|
||||||
|
The returned value must be freed using pa_xfree().
|
||||||
|
*/
|
||||||
|
struct group *pa_getgrnam_malloc(const char *name) {
|
||||||
|
size_t buflen, getgr_buflen;
|
||||||
|
int err;
|
||||||
|
void *buf;
|
||||||
|
void *getgr_buf;
|
||||||
|
struct group *result = NULL;
|
||||||
|
|
||||||
|
buflen = starting_getgr_buflen();
|
||||||
|
buf = pa_xmalloc(buflen);
|
||||||
|
|
||||||
|
getgr_buflen = buflen - sizeof(struct group);
|
||||||
|
getgr_buf = (char *)buf + sizeof(struct group);
|
||||||
|
|
||||||
|
while ((err = getgrnam_r(name, (struct group *)buf, getgr_buf,
|
||||||
|
getgr_buflen, &result)) == ERANGE)
|
||||||
|
{
|
||||||
|
if (expand_buffer_trashcontents(&buf, &buflen))
|
||||||
|
break;
|
||||||
|
|
||||||
|
getgr_buflen = buflen - sizeof(struct group);
|
||||||
|
getgr_buf = (char *)buf + sizeof(struct group);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (err || !result) {
|
||||||
|
result = NULL;
|
||||||
|
if (buf) {
|
||||||
|
pa_xfree(buf);
|
||||||
|
buf = NULL;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pa_assert(result == buf || result == NULL);
|
||||||
|
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
void pa_getgrnam_free(struct group *group) {
|
||||||
|
pa_xfree(group);
|
||||||
|
}
|
||||||
|
|
||||||
|
#else /* !HAVE_GETGRNAM_R */
|
||||||
|
|
||||||
|
struct group *pa_getgrnam_malloc(const char *name) {
|
||||||
|
return getgrnam(name);
|
||||||
|
}
|
||||||
|
|
||||||
|
void pa_getgrnam_free(struct group *group) {
|
||||||
|
/* nothing */
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
#endif /* HAVE_GETGRNAM_R */
|
||||||
|
|
||||||
|
#endif /* HAVE_GRP_H */
|
||||||
|
|
||||||
|
#ifdef HAVE_PWD_H
|
||||||
|
|
||||||
|
#ifdef HAVE_GETPWNAM_R
|
||||||
|
/* Thread-safe getpwnam() function.
|
||||||
|
Returned value should be freed using pa_getpwnam_free() when the caller is
|
||||||
|
finished with the returned passwd data.
|
||||||
|
|
||||||
|
API is the same as getpwnam(), errors are indicated by a NULL return;
|
||||||
|
consult errno for the error cause (zero it before calling).
|
||||||
|
The returned value must be freed using pa_xfree().
|
||||||
|
*/
|
||||||
|
struct passwd *pa_getpwnam_malloc(const char *name) {
|
||||||
|
size_t buflen, getpw_buflen;
|
||||||
|
int err;
|
||||||
|
void *buf;
|
||||||
|
void *getpw_buf;
|
||||||
|
struct passwd *result = NULL;
|
||||||
|
|
||||||
|
buflen = starting_getpw_buflen();
|
||||||
|
buf = pa_xmalloc(buflen);
|
||||||
|
|
||||||
|
getpw_buflen = buflen - sizeof(struct passwd);
|
||||||
|
getpw_buf = (char *)buf + sizeof(struct passwd);
|
||||||
|
|
||||||
|
while ((err = getpwnam_r(name, (struct passwd *)buf, getpw_buf,
|
||||||
|
getpw_buflen, &result)) == ERANGE)
|
||||||
|
{
|
||||||
|
if (expand_buffer_trashcontents(&buf, &buflen))
|
||||||
|
break;
|
||||||
|
|
||||||
|
getpw_buflen = buflen - sizeof(struct passwd);
|
||||||
|
getpw_buf = (char *)buf + sizeof(struct passwd);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (err || !result) {
|
||||||
|
result = NULL;
|
||||||
|
if (buf) {
|
||||||
|
pa_xfree(buf);
|
||||||
|
buf = NULL;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pa_assert(result == buf || result == NULL);
|
||||||
|
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
void pa_getpwnam_free(struct passwd *passwd) {
|
||||||
|
pa_xfree(passwd);
|
||||||
|
}
|
||||||
|
|
||||||
|
#else /* !HAVE_GETPWNAM_R */
|
||||||
|
|
||||||
|
struct passwd *pa_getpwnam_malloc(const char *name) {
|
||||||
|
return getpwnam(name);
|
||||||
|
}
|
||||||
|
|
||||||
|
void pa_getpwnam_free(struct passwd *passwd) {
|
||||||
|
/* nothing */
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
#endif /* !HAVE_GETPWNAM_R */
|
||||||
|
|
||||||
|
#ifdef HAVE_GETPWUID_R
|
||||||
|
/* Thread-safe getpwuid() function.
|
||||||
|
Returned value should be freed using pa_getpwuid_free() when the caller is
|
||||||
|
finished with the returned group data.
|
||||||
|
|
||||||
|
API is the same as getpwuid(), errors are indicated by a NULL return;
|
||||||
|
consult errno for the error cause (zero it before calling).
|
||||||
|
The returned value must be freed using pa_xfree().
|
||||||
|
*/
|
||||||
|
struct passwd *pa_getpwuid_malloc(uid_t uid) {
|
||||||
|
size_t buflen, getpw_buflen;
|
||||||
|
int err;
|
||||||
|
void *buf;
|
||||||
|
void *getpw_buf;
|
||||||
|
struct passwd *result = NULL;
|
||||||
|
|
||||||
|
buflen = starting_getpw_buflen();
|
||||||
|
buf = pa_xmalloc(buflen);
|
||||||
|
|
||||||
|
getpw_buflen = buflen - sizeof(struct passwd);
|
||||||
|
getpw_buf = (char *)buf + sizeof(struct passwd);
|
||||||
|
|
||||||
|
while ((err = getpwuid_r(uid, (struct passwd *)buf, getpw_buf,
|
||||||
|
getpw_buflen, &result)) == ERANGE)
|
||||||
|
{
|
||||||
|
if (expand_buffer_trashcontents(&buf, &buflen))
|
||||||
|
break;
|
||||||
|
|
||||||
|
getpw_buflen = buflen - sizeof(struct passwd);
|
||||||
|
getpw_buf = (char *)buf + sizeof(struct passwd);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (err || !result) {
|
||||||
|
result = NULL;
|
||||||
|
if (buf) {
|
||||||
|
pa_xfree(buf);
|
||||||
|
buf = NULL;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pa_assert(result == buf || result == NULL);
|
||||||
|
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
void pa_getpwuid_free(struct passwd *passwd) {
|
||||||
|
pa_xfree(passwd);
|
||||||
|
}
|
||||||
|
|
||||||
|
#else /* !HAVE_GETPWUID_R */
|
||||||
|
|
||||||
|
struct passwd *pa_getpwuid_malloc(uid_t uid) {
|
||||||
|
return getpwuid(uid);
|
||||||
|
}
|
||||||
|
|
||||||
|
void pa_getpwuid_free(struct passwd *passwd) {
|
||||||
|
/* nothing */
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
#endif /* !HAVE_GETPWUID_R */
|
||||||
|
|
||||||
|
#endif /* HAVE_PWD_H */
|
||||||
51
src/pulsecore/usergroup.h
Normal file
51
src/pulsecore/usergroup.h
Normal file
|
|
@ -0,0 +1,51 @@
|
||||||
|
#ifndef foousergrouphfoo
|
||||||
|
#define foousergrouphfoo
|
||||||
|
|
||||||
|
/***
|
||||||
|
This file is part of PulseAudio.
|
||||||
|
|
||||||
|
Copyright 2009 Ted Percival
|
||||||
|
|
||||||
|
PulseAudio is free software; you can redistribute it and/or modify
|
||||||
|
it under the terms of the GNU Lesser General Public License as
|
||||||
|
published by the Free Software Foundation; either version 2.1 of the
|
||||||
|
License, or (at your option) any later version.
|
||||||
|
|
||||||
|
PulseAudio is distributed in the hope that it will be useful, but
|
||||||
|
WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||||||
|
Lesser General Public License for more details.
|
||||||
|
|
||||||
|
You should have received a copy of the GNU Lesser General Public
|
||||||
|
License along with PulseAudio; if not, write to the Free Software
|
||||||
|
Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307
|
||||||
|
USA.
|
||||||
|
***/
|
||||||
|
|
||||||
|
#include <sys/types.h>
|
||||||
|
|
||||||
|
#ifndef PACKAGE
|
||||||
|
#error "Please include config.h before including this file!"
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#ifdef HAVE_GRP_H
|
||||||
|
|
||||||
|
struct group *pa_getgrgid_malloc(gid_t gid);
|
||||||
|
void pa_getgrgid_free(struct group *grp);
|
||||||
|
|
||||||
|
struct group *pa_getgrnam_malloc(const char *name);
|
||||||
|
void pa_getgrnam_free(struct group *group);
|
||||||
|
|
||||||
|
#endif /* HAVE_GRP_H */
|
||||||
|
|
||||||
|
#ifdef HAVE_PWD_H
|
||||||
|
|
||||||
|
struct passwd *pa_getpwuid_malloc(uid_t uid);
|
||||||
|
void pa_getpwuid_free(struct passwd *passwd);
|
||||||
|
|
||||||
|
struct passwd *pa_getpwnam_malloc(const char *name);
|
||||||
|
void pa_getpwnam_free(struct passwd *passwd);
|
||||||
|
|
||||||
|
#endif /* HAVE_PWD_H */
|
||||||
|
|
||||||
|
#endif /* foousergrouphfoo */
|
||||||
161
src/tests/usergroup-test.c
Normal file
161
src/tests/usergroup-test.c
Normal file
|
|
@ -0,0 +1,161 @@
|
||||||
|
/***
|
||||||
|
This file is part of PulseAudio.
|
||||||
|
|
||||||
|
Copyright 2009 Ted Percival
|
||||||
|
|
||||||
|
PulseAudio is free software; you can redistribute it and/or modify
|
||||||
|
it under the terms of the GNU Lesser General Public License as
|
||||||
|
published by the Free Software Foundation; either version 2.1 of the
|
||||||
|
License, or (at your option) any later version.
|
||||||
|
|
||||||
|
PulseAudio is distributed in the hope that it will be useful, but
|
||||||
|
WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||||||
|
Lesser General Public License for more details.
|
||||||
|
|
||||||
|
You should have received a copy of the GNU Lesser General Public
|
||||||
|
License along with PulseAudio; if not, write to the Free Software
|
||||||
|
Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307
|
||||||
|
USA.
|
||||||
|
***/
|
||||||
|
|
||||||
|
#include <config.h>
|
||||||
|
|
||||||
|
#include <stdio.h>
|
||||||
|
#include <stdlib.h>
|
||||||
|
#include <string.h>
|
||||||
|
#include <sys/types.h>
|
||||||
|
#include <pwd.h>
|
||||||
|
#include <grp.h>
|
||||||
|
#include <errno.h>
|
||||||
|
|
||||||
|
#include <pulsecore/usergroup.h>
|
||||||
|
|
||||||
|
static int load_reference_structs(struct group **gr, struct passwd **pw) {
|
||||||
|
setpwent();
|
||||||
|
*pw = getpwent();
|
||||||
|
endpwent();
|
||||||
|
|
||||||
|
setgrent();
|
||||||
|
*gr = getgrent();
|
||||||
|
endgrent();
|
||||||
|
|
||||||
|
return (*gr && *pw) ? 0 : 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int compare_group(const struct group *a, const struct group *b) {
|
||||||
|
char **amem, **bmem;
|
||||||
|
|
||||||
|
if (strcmp(a->gr_name, b->gr_name)) {
|
||||||
|
fprintf(stderr, "Group name mismatch: [%s] [%s]\n",
|
||||||
|
a->gr_name, b->gr_name);
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (strcmp(a->gr_passwd, b->gr_passwd)) {
|
||||||
|
fprintf(stderr, "Group password mismatch: [%s] [%s]\n",
|
||||||
|
a->gr_passwd, b->gr_passwd);
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (a->gr_gid != b->gr_gid) {
|
||||||
|
fprintf(stderr, "Gid mismatch: [%lu] [%lu]\n",
|
||||||
|
(unsigned long) a->gr_gid, (unsigned long) b->gr_gid);
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* XXX: Assuming the group ordering is identical. */
|
||||||
|
for (amem = a->gr_mem, bmem = b->gr_mem; *amem && *bmem; ++amem, ++bmem) {
|
||||||
|
if (strcmp(*amem, *bmem)) {
|
||||||
|
fprintf(stderr, "Group member mismatch: [%s] [%s]\n",
|
||||||
|
*amem, *bmem);
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (*amem || *bmem) {
|
||||||
|
fprintf(stderr, "Mismatched group count\n");
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int compare_passwd(const struct passwd *a, const struct passwd *b) {
|
||||||
|
if (strcmp(a->pw_name, b->pw_name)) {
|
||||||
|
fprintf(stderr, "pw_name mismatch: [%s] [%s]\n", a->pw_name, b->pw_name);
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (strcmp(a->pw_passwd, b->pw_passwd)) {
|
||||||
|
fprintf(stderr, "pw_passwd mismatch: [%s] [%s]\n", a->pw_passwd, b->pw_passwd);
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (a->pw_uid != b->pw_uid) {
|
||||||
|
fprintf(stderr, "pw_uid mismatch: [%lu] [%lu]\n",
|
||||||
|
(unsigned long) a->pw_uid, (unsigned long) b->pw_uid);
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (a->pw_gid != b->pw_gid) {
|
||||||
|
fprintf(stderr, "pw_gid mismatch: [%lu] [%lu]\n",
|
||||||
|
(unsigned long) a->pw_gid, (unsigned long) b->pw_gid);
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (strcmp(a->pw_gecos, b->pw_gecos)) {
|
||||||
|
fprintf(stderr, "pw_gecos mismatch: [%s] [%s]\n", a->pw_gecos, b->pw_gecos);
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (strcmp(a->pw_dir, b->pw_dir)) {
|
||||||
|
fprintf(stderr, "pw_dir mismatch: [%s] [%s]\n", a->pw_dir, b->pw_dir);
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (strcmp(a->pw_shell, b->pw_shell)) {
|
||||||
|
fprintf(stderr, "pw_shell mismatch: [%s] [%s]\n", a->pw_shell, b->pw_shell);
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
int main(int argc, char *argv[]) {
|
||||||
|
struct group *gr;
|
||||||
|
struct passwd *pw;
|
||||||
|
int err;
|
||||||
|
struct group *reference_group = NULL;
|
||||||
|
struct passwd *reference_passwd = NULL;
|
||||||
|
|
||||||
|
err = load_reference_structs(&reference_group, &reference_passwd);
|
||||||
|
if (err)
|
||||||
|
return 77;
|
||||||
|
|
||||||
|
errno = 0;
|
||||||
|
gr = pa_getgrgid_malloc(reference_group->gr_gid);
|
||||||
|
if (compare_group(reference_group, gr))
|
||||||
|
return 1;
|
||||||
|
pa_getgrgid_free(gr);
|
||||||
|
|
||||||
|
errno = 0;
|
||||||
|
gr = pa_getgrnam_malloc(reference_group->gr_name);
|
||||||
|
if (compare_group(reference_group, gr))
|
||||||
|
return 1;
|
||||||
|
pa_getgrnam_free(gr);
|
||||||
|
|
||||||
|
errno = 0;
|
||||||
|
pw = pa_getpwuid_malloc(reference_passwd->pw_uid);
|
||||||
|
if (compare_passwd(reference_passwd, pw))
|
||||||
|
return 1;
|
||||||
|
pa_getpwuid_free(pw);
|
||||||
|
|
||||||
|
errno = 0;
|
||||||
|
pw = pa_getpwnam_malloc(reference_passwd->pw_name);
|
||||||
|
if (compare_passwd(reference_passwd, pw))
|
||||||
|
return 1;
|
||||||
|
pa_getpwnam_free(pw);
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
Loading…
Add table
Add a link
Reference in a new issue