/* $OpenBSD: identd.c,v 1.46 2007/09/26 02:46:29 ray Exp $ */ /* * This program is in the public domain and may be used freely by anyone * who wants to. * * Please send bug fixes/bug reports to: Peter Eriksson */ #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include "identd.h" extern char *__progname; int verbose_flag; int debug_flag; int syslog_flag; int multi_flag; int unknown_flag; int number_flag; int noident_flag; int userident_flag; int token_flag; int no_user_token_flag; int lport; int fport; const char *opsys_name = "UNIX"; const char *charset_sep = ""; char *charset_name = ""; static pid_t child_pid; void usage(void); void sigchld(int); char * gethost(struct sockaddr_storage *ss); void usage(void) { syslog(LOG_ERR, "usage: %s [-dehlmNnoUv] [-b | -i | -w] [-a address] [-c charset] " "[-g gid] [-p port] [-t seconds] [-u uid]", __progname); exit(2); } /* * Return the name of the connecting host, or the IP number as a string. */ char * gethost4_addr(struct in_addr *addr) { struct hostent *hp; hp = gethostbyaddr(addr, sizeof(struct in_addr), AF_INET); if (hp) return hp->h_name; return inet_ntoa(*addr); } char * gethost(struct sockaddr_storage *ss) { if (ss->ss_family == AF_INET6) return (gethost6((struct sockaddr_in6 *)ss)); return (gethost4((struct sockaddr_in *)ss)); } char * gethost4(struct sockaddr_in *sin) { struct hostent *hp; hp = gethostbyaddr(&sin->sin_addr, sizeof(struct in_addr), AF_INET); if (hp) return hp->h_name; return inet_ntoa(sin->sin_addr); } /* * Return the name of the connecting host, or the IP number as a string. */ char * gethost6(struct sockaddr_in6 *addr) { static char hbuf[2][NI_MAXHOST]; const int niflags = NI_NUMERICHOST; static int bb = 0; int err; bb = (bb+1)%2; err = getnameinfo((struct sockaddr *)addr, addr->sin6_len, hbuf[bb], sizeof(hbuf[bb]), NULL, 0, niflags); if (err != 0) { syslog(LOG_ERR, "getnameinfo failed (%s)", gai_strerror(err)); strlcpy(hbuf[bb], "UNKNOWN", sizeof(hbuf[bb])); } return(hbuf[bb]); } volatile sig_atomic_t alarm_fired; /* * Exit cleanly after our time's up. */ /* ARGSUSED */ static void alarm_handler(int notused) { alarm_fired = 1; } /* * Main entry point into this daemon */ int main(int argc, char *argv[]) { struct sockaddr_storage sa, sa2; /* struct sockaddr_in sin;*/ struct sockaddr_in *sin; struct sockaddr_in6 *sin6; struct in_addr laddr, faddr; struct in6_addr laddr6, faddr6; struct passwd *pwd; struct group *grp; int background_flag = 0, timeout = 0, ch; char *portno = "auth"; char *bind_address = NULL; uid_t set_uid = 0; gid_t set_gid = 0; extern char *optarg; socklen_t len; const char *errstr; openlog(__progname, LOG_PID, LOG_DAEMON); /* runs as _identd if possible, fallback to "nobody" */ if (getuid() == 0) { if ((pwd = getpwnam(DEFAULT_UID)) == NULL) pwd = getpwnam("nobody"); if (pwd == NULL) error("no such user: neither %s nor nobody", DEFAULT_UID); set_uid = pwd->pw_uid; set_gid = pwd->pw_gid; } /* * Parse the command line arguments */ while ((ch = getopt(argc, argv, "hHbwit:p:a:u:g:c:loenvdmNU")) != -1) { switch (ch) { case 'h': token_flag = 1; break; case 'H': no_user_token_flag = token_flag = 1; break; case 'b': /* Start as standalone daemon */ background_flag = 1; break; case 'w': /* Start from Inetd, wait mode */ background_flag = 2; break; case 'i': /* Start from Inetd, nowait mode */ background_flag = 0; break; case 't': timeout = strtonum(optarg, 0, 100000000, &errstr); if (errstr) error("timeout is %s: %s", errstr, optarg); break; case 'p': portno = optarg; break; case 'a': bind_address = optarg; break; case 'u': pwd = getpwnam(optarg); if (pwd == NULL && isdigit(optarg[0])) { set_uid = atoi(optarg); if ((pwd = getpwuid(set_uid)) == NULL) break; } if (pwd == NULL) error("no such user (%s) for -u option", optarg); else { set_uid = pwd->pw_uid; if (set_gid == 0) set_gid = pwd->pw_gid; } break; case 'g': grp = getgrnam(optarg); if (grp == NULL && isdigit(optarg[0])) { set_gid = atoi(optarg); break; } grp = getgrnam(optarg); if (!grp) error("no such group (%s) for -g option", optarg); else set_gid = grp->gr_gid; break; case 'c': charset_name = optarg; charset_sep = " , "; break; case 'l': /* Use the Syslog daemon for logging */ syslog_flag++; break; case 'o': opsys_name = "OTHER"; break; case 'e': unknown_flag = 1; break; case 'n': number_flag = 1; break; case 'v': /* Be verbose */ verbose_flag++; break; case 'd': /* Enable debugging */ debug_flag++; break; case 'm': /* Enable multiline queries */ multi_flag++; break; case 'N': /* Enable users ".noident" files */ noident_flag++; break; case 'U': /* Enable user ".ident" files */ userident_flag++; break; default: usage(); } } /* * Do the special handling needed for the "-b" flag */ if (background_flag == 1) { struct sockaddr_in addr; struct servent *sp; int fd; if (daemon(0, 0) != 0) exit(0); fd = socket(AF_INET, SOCK_STREAM, 0); if (fd == -1) error("main: socket"); if (fd != 0) dup2(fd, 0); memset(&addr, 0, sizeof(addr)); addr.sin_len = sizeof(struct sockaddr_in); addr.sin_family = AF_INET; if (bind_address == NULL) addr.sin_addr.s_addr = htonl(INADDR_ANY); else { if (inet_aton(bind_address, &addr.sin_addr) == 0) { struct hostent *hp; hp = gethostbyname(bind_address); if (!hp) error("no such address (%s) for -a switch", bind_address); memcpy(&addr.sin_addr, hp->h_addr, sizeof(addr.sin_addr)); } } if (isdigit(portno[0])) addr.sin_port = htons(atoi(portno)); else { sp = getservbyname(portno, "tcp"); if (sp == NULL) error("main: getservbyname: %s", portno); addr.sin_port = sp->s_port; } if (bind(0, (struct sockaddr *) &addr, sizeof(addr)) < 0) error("main: bind"); if (listen(0, 3) < 0) error("main: listen"); } if (set_gid) { if (setegid(set_gid) == -1) error("main: setegid"); if (setgid(set_gid) == -1) error("main: setgid"); } if (set_uid) { if (seteuid(set_uid) == -1) error("main: seteuid"); if (setuid(set_uid) == -1) error("main: setuid"); } /* * Do some special handling if the "-b" or "-w" flags are used */ if (background_flag) { int nfds, fd; struct pollfd pfd[1]; signal(SIGCHLD, sigchld); /* * Loop and dispatch client handling processes */ do { /* * Terminate if we've been idle for 'timeout' seconds */ if (background_flag == 2 && timeout) { signal(SIGALRM, alarm_handler); alarm(timeout); } /* * Wait for a connection request to occur. * Ignore EINTR (Interrupted System Call). */ do { if (alarm_fired) { if (syslog_flag) syslog(LOG_DEBUG, "SIGALRM triggered, exiting"); exit(0); } pfd[0].fd = 0; pfd[0].events = POLLIN; if (timeout) nfds = poll(pfd, 1, timeout * 1000); else nfds = poll(pfd, 1, INFTIM); } while (nfds < 0 && errno == EINTR); /* * An error occurred in poll? Just die */ if (nfds < 0) error("main: poll"); /* * Timeout limit reached. Exit nicely */ if (nfds == 0) exit(0); /* * Disable the alarm timeout */ alarm(0); /* * Accept the new client */ fd = accept(0, NULL, NULL); if (fd == -1) error("main: accept. errno = %d", errno); /* * And fork, then close the fd if we are the parent. */ child_pid = fork(); } while (child_pid && (close(fd), 1)); /* * We are now in child, the parent has returned to "do" above. */ if (dup2(fd, 0) == -1) error("main: dup2: failed fd 0"); if (dup2(fd, 1) == -1) error("main: dup2: failed fd 1"); if (dup2(fd, 2) == -1) error("main: dup2: failed fd 2"); } /* * Get foreign internet address */ len = sizeof(sa); if (getpeername(0, (struct sockaddr *) &sa, &len) == -1) { /* * A user has tried to start us from the command line or * the network link died, in which case this message won't * reach to other end anyway, so lets give the poor user some * errors. */ perror("identd: getpeername()"); exit(1); } if (sa.ss_family == AF_INET6) { sin6 = (struct sockaddr_in6 *)&sa; faddr6 = sin6->sin6_addr; } else { sin = (struct sockaddr_in *)&sa; faddr = sin->sin_addr; } /* * Open the connection to the Syslog daemon if requested */ if (syslog_flag) syslog(LOG_INFO, "Connection from %s", gethost(&sa)); /* * Get local internet address */ len = sizeof(sa2); if (getsockname(0, (struct sockaddr *) &sa2, &len) == -1) { /* * We can just die here, because if this fails then the * network has died and we haven't got anyone to return * errors to. */ exit(1); } /* are we V4 or V6? */ if (sa2.ss_family == AF_INET6) { sin6 = (struct sockaddr_in6 *)&sa2; laddr6 = sin6->sin6_addr; /* * Get the local/foreign port pair from the luser */ parse6(STDIN_FILENO, (struct sockaddr_in6 *)&sa2, (struct sockaddr_in6 *)&sa); } else { sin = (struct sockaddr_in *)&sa2; laddr = sin->sin_addr; /* * Get the local/foreign port pair from the luser */ parse(STDIN_FILENO, &laddr, &faddr); } exit(0); } void error(char *fmt, ...) { va_list ap, ap2; va_start(ap, fmt); if (syslog_flag) { va_copy(ap2, ap); vsyslog(LOG_ERR, fmt, ap2); va_end(ap2); } if (debug_flag) { fprintf(stderr, "%d , %d : ERROR : X-DBG : ", lport, fport); vfprintf(stderr, fmt, ap); perror(": "); } else printf("%d , %d : ERROR : UNKNOWN-ERROR\r\n", lport, fport); va_end(ap); exit(1); } void sigchld(int signo) { pid_t pid; do { pid = waitpid(-1, NULL, WNOHANG); } while (pid > 0 || (pid == -1 && errno == EINTR)); }