2 * SPDX-License-Identifier: BSD-2-Clause-FreeBSD
4 * Copyright (c) 1997 Robert Nordier
7 * Redistribution and use in source and binary forms, with or without
8 * modification, are permitted provided that the following conditions
10 * 1. Redistributions of source code must retain the above copyright
11 * notice, this list of conditions and the following disclaimer.
12 * 2. Redistributions in binary form must reproduce the above copyright
13 * notice, this list of conditions and the following disclaimer in
14 * the documentation and/or other materials provided with the
17 * THIS SOFTWARE IS PROVIDED BY THE AUTHOR(S) ``AS IS'' AND ANY EXPRESS
18 * OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
19 * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
20 * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR(S) BE LIABLE FOR ANY
21 * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
22 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE
23 * GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
24 * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER
25 * IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR
26 * OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN
27 * IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
31 static const char rcsid[] =
35 #include <sys/types.h>
49 extern int crc(int fd, uint32_t *cval, off_t *clen);
51 #define DISTMD5 1 /* MD5 format */
52 #define DISTINF 2 /* .inf format */
53 #define DISTTYPES 2 /* types supported */
55 #define E_UNKNOWN 1 /* Unknown format */
56 #define E_BADMD5 2 /* Invalid MD5 format */
57 #define E_BADINF 3 /* Invalid .inf format */
58 #define E_NAME 4 /* Can't derive component name */
59 #define E_LENGTH 5 /* Length mismatch */
60 #define E_CHKSUM 6 /* Checksum mismatch */
61 #define E_ERRNO 7 /* sys_errlist[errno] */
63 #define isfatal(err) ((err) && (err) <= E_NAME)
65 #define NAMESIZE 256 /* filename buffer size */
66 #define MDSUMLEN 32 /* length of MD5 message digest */
68 #define isstdin(path) ((path)[0] == '-' && !(path)[1])
70 static const char *opt_dir; /* where to look for components */
71 static const char *opt_name; /* name for accessing components */
72 static int opt_all; /* report on all components */
73 static int opt_ignore; /* ignore missing components */
74 static int opt_recurse; /* search directories recursively */
75 static int opt_silent; /* silent about inaccessible files */
76 static int opt_type; /* dist type: md5 or inf */
77 static int opt_exist; /* just verify existence */
79 static int ckdist(const char *path, int type);
80 static int chkmd5(FILE * fp, const char *path);
81 static int chkinf(FILE * fp, const char *path);
82 static int report(const char *path, const char *name, int error);
83 static const char *distname(const char *path, const char *name,
85 static const char *stripath(const char *path);
86 static int distfile(const char *path);
87 static int disttype(const char *name);
88 static int fail(const char *path, const char *msg);
89 static void usage(void);
92 main(int argc, char *argv[])
100 while ((c = getopt(argc, argv, "ad:in:rst:x")) != -1)
121 if ((opt_type = disttype(optarg)) == 0) {
122 warnx("illegal argument to -t option");
137 if (stat(opt_dir, &sb))
138 err(2, "%s", opt_dir);
139 if (!S_ISDIR(sb.st_mode))
140 errx(2, "%s: not a directory", opt_dir);
145 rval |= ckdist(*argv, opt_type);
146 else if (stat(*argv, &sb))
147 rval |= fail(*argv, NULL);
148 else if (S_ISREG(sb.st_mode))
149 rval |= ckdist(*argv, opt_type);
152 if ((ftsp = fts_open(arg, FTS_LOGICAL, NULL)) == NULL)
154 while ((f = fts_read(ftsp)) != NULL)
155 switch (f->fts_info) {
157 rval = fail(f->fts_path, "Directory causes a cycle");
162 rval = fail(f->fts_path, sys_errlist[f->fts_errno]);
165 if (!opt_recurse && f->fts_level > FTS_ROOTLEVEL &&
166 fts_set(ftsp, f, FTS_SKIP))
170 if ((type = distfile(f->fts_name)) != 0 &&
171 (!opt_type || type == opt_type))
172 rval |= ckdist(f->fts_path, type);
186 ckdist(const char *path, int type)
194 } else if ((fp = fopen(path, "r")) == NULL)
195 return fail(path, NULL);
198 type = distfile(path);
200 if ((c = fgetc(fp)) != EOF) {
201 type = c == 'M' ? DISTMD5 : c == 'P' ? DISTINF : 0;
207 rval = chkmd5(fp, path);
210 rval = chkinf(fp, path);
213 rval = report(path, NULL, E_UNKNOWN);
217 if (fp != stdin && fclose(fp))
223 chkmd5(FILE * fp, const char *path)
225 char buf[298]; /* "MD5 (NAMESIZE = MDSUMLEN" */
226 char name[NAMESIZE + 1];
227 char sum[MDSUMLEN + 1], chk[MDSUMLEN + 1];
230 int rval, error, c, fd;
234 while (fgets(buf, sizeof(buf), fp)) {
237 if (((c = sscanf(buf, "MD5 (%256s = %32s%c", name, sum,
238 &ch)) != 3 && (!feof(fp) || c != 2)) ||
239 (c == 3 && ch != '\n') ||
240 (s = strrchr(name, ')')) == NULL ||
241 strlen(sum) != MDSUMLEN)
245 if ((dname = distname(path, name, NULL)) == NULL)
247 else if (opt_exist) {
248 if ((fd = open(dname, O_RDONLY)) == -1)
252 } else if (!MD5File(dname, chk))
254 else if (strcmp(chk, sum))
257 if (opt_ignore && error == E_ERRNO && errno == ENOENT)
259 if (error || opt_all)
260 rval |= report(path, dname, error);
268 chkinf(FILE * fp, const char *path)
270 char buf[30]; /* "cksum.2 = 10 6" */
278 int rval, error, c, pieces, cnt, fd;
282 for (cnt = -1; fgets(buf, sizeof(buf), fp); cnt++) {
287 if ((c = sscanf(buf, "Pieces = %d%c", &pieces, &ch)) != 2 ||
288 ch != '\n' || pieces < 1)
290 } else if (((c = sscanf(buf, "cksum.%2s = %lu %jd%c", ext, &sum,
291 &sumlen, &ch)) != 4 &&
292 (!feof(fp) || c != 3)) || (c == 4 && ch != '\n') ||
293 ext[0] != 'a' + cnt / 26 || ext[1] != 'a' + cnt % 26)
295 else if ((dname = distname(fp == stdin ? NULL : path, NULL,
298 else if ((fd = open(dname, O_RDONLY)) == -1)
300 else if (fstat(fd, &sb))
302 else if (sb.st_size != (off_t)sumlen)
304 else if (!opt_exist) {
305 if (crc(fd, &chk, &len))
310 if (fd != -1 && close(fd))
312 if (opt_ignore && error == E_ERRNO && errno == ENOENT)
314 if (error || (opt_all && cnt >= 0))
315 rval |= report(path, dname, error);
323 report(const char *path, const char *name, int error)
326 name = stripath(name);
329 printf("%s: Unknown format\n", path);
332 printf("%s: Invalid MD5 format\n", path);
335 printf("%s: Invalid .inf format\n", path);
338 printf("%s: Can't derive component name\n", path);
341 printf("%s: %s: Size mismatch\n", path, name);
344 printf("%s: %s: Checksum mismatch\n", path, name);
347 printf("%s: %s: %s\n", path, name, sys_errlist[errno]);
350 printf("%s: %s: OK\n", path, name);
356 distname(const char *path, const char *name, const char *ext)
358 static char buf[NAMESIZE];
367 name = stripath(path);
370 if (ext && nlen > 4 && name[nlen - 4] == '.' &&
371 disttype(name + nlen - 3) == DISTINF)
377 plen = path && (s = strrchr(path, '/')) != NULL ?
378 (size_t)(s - path) : 0;
379 if (plen + (plen > 0) + nlen + (ext ? 3 : 0) >= sizeof(buf))
383 memcpy(s, path, plen);
387 memcpy(s, name, nlen);
399 stripath(const char *path)
403 return ((s = strrchr(path, '/')) != NULL && s[1] ?
408 distfile(const char *path)
413 if ((type = disttype(path)) == DISTMD5 ||
414 ((s = strrchr(path, '.')) != NULL && s > path &&
415 (type = disttype(s + 1)) != 0))
421 disttype(const char *name)
423 static const char dname[DISTTYPES][4] = {"md5", "inf"};
426 for (i = 0; i < DISTTYPES; i++)
427 if (!strcmp(dname[i], name))
433 fail(const char *path, const char *msg)
437 warnx("%s: %s", path, msg ? msg : sys_errlist[errno]);
445 "usage: ckdist [-airsx] [-d dir] [-n name] [-t type] file ...\n");