mirror of
https://github.com/jart/cosmopolitan.git
synced 2025-01-31 11:37:35 +00:00
244 lines
7.4 KiB
C
244 lines
7.4 KiB
C
/*-*- mode:c;indent-tabs-mode:nil;c-basic-offset:2;tab-width:8;coding:utf-8 -*-│
|
|
│vi: set net ft=c ts=2 sts=2 sw=2 fenc=utf-8 :vi│
|
|
╞══════════════════════════════════════════════════════════════════════════════╡
|
|
│ Copyright 2022 Justine Alexandra Roberts Tunney │
|
|
│ │
|
|
│ Permission to use, copy, modify, and/or distribute this software for │
|
|
│ any purpose with or without fee is hereby granted, provided that the │
|
|
│ above copyright notice and this permission notice appear in all copies. │
|
|
│ │
|
|
│ THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL │
|
|
│ WARRANTIES WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED │
|
|
│ WARRANTIES OF MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE │
|
|
│ AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL │
|
|
│ DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR │
|
|
│ PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER │
|
|
│ TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR │
|
|
│ PERFORMANCE OF THIS SOFTWARE. │
|
|
╚─────────────────────────────────────────────────────────────────────────────*/
|
|
#include "libc/assert.h"
|
|
#include "libc/calls/calls.h"
|
|
#include "libc/errno.h"
|
|
#include "libc/fmt/itoa.h"
|
|
#include "libc/fmt/magnumstrs.internal.h"
|
|
#include "libc/runtime/runtime.h"
|
|
#include "libc/stdio/stdio.h"
|
|
#include "libc/str/str.h"
|
|
#include "libc/str/tab.internal.h"
|
|
#include "third_party/getopt/getopt.h"
|
|
#include "third_party/mbedtls/sha256.h"
|
|
|
|
#define PROG "sha256sum"
|
|
#define USAGE \
|
|
"\
|
|
Usage: " PROG " [-?hbctw] [PATH...]\n\
|
|
-h help\n\
|
|
-c check mode\n\
|
|
-b binary mode\n\
|
|
-t textual mode\n\
|
|
-w warning mode\n\
|
|
\n\
|
|
cosmopolitan " PROG " v1.0\n\
|
|
copyright 2022 justine alexandra roberts tunney\n\
|
|
notice licenses are embedded in the binary\n\
|
|
https://twitter.com/justinetunney\n\
|
|
https://github.com/jart\n\
|
|
\n\
|
|
This program is an Actually Portable Executable (APE) rewrite of the\n\
|
|
GNU coreutils sha256sum command. Our version is better because it goes\n\
|
|
400% faster than coreutils if the Intel® SHA-NI™ ISA is available,\n\
|
|
otherwise this command goes 50% faster than coreutils. This executable\n\
|
|
will work consistently on Linux/Mac/Windows/FreeBSD/NetBSD/OpenBSD, so\n\
|
|
consider vendoring it in your repo to avoid platform portability toil.\n\
|
|
"
|
|
|
|
static bool g_warn;
|
|
static char g_mode;
|
|
static bool g_check;
|
|
static int g_mismatches;
|
|
|
|
static void GetOpts(int argc, char *argv[]) {
|
|
int opt;
|
|
g_mode = ' ';
|
|
while ((opt = getopt(argc, argv, "?hbctw")) != -1) {
|
|
switch (opt) {
|
|
case 'w':
|
|
g_warn = true;
|
|
break;
|
|
case 'c':
|
|
g_check = true;
|
|
break;
|
|
case 't':
|
|
g_mode = ' ';
|
|
break;
|
|
case 'b':
|
|
g_mode = '*';
|
|
break;
|
|
case 'h':
|
|
case '?':
|
|
write(1, USAGE, sizeof(USAGE) - 1);
|
|
exit(0);
|
|
default:
|
|
write(2, USAGE, sizeof(USAGE) - 1);
|
|
exit(64);
|
|
}
|
|
}
|
|
}
|
|
|
|
static ssize_t Write(int fd, const char *s, ...) {
|
|
va_list va;
|
|
char buf[512];
|
|
buf[0] = 0;
|
|
va_start(va, s);
|
|
do {
|
|
strlcat(buf, s, sizeof(buf));
|
|
} while ((s = va_arg(va, const char *)));
|
|
va_end(va);
|
|
return write(fd, buf, strlen(buf));
|
|
}
|
|
|
|
static bool IsModeCharacter(char c) {
|
|
switch (c) {
|
|
case ' ':
|
|
case '*':
|
|
return true;
|
|
default:
|
|
return false;
|
|
}
|
|
}
|
|
|
|
static bool IsSupportedPath(const char *path) {
|
|
size_t i;
|
|
for (i = 0;; ++i) {
|
|
switch (path[i]) {
|
|
case 0:
|
|
if (i) return true;
|
|
// fallthrough
|
|
case '\r':
|
|
case '\n':
|
|
case '\\':
|
|
Write(2, PROG, ": ", path, ": unsupported path\n", NULL);
|
|
return false;
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
static bool GetDigest(const char *path, FILE *f, unsigned char digest[32]) {
|
|
size_t got;
|
|
unsigned char buf[512];
|
|
mbedtls_sha256_context ctx;
|
|
mbedtls_sha256_init(&ctx);
|
|
_unassert(!mbedtls_sha256_starts_ret(&ctx, false));
|
|
while ((got = fread(buf, 1, sizeof(buf), f))) {
|
|
_unassert(!mbedtls_sha256_update_ret(&ctx, buf, got));
|
|
}
|
|
if (ferror(f)) {
|
|
Write(2, PROG, ": ", path, ": ", _strerdoc(errno), "\n", NULL);
|
|
return false;
|
|
}
|
|
_unassert(!mbedtls_sha256_finish_ret(&ctx, digest));
|
|
mbedtls_sha256_free(&ctx);
|
|
return true;
|
|
}
|
|
|
|
static bool ProduceDigest(const char *path, FILE *f) {
|
|
char hexdigest[65];
|
|
char mode[2] = {g_mode};
|
|
unsigned char digest[32];
|
|
if (!IsSupportedPath(path)) return false;
|
|
if (!GetDigest(path, f, digest)) return false;
|
|
hexpcpy(hexdigest, digest, 32);
|
|
Write(1, hexdigest, " ", mode, path, "\n", NULL);
|
|
return true;
|
|
}
|
|
|
|
static bool CheckDigests(const char *path, FILE *f) {
|
|
FILE *f2;
|
|
bool k = true;
|
|
int a, b, i, line;
|
|
const char *path2, *status;
|
|
unsigned char wantdigest[32], gotdigest[32];
|
|
char buf[64 + 2 + PATH_MAX + 1 + 1], *p;
|
|
for (line = 0; fgets(buf, sizeof(buf), f); ++line) {
|
|
if (!*_chomp(buf)) continue;
|
|
for (p = buf, i = 0; i < 32; ++i) {
|
|
if ((a = kHexToInt[*p++ & 255]) == -1) goto InvalidLine;
|
|
if ((b = kHexToInt[*p++ & 255]) == -1) goto InvalidLine;
|
|
wantdigest[i] = a << 4 | b;
|
|
}
|
|
if (*p++ != ' ') goto InvalidLine;
|
|
if (!IsModeCharacter(*p++)) goto InvalidLine;
|
|
path2 = p;
|
|
if (!*path2) goto InvalidLine;
|
|
if (!IsSupportedPath(path2)) continue;
|
|
if ((f2 = fopen(path2, "rb"))) {
|
|
if (GetDigest(path2, f2, gotdigest)) {
|
|
if (!memcmp(wantdigest, gotdigest, 32)) {
|
|
status = "OK";
|
|
} else {
|
|
status = "FAILED";
|
|
++g_mismatches;
|
|
k = false;
|
|
}
|
|
Write(1, path2, ": ", status, "\n", NULL);
|
|
} else {
|
|
k = false;
|
|
}
|
|
fclose(f2);
|
|
} else {
|
|
Write(2, PROG, ": ", path2, ": ", _strerdoc(errno), "\n", NULL);
|
|
k = false;
|
|
}
|
|
continue;
|
|
InvalidLine:
|
|
if (g_warn) {
|
|
char linestr[12];
|
|
FormatInt32(linestr, line + 1);
|
|
Write(2, PROG, ": ", path, ":", linestr, ": ",
|
|
"improperly formatted checksum line", "\n", NULL);
|
|
}
|
|
}
|
|
if (ferror(f)) {
|
|
Write(2, PROG, ": ", path, ": ", _strerdoc(errno), "\n", NULL);
|
|
k = false;
|
|
}
|
|
return k;
|
|
}
|
|
|
|
static bool Process(const char *path, FILE *f) {
|
|
if (g_check) {
|
|
return CheckDigests(path, f);
|
|
} else {
|
|
return ProduceDigest(path, f);
|
|
}
|
|
}
|
|
|
|
int main(int argc, char *argv[]) {
|
|
int i;
|
|
FILE *f;
|
|
bool k = true;
|
|
GetOpts(argc, argv);
|
|
if (optind == argc) {
|
|
f = stdin;
|
|
k &= Process("-", f);
|
|
} else {
|
|
for (i = optind; i < argc; ++i) {
|
|
if ((f = fopen(argv[i], "rb"))) {
|
|
k &= Process(argv[i], f);
|
|
fclose(f);
|
|
} else {
|
|
Write(2, PROG, ": ", argv[i], ": ", _strerdoc(errno), "\n", NULL);
|
|
k = false;
|
|
}
|
|
}
|
|
}
|
|
if (g_mismatches) {
|
|
char ibuf[12];
|
|
FormatInt32(ibuf, g_mismatches);
|
|
Write(2, PROG, ": WARNING: ", ibuf, " computed checksum did NOT match\n",
|
|
NULL);
|
|
}
|
|
return !k;
|
|
}
|