#include <sys/types.h>
#include <sys/ioctl.h>
#include <err.h>
#include <errno.h>
#include <fcntl.h>
#include <limits.h>
#include <paths.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <ttyent.h>
#include <unistd.h>
int all(int);
int ttys(char **, int);
int ttyflags(struct ttyent *, int);
void usage(void);
int nflag, vflag;
int
main(int argc, char *argv[])
{
int aflag, ch, rval, pflag = 0;
aflag = nflag = vflag = 0;
while ((ch = getopt(argc, argv, "panv")) != -1)
switch (ch) {
case 'a':
aflag = 1;
break;
case 'n':
nflag = 1;
break;
case 'p':
pflag = 1;
break;
case 'v':
vflag = 1;
break;
default:
usage();
}
argc -= optind;
argv += optind;
if (aflag && argc != 0)
usage();
if (setttyent() == 0)
err(1, "setttyent");
if (aflag)
rval = all(pflag);
else
rval = ttys(argv, pflag);
if (endttyent() == 0)
warn("endttyent");
exit(rval);
}
int
all(int print)
{
struct ttyent *tep;
int rval;
rval = 0;
for (tep = getttyent(); tep != NULL; tep = getttyent()) {
if (tep->ty_type == NULL ||
strcmp(tep->ty_type, "network") == 0)
continue;
if (ttyflags(tep, print))
rval = 1;
}
return (rval);
}
int
ttys(char **ttylist, int print)
{
struct ttyent *tep;
int rval;
rval = 0;
for (; *ttylist != NULL; ttylist++) {
tep = getttynam(*ttylist);
if (tep == NULL) {
warnx("couldn't find an entry in %s for \"%s\"",
_PATH_TTYS, *ttylist);
rval = 1;
continue;
}
if (ttyflags(tep, print))
rval = 1;
}
return (rval);
}
int
ttyflags(struct ttyent *tep, int print)
{
int fd, flags = 0, rval = 0, st, sep = 0;
char path[PATH_MAX];
char strflags[256];
st = tep->ty_status;
strflags[0] = '\0';
(void)snprintf(path, sizeof path, "%s%s", _PATH_DEV, tep->ty_name);
if (print == 0) {
if (st & TTY_LOCAL) {
flags |= TIOCFLAG_CLOCAL;
(void)strlcat(strflags, "local", sizeof strflags);
sep++;
}
if (st & TTY_RTSCTS) {
flags |= TIOCFLAG_CRTSCTS;
if (sep++)
(void)strlcat(strflags, "|", sizeof strflags);
(void)strlcat(strflags, "rtscts", sizeof strflags);
}
if (st & TTY_SOFTCAR) {
flags |= TIOCFLAG_SOFTCAR;
if (sep++)
(void)strlcat(strflags, "|", sizeof strflags);
(void)strlcat(strflags, "softcar", sizeof strflags);
}
if (st & TTY_MDMBUF) {
flags |= TIOCFLAG_MDMBUF;
if (sep++)
(void)strlcat(strflags, "|", sizeof strflags);
(void)strlcat(strflags, "mdmbuf", sizeof strflags);
}
if (vflag)
printf("%s setting flags to: %s\n", path, strflags);
}
if (nflag)
return (0);
if ((fd = open(path, O_RDONLY | O_NONBLOCK)) == -1) {
if (!(errno == ENXIO ||
(errno == ENOENT && (st & TTY_ON) == 0)))
rval = 1;
if (vflag)
warn("open %s", path);
return (rval);
}
if (print == 0) {
if (ioctl(fd, TIOCSFLAGS, &flags) == -1)
if (errno != ENOTTY || vflag) {
warn("TIOCSFLAGS on %s", path);
rval = (errno != ENOTTY);
}
} else {
if (ioctl(fd, TIOCGFLAGS, &flags) == -1)
if (errno != ENOTTY || vflag) {
warn("TIOCGFLAGS on %s", path);
rval = (errno != ENOTTY);
}
if (flags & TIOCFLAG_CLOCAL) {
(void)strlcat(strflags, "local", sizeof strflags);
sep++;
}
if (flags & TIOCFLAG_CRTSCTS) {
if (sep++)
(void)strlcat(strflags, "|", sizeof strflags);
(void)strlcat(strflags, "rtscts", sizeof strflags);
}
if (flags & TIOCFLAG_SOFTCAR) {
if (sep++)
(void)strlcat(strflags, "|", sizeof strflags);
(void)strlcat(strflags, "softcar", sizeof strflags);
}
if (flags & TIOCFLAG_MDMBUF) {
if (sep++)
(void)strlcat(strflags, "|", sizeof strflags);
(void)strlcat(strflags, "mdmbuf", sizeof strflags);
}
printf("%s flags are: %s\n", path, strflags);
}
if (close(fd) == -1) {
warn("close %s", path);
return (1);
}
return (rval);
}
void
usage(void)
{
(void)fprintf(stderr, "usage: ttyflags [-pv] [-a | tty ...]\n");
exit(1);
}