logo

utils-std

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

strings.c (4163B)


  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 <ctype.h> /* isprint() */
  7. #include <errno.h> /* errno */
  8. #include <fcntl.h> /* open(), O_RDONLY */
  9. #include <limits.h> /* LONG_MAX */
  10. #include <stdio.h> /* fprintf(), BUFSIZ */
  11. #include <stdlib.h> /* strtol() */
  12. #include <string.h> /* strerror(), strncmp() */
  13. #include <unistd.h> /* read(), write(), close() */
  14. const char *argv0 = "strings";
  15. size_t opt_min_strlen = 4;
  16. const char *opt_offset_format = NULL;
  17. int delim = '\n';
  18. static int
  19. print_string(char *buffer, size_t offset)
  20. {
  21. int ret = 0;
  22. if(opt_offset_format == NULL)
  23. {
  24. ret = printf("%s%c", buffer, delim);
  25. }
  26. else
  27. {
  28. ret = printf(opt_offset_format, offset, buffer, delim);
  29. }
  30. if(ret < 0)
  31. {
  32. fprintf(stderr, "strings: error: Failed writing: %s\n", strerror(errno));
  33. errno = 0;
  34. return 1;
  35. }
  36. else
  37. {
  38. return 0;
  39. }
  40. }
  41. static int
  42. concat(int fd, const char *fdname)
  43. {
  44. ssize_t c;
  45. static char read_buf[4096];
  46. static char write_buf[4096];
  47. size_t write_pos = 0;
  48. size_t offset = 0;
  49. while((c = read(fd, read_buf, sizeof(read_buf))) > 0)
  50. {
  51. int read_pos = 0;
  52. char b = 0;
  53. for(; read_pos < c; read_pos++)
  54. {
  55. b = read_buf[read_pos];
  56. if(isprint(b) && write_pos < 4096)
  57. {
  58. write_buf[write_pos++] = b;
  59. }
  60. else
  61. {
  62. if(write_pos >= opt_min_strlen)
  63. {
  64. write_buf[write_pos + 1] = 0;
  65. if(print_string(write_buf, offset) != 0)
  66. {
  67. return 1;
  68. }
  69. }
  70. offset += write_pos;
  71. offset++;
  72. write_pos = 0;
  73. }
  74. }
  75. }
  76. if(c < 0)
  77. {
  78. fprintf(stderr, "strings: error: Failed reading ‘%s’: %s\n", fdname, strerror(errno));
  79. errno = 0;
  80. return 1;
  81. }
  82. return 0;
  83. }
  84. static void
  85. usage(void)
  86. {
  87. fprintf(stderr, "strings: [-az] [-t format] [-n number] [file...]\n");
  88. }
  89. int
  90. main(int argc, char *argv[])
  91. {
  92. for(int c = -1; (c = getopt_nolong(argc, argv, ":an:t:z")) != -1;)
  93. {
  94. char *endptr = NULL;
  95. switch(c)
  96. {
  97. case 'a':
  98. /* Structure is always ignored */
  99. break;
  100. case 'n':
  101. opt_min_strlen = strtol(optarg, &endptr, 10);
  102. if(endptr && *endptr != 0)
  103. {
  104. // extraneous characters is invalid
  105. errno = EINVAL;
  106. }
  107. if(errno != 0)
  108. {
  109. fprintf(stderr, "strings: error: Option `-n %s`: %s\n", optarg, strerror(errno));
  110. usage();
  111. return 1;
  112. }
  113. if(opt_min_strlen < 1)
  114. {
  115. fprintf(stderr, "strings: error: Option `-n %s` is too small\n", optarg);
  116. usage();
  117. return 1;
  118. }
  119. if(opt_min_strlen == LONG_MAX || opt_min_strlen > 4096)
  120. {
  121. fprintf(stderr, "strings: error: Option `-n %s` is too large\n", optarg);
  122. usage();
  123. return 1;
  124. }
  125. break;
  126. case 't':
  127. if(strnlen(optarg, 2) > 1)
  128. {
  129. usage();
  130. return 1;
  131. }
  132. switch(optarg[0])
  133. {
  134. case 'o':
  135. opt_offset_format = "%zo %s%c";
  136. break;
  137. case 'x':
  138. opt_offset_format = "%zx %s%c";
  139. break;
  140. case 'd':
  141. opt_offset_format = "%zd %s%c";
  142. break;
  143. default:
  144. fprintf(stderr, "strings: error: Unknown format: %s\n", optarg);
  145. usage();
  146. return 1;
  147. }
  148. break;
  149. case 'z':
  150. delim = '\0';
  151. break;
  152. case ':':
  153. fprintf(stderr, "strings: error: Missing operand for option: '-%c'\n", optopt);
  154. usage();
  155. return 1;
  156. case '?':
  157. if(!got_long_opt) fprintf(stderr, "strings: error: Unrecognised option: '-%c'\n", optopt);
  158. usage();
  159. return 1;
  160. default:
  161. abort();
  162. }
  163. }
  164. argc -= optind;
  165. argv += optind;
  166. if(argc < 1)
  167. {
  168. return concat(0, "<stdin>");
  169. }
  170. for(int argi = 0; argi < argc; argi++)
  171. {
  172. if(strncmp(argv[argi], "-", 2) == 0)
  173. {
  174. if(concat(0, "<stdin>") != 0)
  175. {
  176. return 1;
  177. }
  178. }
  179. else
  180. {
  181. int fd = open(argv[argi], O_RDONLY);
  182. if(fd <= 0)
  183. {
  184. fprintf(stderr, "strings: error: Failed opening ‘%s’: %s\n", argv[argi], strerror(errno));
  185. return 1;
  186. }
  187. if(concat(fd, argv[argi]) != 0)
  188. {
  189. return 1;
  190. }
  191. if(close(fd) < 0)
  192. {
  193. fprintf(stderr, "strings: error: Failed closing ‘%s’: %s\n", argv[argi], strerror(errno));
  194. return 1;
  195. }
  196. }
  197. }
  198. return 0;
  199. }