diff options
author | Tobias Fella <[email protected]> | 2023-10-11 10:07:34 +0000 |
---|---|---|
committer | Tobias Fella <[email protected]> | 2023-10-24 13:16:58 +0000 |
commit | 3e0d1c229b8a15db066fd57cec5bce6dadb1af6b (patch) | |
tree | b4b8a710442aefdfacfb9b95ee4e0a8ed73f2528 /tools/gpgsum.c | |
parent | tests:tpm2dtests: Fix tests with SWTPM. (diff) | |
download | gnupg-3e0d1c229b8a15db066fd57cec5bce6dadb1af6b.tar.gz gnupg-3e0d1c229b8a15db066fd57cec5bce6dadb1af6b.zip |
tools: Add gpgsum tooltobias/gpgsum
* Makefile.am: Add option for building gpgsum
* autogen.rc: Add option for building gpgsum
* configure.ac: Add option for building gpgsum
* tools/Makefile.am: Add sources for gpgsum
* tools/gpgsum-w32info.rc: Add new file
* tools/gpgsum.c: Add gpsum tool
* tools/gpgsum.w32-manifest.in: Add new file
Diffstat (limited to 'tools/gpgsum.c')
-rw-r--r-- | tools/gpgsum.c | 528 |
1 files changed, 528 insertions, 0 deletions
diff --git a/tools/gpgsum.c b/tools/gpgsum.c new file mode 100644 index 000000000..967b40c19 --- /dev/null +++ b/tools/gpgsum.c @@ -0,0 +1,528 @@ +/* gpgsum.c - A simple hash sum tool mainly useful for Windows. + * Copyright (C) 2023 g10 Code GmbH + * + * This file is part of GnuPG. + * + * GnuPG is free software; you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation; either version 3 of the License, or + * (at your option) any later version. + * + * GnuPG is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program; if not, see <https://www.gnu.org/licenses/>. + * SPDX-License-Identifier: GPL-3.0-or-later + */ + +#include <config.h> + +#include <gpg-error.h> + +#include <ctype.h> +#include <errno.h> +#include <stdio.h> +#include <stdlib.h> +#include <string.h> +#ifdef _WIN32 +# include <fcntl.h> +# include <windows.h> +#endif + +#define INCLUDED_BY_MAIN_MODULE 1 +#include "../common/util.h" +#include "../common/init.h" +#include "../common/i18n.h" +#include "../common/sysutils.h" + +#include <gcrypt.h> + +static unsigned int filecount; +static unsigned int readerrors; +static unsigned int checkcount; +static unsigned int matcherrors; + +struct { + int algo; + int check; + int filenames; +} opt; + +enum cmd_and_opt_values + { + aNull = 0, + aCheck = 'c', + oFileNamesFromStdIn = '0', + }; + +static gpgrt_opt_t opts[] = { + ARGPARSE_group (300, N_("@Commands:\n ")), + + ARGPARSE_c (aCheck, "check", + N_("Read checksums from files and check them")), + ARGPARSE_group (301, N_("@\nOptions:\n ")), + ARGPARSE_s_n (oFileNamesFromStdIn, "filenames", + N_("Read file names from stdin")), + ARGPARSE_end () +}; + +static void +parse_arguments (gpgrt_argparse_t *pargs, gpgrt_opt_t *popts) +{ + while (gpgrt_argparse (NULL, pargs, popts)) + { + switch (pargs->r_opt) + { + case aCheck: + opt.check = 1; + break; + case oFileNamesFromStdIn: + opt.filenames = 1; + break; + default: pargs->err = 2; break; + } + } +} + +static const char * +my_strusage( int level ) +{ + const char *p; + + switch (level) + { + case 9: p = "GPL-3.0-or-later"; break; + case 11: p = "gpgsum (@GNUPG@)"; + break; + case 13: p = VERSION; break; + //case 14: p = GNUPG_DEF_COPYRIGHT_LINE; break; + //case 17: p = PRINTABLE_OS_NAME; break; + case 19: p = _("Please report bugs to <@EMAIL@>.\n"); break; + + case 1: + case 40: + p = _("Usage: gpgsum [-c|-0] [--] FILENAMES|-"); + break; + case 41: + p = _("Syntax: gpgsum [-c|-0] [--] FILENAMES|-\n" + "Create or verify hash sums for files.\n" + "The executable must be named for the desired hash algorithm " + "(i.e., \"sha2sum\")."); + break; + + default: p = NULL; break; + } + return p; +} + +/* We need to escape the fname so that included linefeeds etc don't + mess up the the output file. On windows we also turn backslashes + into slashes so that we don't get into conflicts with the escape + character. Note that the GNU version escapes the backslash and the + LF but we also escape the CR. */ +static char * +escapefname (const char *fname, int *escaped) +{ + const char *s; + char *buffer; + char *d; + size_t n; + + *escaped = 0; + for (n = 0, s = fname; *s; s++) + { + if (*s == '\n' || *s == '\r') + n += 2; + else if (*s == '\\') + { +#ifdef _WIN32 + n++; +#else + n += 2; +#endif + } + else + n++; + } + n++; + buffer = xmalloc (n); + d = buffer; + for (s = fname; *s; s++) + { + if (*s == '\n') + { + *d++ = '\\'; + *d++ = 'n' ; + *escaped = 1; + } + else if (*s == '\r') + { + *d++ = '\\'; + *d++ = 'r' ; + *escaped = 1; + } + else if (*s == '\\') + { +#ifdef _WIN32 + *d++ = '/'; +#else + *d++ = '\\'; + *d++ = '\\' ; + *escaped = 1; +#endif + } + else + *d++ = *s; + } + *d = 0; + return buffer; +} + + +/* Revert the escaping in-place. We handle some more of the standard + escaping characters but not all. */ +static void +unescapefname (char *fname) +{ + char *s, *d; + + for (s=d=fname; *s; s++) + { + if (*s == '\\' && s[1]) + { + s++; + switch (*s) + { + case '\\': *d++ = '\\'; break; + case 'n': *d++ = '\n'; break; + case 'r': *d++ = '\r'; break; + case 'f': *d++ = '\f'; break; + case 'v': *d++ = '\v'; break; + case 'b': *d++ = '\b'; break; + default: *d++ = '\\'; *d++ = *s; break; + } + } + else + *d++ = *s; + } + *d = 0; +} + +static gpg_error_t +hash_file (const char *fname, const char *expected) +{ + gpg_error_t err; + estream_t fp; + char buffer[4096]; + size_t n; + char *p; + char *fnamebuf; + int escaped; + gcry_md_hd_t hd; + unsigned char *result; + unsigned int digest_length; + + + digest_length = gcry_md_get_algo_dlen(opt.algo); + + filecount++; + if (!expected && *fname == '-' && !fname[1]) + { + /* Not in check mode and asked to read from stdin. */ + fp = es_stdin; + es_set_binary (es_stdin); + } + else + fp = es_fopen (fname, "rb"); + + if (!fp) + { + err = gpg_error_from_syserror (); + log_error ("Can't open '%s': %s\n", + fname, gpg_strerror (err)); + if (expected) + log_error ("%s: FAILED open\n", fname); + readerrors++; + return gpg_error(GPG_ERR_GENERAL); + } + + err = gcry_md_open(&hd, opt.algo, 0); + if (err) { + log_error("Failed to open md: %s\n", gcry_strerror(err)); + } + + while ( (n = es_fread (buffer, 1, sizeof buffer, fp))) + gcry_md_write(hd, buffer, n); + if (es_ferror (fp)) + { + log_error ("Error reading `%s': %s\n", + fname, strerror (errno)); + if (fp != es_stdin) + es_fclose (fp); + if (expected) + es_printf ("%s: FAILED read\n", fname); + readerrors++; + return gpg_error(GPG_ERR_GENERAL); + } + if (fp != es_stdin) + es_fclose (fp); + + fnamebuf = escapefname (fname, &escaped); + fname = fnamebuf; + + result = gcry_md_read(hd, opt.algo); + if (!result) { + log_error("Failed to read digest\n"); + return gpg_error(GPG_ERR_GENERAL); + } + checkcount++; + bin2hex(result, digest_length, buffer); + if (expected) + { + /* Lowercase the checksum. */ + buffer[strlen(expected)] = 0; + for (p=buffer; *p; p++) + if (*p >= 'A' && *p <= 'Z') + *p |= 0x20; + if (strcmp (buffer, expected)) + { + es_printf ("%s: FAILED\n", fname); + matcherrors++; + return -1; + } + es_printf ("%s: OK\n", fname); + } + else + es_printf ("%s%s %s\n", escaped? "\\":"", buffer, fname); + xfree (fnamebuf); + return 0; +} + +static gpg_error_t +check_file (const char *fname) +{ + estream_t fp; + char *linebuf = NULL; + char *line; + char *p; + size_t n; + int rc = 0; + int escaped; + unsigned int digest_length; + unsigned int name_offset; + size_t line_length; + size_t max_length; + + digest_length = gcry_md_get_algo_dlen(opt.algo); + name_offset = digest_length * 2 + 2; + + if (*fname == '-' && !fname[1]) + fp = es_stdin; + else + fp = es_fopen (fname, "r"); + if (!fp) + { + log_error ("Can't open '%s': %s\n", fname, strerror(errno)); + return -1; + } + + max_length = 4096; + while ( es_read_line (fp, &linebuf, &line_length, &max_length) ) + { + escaped = (*linebuf == '\\'); + line = linebuf + escaped; + n = strlen(line); + if (!n || line[n-1] != '\n') + { + log_error ("Error reading '%s': %s\n", fname, + es_feof (fp)? "last linefeed missing":"line too long"); + rc = -1; + break; + } + line[--n] = 0; + if (n && line[n-1] == '\r') + line[--n] = 0; + if (!*line) + continue; /* Ignore empty lines. */ + if (n < name_offset || line[name_offset-2] != ' ') + { + fprintf (stderr, "Error parsing `%s': %s\n", fname, + "invalid line"); + rc = -1; + continue; + } + + /* Note that we ignore the binary flag ('*') used by GNU + versions of this tool: It does not make sense to compute a + digest over some transformation of a file - we always want a + reliable checksum. The flag does not work: On Unix a + checksum file is created without the flag because it is the + default there. When checking it on Windows the missing flag + would indicate that it has been created in text mode and thus + the checksums will differ. */ + + /* Lowercase the checksum. */ + line[name_offset-2] = 0; + for (p=line; *p; p++) + if (*p >= 'A' && *p <= 'Z') + *p |= 0x20; + /* Unescape the fname. */ + if (escaped) + unescapefname (line+name_offset); + /* Hash the file. */ + if (hash_file (line+name_offset, line)) + rc = -1; + } + + if (es_ferror (fp)) + { + fprintf (stderr, "Error reading `%s': %s\n", + fname, strerror (errno)); + rc = -1; + } + if (fp != stdin) + es_fclose (fp); + + return rc; +} + +static gpg_error_t +hash_list (void) +{ + int rc = 0; + int ready = 0; + int c; + char namebuf[4096]; + size_t n = 0; + unsigned long lastoff = 0; + unsigned long off = 0; + + es_set_binary(es_stdin); + do + { + if ((c = es_getc (es_stdin)) == EOF) + { + if (es_ferror (es_stdin)) + { + log_error ("Error reading '%s' at offset %lu: %s\n", + "[stdin]", off, strerror (errno)); + rc = -1; + break; + } + /* Note: The Nul is a delimiter and not a terminator. */ + c = 0; + ready = 1; + } + if (n >= sizeof namebuf) + { + log_error ("Error reading '%s': " + "filename at offset %lu too long\n", + "[stdin]", lastoff); + rc = -1; + break; + } + namebuf[n++] = c; + off++; + if (!c) + { + if (*namebuf && hash_file (namebuf, NULL)) + rc = -1; + n = 0; + lastoff = off; + } + } + while (!ready); + + return rc; +} + +int +main (int argc, char **argv) +{ + gpgrt_argparse_t pargs; + char *executable = argv[0]; + char *maybe_executable; + char *algo_str; + + int rc = 0; + + gnupg_reopen_std ("gpgsum"); + i18n_init(); + init_common_subsystems(&argc, &argv); + + gpgrt_set_strusage (my_strusage); + pargs.argc = &argc; + pargs.argv = &argv; + pargs.flags = ARGPARSE_FLAG_KEEP; + parse_arguments (&pargs, opts); + gpgrt_argparse (NULL, &pargs, NULL); + +#ifdef _WIN32 + maybe_executable = strrchr(executable, '\\'); +#else + maybe_executable = strrchr(executable, '/'); +#endif + if (maybe_executable) + { + executable = ++maybe_executable; + } + + algo_str = executable; + if (strlen(executable) > 3) + { + algo_str = gpgrt_strdup(executable); + if (strstr(algo_str, ".exe") != NULL) + algo_str[strlen(algo_str) - 7] = 0; + else + algo_str[strlen(algo_str) - 3] = 0; + } + + opt.algo = gcry_md_map_name(algo_str); + + if (!opt.algo) + { + //unknown algo + gpgrt_usage (1); + } + gpgrt_free(algo_str); + if (opt.filenames && opt.check) { + gpgrt_usage (1); + } + + if (opt.filenames) + { + /* With option -0 a dash must be given as filename. */ + if (argc != 1 || strcmp (argv[0], "-")) + gpgrt_usage (1); + if (hash_list ()) + rc = 1; + } + else + { + for (; argc; argv++, argc--) + { + if (opt.check) + { + if (check_file (*argv)) + rc = 1; + } + else + { + if (hash_file (*argv, NULL)) + rc = 1; + } + } + } + + if (opt.check && readerrors) + log_error ("WARNING: %u of %u listed files could not be read\n", + readerrors, filecount); + if (opt.check && matcherrors) + log_error ("WARNING: %u of %u computed checksums did NOT match\n", + matcherrors, checkcount); + + return rc; +} |