diff --git a/common/openpgp-fpr.c b/common/openpgp-fpr.c index de28c253b..7b110085f 100644 --- a/common/openpgp-fpr.c +++ b/common/openpgp-fpr.c @@ -1,283 +1,283 @@ /* openpgp-fpr.c - OpenPGP Fingerprint computation * Copyright (C) 2021 g10 Code GmbH * * This file is part of GnuPG. * * This file is free software; you can redistribute it and/or modify * it under the terms of either * * - the GNU Lesser General Public License as published by the Free * Software Foundation; either version 3 of the License, or (at * your option) any later version. * * or * * - the GNU General Public License as published by the Free * Software Foundation; either version 2 of the License, or (at * your option) any later version. * * or both in parallel, as here. * * This file 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 . * SPDX-License-Identifier: (LGPL-3.0-or-later OR GPL-2.0-or-later) */ #include #include #include #include #include "util.h" #include "openpgpdefs.h" /* Count the number of bits, assuming the A represents an unsigned big * integer of length LEN bytes. */ static unsigned int count_bits (const unsigned char *a, size_t len) { unsigned int n = len * 8; int i; for (; len && !*a; len--, a++, n -=8) ; if (len) { for (i=7; i && !(*a & (1<> 24); prefix[i++] = (n >> 16); } else if (keyversion == 4) { hashalgo = GCRY_MD_SHA1; n += 6; /* Add the prefix length. */ prefix[i++] = 0x99; } else return gpg_error (GPG_ERR_UNKNOWN_VERSION); prefix[i++] = (n >> 8); prefix[i++] = n; prefix[i++] = keyversion; prefix[i++] = (timestamp >> 24); prefix[i++] = (timestamp >> 16); prefix[i++] = (timestamp >> 8); prefix[i++] = (timestamp); prefix[i++] = pgpalgo; if (keyversion == 5) { prefix[i++] = ((n-10) >> 24); prefix[i++] = ((n-10) >> 16); prefix[i++] = ((n-10) >> 8); prefix[i++] = (n-10); } log_assert (i <= sizeof prefix); /* The first element is reserved for our use; set it. */ iov[0].size = 0; iov[0].off = 0; iov[0].len = i; iov[0].data = prefix; /* for (i=0; i < iovcnt; i++) */ /* log_printhex (iov[i].data, iov[i].len, "cmpfpr i=%d: ", i); */ err = gcry_md_hash_buffers (hashalgo, 0, result, iov, iovcnt); /* log_printhex (result, 20, "fingerpint: "); */ /* Better clear the first element because it was set by us. */ iov[0].size = 0; iov[0].off = 0; iov[0].len = 0; iov[0].data = NULL; if (!err && r_resultlen) *r_resultlen = (hashalgo == GCRY_MD_SHA1)? 20 : 32; return err; } gpg_error_t compute_openpgp_fpr_rsa (int keyversion, unsigned long timestamp, const unsigned char *m, unsigned int mlen, const unsigned char *e, unsigned int elen, unsigned char *result, unsigned int *r_resultlen) { gcry_buffer_t iov[5] = { {0} }; unsigned char nbits_m[2], nbits_e[2]; unsigned int n; /* Strip leading zeroes. */ for (; mlen && !*m; mlen--, m++) ; for (; elen && !*e; elen--, e++) ; /* Count bits. */ n = count_bits (m, mlen); nbits_m[0] = n >> 8; nbits_m[1] = n; n = count_bits (e, elen); nbits_e[0] = n >> 8; nbits_e[1] = n; /* Put parms into the array. Note that iov[0] is reserved. */ iov[1].len = 2; iov[1].data = nbits_m; iov[2].len = mlen; iov[2].data = (void*)m; iov[3].len = 2; iov[3].data = nbits_e; iov[4].len = elen; iov[4].data = (void*)e; return compute_openpgp_fpr (keyversion, PUBKEY_ALGO_RSA, timestamp, iov, 5, result, r_resultlen); } /* Determine KDF hash algorithm and KEK encryption algorithm by CURVE. * The returned buffer has a length of 4. * Note: This needs to be kept in sync with the table in g10/ecdh.c */ static const unsigned char* default_ecdh_params (unsigned int nbits) { /* See RFC-6637 for those constants. 0x03: Number of bytes 0x01: Version for this parameter format KEK digest algorithm KEK cipher algorithm */ if (nbits <= 256) return (const unsigned char*)"\x03\x01\x08\x07"; else if (nbits <= 384) return (const unsigned char*)"\x03\x01\x09\x09"; else return (const unsigned char*)"\x03\x01\x0a\x09"; } gpg_error_t compute_openpgp_fpr_ecc (int keyversion, unsigned long timestamp, const char *curvename, int for_encryption, const unsigned char *q, unsigned int qlen, const unsigned char *kdf, unsigned int kdflen, unsigned char *result, unsigned int *r_resultlen) { gpg_error_t err; const char *curveoidstr; gcry_mpi_t curveoid = NULL; unsigned int curvebits; int pgpalgo; const unsigned char *oidraw; size_t oidrawlen; gcry_buffer_t iov[5] = { {0} }; unsigned int iovlen; unsigned char nbits_q[2]; unsigned int n; curveoidstr = openpgp_curve_to_oid (curvename, &curvebits, &pgpalgo); err = openpgp_oid_from_str (curveoidstr, &curveoid); if (err) goto leave; oidraw = gcry_mpi_get_opaque (curveoid, &n); if (!oidraw) { err = gpg_error_from_syserror (); goto leave; } oidrawlen = (n+7)/8; /* If the curve does not enforce a certain algorithm, we use the * for_encryption flag to decide which algo to use. */ if (!pgpalgo) pgpalgo = for_encryption? PUBKEY_ALGO_ECDH : PUBKEY_ALGO_ECDSA; /* Count bits. */ n = count_sos_bits (q, qlen); nbits_q[0] = n >> 8; nbits_q[1] = n; /* Put parms into the array. Note that iov[0] is reserved. */ iov[1].len = oidrawlen; iov[1].data = (void*)oidraw; iov[2].len = 2; iov[2].data = nbits_q; iov[3].len = qlen; iov[3].data = (void*)q; iovlen = 4; if (pgpalgo == PUBKEY_ALGO_ECDH) { if (!kdf || !kdflen || !kdf[0]) { - /* No KDF givem - use the default. */ + /* No KDF given - use the default. */ kdflen = 4; kdf = default_ecdh_params (curvebits); } iov[4].len = kdflen; iov[4].data = (void*)kdf; iovlen++; } err = compute_openpgp_fpr (keyversion, pgpalgo, timestamp, iov, iovlen, result, r_resultlen); leave: gcry_mpi_release (curveoid); return err; } diff --git a/g10/openfile.c b/g10/openfile.c index d82858f5a..6f4e889e4 100644 --- a/g10/openfile.c +++ b/g10/openfile.c @@ -1,403 +1,403 @@ /* openfile.c * Copyright (C) 1998, 1999, 2000, 2001, 2002, 2003, 2004, 2005, 2009, * 2010 Free Software Foundation, Inc. * * 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 . */ #include #include #include #include #include #include #include #include #include #include "gpg.h" #include "../common/util.h" #include "../common/ttyio.h" #include "options.h" #include "main.h" #include "../common/status.h" #include "../common/i18n.h" #ifdef HAVE_W32_SYSTEM #define NAME_OF_DEV_NULL "nul" #else #define NAME_OF_DEV_NULL "/dev/null" #endif #if defined (HAVE_DRIVE_LETTERS) || defined (__riscos__) #define CMP_FILENAME(a,b) ascii_strcasecmp( (a), (b) ) #else #define CMP_FILENAME(a,b) strcmp( (a), (b) ) #endif /* FIXME: Implement opt.interactive. */ /* * Check whether FNAME exists and ask if it's okay to overwrite an * existing one. * Returns: True: it's okay to overwrite or the file does not exist * False: Do not overwrite */ int overwrite_filep( const char *fname ) { if ( iobuf_is_pipe_filename (fname) ) return 1; /* Writing to stdout is always okay. */ - if ( access( fname, F_OK ) ) + if ( gnupg_access( fname, F_OK ) ) return 1; /* Does not exist. */ if ( !compare_filenames (fname, NAME_OF_DEV_NULL) ) return 1; /* Does not do any harm. */ if (opt.answer_yes) return 1; if (opt.answer_no || opt.batch) return 0; /* Do not overwrite. */ tty_printf (_("File '%s' exists. "), fname); if (cpr_enabled ()) tty_printf ("\n"); if (cpr_get_answer_is_yes ("openfile.overwrite.okay", _("Overwrite? (y/N) ")) ) return 1; return 0; } /* * Strip known extensions from iname and return a newly allocated * filename. Return NULL if we can't do that. */ char * make_outfile_name (const char *iname) { size_t n; if (iobuf_is_pipe_filename (iname)) return xstrdup ("-"); n = strlen (iname); if (n > 4 && (!CMP_FILENAME(iname+n-4, EXTSEP_S GPGEXT_GPG) || !CMP_FILENAME(iname+n-4, EXTSEP_S "pgp") || !CMP_FILENAME(iname+n-4, EXTSEP_S "sig") || !CMP_FILENAME(iname+n-4, EXTSEP_S "asc"))) { char *buf = xstrdup (iname); buf[n-4] = 0; return buf; } else if (n > 5 && !CMP_FILENAME(iname+n-5, EXTSEP_S "sign")) { char *buf = xstrdup (iname); buf[n-5] = 0; return buf; } log_info (_("%s: unknown suffix\n"), iname); return NULL; } /* Ask for an output filename; use the given one as default. Return NULL if no file has been given or if it is not possible to ask the user. NAME is the template len which might contain enbedded Nuls. NAMELEN is its actual length. */ char * ask_outfile_name( const char *name, size_t namelen ) { size_t n; const char *s; char *prompt; char *fname; char *defname; if ( opt.batch ) return NULL; defname = name && namelen? make_printable_string (name, namelen, 0) : NULL; s = _("Enter new filename"); n = strlen(s) + (defname?strlen (defname):0) + 10; prompt = xmalloc (n); if (defname) snprintf (prompt, n, "%s [%s]: ", s, defname ); else snprintf (prompt, n, "%s: ", s ); tty_enable_completion(NULL); fname = cpr_get ("openfile.askoutname", prompt ); cpr_kill_prompt (); tty_disable_completion (); xfree (prompt); if ( !*fname ) { xfree (fname); fname = defname; defname = NULL; } xfree (defname); if (fname) trim_spaces (fname); return fname; } /* * Make an output filename for the inputfile INAME. * Returns an IOBUF and an errorcode * Mode 0 = use ".gpg" * 1 = use ".asc" * 2 = use ".sig" * 3 = use ".rev" * * If INP_FD is not -1 the function simply creates an IOBUF for that * file descriptor and ignore INAME and MODE. Note that INP_FD won't * be closed if the returned IOBUF is closed. With RESTRICTEDPERM a * file will be created with mode 700 if possible. */ int open_outfile (int inp_fd, const char *iname, int mode, int restrictedperm, iobuf_t *a) { int rc = 0; *a = NULL; if (inp_fd != -1) { char xname[64]; *a = iobuf_fdopen_nc (inp_fd, "wb"); if (!*a) { rc = gpg_error_from_syserror (); snprintf (xname, sizeof xname, "[fd %d]", inp_fd); log_error (_("can't open '%s': %s\n"), xname, gpg_strerror (rc)); } else if (opt.verbose) { snprintf (xname, sizeof xname, "[fd %d]", inp_fd); log_info (_("writing to '%s'\n"), xname); } } else if (iobuf_is_pipe_filename (iname) && !opt.outfile) { *a = iobuf_create (NULL, 0); if ( !*a ) { rc = gpg_error_from_syserror (); log_error (_("can't open '%s': %s\n"), "[stdout]", strerror(errno) ); } else if ( opt.verbose ) log_info (_("writing to stdout\n")); } else { char *buf = NULL; const char *name; if (opt.dry_run) name = NAME_OF_DEV_NULL; else if (opt.outfile) name = opt.outfile; else { #ifdef USE_ONLY_8DOT3 if (opt.mangle_dos_filenames) { /* It is quite common for DOS systems to have only one dot in a filename. If we have something like this, we simple replace the suffix except in cases where the suffix is larger than 3 characters and not the same as the new one. We don't map the filenames to 8.3 because this is a duty of the file system. */ char *dot; const char *newsfx; newsfx = (mode==1 ? ".asc" : mode==2 ? ".sig" : mode==3 ? ".rev" : ".gpg"); buf = xmalloc (strlen(iname)+4+1); strcpy (buf, iname); dot = strchr (buf, '.' ); if ( dot && dot > buf && dot[1] && strlen(dot) <= 4 && CMP_FILENAME (newsfx, dot) ) strcpy (dot, newsfx); else if (dot && !dot[1]) /* Do not duplicate a dot. */ strcpy (dot, newsfx+1); else strcat (buf, newsfx); } if (!buf) #endif /* USE_ONLY_8DOT3 */ { buf = xstrconcat (iname, (mode==1 ? EXTSEP_S "asc" : mode==2 ? EXTSEP_S "sig" : mode==3 ? EXTSEP_S "rev" : /* */ EXTSEP_S GPGEXT_GPG), NULL); } name = buf; } rc = 0; while ( !overwrite_filep (name) ) { char *tmp = ask_outfile_name (NULL, 0); if ( !tmp || !*tmp ) { xfree (tmp); rc = gpg_error (GPG_ERR_EEXIST); break; } xfree (buf); name = buf = tmp; } if ( !rc ) { if (is_secured_filename (name) ) { *a = NULL; gpg_err_set_errno (EPERM); } else *a = iobuf_create (name, restrictedperm); if (!*a) { rc = gpg_error_from_syserror (); log_error(_("can't create '%s': %s\n"), name, strerror(errno) ); } else if( opt.verbose ) log_info (_("writing to '%s'\n"), name ); } xfree(buf); } if (*a) iobuf_ioctl (*a, IOBUF_IOCTL_NO_CACHE, 1, NULL); return rc; } /* Find a matching data file for the signature file SIGFILENAME and return it as a malloced string. If no matching data file is found, return NULL. */ char * get_matching_datafile (const char *sigfilename) { char *fname = NULL; size_t len; if (iobuf_is_pipe_filename (sigfilename)) return NULL; len = strlen (sigfilename); if (len > 4 && (!strcmp (sigfilename + len - 4, EXTSEP_S "sig") || (len > 5 && !strcmp(sigfilename + len - 5, EXTSEP_S "sign")) || !strcmp(sigfilename + len - 4, EXTSEP_S "asc"))) { fname = xstrdup (sigfilename); fname[len-(fname[len-1]=='n'?5:4)] = 0 ; if (gnupg_access (fname, R_OK )) { /* Not found or other error. */ xfree (fname); fname = NULL; } } return fname; } /* * Try to open a file without the extension ".sig" or ".asc" * Return NULL if such a file is not available. */ iobuf_t open_sigfile (const char *sigfilename, progress_filter_context_t *pfx) { iobuf_t a = NULL; char *buf; buf = get_matching_datafile (sigfilename); if (buf) { a = iobuf_open (buf); if (a && is_secured_file (iobuf_get_fd (a))) { iobuf_close (a); a = NULL; gpg_err_set_errno (EPERM); } if (a) log_info (_("assuming signed data in '%s'\n"), buf); if (a && pfx) handle_progress (pfx, a, buf); xfree (buf); } return a; } /* Create the directory only if the supplied directory name is the same as the default one. This way we avoid to create arbitrary directories when a non-default home directory is used. To cope with HOME, we do compare only the suffix if we see that the default homedir does start with a tilde. */ void try_make_homedir (const char *fname) { if ( opt.dry_run || opt.no_homedir_creation ) return; gnupg_maybe_make_homedir (fname, opt.quiet); } /* Get and if needed create a string with the directory used to store openpgp revocations. */ char * get_openpgp_revocdir (const char *home) { char *fname; struct stat statbuf; fname = make_filename (home, GNUPG_OPENPGP_REVOC_DIR, NULL); if (gnupg_stat (fname, &statbuf) && errno == ENOENT) { if (gnupg_mkdir (fname, "-rwx")) log_error (_("can't create directory '%s': %s\n"), fname, strerror (errno) ); else if (!opt.quiet) log_info (_("directory '%s' created\n"), fname); } return fname; }