X-Git-Url: https://git.octo.it/?a=blobdiff_plain;f=src%2Fexec.c;h=2c99ce2a7f296395b3e682d2f85ada85af95a5a9;hb=436dacdb577300ad6ed4692c95089f22ae98de3c;hp=22da3160be6c6bb7054c80f9533e8089d421910f;hpb=43ee9ced15521d541dd3fb0698c9c490339dd35f;p=collectd.git diff --git a/src/exec.c b/src/exec.c index 22da3160..2c99ce2a 100644 --- a/src/exec.c +++ b/src/exec.c @@ -39,6 +39,7 @@ #include #include +#include #ifdef HAVE_SYS_CAPABILITY_H #include #endif @@ -78,9 +79,14 @@ typedef struct program_list_and_notification_s { } program_list_and_notification_t; /* + * constants + */ +const long int MAX_GRBUF_SIZE = 65536; + +/* * Private variables */ -static program_list_t *pl_head = NULL; +static program_list_t *pl_head; static pthread_mutex_t pl_lock = PTHREAD_MUTEX_INITIALIZER; /* @@ -265,7 +271,6 @@ __attribute__((noreturn)) static void exec_child(program_list_t *pl, int uid, int gid, int egid) /* {{{ */ { int status; - char errbuf[1024]; #if HAVE_SETGROUPS if (getuid() == 0) { @@ -286,31 +291,27 @@ __attribute__((noreturn)) static void exec_child(program_list_t *pl, int uid, status = setgid(gid); if (status != 0) { - ERROR("exec plugin: setgid (%i) failed: %s", gid, - sstrerror(errno, errbuf, sizeof(errbuf))); + ERROR("exec plugin: setgid (%i) failed: %s", gid, STRERRNO); exit(-1); } if (egid != -1) { status = setegid(egid); if (status != 0) { - ERROR("exec plugin: setegid (%i) failed: %s", egid, - sstrerror(errno, errbuf, sizeof(errbuf))); + ERROR("exec plugin: setegid (%i) failed: %s", egid, STRERRNO); exit(-1); } } status = setuid(uid); if (status != 0) { - ERROR("exec plugin: setuid (%i) failed: %s", uid, - sstrerror(errno, errbuf, sizeof(errbuf))); + ERROR("exec plugin: setuid (%i) failed: %s", uid, STRERRNO); exit(-1); } execvp(pl->exec, pl->argv); - ERROR("exec plugin: Failed to execute ``%s'': %s", pl->exec, - sstrerror(errno, errbuf, sizeof(errbuf))); + ERROR("exec plugin: Failed to execute ``%s'': %s", pl->exec, STRERRNO); exit(-1); } /* void exec_child }}} */ @@ -324,13 +325,11 @@ static void reset_signal_mask(void) /* {{{ */ static int create_pipe(int fd_pipe[2]) /* {{{ */ { - char errbuf[1024]; int status; status = pipe(fd_pipe); if (status != 0) { - ERROR("exec plugin: pipe failed: %s", - sstrerror(errno, errbuf, sizeof(errbuf))); + ERROR("exec plugin: pipe failed: %s", STRERRNO); return -1; } @@ -358,7 +357,6 @@ static int fork_child(program_list_t *pl, int *fd_in, int *fd_out, int fd_pipe_in[2] = {-1, -1}; int fd_pipe_out[2] = {-1, -1}; int fd_pipe_err[2] = {-1, -1}; - char errbuf[1024]; int status; int pid; @@ -368,11 +366,17 @@ static int fork_child(program_list_t *pl, int *fd_in, int *fd_out, struct passwd *sp_ptr; struct passwd sp; - char nambuf[4096]; if (pl->pid != 0) return -1; + long int nambuf_size = sysconf(_SC_GETPW_R_SIZE_MAX); + if (nambuf_size <= 0) + nambuf_size = sysconf(_SC_PAGESIZE); + if (nambuf_size <= 0) + nambuf_size = 4096; + char nambuf[nambuf_size]; + if ((create_pipe(fd_pipe_in) == -1) || (create_pipe(fd_pipe_out) == -1) || (create_pipe(fd_pipe_err) == -1)) goto failed; @@ -381,7 +385,7 @@ static int fork_child(program_list_t *pl, int *fd_in, int *fd_out, status = getpwnam_r(pl->user, &sp, nambuf, sizeof(nambuf), &sp_ptr); if (status != 0) { ERROR("exec plugin: Failed to get user information for user ``%s'': %s", - pl->user, sstrerror(status, errbuf, sizeof(errbuf))); + pl->user, STRERROR(status)); goto failed; } @@ -405,19 +409,70 @@ static int fork_child(program_list_t *pl, int *fd_in, int *fd_out, struct group *gr_ptr = NULL; struct group gr; - status = getgrnam_r(pl->group, &gr, nambuf, sizeof(nambuf), &gr_ptr); - if (status != 0) { - ERROR("exec plugin: Failed to get group information " - "for group ``%s'': %s", - pl->group, sstrerror(status, errbuf, sizeof(errbuf))); - goto failed; + long int grbuf_size = sysconf(_SC_GETGR_R_SIZE_MAX); + if (grbuf_size <= 0) + grbuf_size = sysconf(_SC_PAGESIZE); + if (grbuf_size <= 0) + grbuf_size = 4096; + + long int size; + size = grbuf_size; + char *temp = NULL; + char *grbuf = NULL; + int getgr_failed = 0; + grbuf = malloc(size); + if (grbuf == NULL) { + ERROR("exec plugin: get group information for '%s' failed: buffer " + "malloc() failed", + pl->group); + getgr_failed = 1; + goto gr_finally; + } + int status; + while ((status = getgrnam_r(pl->group, &gr, grbuf, size, &gr_ptr)) != 0) { + switch (errno) { + case ERANGE: + if ((size + grbuf_size) < size || + (size + grbuf_size) > MAX_GRBUF_SIZE) { + ERROR("exec plugin: get group information for '%s' max buffer " + "limit (%ld) reached \n", + pl->group, MAX_GRBUF_SIZE); + getgr_failed = 1; + goto gr_finally; + } + /* grow the buffer by 'grbuf_size' each time getgrnamr fails */ + size += grbuf_size; + temp = realloc(grbuf, size); + if (temp == NULL) { + ERROR("exec plugin: get group information for '%s' realloc() " + "buffer to (%ld) failed ", + pl->group, size); + getgr_failed = 1; + goto gr_finally; + } + grbuf = temp; + break; + default: + ERROR("exec plugin: default errno: get group information for '%s' " + "failed : %s", + pl->group, sstrerror(status, errbuf, sizeof(errbuf))); + getgr_failed = 1; + goto gr_finally; + } } if (gr_ptr == NULL) { ERROR("exec plugin: No such group: `%s'", pl->group); - goto failed; + getgr_failed = 1; + goto gr_finally; } - egid = gr.gr_gid; + gr_finally: + free(grbuf); + DEBUG("exec plugin: release grbuf memory "); + grbuf = NULL; + if (getgr_failed > 0) { + goto failed; + } } else { egid = gid; } @@ -425,8 +480,7 @@ static int fork_child(program_list_t *pl, int *fd_in, int *fd_out, pid = fork(); if (pid < 0) { - ERROR("exec plugin: fork failed: %s", - sstrerror(errno, errbuf, sizeof(errbuf))); + ERROR("exec plugin: fork failed: %s", STRERRNO); goto failed; } else if (pid == 0) { int fd_num; @@ -675,9 +729,7 @@ static void *exec_notification_one(void *arg) /* {{{ */ fh = fdopen(fd, "w"); if (fh == NULL) { - char errbuf[1024]; - ERROR("exec plugin: fdopen (%i) failed: %s", fd, - sstrerror(errno, errbuf, sizeof(errbuf))); + ERROR("exec plugin: fdopen (%i) failed: %s", fd, STRERRNO); kill(pid, SIGTERM); close(fd); sfree(arg); @@ -786,7 +838,11 @@ static int exec_read(void) /* {{{ */ pthread_attr_init(&attr); pthread_attr_setdetachstate(&attr, PTHREAD_CREATE_DETACHED); - plugin_thread_create(&t, &attr, exec_read_one, (void *)pl, "exec read"); + int status = + plugin_thread_create(&t, &attr, exec_read_one, (void *)pl, "exec read"); + if (status != 0) { + ERROR("exec plugin: plugin_thread_create failed."); + } pthread_attr_destroy(&attr); } /* for (pl) */ @@ -825,8 +881,11 @@ static int exec_notification(const notification_t *n, /* {{{ */ pthread_attr_init(&attr); pthread_attr_setdetachstate(&attr, PTHREAD_CREATE_DETACHED); - plugin_thread_create(&t, &attr, exec_notification_one, (void *)pln, - "exec notify"); + int status = plugin_thread_create(&t, &attr, exec_notification_one, + (void *)pln, "exec notify"); + if (status != 0) { + ERROR("exec plugin: plugin_thread_create failed."); + } pthread_attr_destroy(&attr); } /* for (pl) */