123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147 |
- /*
- * This file implements the following functions:
- * pam_modutil_sanitize_helper_fds:
- * redirects standard descriptors, closes all other descriptors.
- */
- #include "pam_modutil_private.h"
- #include <security/pam_ext.h>
- #include <unistd.h>
- #include <fcntl.h>
- #include <syslog.h>
- #include <sys/resource.h>
- /*
- * Creates a pipe, closes its write end, redirects fd to its read end.
- * Returns fd on success, -1 otherwise.
- */
- static int
- redirect_in_pipe(pam_handle_t *pamh, int fd, const char *name)
- {
- int in[2];
- if (pipe(in) < 0) {
- pam_syslog(pamh, LOG_ERR, "Could not create pipe: %m");
- return -1;
- }
- close(in[1]);
- if (in[0] == fd)
- return fd;
- if (dup2(in[0], fd) != fd) {
- pam_syslog(pamh, LOG_ERR, "dup2 of %s failed: %m", name);
- fd = -1;
- }
- close(in[0]);
- return fd;
- }
- /*
- * Opens /dev/null for writing, redirects fd there.
- * Returns fd on success, -1 otherwise.
- */
- static int
- redirect_out_null(pam_handle_t *pamh, int fd, const char *name)
- {
- int null = open("/dev/null", O_WRONLY);
- if (null < 0) {
- pam_syslog(pamh, LOG_ERR, "open of %s failed: %m", "/dev/null");
- return -1;
- }
- if (null == fd)
- return fd;
- if (dup2(null, fd) != fd) {
- pam_syslog(pamh, LOG_ERR, "dup2 of %s failed: %m", name);
- fd = -1;
- }
- close(null);
- return fd;
- }
- static int
- redirect_out(pam_handle_t *pamh, enum pam_modutil_redirect_fd mode,
- int fd, const char *name)
- {
- switch (mode) {
- case PAM_MODUTIL_PIPE_FD:
- if (redirect_in_pipe(pamh, fd, name) < 0)
- return -1;
- break;
- case PAM_MODUTIL_NULL_FD:
- if (redirect_out_null(pamh, fd, name) < 0)
- return -1;
- break;
- case PAM_MODUTIL_IGNORE_FD:
- break;
- }
- return fd;
- }
- /* Closes all descriptors after stderr. */
- static void
- close_fds(void)
- {
- /*
- * An arbitrary upper limit for the maximum file descriptor number
- * returned by RLIMIT_NOFILE.
- */
- const int MAX_FD_NO = 65535;
- /* The lower limit is the same as for _POSIX_OPEN_MAX. */
- const int MIN_FD_NO = 20;
- int fd;
- struct rlimit rlim;
- if (getrlimit(RLIMIT_NOFILE, &rlim) || rlim.rlim_max > (rlim_t)MAX_FD_NO)
- fd = MAX_FD_NO;
- else if (rlim.rlim_max < (rlim_t)MIN_FD_NO)
- fd = MIN_FD_NO;
- else
- fd = (int)rlim.rlim_max - 1;
- for (; fd > STDERR_FILENO; --fd)
- close(fd);
- }
- int
- pam_modutil_sanitize_helper_fds(pam_handle_t *pamh,
- enum pam_modutil_redirect_fd stdin_mode,
- enum pam_modutil_redirect_fd stdout_mode,
- enum pam_modutil_redirect_fd stderr_mode)
- {
- if (stdin_mode != PAM_MODUTIL_IGNORE_FD &&
- redirect_in_pipe(pamh, STDIN_FILENO, "stdin") < 0) {
- return -1;
- }
- if (redirect_out(pamh, stdout_mode, STDOUT_FILENO, "stdout") < 0)
- return -1;
- /*
- * If stderr should not be ignored and
- * redirect mode for stdout and stderr are the same,
- * optimize by redirecting stderr to stdout.
- */
- if (stderr_mode != PAM_MODUTIL_IGNORE_FD &&
- stdout_mode == stderr_mode) {
- if (dup2(STDOUT_FILENO, STDERR_FILENO) != STDERR_FILENO) {
- pam_syslog(pamh, LOG_ERR,
- "dup2 of %s failed: %m", "stderr");
- return -1;
- }
- } else {
- if (redirect_out(pamh, stderr_mode, STDERR_FILENO, "stderr") < 0)
- return -1;
- }
- close_fds();
- return 0;
- }
|