#include <config.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <stdio.h>
#ifdef HAVE_STDBOOL_H
# include <stdbool.h>
#else
# include <compat/stdbool.h>
#endif
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#include <limits.h>
#include <errno.h>
#include <sudo_compat.h>
#include <sudo_fatal.h>
#include <sudo_util.h>
#include <sudo_debug.h>
sudo_dso_public int main(int argc, char *argv[]);
int sudo_debug_instance = SUDO_DEBUG_INSTANCE_INITIALIZER;
extern dev_t get_process_ttyname(char *name, size_t namelen);
static int
match_ttys(const char *tty1, const char *tty2)
{
struct stat sb1, sb2;
if (tty1 != NULL && tty2 != NULL) {
if (strcmp(tty1, tty2) == 0)
return 0;
if (stat(tty1, &sb1) == 0 && S_ISCHR(sb1.st_mode) &&
stat(tty2, &sb2) == 0 && S_ISCHR(sb2.st_mode)) {
if (sb1.st_rdev == sb2.st_rdev)
return 0;
}
} else if (tty1 == NULL && tty2 == NULL) {
return 0;
}
return 1;
}
int
main(int argc, char *argv[])
{
char *tty_libc = NULL, *tty_sudo = NULL;
char pathbuf[PATH_MAX];
bool verbose = false;
dev_t ttydev = NODEV;
int ch, errors = 0, ntests = 1;
initprogname(argc > 0 ? argv[0] : "check_ttyname");
while ((ch = getopt(argc, argv, "v")) != -1) {
switch (ch) {
case 'v':
verbose = true;
break;
default:
fprintf(stderr, "usage: %s [-v]\n", getprogname());
return EXIT_FAILURE;
}
}
ttydev = get_process_ttyname(pathbuf, sizeof(pathbuf));
if (ttydev != NODEV) {
char numbuf[STRLEN_MAX_SIGNED(long long) + 1];
const char *errstr;
dev_t newdev;
tty_sudo = pathbuf;
ntests++;
(void)snprintf(numbuf, sizeof(numbuf), "%lld", (long long)ttydev);
newdev = sudo_strtonum(numbuf, LLONG_MIN, LLONG_MAX, &errstr);
if (errstr != NULL) {
printf("%s: FAIL unable to parse device number %s: %s",
getprogname(), numbuf, errstr);
errors++;
} else if (ttydev != newdev) {
printf("%s: FAIL device mismatch for %s, %s != %lld",
getprogname(), pathbuf, numbuf, (long long)ttydev);
errors++;
}
}
#if defined(HAVE_KINFO_PROC2_NETBSD) || \
defined(HAVE_KINFO_PROC_OPENBSD) || \
defined(HAVE_KINFO_PROC_FREEBSD) || \
defined(HAVE_KINFO_PROC_DFLY) || \
defined(HAVE_KINFO_PROC_44BSD) || \
defined(HAVE__TTYNAME_DEV) || defined(HAVE_STRUCT_PSINFO_PR_TTYDEV) || \
defined(HAVE_PSTAT_GETPROC) || defined(__linux__)
tty_libc = ttyname(STDIN_FILENO);
#endif
if (match_ttys(tty_libc, tty_sudo) == 0) {
if (verbose)
printf("%s: OK (%s)\n", getprogname(), tty_sudo ? tty_sudo : "none");
} else if (tty_libc == NULL) {
if (verbose)
printf("%s: SKIP (%s)\n", getprogname(), tty_sudo ? tty_sudo : "none");
ntests = 0;
} else {
printf("%s: FAIL %s (sudo) vs. %s (libc)\n", getprogname(),
tty_sudo ? tty_sudo : "none", tty_libc ? tty_libc : "none");
errors++;
}
if (ntests != 0) {
printf("%s: %d tests run, %d errors, %d%% success rate\n",
getprogname(), ntests, errors, (ntests - errors) * 100 / ntests);
}
return errors;
}