#include <sys/types.h>
#include <sys/stat.h>
#include <sys/queue.h>
#include <err.h>
#include <errno.h>
#include <fcntl.h>
#include <signal.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#define BSIZE (64 * 1024)
struct list {
SLIST_ENTRY(list) next;
int fd;
char *name;
};
SLIST_HEAD(, list) head;
static void
add(int fd, char *name)
{
struct list *p;
if ((p = malloc(sizeof(*p))) == NULL)
err(1, NULL);
p->fd = fd;
p->name = name;
SLIST_INSERT_HEAD(&head, p, next);
}
int
main(int argc, char *argv[])
{
struct list *p;
int fd;
ssize_t n, rval, wval;
int append, ch, exitval;
char *buf;
if (pledge("stdio wpath cpath", NULL) == -1)
err(1, "pledge");
SLIST_INIT(&head);
append = 0;
while ((ch = getopt(argc, argv, "ai")) != -1) {
switch(ch) {
case 'a':
append = 1;
break;
case 'i':
(void)signal(SIGINT, SIG_IGN);
break;
default:
(void)fprintf(stderr, "usage: tee [-ai] [file ...]\n");
return 1;
}
}
argv += optind;
argc -= optind;
add(STDOUT_FILENO, "stdout");
exitval = 0;
while (*argv) {
if ((fd = open(*argv, O_WRONLY | O_CREAT |
(append ? O_APPEND : O_TRUNC), DEFFILEMODE)) == -1) {
warn("%s", *argv);
exitval = 1;
} else
add(fd, *argv);
argv++;
}
if (pledge("stdio", NULL) == -1)
err(1, "pledge");
buf = malloc(BSIZE);
if (buf == NULL)
err(1, NULL);
while ((rval = read(STDIN_FILENO, buf, BSIZE)) != 0 && rval != -1) {
SLIST_FOREACH(p, &head, next) {
for (n = 0; n < rval; n += wval) {
wval = write(p->fd, buf + n, rval - n);
if (wval == -1) {
warn("%s", p->name);
exitval = 1;
break;
}
}
}
}
free(buf);
if (rval == -1) {
warn("read");
exitval = 1;
}
SLIST_FOREACH(p, &head, next) {
if (close(p->fd) == -1) {
warn("%s", p->name);
exitval = 1;
}
}
return exitval;
}