logo

utils-std

Collection of commonly available Unix tools git clone https://anongit.hacktivis.me/git/utils-std.git/

tee.c (2459B)


  1. // utils-std: Collection of commonly available Unix tools
  2. // SPDX-FileCopyrightText: 2017 Haelwenn (lanodan) Monnier <contact+utils@hacktivis.me>
  3. // SPDX-License-Identifier: MPL-2.0
  4. #define _POSIX_C_SOURCE 200809L
  5. #include "../lib/getopt_nolong.h"
  6. #include <assert.h> /* assert() */
  7. #include <errno.h> /* errno */
  8. #include <signal.h> /* signal() */
  9. #include <stdio.h> /* fprintf(), fgetc(), fputc(), fclose(), fopen() */
  10. #include <stdlib.h> /* calloc(), free(), abort() */
  11. #include <string.h> /* strerror() */
  12. #include <unistd.h> /* getopt(), opt… */
  13. const char *argv0 = "tee";
  14. static void
  15. cleanup(FILE **fds)
  16. {
  17. if(fds != NULL)
  18. {
  19. free(fds);
  20. }
  21. }
  22. int
  23. main(int argc, char *argv[])
  24. {
  25. const char *mode = "w";
  26. FILE **fds = {NULL}; // Shut up GCC
  27. int c;
  28. for(int c = -1; (c = getopt_nolong(argc, argv, ":ai")) != -1;)
  29. {
  30. switch(c)
  31. {
  32. case 'a':
  33. mode = "a";
  34. break;
  35. case 'i': /* ignore SIGINT */;
  36. signal(SIGINT, SIG_IGN);
  37. break;
  38. case ':':
  39. fprintf(stderr, "tee: error: Missing operand for option: '-%c'\n", optopt);
  40. return 1;
  41. case '?':
  42. if(!got_long_opt) fprintf(stderr, "tee: error: Unrecognised option: '-%c'\n", optopt);
  43. return 1;
  44. default:
  45. abort();
  46. }
  47. }
  48. argc -= optind;
  49. argv += optind;
  50. if(argc > 0)
  51. {
  52. fds = calloc(argc, sizeof(*fds));
  53. if(!fds)
  54. {
  55. fprintf(stderr, "tee: error: Failed to allocate fd array: %s\n", strerror(errno));
  56. return 1;
  57. }
  58. }
  59. for(int argi = 0; argi < argc; argi++)
  60. {
  61. assert(argv[argi]);
  62. // POSIX: implementations shouldn't treat '-' as stdin
  63. fds[argi] = fopen(argv[argi], mode);
  64. if(fds[argi] == NULL)
  65. {
  66. fprintf(stderr, "tee: error: Failed opening file ‘%s’: %s\n", argv[argi], strerror(errno));
  67. cleanup(fds);
  68. return 1;
  69. }
  70. }
  71. // main loop, note that failed writes shouldn't make tee exit
  72. int err = 0;
  73. while((c = fgetc(stdin)) != EOF)
  74. {
  75. if(fputc(c, stdout) == EOF)
  76. {
  77. fprintf(stderr, "tee: error: Failed writing to stdout: %s\n", strerror(errno));
  78. err = 1;
  79. errno = 0;
  80. }
  81. for(int argi = 0; argi < argc; argi++)
  82. {
  83. if(fputc(c, fds[argi]) == EOF)
  84. {
  85. fprintf(stderr, "tee: error: Failed writing to argument %d: %s\n", argi, strerror(errno));
  86. err = 1;
  87. errno = 0;
  88. }
  89. }
  90. }
  91. // cleanup
  92. for(int argi = 0; argi < argc; argi++)
  93. {
  94. if(fclose(fds[argi]) != 0)
  95. {
  96. fprintf(stderr, "tee: error: Failed closing file '%s': %s\n", argv[argi], strerror(errno));
  97. err++;
  98. }
  99. }
  100. cleanup(fds);
  101. return err;
  102. }