| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172 | #include <errno.h>#include <unistd.h>#include <string.h>#include <sys/stat.h>#include <dirent.h>static char * __check_dir_for_tty_match(char * dirname, struct stat *st){    DIR *fp;    struct stat dst;    struct dirent *d;    static char name[NAME_MAX];    fp = opendir(dirname);    if (fp == 0)	return 0;    strcpy(name, dirname);    strcat(name, "/");    while ((d = readdir(fp)) != 0) {	strcpy(name + strlen(dirname) + 1, d->d_name);	if (stat(name, &dst) == 0		&& st->st_dev == dst.st_dev && st->st_ino == dst.st_ino) {	    closedir(fp);	    return name;	}    }    closedir(fp);    return NULL;}/* This is a failly slow approach.  We do a linear search through * some directories looking for a match.  Yes this is lame.  But  * it should work, should be small, and will return names that match * what is on disk.   * * Another approach we could use would be to use the info in /proc/self/fd */char *ttyname(fd)int fd;{    char *the_name = NULL;    struct stat st;    int noerr = errno;    if (fstat(fd, &st) < 0)	return 0;    if (!isatty(fd)) {	noerr = ENOTTY;	goto cool_found_it;    }    /* Lets try /dev/vc first (be devfs compatible) */    if ( (the_name=__check_dir_for_tty_match("/dev/vc", &st))) 	goto cool_found_it;    /* Lets try /dev/tts next (be devfs compatible) */    if ( (the_name=__check_dir_for_tty_match("/dev/tts", &st))) 	goto cool_found_it;    /* Lets try /dev/pts next */    if ( (the_name=__check_dir_for_tty_match("/dev/pts", &st))) 	goto cool_found_it;    /* Lets try walking through /dev last */    if ( (the_name=__check_dir_for_tty_match("/dev", &st))) 	goto cool_found_it;cool_found_it:    __set_errno(noerr);    return the_name;}
 |