/* Copyright (C) 2008 Emmanuel Varoquaux This file is part of XOS. This program is free software: you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation, either version 3 of the License, or (at your option) any later version. This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with this program. If not, see . */ #include "common.h" #include #include #include #include #include #include #include #include #include #include static size_t output_io_size; static void __attribute__ ((noreturn)) die_bad_usage() { fprintf(stderr, "Try `%s --help' for more information.\n", program_invocation_name); exit(EXIT_FAILURE); } static void print_help() { printf("Usage: %s [OPTION] [FILE]...\n", program_invocation_name); puts("Concatenate FILE(s), or standard input, to standard output."); putchar('\n'); puts(" -h, --help display this help and exit"); putchar('\n'); puts("With no FILE, or when FILE is -, read standard input."); } static int cat(int fd, const char *filename) { int err; struct stat statbuf; size_t input_io_size; size_t buf_size; char *buf; ssize_t n_read; ssize_t n_write; err = 0; if (fstat(fd, &statbuf) == -1) { error(0, errno, filename); err = 1; goto end; } if (S_ISDIR(statbuf.st_mode)) { error(0, EISDIR, filename); err = 1; goto end; } input_io_size = statbuf.st_blksize > 0 ? statbuf.st_blksize : 4096; buf_size = input_io_size > output_io_size ? input_io_size : output_io_size; if (!(buf = malloc(buf_size))) error(EXIT_FAILURE, errno, "malloc"); while (1) { if ((n_read = read(fd, buf, buf_size)) == -1) { error(0, errno, filename); err = 1; goto free_buf; } if (n_read == 0) break; if ((n_write = write(STDOUT_FILENO, buf, n_read)) == -1) error(EXIT_FAILURE, errno, "write error"); if (n_write != n_read) error(EXIT_FAILURE, ENOSPC, "write error"); } free_buf: free(buf); end: return err ? -1 : 0; } int main(int argc, char **argv) { static const struct option longopts[] = { {"help", no_argument, NULL, 'h'}, {NULL, 0, NULL, 0} }; int c; struct stat statbuf; int err; int fd; atexit(close_stdout); while ((c = getopt_long(argc, argv, "h", longopts, NULL)) != -1) switch (c) { case 'h': print_help(); return EXIT_SUCCESS; case '?': die_bad_usage(); } if (fstat(STDOUT_FILENO, &statbuf) == -1) error(EXIT_FAILURE, errno, "standard output"); output_io_size = statbuf.st_blksize > 0 ? statbuf.st_blksize : 4096; err = 0; if (optind == argc) { if (cat(STDIN_FILENO, "standard input") == -1) err = 1; } else do if (!strcmp(argv[optind], "-")) { if (cat(STDIN_FILENO, "standard input") == -1) err = 1; } else { if ((fd = open(argv[optind], O_RDONLY)) == -1) { error(0, errno, argv[optind]); err = 1; continue; } if (cat(fd, argv[optind]) == -1) err = 1; if (close(fd) == -1) { error(0, errno, argv[optind]); err = 1; continue; } } while (++optind < argc); return err ? EXIT_FAILURE : EXIT_SUCCESS; }