logo

utils-std

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

sha256sum.c (5794B)


  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 "../lib/sha256.h" // sha256_*
  7. #include "../lib/strconv.h" // bytes2hex
  8. #include <ctype.h> // isxdigit
  9. #include <errno.h>
  10. #include <fcntl.h> // open, O_*
  11. #include <stdbool.h>
  12. #include <stdio.h> // fprintf
  13. #include <stdlib.h> // free
  14. #include <string.h> // strerror
  15. #include <unistd.h> // read, write, close, getopt
  16. #define SHA256SUM_LEN SHA256_DIGEST_LENGTH * 2 + 1
  17. const char *argv0 = "sha256sum";
  18. static int
  19. sha256sum(int fd, const char *fdname, char sum[SHA256SUM_LEN])
  20. {
  21. struct sha256 ctx;
  22. sha256_init(&ctx);
  23. uint8_t buf[BUFSIZ];
  24. ssize_t nread = -1;
  25. while((nread = read(fd, buf, BUFSIZ)) > 0)
  26. {
  27. sha256_update(&ctx, buf, nread);
  28. }
  29. if(nread < 0)
  30. {
  31. fprintf(stderr,
  32. "%s: error: Failed reading file '%s': %s\n",
  33. argv0,
  34. fdname ? fdname : "<stdin>",
  35. strerror(errno));
  36. return -1;
  37. }
  38. uint8_t res[SHA256_DIGEST_LENGTH] = "";
  39. sha256_sum(&ctx, res);
  40. bytes2hex(res, SHA256_DIGEST_LENGTH, sum, SHA256SUM_LEN);
  41. return 0;
  42. }
  43. #define STR(s) #s
  44. #define XSTR(s) STR(s)
  45. static int
  46. check(FILE *file, const char *filename)
  47. {
  48. int err = 0;
  49. ssize_t nread = -1;
  50. char *line = NULL;
  51. size_t len = 0;
  52. errno = 0;
  53. while((nread = getline(&line, &len, file)) > 0)
  54. {
  55. if(line[nread - 1] == '\n') line[nread - 1] = '\0';
  56. ssize_t i = 0;
  57. for(; i < nread; i++)
  58. {
  59. if(isxdigit(line[i])) continue;
  60. if(line[i] == ' ')
  61. {
  62. line[i] = '\0';
  63. break;
  64. }
  65. fprintf(stderr,
  66. "%s: error: Invalid character '%c' while reading hash in line: %s\n",
  67. argv0,
  68. line[i],
  69. line);
  70. if(len > 0) free(line);
  71. return -1;
  72. }
  73. if(line[i++] != '\0')
  74. {
  75. fprintf(stderr, "%s: error: Invalid line: %s\n", argv0, line);
  76. if(len > 0) free(line);
  77. return -1;
  78. }
  79. if(i != SHA256SUM_LEN)
  80. {
  81. fprintf(stderr,
  82. "%s: error: Got %zd hexadecimal digits while expected %d for a SHA256\n",
  83. argv0,
  84. i,
  85. SHA256SUM_LEN);
  86. if(len > 0) free(line);
  87. return -1;
  88. }
  89. while(i < nread && line[i] == ' ')
  90. i++;
  91. if(i < nread && line[i] == '*') i++;
  92. char *target = line + i;
  93. int fd = open(target, O_RDONLY | O_NOCTTY);
  94. if(fd < 0)
  95. {
  96. fprintf(stderr, "%s: error: Failed opening file '%s': %s\n", argv0, target, strerror(errno));
  97. if(len > 0) free(line);
  98. return -1;
  99. }
  100. int ret = posix_fadvise(fd, 0, 0, POSIX_FADV_SEQUENTIAL);
  101. if(ret != 0)
  102. fprintf(stderr,
  103. "%s: warning: posix_fadvise failed on file '%s': %s\n",
  104. argv0,
  105. target,
  106. strerror(ret));
  107. char got[SHA256SUM_LEN] = "";
  108. if(sha256sum(fd, target, got) < 0) err = 1;
  109. if(memcmp(line, got, SHA256SUM_LEN) == 0)
  110. {
  111. printf("%s: OK\n", target);
  112. }
  113. else
  114. {
  115. err = 1;
  116. printf("%s: FAILED\n", target);
  117. }
  118. if(close(fd) < 0)
  119. {
  120. fprintf(
  121. stderr, "%s: error: Failed closing file '%s': %s\n", argv0, filename, strerror(errno));
  122. if(len > 0) free(line);
  123. return -1;
  124. }
  125. }
  126. if(nread < 0 && errno != 0)
  127. {
  128. err = 1;
  129. fprintf(stderr,
  130. "%s: error: Failed reading line from file '%s': %s\n",
  131. argv0,
  132. filename,
  133. strerror(errno));
  134. }
  135. if(len > 0) free(line);
  136. return err;
  137. }
  138. int
  139. main(int argc, char *argv[])
  140. {
  141. bool opt_c = false;
  142. for(int c = -1; (c = getopt_nolong(argc, argv, ":c")) != -1;)
  143. {
  144. switch(c)
  145. {
  146. case 'c':
  147. opt_c = true;
  148. break;
  149. case ':':
  150. fprintf(stderr, "%s: error: Missing operand for option: '-%c'\n", argv0, optopt);
  151. return 1;
  152. case '?':
  153. if(!got_long_opt) fprintf(stderr, "%s: error: Unrecognised option: '-%c'\n", argv0, optopt);
  154. return 1;
  155. default:
  156. abort();
  157. }
  158. }
  159. argc -= optind;
  160. argv += optind;
  161. if(opt_c)
  162. {
  163. if(argc == 0)
  164. {
  165. if(check(stdin, "<stdin>") != 0) return 1;
  166. return 0;
  167. }
  168. int err = 0;
  169. for(int i = 0; i < argc; i++)
  170. {
  171. FILE *file = NULL;
  172. const char *filename = argv[i];
  173. if(filename[0] == '-' && filename[1] == '\0')
  174. {
  175. filename = "<stdin>";
  176. file = stdin;
  177. }
  178. else
  179. {
  180. file = fopen(filename, "rb");
  181. if(file == NULL)
  182. {
  183. fprintf(stderr,
  184. "%s: error: Failed opening file '%s': %s\n",
  185. argv0,
  186. filename,
  187. strerror(errno));
  188. return 1;
  189. }
  190. }
  191. if(check(file, filename) != 0) err = 1;
  192. if(fclose(file) < 0)
  193. {
  194. fprintf(
  195. stderr, "%s: error: Failed closing file '%s': %s\n", argv0, filename, strerror(errno));
  196. return 1;
  197. }
  198. }
  199. return err;
  200. }
  201. if(argc == 0)
  202. {
  203. char sum[SHA256SUM_LEN] = "";
  204. if(sha256sum(STDIN_FILENO, NULL, sum) < 0) return 1;
  205. puts(sum);
  206. return 0;
  207. }
  208. for(int i = 0; i < argc; i++)
  209. {
  210. int fd = -1;
  211. const char *filename = argv[i];
  212. if(filename[0] == '-' && filename[1] == '\0')
  213. {
  214. filename = "<stdin>";
  215. fd = STDIN_FILENO;
  216. }
  217. else
  218. {
  219. fd = open(filename, O_RDONLY | O_NOCTTY);
  220. if(fd < 0)
  221. {
  222. fprintf(
  223. stderr, "%s: error: Failed opening file '%s': %s\n", argv0, filename, strerror(errno));
  224. return 1;
  225. }
  226. int ret = posix_fadvise(fd, 0, 0, POSIX_FADV_SEQUENTIAL);
  227. if(ret != 0)
  228. fprintf(stderr,
  229. "%s: warning: posix_fadvise failed on file '%s': %s\n",
  230. argv0,
  231. filename,
  232. strerror(ret));
  233. }
  234. int err = 0;
  235. char sum[SHA256SUM_LEN] = "";
  236. if(sha256sum(fd, filename, sum) < 0) err = 1;
  237. printf("%s %s\n", sum, filename);
  238. if(close(fd) < 0)
  239. {
  240. fprintf(
  241. stderr, "%s: error: Failed closing file '%s': %s\n", argv0, filename, strerror(errno));
  242. return 1;
  243. }
  244. if(err == 1) return 1;
  245. }
  246. return 0;
  247. }