logo

utils-std

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

cmp.c (3790B)


  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>
  7. #include <errno.h>
  8. #include <stdbool.h>
  9. #include <stdio.h> // fopen, fprintf, getline
  10. #include <stdlib.h> // abort, strtoul
  11. #include <string.h> // strerror
  12. #include <unistd.h> // getopt
  13. static bool opt_s = false, opt_l = false;
  14. static unsigned long max_bytes = 0;
  15. const char *argv0 = "cmp";
  16. #undef MIN
  17. #define MIN(a, b) (((a) < (b)) ? (a) : (b))
  18. static int
  19. do_cmp(FILE *file1, const char *name1, FILE *file2, const char *name2)
  20. {
  21. char *line1 = NULL, *line2 = NULL;
  22. size_t len1 = 0, len2 = 0;
  23. unsigned long pos = 1, ln = 1;
  24. while(true)
  25. {
  26. ssize_t nread1 = getline(&line1, &len1, file1);
  27. if(nread1 < 0)
  28. {
  29. if(!ferror(file1)) return 0;
  30. fprintf(stderr,
  31. "%s: error: Failed to read line %ld from file '%s': %s\n",
  32. argv0,
  33. ln,
  34. name1,
  35. strerror(errno));
  36. return 1;
  37. }
  38. ssize_t nread2 = getline(&line2, &len2, file2);
  39. if(nread2 < 0)
  40. {
  41. if(!ferror(file2))
  42. {
  43. if(!opt_s) fprintf(stderr, "%s: error: EOF on %s line %ld\n", argv0, name2, ln);
  44. return 1;
  45. }
  46. fprintf(stderr,
  47. "%s: error: Failed to read line %ld from file '%s': %s\n",
  48. argv0,
  49. ln,
  50. name1,
  51. strerror(errno));
  52. return 2;
  53. }
  54. for(ssize_t i = 0; i < MIN(nread1, nread2); i++)
  55. {
  56. if(max_bytes != 0 && pos + i >= max_bytes) return 0;
  57. if(line1[i] != line2[i])
  58. {
  59. if(opt_s) return 1;
  60. if(opt_l)
  61. printf("%ld %o %o\n", pos + i, line1[i], line2[i]);
  62. else
  63. printf("%s %s differ: char %zd, line %ld\n", name1, name2, i + 1, ln);
  64. return 1;
  65. }
  66. }
  67. assert(nread1 == nread2);
  68. pos += nread1;
  69. ln++;
  70. }
  71. return 0;
  72. }
  73. static void
  74. usage(void)
  75. {
  76. fprintf(stderr, "Usage: cmp [-l|-s] [-n max_bytes] file1 file2\n");
  77. }
  78. int
  79. main(int argc, char *argv[])
  80. {
  81. char *endptr = NULL;
  82. for(int c = -1; (c = getopt_nolong(argc, argv, ":ln:s")) != -1;)
  83. {
  84. switch(c)
  85. {
  86. case 'l':
  87. opt_l = true;
  88. break;
  89. case 's':
  90. opt_s = true;
  91. break;
  92. case 'n':
  93. errno = 0;
  94. max_bytes = strtoul(optarg, &endptr, 0);
  95. if(errno != 0)
  96. {
  97. fprintf(stderr, "%s: error: Failed parsing '-n %s': %s\n", argv0, optarg, strerror(errno));
  98. return 1;
  99. }
  100. if(endptr != NULL && endptr[0] != 0)
  101. {
  102. fprintf(stderr,
  103. "%s: error: Non-numeric characters passed to '-n %s': %s\n",
  104. argv0,
  105. optarg,
  106. endptr);
  107. return 1;
  108. }
  109. break;
  110. case ':':
  111. fprintf(stderr, "%s: error: Missing operand for option: '-%c'\n", argv0, optopt);
  112. usage();
  113. return 1;
  114. case '?':
  115. if(!got_long_opt) fprintf(stderr, "%s: error: Unrecognised option: '-%c'\n", argv0, optopt);
  116. usage();
  117. return 1;
  118. default:
  119. abort();
  120. }
  121. }
  122. argc -= optind;
  123. argv += optind;
  124. if(argc != 2)
  125. {
  126. fprintf(stderr, "%s: error: Expected 2 arguments, got %d arguments\n", argv0, argc);
  127. return 1;
  128. }
  129. if(strcmp(argv[0], argv[1]) == 0) return 0;
  130. FILE *file1 = NULL;
  131. if(argv[0][0] == '-' && argv[0][1] == 0)
  132. file1 = stdin;
  133. else
  134. {
  135. file1 = fopen(argv[0], "r");
  136. if(file1 == NULL)
  137. {
  138. fprintf(stderr, "%s: error: Failed opening file '%s': %s\n", argv0, argv[0], strerror(errno));
  139. return 1;
  140. }
  141. }
  142. FILE *file2 = NULL;
  143. if(argv[1][0] == '-' && argv[1][1] == 0)
  144. file2 = stdin;
  145. else
  146. {
  147. file2 = fopen(argv[1], "r");
  148. if(file2 == NULL)
  149. {
  150. fprintf(stderr, "%s: error: Failed opening file '%s': %s\n", argv0, argv[1], strerror(errno));
  151. return 1;
  152. }
  153. }
  154. int ret = do_cmp(file1, argv[0], file2, argv[1]);
  155. fclose(file1);
  156. fclose(file2);
  157. return ret;
  158. }