2019-01-22 09:07:24 +01:00
|
|
|
|
/* gpg-card-tool.c - An interactive tool to work with cards.
|
2019-01-27 20:12:00 +01:00
|
|
|
|
* Copyright (C) 2019 g10 Code GmbH
|
2019-01-22 09:07:24 +01:00
|
|
|
|
*
|
|
|
|
|
* This file is part of GnuPG.
|
|
|
|
|
*
|
|
|
|
|
* This file 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.
|
|
|
|
|
*
|
|
|
|
|
* 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 Lesser 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://gnu.org/licenses/>.
|
2019-01-27 20:12:00 +01:00
|
|
|
|
* SPDX-License-Identifier: GPL-3.0-or-later
|
2019-01-22 09:07:24 +01:00
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
#include <config.h>
|
|
|
|
|
#include <stdio.h>
|
|
|
|
|
#include <stdlib.h>
|
|
|
|
|
#include <string.h>
|
|
|
|
|
#ifdef HAVE_LIBREADLINE
|
|
|
|
|
# define GNUPG_LIBREADLINE_H_INCLUDED
|
|
|
|
|
# include <readline/readline.h>
|
|
|
|
|
#endif /*HAVE_LIBREADLINE*/
|
|
|
|
|
|
|
|
|
|
#include "../common/util.h"
|
|
|
|
|
#include "../common/status.h"
|
|
|
|
|
#include "../common/i18n.h"
|
|
|
|
|
#include "../common/init.h"
|
|
|
|
|
#include "../common/sysutils.h"
|
|
|
|
|
#include "../common/asshelp.h"
|
|
|
|
|
#include "../common/userids.h"
|
|
|
|
|
#include "../common/ccparray.h"
|
|
|
|
|
#include "../common/exectool.h"
|
|
|
|
|
#include "../common/ttyio.h"
|
2019-01-27 20:12:00 +01:00
|
|
|
|
#include "../common/server-help.h"
|
|
|
|
|
#include "../common/openpgpdefs.h"
|
|
|
|
|
|
|
|
|
|
#include "card-tool.h"
|
|
|
|
|
|
2019-01-22 09:07:24 +01:00
|
|
|
|
|
|
|
|
|
#define CONTROL_D ('D' - 'A' + 1)
|
|
|
|
|
|
|
|
|
|
/* Constants to identify the commands and options. */
|
|
|
|
|
enum cmd_and_opt_values
|
|
|
|
|
{
|
|
|
|
|
aNull = 0,
|
|
|
|
|
|
|
|
|
|
oQuiet = 'q',
|
|
|
|
|
oVerbose = 'v',
|
|
|
|
|
|
|
|
|
|
oDebug = 500,
|
|
|
|
|
|
|
|
|
|
oGpgProgram,
|
|
|
|
|
oGpgsmProgram,
|
|
|
|
|
oStatusFD,
|
|
|
|
|
oWithColons,
|
2019-01-27 20:12:00 +01:00
|
|
|
|
oNoAutostart,
|
|
|
|
|
oAgentProgram,
|
|
|
|
|
|
|
|
|
|
oDisplay,
|
|
|
|
|
oTTYname,
|
|
|
|
|
oTTYtype,
|
|
|
|
|
oXauthority,
|
|
|
|
|
oLCctype,
|
|
|
|
|
oLCmessages,
|
2019-01-22 09:07:24 +01:00
|
|
|
|
|
|
|
|
|
oDummy
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* The list of commands and options. */
|
|
|
|
|
static ARGPARSE_OPTS opts[] = {
|
|
|
|
|
ARGPARSE_group (300, ("@Commands:\n ")),
|
|
|
|
|
|
|
|
|
|
ARGPARSE_group (301, ("@\nOptions:\n ")),
|
|
|
|
|
|
|
|
|
|
ARGPARSE_s_n (oVerbose, "verbose", ("verbose")),
|
|
|
|
|
ARGPARSE_s_n (oQuiet, "quiet", ("be somewhat more quiet")),
|
|
|
|
|
ARGPARSE_s_s (oDebug, "debug", "@"),
|
|
|
|
|
ARGPARSE_s_s (oGpgProgram, "gpg", "@"),
|
|
|
|
|
ARGPARSE_s_s (oGpgsmProgram, "gpgsm", "@"),
|
|
|
|
|
ARGPARSE_s_i (oStatusFD, "status-fd", N_("|FD|write status info to this FD")),
|
|
|
|
|
ARGPARSE_s_n (oWithColons, "with-colons", "@"),
|
2019-01-27 20:12:00 +01:00
|
|
|
|
ARGPARSE_s_n (oNoAutostart, "no-autostart", "@"),
|
|
|
|
|
ARGPARSE_s_s (oAgentProgram, "agent-program", "@"),
|
|
|
|
|
ARGPARSE_s_s (oDisplay, "display", "@"),
|
|
|
|
|
ARGPARSE_s_s (oTTYname, "ttyname", "@"),
|
|
|
|
|
ARGPARSE_s_s (oTTYtype, "ttytype", "@"),
|
|
|
|
|
ARGPARSE_s_s (oXauthority, "xauthority", "@"),
|
|
|
|
|
ARGPARSE_s_s (oLCctype, "lc-ctype", "@"),
|
|
|
|
|
ARGPARSE_s_s (oLCmessages, "lc-messages","@"),
|
2019-01-22 09:07:24 +01:00
|
|
|
|
|
|
|
|
|
ARGPARSE_end ()
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
/* The list of supported debug flags. */
|
|
|
|
|
static struct debug_flags_s debug_flags [] =
|
|
|
|
|
{
|
|
|
|
|
{ DBG_IPC_VALUE , "ipc" },
|
|
|
|
|
{ DBG_EXTPROG_VALUE, "extprog" },
|
|
|
|
|
{ 0, NULL }
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
2019-01-29 08:48:53 +01:00
|
|
|
|
/* An object to create lists of labels and keyrefs. */
|
|
|
|
|
struct keyinfolabel_s
|
|
|
|
|
{
|
|
|
|
|
const char *label;
|
|
|
|
|
const char *keyref;
|
|
|
|
|
};
|
|
|
|
|
typedef struct keyinfolabel_s *keyinfolabel_t;
|
|
|
|
|
|
|
|
|
|
|
2019-01-27 20:12:00 +01:00
|
|
|
|
/* Limit of size of data we read from a file for certain commands. */
|
|
|
|
|
#define MAX_GET_DATA_FROM_FILE 16384
|
|
|
|
|
|
2019-01-29 08:48:53 +01:00
|
|
|
|
/* Constants for OpenPGP cards. */
|
2019-01-27 20:12:00 +01:00
|
|
|
|
#define OPENPGP_USER_PIN_DEFAULT "123456"
|
|
|
|
|
#define OPENPGP_ADMIN_PIN_DEFAULT "12345678"
|
|
|
|
|
#define OPENPGP_KDF_DATA_LENGTH_MIN 90
|
|
|
|
|
#define OPENPGP_KDF_DATA_LENGTH_MAX 110
|
|
|
|
|
|
2019-01-22 09:07:24 +01:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
2019-01-27 20:12:00 +01:00
|
|
|
|
/* Local prototypes. */
|
2019-01-22 09:07:24 +01:00
|
|
|
|
static void wrong_args (const char *text) GPGRT_ATTR_NORETURN;
|
|
|
|
|
static void interactive_loop (void);
|
|
|
|
|
#ifdef HAVE_LIBREADLINE
|
|
|
|
|
static char **command_completion (const char *text, int start, int end);
|
|
|
|
|
#endif /*HAVE_LIBREADLINE*/
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Print usage information and provide strings for help. */
|
|
|
|
|
static const char *
|
|
|
|
|
my_strusage( int level )
|
|
|
|
|
{
|
|
|
|
|
const char *p;
|
|
|
|
|
|
|
|
|
|
switch (level)
|
|
|
|
|
{
|
|
|
|
|
case 11: p = "gpg-card-tool"; break;
|
|
|
|
|
case 12: p = "@GNUPG@"; break;
|
|
|
|
|
case 13: p = VERSION; break;
|
|
|
|
|
case 17: p = PRINTABLE_OS_NAME; break;
|
|
|
|
|
case 19: p = ("Please report bugs to <@EMAIL@>.\n"); break;
|
|
|
|
|
|
|
|
|
|
case 1:
|
|
|
|
|
case 40:
|
|
|
|
|
p = ("Usage: gpg-card-tool [command] [options] [args] (-h for help)");
|
|
|
|
|
break;
|
|
|
|
|
case 41:
|
|
|
|
|
p = ("Syntax: gpg-card-tool [command] [options] [args]\n"
|
|
|
|
|
"Tool to configure cards and tokens\n");
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
default: p = NULL; break;
|
|
|
|
|
}
|
|
|
|
|
return p;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
wrong_args (const char *text)
|
|
|
|
|
{
|
|
|
|
|
es_fprintf (es_stderr, _("usage: %s [options] %s\n"), strusage (11), text);
|
|
|
|
|
exit (2);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2019-01-27 20:12:00 +01:00
|
|
|
|
static void
|
|
|
|
|
set_opt_session_env (const char *name, const char *value)
|
|
|
|
|
{
|
|
|
|
|
gpg_error_t err;
|
|
|
|
|
|
|
|
|
|
err = session_env_setenv (opt.session_env, name, value);
|
|
|
|
|
if (err)
|
|
|
|
|
log_fatal ("error setting session environment: %s\n",
|
|
|
|
|
gpg_strerror (err));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2019-01-22 09:07:24 +01:00
|
|
|
|
|
|
|
|
|
/* Command line parsing. */
|
|
|
|
|
static enum cmd_and_opt_values
|
|
|
|
|
parse_arguments (ARGPARSE_ARGS *pargs, ARGPARSE_OPTS *popts)
|
|
|
|
|
{
|
|
|
|
|
enum cmd_and_opt_values cmd = 0;
|
|
|
|
|
int no_more_options = 0;
|
|
|
|
|
|
|
|
|
|
while (!no_more_options && optfile_parse (NULL, NULL, NULL, pargs, popts))
|
|
|
|
|
{
|
|
|
|
|
switch (pargs->r_opt)
|
|
|
|
|
{
|
|
|
|
|
case oQuiet: opt.quiet = 1; break;
|
|
|
|
|
case oVerbose: opt.verbose++; break;
|
|
|
|
|
case oDebug:
|
|
|
|
|
if (parse_debug_flag (pargs->r.ret_str, &opt.debug, debug_flags))
|
|
|
|
|
{
|
|
|
|
|
pargs->r_opt = ARGPARSE_INVALID_ARG;
|
|
|
|
|
pargs->err = ARGPARSE_PRINT_ERROR;
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
|
2019-01-27 20:12:00 +01:00
|
|
|
|
case oGpgProgram: opt.gpg_program = pargs->r.ret_str; break;
|
|
|
|
|
case oGpgsmProgram: opt.gpgsm_program = pargs->r.ret_str; break;
|
|
|
|
|
case oAgentProgram: opt.agent_program = pargs->r.ret_str; break;
|
|
|
|
|
|
2019-01-22 09:07:24 +01:00
|
|
|
|
case oStatusFD:
|
|
|
|
|
gnupg_set_status_fd (translate_sys2libc_fd_int (pargs->r.ret_int, 1));
|
|
|
|
|
break;
|
2019-01-27 20:12:00 +01:00
|
|
|
|
|
|
|
|
|
case oWithColons: opt.with_colons = 1; break;
|
|
|
|
|
case oNoAutostart: opt.autostart = 0; break;
|
|
|
|
|
|
|
|
|
|
case oDisplay: set_opt_session_env ("DISPLAY", pargs->r.ret_str); break;
|
|
|
|
|
case oTTYname: set_opt_session_env ("GPG_TTY", pargs->r.ret_str); break;
|
|
|
|
|
case oTTYtype: set_opt_session_env ("TERM", pargs->r.ret_str); break;
|
|
|
|
|
case oXauthority: set_opt_session_env ("XAUTHORITY",
|
|
|
|
|
pargs->r.ret_str); break;
|
|
|
|
|
case oLCctype: opt.lc_ctype = pargs->r.ret_str; break;
|
|
|
|
|
case oLCmessages: opt.lc_messages = pargs->r.ret_str; break;
|
2019-01-22 09:07:24 +01:00
|
|
|
|
|
|
|
|
|
default: pargs->err = 2; break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return cmd;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* gpg-card-tool main. */
|
|
|
|
|
int
|
|
|
|
|
main (int argc, char **argv)
|
|
|
|
|
{
|
|
|
|
|
gpg_error_t err;
|
|
|
|
|
ARGPARSE_ARGS pargs;
|
|
|
|
|
enum cmd_and_opt_values cmd;
|
|
|
|
|
|
|
|
|
|
gnupg_reopen_std ("gpg-card-tool");
|
|
|
|
|
set_strusage (my_strusage);
|
|
|
|
|
gnupg_rl_initialize ();
|
|
|
|
|
log_set_prefix ("gpg-card-tool", GPGRT_LOG_WITH_PREFIX);
|
|
|
|
|
|
|
|
|
|
/* Make sure that our subsystems are ready. */
|
|
|
|
|
i18n_init();
|
|
|
|
|
init_common_subsystems (&argc, &argv);
|
|
|
|
|
|
|
|
|
|
assuan_set_gpg_err_source (GPG_ERR_SOURCE_DEFAULT);
|
|
|
|
|
setup_libassuan_logging (&opt.debug, NULL);
|
|
|
|
|
|
2019-01-27 20:12:00 +01:00
|
|
|
|
/* Setup default options. */
|
|
|
|
|
opt.autostart = 1;
|
|
|
|
|
opt.session_env = session_env_new ();
|
|
|
|
|
if (!opt.session_env)
|
|
|
|
|
log_fatal ("error allocating session environment block: %s\n",
|
|
|
|
|
gpg_strerror (gpg_error_from_syserror ()));
|
|
|
|
|
|
|
|
|
|
|
2019-01-22 09:07:24 +01:00
|
|
|
|
/* Parse the command line. */
|
|
|
|
|
pargs.argc = &argc;
|
|
|
|
|
pargs.argv = &argv;
|
|
|
|
|
pargs.flags = ARGPARSE_FLAG_KEEP;
|
|
|
|
|
cmd = parse_arguments (&pargs, opts);
|
|
|
|
|
|
|
|
|
|
if (log_get_errorcount (0))
|
|
|
|
|
exit (2);
|
|
|
|
|
|
|
|
|
|
/* Print a warning if an argument looks like an option. */
|
|
|
|
|
if (!opt.quiet && !(pargs.flags & ARGPARSE_FLAG_STOP_SEEN))
|
|
|
|
|
{
|
|
|
|
|
int i;
|
|
|
|
|
|
|
|
|
|
for (i=0; i < argc; i++)
|
|
|
|
|
if (argv[i][0] == '-' && argv[i][1] == '-')
|
|
|
|
|
log_info (("NOTE: '%s' is not considered an option\n"), argv[i]);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Set defaults for non given options. */
|
|
|
|
|
if (!opt.gpg_program)
|
|
|
|
|
opt.gpg_program = gnupg_module_name (GNUPG_MODULE_NAME_GPG);
|
|
|
|
|
if (!opt.gpgsm_program)
|
|
|
|
|
opt.gpgsm_program = gnupg_module_name (GNUPG_MODULE_NAME_GPGSM);
|
|
|
|
|
|
|
|
|
|
/* Run the selected command. */
|
|
|
|
|
switch (cmd)
|
|
|
|
|
{
|
|
|
|
|
default:
|
|
|
|
|
interactive_loop ();
|
|
|
|
|
err = 0;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (err)
|
|
|
|
|
gnupg_status_printf (STATUS_FAILURE, "- %u", err);
|
|
|
|
|
else if (log_get_errorcount (0))
|
|
|
|
|
gnupg_status_printf (STATUS_FAILURE, "- %u", GPG_ERR_GENERAL);
|
|
|
|
|
else
|
|
|
|
|
gnupg_status_printf (STATUS_SUCCESS, NULL);
|
|
|
|
|
return log_get_errorcount (0)? 1:0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2019-01-27 20:12:00 +01:00
|
|
|
|
/* Read data from file FNAME up to MAX_GET_DATA_FROM_FILE characters.
|
|
|
|
|
* On error return an error code and stores NULL at R_BUFFER; on
|
|
|
|
|
* success returns 0, stpres the number of bytes read at R_BUFLEN and
|
|
|
|
|
* the address of a newly allocated buffer at R_BUFFER. */
|
|
|
|
|
static gpg_error_t
|
|
|
|
|
get_data_from_file (const char *fname, char **r_buffer, size_t *r_buflen)
|
|
|
|
|
{
|
|
|
|
|
gpg_error_t err;
|
|
|
|
|
estream_t fp;
|
|
|
|
|
char *data;
|
|
|
|
|
int n;
|
2019-01-22 09:07:24 +01:00
|
|
|
|
|
2019-01-27 20:12:00 +01:00
|
|
|
|
*r_buffer = NULL;
|
|
|
|
|
*r_buflen = 0;
|
2019-01-22 09:07:24 +01:00
|
|
|
|
|
2019-01-27 20:12:00 +01:00
|
|
|
|
fp = es_fopen (fname, "rb");
|
|
|
|
|
if (!fp)
|
|
|
|
|
{
|
|
|
|
|
err = gpg_error_from_syserror ();
|
|
|
|
|
log_error (_("can't open '%s': %s\n"), fname, gpg_strerror (err));
|
|
|
|
|
return err;
|
|
|
|
|
}
|
2019-01-22 09:07:24 +01:00
|
|
|
|
|
2019-01-27 20:12:00 +01:00
|
|
|
|
data = xtrymalloc (MAX_GET_DATA_FROM_FILE);
|
|
|
|
|
if (!data)
|
|
|
|
|
{
|
|
|
|
|
err = gpg_error_from_syserror ();
|
|
|
|
|
log_error (_("error allocating enough memory: %s\n"), gpg_strerror (err));
|
|
|
|
|
es_fclose (fp);
|
|
|
|
|
return err;
|
|
|
|
|
}
|
2019-01-22 09:07:24 +01:00
|
|
|
|
|
2019-01-27 20:12:00 +01:00
|
|
|
|
n = es_fread (data, 1, MAX_GET_DATA_FROM_FILE, fp);
|
|
|
|
|
es_fclose (fp);
|
|
|
|
|
if (n < 0)
|
|
|
|
|
{
|
|
|
|
|
err = gpg_error_from_syserror ();
|
|
|
|
|
tty_printf (_("error reading '%s': %s\n"), fname, gpg_strerror (err));
|
|
|
|
|
xfree (data);
|
|
|
|
|
return err;
|
|
|
|
|
}
|
|
|
|
|
*r_buffer = data;
|
|
|
|
|
*r_buflen = n;
|
|
|
|
|
return 0;
|
2019-01-22 09:07:24 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2019-01-27 20:12:00 +01:00
|
|
|
|
/* Write LENGTH bytes from BUFFER to file FNAME. Return 0 on
|
|
|
|
|
* success. */
|
|
|
|
|
static gpg_error_t
|
|
|
|
|
put_data_to_file (const char *fname, const void *buffer, size_t length)
|
2019-01-22 09:07:24 +01:00
|
|
|
|
{
|
2019-01-27 20:12:00 +01:00
|
|
|
|
gpg_error_t err;
|
|
|
|
|
estream_t fp;
|
2019-01-22 09:07:24 +01:00
|
|
|
|
|
2019-01-27 20:12:00 +01:00
|
|
|
|
fp = es_fopen (fname, "wb");
|
|
|
|
|
if (!fp)
|
|
|
|
|
{
|
|
|
|
|
err = gpg_error_from_syserror ();
|
|
|
|
|
log_error (_("can't create '%s': %s\n"), fname, gpg_strerror (err));
|
|
|
|
|
return err;
|
|
|
|
|
}
|
2019-01-22 09:07:24 +01:00
|
|
|
|
|
2019-01-27 20:12:00 +01:00
|
|
|
|
if (length && es_fwrite (buffer, length, 1, fp) != 1)
|
|
|
|
|
{
|
|
|
|
|
err = gpg_error_from_syserror ();
|
|
|
|
|
log_error (_("error writing '%s': %s\n"), fname, gpg_strerror (err));
|
|
|
|
|
es_fclose (fp);
|
|
|
|
|
return err;
|
|
|
|
|
}
|
|
|
|
|
if (es_fclose (fp))
|
|
|
|
|
{
|
|
|
|
|
err = gpg_error_from_syserror ();
|
|
|
|
|
log_error (_("error writing '%s': %s\n"), fname, gpg_strerror (err));
|
|
|
|
|
return err;
|
|
|
|
|
}
|
|
|
|
|
return 0;
|
2019-01-22 09:07:24 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2019-01-27 20:12:00 +01:00
|
|
|
|
|
|
|
|
|
/* Simply prints TEXT to the output. Returns 0 as a convenience.
|
|
|
|
|
* This is a separate fucntion so that it can be extended to run
|
|
|
|
|
* less(1) or so. The extra arguments are int values terminated by a
|
|
|
|
|
* 0 to indicate card application types supported with this command.
|
|
|
|
|
* If none are given (just teh final 0), this is a general
|
|
|
|
|
* command. */
|
|
|
|
|
static gpg_error_t
|
|
|
|
|
print_help (const char *text, ...)
|
2019-01-22 09:07:24 +01:00
|
|
|
|
{
|
2019-01-27 20:12:00 +01:00
|
|
|
|
va_list arg_ptr;
|
|
|
|
|
int value;
|
|
|
|
|
int any = 0;
|
2019-01-22 09:07:24 +01:00
|
|
|
|
|
2019-01-27 20:12:00 +01:00
|
|
|
|
tty_fprintf (NULL, "%s\n", text);
|
2019-01-22 09:07:24 +01:00
|
|
|
|
|
2019-01-27 20:12:00 +01:00
|
|
|
|
va_start (arg_ptr, text);
|
|
|
|
|
while ((value = va_arg (arg_ptr, int)))
|
|
|
|
|
{
|
|
|
|
|
if (!any)
|
|
|
|
|
tty_fprintf (NULL, "[Supported by: ");
|
|
|
|
|
tty_fprintf (NULL, "%s%s", any?", ":"", app_type_string (value));
|
|
|
|
|
any = 1;
|
|
|
|
|
}
|
|
|
|
|
if (any)
|
|
|
|
|
tty_fprintf (NULL, "]\n");
|
2019-01-22 09:07:24 +01:00
|
|
|
|
|
2019-01-27 20:12:00 +01:00
|
|
|
|
va_end (arg_ptr);
|
|
|
|
|
return 0;
|
2019-01-22 09:07:24 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2019-01-27 20:12:00 +01:00
|
|
|
|
/* Return the OpenPGP card manufacturer name. */
|
|
|
|
|
static const char *
|
|
|
|
|
get_manufacturer (unsigned int no)
|
2019-01-22 09:07:24 +01:00
|
|
|
|
{
|
2019-01-27 20:12:00 +01:00
|
|
|
|
/* Note: Make sure that there is no colon or linefeed in the string. */
|
|
|
|
|
switch (no)
|
|
|
|
|
{
|
|
|
|
|
case 0x0001: return "PPC Card Systems";
|
|
|
|
|
case 0x0002: return "Prism";
|
|
|
|
|
case 0x0003: return "OpenFortress";
|
|
|
|
|
case 0x0004: return "Wewid";
|
|
|
|
|
case 0x0005: return "ZeitControl";
|
|
|
|
|
case 0x0006: return "Yubico";
|
|
|
|
|
case 0x0007: return "OpenKMS";
|
|
|
|
|
case 0x0008: return "LogoEmail";
|
|
|
|
|
case 0x0009: return "Fidesmo";
|
|
|
|
|
case 0x000A: return "Dangerous Things";
|
|
|
|
|
|
|
|
|
|
case 0x002A: return "Magrathea";
|
|
|
|
|
case 0x0042: return "GnuPG e.V.";
|
|
|
|
|
|
|
|
|
|
case 0x1337: return "Warsaw Hackerspace";
|
|
|
|
|
case 0x2342: return "warpzone"; /* hackerspace Muenster. */
|
|
|
|
|
case 0x4354: return "Confidential Technologies"; /* cotech.de */
|
|
|
|
|
case 0x63AF: return "Trustica";
|
|
|
|
|
case 0xBD0E: return "Paranoidlabs";
|
|
|
|
|
case 0xF517: return "FSIJ";
|
|
|
|
|
|
|
|
|
|
/* 0x0000 and 0xFFFF are defined as test cards per spec,
|
|
|
|
|
* 0xFF00 to 0xFFFE are assigned for use with randomly created
|
|
|
|
|
* serial numbers. */
|
|
|
|
|
case 0x0000:
|
|
|
|
|
case 0xffff: return "test card";
|
|
|
|
|
default: return (no & 0xff00) == 0xff00? "unmanaged S/N range":"unknown";
|
|
|
|
|
}
|
2019-01-22 09:07:24 +01:00
|
|
|
|
}
|
|
|
|
|
|
2019-01-27 20:12:00 +01:00
|
|
|
|
/* Print an (OpenPGP) fingerprint. */
|
2019-01-22 09:07:24 +01:00
|
|
|
|
static void
|
2019-01-27 20:12:00 +01:00
|
|
|
|
print_shax_fpr (estream_t fp, const unsigned char *fpr, unsigned int fprlen)
|
2019-01-22 09:07:24 +01:00
|
|
|
|
{
|
2019-01-27 20:12:00 +01:00
|
|
|
|
int i;
|
2019-01-22 09:07:24 +01:00
|
|
|
|
|
2019-01-27 20:12:00 +01:00
|
|
|
|
if (fpr)
|
|
|
|
|
{
|
|
|
|
|
/* FIXME: Fix formatting for FPRLEN != 20 */
|
|
|
|
|
for (i=0; i < fprlen ; i+=2, fpr += 2 )
|
|
|
|
|
{
|
|
|
|
|
if (i == 10 )
|
|
|
|
|
tty_fprintf (fp, " ");
|
|
|
|
|
tty_fprintf (fp, " %02X%02X", *fpr, fpr[1]);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
tty_fprintf (fp, " [none]");
|
|
|
|
|
tty_fprintf (fp, "\n");
|
2019-01-22 09:07:24 +01:00
|
|
|
|
}
|
|
|
|
|
|
2019-01-27 20:12:00 +01:00
|
|
|
|
/* Print the keygrip GRP. */
|
2019-01-22 09:07:24 +01:00
|
|
|
|
static void
|
2019-01-27 20:12:00 +01:00
|
|
|
|
print_keygrip (estream_t fp, const unsigned char *grp)
|
2019-01-22 09:07:24 +01:00
|
|
|
|
{
|
2019-01-27 20:12:00 +01:00
|
|
|
|
int i;
|
|
|
|
|
|
|
|
|
|
for (i=0; i < 20 ; i++, grp++)
|
|
|
|
|
tty_fprintf (fp, "%02X", *grp);
|
|
|
|
|
tty_fprintf (fp, "\n");
|
2019-01-22 09:07:24 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2019-01-27 20:12:00 +01:00
|
|
|
|
/* Print a string but avoid printing control characters. */
|
2019-01-22 09:07:24 +01:00
|
|
|
|
static void
|
2019-01-27 20:12:00 +01:00
|
|
|
|
print_string (estream_t fp, const char *text, const char *name)
|
2019-01-22 09:07:24 +01:00
|
|
|
|
{
|
2019-01-27 20:12:00 +01:00
|
|
|
|
tty_fprintf (fp, "%s", text);
|
|
|
|
|
|
|
|
|
|
/* FIXME: tty_printf_utf8_string2 eats everything after and
|
|
|
|
|
including an @ - e.g. when printing an url. */
|
|
|
|
|
if (name && *name)
|
|
|
|
|
{
|
|
|
|
|
if (fp)
|
|
|
|
|
print_utf8_buffer2 (fp, name, strlen (name), '\n');
|
|
|
|
|
else
|
|
|
|
|
tty_print_utf8_string2 (NULL, name, strlen (name), 0);
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
tty_fprintf (fp, _("[not set]"));
|
|
|
|
|
tty_fprintf (fp, "\n");
|
2019-01-22 09:07:24 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2019-01-27 20:12:00 +01:00
|
|
|
|
/* Print an ISO formatted name or "[not set]". */
|
2019-01-22 09:07:24 +01:00
|
|
|
|
static void
|
2019-01-27 20:12:00 +01:00
|
|
|
|
print_isoname (estream_t fp, const char *name)
|
2019-01-22 09:07:24 +01:00
|
|
|
|
{
|
2019-01-27 20:12:00 +01:00
|
|
|
|
if (name && *name)
|
|
|
|
|
{
|
|
|
|
|
char *p, *given, *buf;
|
|
|
|
|
|
|
|
|
|
buf = xstrdup (name);
|
|
|
|
|
given = strstr (buf, "<<");
|
|
|
|
|
for (p=buf; *p; p++)
|
|
|
|
|
if (*p == '<')
|
|
|
|
|
*p = ' ';
|
|
|
|
|
if (given && given[2])
|
|
|
|
|
{
|
|
|
|
|
*given = 0;
|
|
|
|
|
given += 2;
|
|
|
|
|
if (fp)
|
|
|
|
|
print_utf8_buffer2 (fp, given, strlen (given), '\n');
|
|
|
|
|
else
|
|
|
|
|
tty_print_utf8_string2 (NULL, given, strlen (given), 0);
|
|
|
|
|
|
|
|
|
|
if (*buf)
|
|
|
|
|
tty_fprintf (fp, " ");
|
|
|
|
|
}
|
2019-01-22 09:07:24 +01:00
|
|
|
|
|
2019-01-27 20:12:00 +01:00
|
|
|
|
if (fp)
|
|
|
|
|
print_utf8_buffer2 (fp, buf, strlen (buf), '\n');
|
|
|
|
|
else
|
|
|
|
|
tty_print_utf8_string2 (NULL, buf, strlen (buf), 0);
|
2019-01-22 09:07:24 +01:00
|
|
|
|
|
2019-01-27 20:12:00 +01:00
|
|
|
|
xfree (buf);
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
tty_fprintf (fp, _("[not set]"));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
tty_fprintf (fp, "\n");
|
2019-01-22 09:07:24 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2019-01-29 08:48:53 +01:00
|
|
|
|
/* Return true if the buffer MEM of length memlen consists only of zeroes. */
|
2019-01-27 20:12:00 +01:00
|
|
|
|
static int
|
2019-01-29 08:48:53 +01:00
|
|
|
|
mem_is_zero (const char *mem, unsigned int memlen)
|
2019-01-22 09:07:24 +01:00
|
|
|
|
{
|
2019-01-27 20:12:00 +01:00
|
|
|
|
int i;
|
|
|
|
|
|
2019-01-29 08:48:53 +01:00
|
|
|
|
for (i=0; i < memlen && !mem[i]; i++)
|
2019-01-27 20:12:00 +01:00
|
|
|
|
;
|
2019-01-29 08:48:53 +01:00
|
|
|
|
return (i == memlen);
|
2019-01-22 09:07:24 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2019-01-29 08:48:53 +01:00
|
|
|
|
/* Return true if the buffer MEM or length MEMLEN consists only of 0xFF. */
|
2019-01-27 20:12:00 +01:00
|
|
|
|
static int
|
2019-01-29 08:48:53 +01:00
|
|
|
|
mem_is_ff (const char *mem, unsigned int memlen)
|
2019-01-22 09:07:24 +01:00
|
|
|
|
{
|
2019-01-27 20:12:00 +01:00
|
|
|
|
int i;
|
|
|
|
|
|
2019-01-29 08:48:53 +01:00
|
|
|
|
for (i=0; i < memlen && mem[i] == '\xff'; i++)
|
2019-01-27 20:12:00 +01:00
|
|
|
|
;
|
2019-01-29 08:48:53 +01:00
|
|
|
|
return (i == memlen);
|
2019-01-22 09:07:24 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2019-01-27 20:12:00 +01:00
|
|
|
|
|
2019-01-29 08:48:53 +01:00
|
|
|
|
/* Helper to list a single keyref. */
|
|
|
|
|
static void
|
|
|
|
|
list_one_kinfo (key_info_t kinfo, estream_t fp)
|
|
|
|
|
{
|
|
|
|
|
if (kinfo)
|
|
|
|
|
{
|
|
|
|
|
tty_fprintf (fp, " ");
|
|
|
|
|
if (mem_is_zero (kinfo->grip, sizeof kinfo->grip))
|
|
|
|
|
tty_fprintf (fp, "[none]\n");
|
|
|
|
|
else
|
|
|
|
|
print_keygrip (fp, kinfo->grip);
|
|
|
|
|
|
|
|
|
|
if (kinfo->fprlen && kinfo->created)
|
|
|
|
|
{
|
|
|
|
|
tty_fprintf (fp, " fingerprint :");
|
|
|
|
|
print_shax_fpr (fp, kinfo->fpr, kinfo->fprlen);
|
|
|
|
|
tty_fprintf (fp, " created ....: %s\n",
|
|
|
|
|
isotimestamp (kinfo->created));
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
tty_fprintf (fp, " [none]\n");
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* List all keyinfo in INFO using the list of LABELS. */
|
|
|
|
|
static void
|
|
|
|
|
list_all_kinfo (card_info_t info, keyinfolabel_t labels, estream_t fp)
|
|
|
|
|
{
|
|
|
|
|
key_info_t kinfo;
|
|
|
|
|
int idx, i;
|
|
|
|
|
|
|
|
|
|
/* Print the keyinfo. We first print those we known and then all
|
|
|
|
|
* remaining item. */
|
|
|
|
|
for (kinfo = info->kinfo; kinfo; kinfo = kinfo->next)
|
|
|
|
|
kinfo->xflag = 0;
|
|
|
|
|
if (labels)
|
|
|
|
|
{
|
|
|
|
|
for (idx=0; labels[idx].label; idx++)
|
|
|
|
|
{
|
|
|
|
|
tty_fprintf (fp, "%s", labels[idx].label);
|
|
|
|
|
kinfo = find_kinfo (info, labels[idx].keyref);
|
|
|
|
|
list_one_kinfo (kinfo, fp);
|
|
|
|
|
if (kinfo)
|
|
|
|
|
kinfo->xflag = 1;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
for (kinfo = info->kinfo; kinfo; kinfo = kinfo->next)
|
|
|
|
|
{
|
|
|
|
|
if (kinfo->xflag)
|
|
|
|
|
continue;
|
|
|
|
|
tty_fprintf (fp, "Key %s ", kinfo->keyref);
|
|
|
|
|
for (i=5+strlen (kinfo->keyref); i < 18; i++)
|
|
|
|
|
tty_fprintf (fp, ".");
|
|
|
|
|
tty_fprintf (fp, ":");
|
|
|
|
|
list_one_kinfo (kinfo, fp);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2019-01-27 20:12:00 +01:00
|
|
|
|
/* List OpenPGP card specific data. */
|
2019-01-22 09:07:24 +01:00
|
|
|
|
static void
|
2019-01-27 20:12:00 +01:00
|
|
|
|
list_openpgp (card_info_t info, estream_t fp)
|
2019-01-22 09:07:24 +01:00
|
|
|
|
{
|
2019-01-29 08:48:53 +01:00
|
|
|
|
static struct keyinfolabel_s keyinfolabels[] = {
|
|
|
|
|
{ "Signature key ....:", "OPENPGP.1" },
|
|
|
|
|
{ "Encryption key....:", "OPENPGP.2" },
|
|
|
|
|
{ "Authentication key:", "OPENPGP.3" },
|
|
|
|
|
{ NULL, NULL }
|
|
|
|
|
};
|
2019-01-27 20:12:00 +01:00
|
|
|
|
int i;
|
2019-01-22 09:07:24 +01:00
|
|
|
|
|
2019-01-27 20:12:00 +01:00
|
|
|
|
if (!info->serialno
|
|
|
|
|
|| strncmp (info->serialno, "D27600012401", 12)
|
|
|
|
|
|| strlen (info->serialno) != 32 )
|
|
|
|
|
{
|
|
|
|
|
tty_fprintf (fp, "invalid OpenPGP card\n");
|
|
|
|
|
return;
|
|
|
|
|
}
|
2019-01-22 09:07:24 +01:00
|
|
|
|
|
2019-01-27 20:12:00 +01:00
|
|
|
|
tty_fprintf (fp, "Version ..........: %.1s%c.%.1s%c\n",
|
|
|
|
|
info->serialno[12] == '0'?"":info->serialno+12,
|
|
|
|
|
info->serialno[13],
|
|
|
|
|
info->serialno[14] == '0'?"":info->serialno+14,
|
|
|
|
|
info->serialno[15]);
|
|
|
|
|
tty_fprintf (fp, "Manufacturer .....: %s\n",
|
|
|
|
|
get_manufacturer (xtoi_2(info->serialno+16)*256
|
|
|
|
|
+ xtoi_2 (info->serialno+18)));
|
|
|
|
|
tty_fprintf (fp, "Name of cardholder: ");
|
|
|
|
|
print_isoname (fp, info->disp_name);
|
|
|
|
|
|
|
|
|
|
print_string (fp, "Language prefs ...: ", info->disp_lang);
|
|
|
|
|
tty_fprintf (fp, "Salutation .......: %s\n",
|
|
|
|
|
info->disp_sex == 1? _("Mr."):
|
|
|
|
|
info->disp_sex == 2? _("Mrs.") : "");
|
|
|
|
|
print_string (fp, "URL of public key : ", info->pubkey_url);
|
|
|
|
|
print_string (fp, "Login data .......: ", info->login_data);
|
|
|
|
|
if (info->private_do[0])
|
|
|
|
|
print_string (fp, "Private DO 1 .....: ", info->private_do[0]);
|
|
|
|
|
if (info->private_do[1])
|
|
|
|
|
print_string (fp, "Private DO 2 .....: ", info->private_do[1]);
|
|
|
|
|
if (info->private_do[2])
|
|
|
|
|
print_string (fp, "Private DO 3 .....: ", info->private_do[2]);
|
|
|
|
|
if (info->private_do[3])
|
|
|
|
|
print_string (fp, "Private DO 4 .....: ", info->private_do[3]);
|
|
|
|
|
if (info->cafpr1len)
|
|
|
|
|
{
|
|
|
|
|
tty_fprintf (fp, "CA fingerprint %d .:", 1);
|
|
|
|
|
print_shax_fpr (fp, info->cafpr1, info->cafpr1len);
|
|
|
|
|
}
|
|
|
|
|
if (info->cafpr2len)
|
|
|
|
|
{
|
|
|
|
|
tty_fprintf (fp, "CA fingerprint %d .:", 2);
|
|
|
|
|
print_shax_fpr (fp, info->cafpr2, info->cafpr2len);
|
|
|
|
|
}
|
|
|
|
|
if (info->cafpr3len)
|
|
|
|
|
{
|
|
|
|
|
tty_fprintf (fp, "CA fingerprint %d .:", 3);
|
|
|
|
|
print_shax_fpr (fp, info->cafpr3, info->cafpr3len);
|
|
|
|
|
}
|
|
|
|
|
tty_fprintf (fp, "Signature PIN ....: %s\n",
|
|
|
|
|
info->chv1_cached? _("not forced"): _("forced"));
|
|
|
|
|
if (info->key_attr[0].algo)
|
|
|
|
|
{
|
|
|
|
|
tty_fprintf (fp, "Key attributes ...:");
|
|
|
|
|
for (i=0; i < DIM (info->key_attr); i++)
|
|
|
|
|
if (info->key_attr[i].algo == PUBKEY_ALGO_RSA)
|
|
|
|
|
tty_fprintf (fp, " rsa%u", info->key_attr[i].nbits);
|
|
|
|
|
else if (info->key_attr[i].algo == PUBKEY_ALGO_ECDH
|
|
|
|
|
|| info->key_attr[i].algo == PUBKEY_ALGO_ECDSA
|
|
|
|
|
|| info->key_attr[i].algo == PUBKEY_ALGO_EDDSA)
|
|
|
|
|
{
|
|
|
|
|
const char *curve_for_print = "?";
|
|
|
|
|
const char *oid;
|
|
|
|
|
|
|
|
|
|
if (info->key_attr[i].curve
|
|
|
|
|
&& (oid = openpgp_curve_to_oid (info->key_attr[i].curve, NULL)))
|
|
|
|
|
curve_for_print = openpgp_oid_to_curve (oid, 0);
|
|
|
|
|
tty_fprintf (fp, " %s", curve_for_print);
|
|
|
|
|
}
|
|
|
|
|
tty_fprintf (fp, "\n");
|
|
|
|
|
}
|
|
|
|
|
tty_fprintf (fp, "Max. PIN lengths .: %d %d %d\n",
|
|
|
|
|
info->chvmaxlen[0], info->chvmaxlen[1], info->chvmaxlen[2]);
|
|
|
|
|
tty_fprintf (fp, "PIN retry counter : %d %d %d\n",
|
2019-01-29 09:30:15 +01:00
|
|
|
|
info->chvinfo[0], info->chvinfo[1], info->chvinfo[2]);
|
2019-01-27 20:12:00 +01:00
|
|
|
|
tty_fprintf (fp, "Signature counter : %lu\n", info->sig_counter);
|
|
|
|
|
if (info->extcap.kdf)
|
|
|
|
|
{
|
|
|
|
|
tty_fprintf (fp, "KDF setting ......: %s\n",
|
|
|
|
|
info->kdf_do_enabled ? "on" : "off");
|
|
|
|
|
}
|
|
|
|
|
if (info->extcap.bt)
|
|
|
|
|
{
|
|
|
|
|
tty_fprintf (fp, "UIF setting ......: Sign=%s Decrypt=%s Auth=%s\n",
|
|
|
|
|
info->uif[0] ? "on" : "off", info->uif[1] ? "on" : "off",
|
|
|
|
|
info->uif[2] ? "on" : "off");
|
|
|
|
|
}
|
2019-01-29 08:48:53 +01:00
|
|
|
|
|
|
|
|
|
list_all_kinfo (info, keyinfolabels, fp);
|
2019-01-27 20:12:00 +01:00
|
|
|
|
|
|
|
|
|
/* tty_fprintf (fp, "General key info->.: "); */
|
|
|
|
|
/* thefpr = (info->fpr1len? info->fpr1 : info->fpr2len? info->fpr2 : */
|
|
|
|
|
/* info->fpr3len? info->fpr3 : NULL); */
|
|
|
|
|
/* thefprlen = (info->fpr1len? info->fpr1len : info->fpr2len? info->fpr2len : */
|
|
|
|
|
/* info->fpr3len? info->fpr3len : 0); */
|
|
|
|
|
/* If the fingerprint is all 0xff, the key has no associated
|
|
|
|
|
OpenPGP certificate. */
|
2019-01-29 08:48:53 +01:00
|
|
|
|
/* if ( thefpr && !mem_is_ff (thefpr, thefprlen) */
|
2019-01-27 20:12:00 +01:00
|
|
|
|
/* && !get_pubkey_byfprint (ctrl, pk, &keyblock, thefpr, thefprlen)) */
|
|
|
|
|
/* { */
|
|
|
|
|
/* print_pubkey_info (ctrl, fp, pk); */
|
|
|
|
|
/* if (keyblock) */
|
|
|
|
|
/* print_card_key_info (fp, keyblock); */
|
|
|
|
|
/* } */
|
|
|
|
|
/* else */
|
|
|
|
|
/* tty_fprintf (fp, "[none]\n"); */
|
2019-01-22 09:07:24 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2019-01-29 09:30:15 +01:00
|
|
|
|
/* List PIV card specific data. */
|
|
|
|
|
static void
|
|
|
|
|
list_piv (card_info_t info, estream_t fp)
|
|
|
|
|
{
|
|
|
|
|
static struct keyinfolabel_s keyinfolabels[] = {
|
2019-01-29 13:28:10 +01:00
|
|
|
|
{ "PIV authentication:", "PIV.9A" },
|
|
|
|
|
{ "Card authenticat. :", "PIV.9E" },
|
|
|
|
|
{ "Digital signature :", "PIV.9C" },
|
|
|
|
|
{ "Key management ...:", "PIV.9D" },
|
2019-01-29 09:30:15 +01:00
|
|
|
|
{ NULL, NULL }
|
|
|
|
|
};
|
|
|
|
|
const char *s;
|
|
|
|
|
int i;
|
|
|
|
|
|
2019-01-29 13:28:10 +01:00
|
|
|
|
if (info->chvusage[0] || info->chvusage[1])
|
|
|
|
|
{
|
|
|
|
|
tty_fprintf (fp, "PIN usage policy .:");
|
|
|
|
|
if ((info->chvusage[0] & 0x40))
|
|
|
|
|
tty_fprintf (fp, " app-pin");
|
|
|
|
|
if ((info->chvusage[0] & 0x20))
|
|
|
|
|
tty_fprintf (fp, " global-pin");
|
|
|
|
|
if ((info->chvusage[0] & 0x10))
|
|
|
|
|
tty_fprintf (fp, " occ");
|
|
|
|
|
if ((info->chvusage[0] & 0x08))
|
|
|
|
|
tty_fprintf (fp, " vci");
|
|
|
|
|
if ((info->chvusage[0] & 0x08) && !(info->chvusage[0] & 0x04))
|
|
|
|
|
tty_fprintf (fp, " pairing");
|
|
|
|
|
|
|
|
|
|
if (info->chvusage[1] == 0x10)
|
|
|
|
|
tty_fprintf (fp, " primary:card");
|
|
|
|
|
else if (info->chvusage[1] == 0x20)
|
|
|
|
|
tty_fprintf (fp, " primary:global");
|
|
|
|
|
|
|
|
|
|
tty_fprintf (fp, "\n");
|
|
|
|
|
}
|
|
|
|
|
|
2019-01-29 09:30:15 +01:00
|
|
|
|
tty_fprintf (fp, "PIN retry counter :");
|
|
|
|
|
for (i=0; i < DIM (info->chvinfo); i++)
|
|
|
|
|
{
|
|
|
|
|
if (info->chvinfo[i] > 0)
|
|
|
|
|
tty_fprintf (fp, " %d", info->chvinfo[i]);
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
switch (info->chvinfo[i])
|
|
|
|
|
{
|
|
|
|
|
case -1: s = "[error]"; break;
|
|
|
|
|
case -2: s = "-"; break; /* No such PIN */
|
|
|
|
|
case -3: s = "[blocked]"; break;
|
|
|
|
|
case -5: s = "[verified]"; break;
|
|
|
|
|
default: s = "[?]"; break;
|
|
|
|
|
}
|
|
|
|
|
tty_fprintf (fp, " %s", s);
|
|
|
|
|
}
|
|
|
|
|
}
|
2019-01-29 13:28:10 +01:00
|
|
|
|
tty_fprintf (fp, "\n");
|
2019-01-29 09:30:15 +01:00
|
|
|
|
list_all_kinfo (info, keyinfolabels, fp);
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2019-01-27 20:12:00 +01:00
|
|
|
|
/* Print all available information about the current card. */
|
2019-01-22 09:07:24 +01:00
|
|
|
|
static void
|
2019-01-27 20:12:00 +01:00
|
|
|
|
list_card (card_info_t info)
|
2019-01-22 09:07:24 +01:00
|
|
|
|
{
|
2019-01-27 20:12:00 +01:00
|
|
|
|
estream_t fp = NULL;
|
|
|
|
|
|
|
|
|
|
tty_fprintf (fp, "Reader ...........: %s\n",
|
|
|
|
|
info->reader? info->reader : "[none]");
|
2019-01-29 13:28:10 +01:00
|
|
|
|
if (info->cardtype)
|
|
|
|
|
tty_fprintf (fp, "Card type ........: %s\n", info->cardtype);
|
2019-01-27 20:12:00 +01:00
|
|
|
|
tty_fprintf (fp, "Serial number ....: %s\n",
|
|
|
|
|
info->serialno? info->serialno : "[none]");
|
2019-01-29 13:28:10 +01:00
|
|
|
|
tty_fprintf (fp, "Application type .: %s%s%s%s\n",
|
2019-01-27 20:12:00 +01:00
|
|
|
|
app_type_string (info->apptype),
|
|
|
|
|
info->apptype == APP_TYPE_UNKNOWN && info->apptypestr? "(":"",
|
|
|
|
|
info->apptype == APP_TYPE_UNKNOWN && info->apptypestr
|
|
|
|
|
? info->apptypestr:"",
|
|
|
|
|
info->apptype == APP_TYPE_UNKNOWN && info->apptypestr? ")":"");
|
|
|
|
|
if (info->serialno && info->dispserialno
|
|
|
|
|
&& strcmp (info->serialno, info->dispserialno))
|
|
|
|
|
tty_fprintf (fp, "Displayed S/N ....: %s\n", info->dispserialno);
|
|
|
|
|
|
|
|
|
|
switch (info->apptype)
|
|
|
|
|
{
|
|
|
|
|
case APP_TYPE_OPENPGP: list_openpgp (info, fp); break;
|
2019-01-29 09:30:15 +01:00
|
|
|
|
case APP_TYPE_PIV: list_piv (info, fp); break;
|
2019-01-27 20:12:00 +01:00
|
|
|
|
default: break;
|
|
|
|
|
}
|
2019-01-22 09:07:24 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2019-01-27 20:12:00 +01:00
|
|
|
|
|
|
|
|
|
/* The VERIFY command. */
|
|
|
|
|
static gpg_error_t
|
|
|
|
|
cmd_verify (card_info_t info, char *argstr)
|
2019-01-22 09:07:24 +01:00
|
|
|
|
{
|
2019-01-27 20:12:00 +01:00
|
|
|
|
gpg_error_t err;
|
2019-01-22 09:07:24 +01:00
|
|
|
|
|
2019-01-27 20:12:00 +01:00
|
|
|
|
if (!info)
|
|
|
|
|
return print_help ("verify [chvid]", 0);
|
2019-01-22 09:07:24 +01:00
|
|
|
|
|
2019-01-27 20:12:00 +01:00
|
|
|
|
if (info->apptype == APP_TYPE_OPENPGP)
|
|
|
|
|
err = scd_checkpin (info->serialno);
|
|
|
|
|
else
|
|
|
|
|
err = scd_checkpin (argstr);
|
2019-01-22 09:07:24 +01:00
|
|
|
|
|
2019-01-27 20:12:00 +01:00
|
|
|
|
if (err)
|
|
|
|
|
log_error ("verify failed: %s <%s>\n",
|
|
|
|
|
gpg_strerror (err), gpg_strsource (err));
|
|
|
|
|
return err;
|
|
|
|
|
}
|
2019-01-22 09:07:24 +01:00
|
|
|
|
|
|
|
|
|
|
2019-01-27 20:12:00 +01:00
|
|
|
|
/* Helper for cmd_name to qyery a part of name. */
|
|
|
|
|
static char *
|
|
|
|
|
ask_one_name (const char *prompt)
|
2019-01-22 09:07:24 +01:00
|
|
|
|
{
|
2019-01-27 20:12:00 +01:00
|
|
|
|
char *name;
|
2019-01-22 09:07:24 +01:00
|
|
|
|
int i;
|
|
|
|
|
|
|
|
|
|
for (;;)
|
|
|
|
|
{
|
2019-01-27 20:12:00 +01:00
|
|
|
|
name = tty_get (prompt);
|
|
|
|
|
trim_spaces (name);
|
|
|
|
|
tty_kill_prompt ();
|
|
|
|
|
if (!*name || *name == CONTROL_D)
|
2019-01-22 09:07:24 +01:00
|
|
|
|
{
|
2019-01-27 20:12:00 +01:00
|
|
|
|
if (*name == CONTROL_D)
|
|
|
|
|
tty_fprintf (NULL, "\n");
|
|
|
|
|
xfree (name);
|
|
|
|
|
return NULL;
|
|
|
|
|
}
|
|
|
|
|
for (i=0; name[i] && name[i] >= ' ' && name[i] <= 126; i++)
|
|
|
|
|
;
|
|
|
|
|
|
|
|
|
|
/* The name must be in Latin-1 and not UTF-8 - lacking the code
|
|
|
|
|
* to ensure this we restrict it to ASCII. */
|
|
|
|
|
if (name[i])
|
|
|
|
|
tty_printf (_("Error: Only plain ASCII is currently allowed.\n"));
|
|
|
|
|
else if (strchr (name, '<'))
|
|
|
|
|
tty_printf (_("Error: The \"<\" character may not be used.\n"));
|
|
|
|
|
else if (strstr (name, " "))
|
|
|
|
|
tty_printf (_("Error: Double spaces are not allowed.\n"));
|
|
|
|
|
else
|
|
|
|
|
return name;
|
|
|
|
|
xfree (name);
|
|
|
|
|
}
|
|
|
|
|
}
|
2019-01-22 09:07:24 +01:00
|
|
|
|
|
|
|
|
|
|
2019-01-27 20:12:00 +01:00
|
|
|
|
/* The NAME command. */
|
|
|
|
|
static gpg_error_t
|
|
|
|
|
cmd_name (card_info_t info, const char *argstr)
|
|
|
|
|
{
|
|
|
|
|
gpg_error_t err;
|
|
|
|
|
char *surname, *givenname;
|
|
|
|
|
char *isoname, *p;
|
2019-01-22 09:07:24 +01:00
|
|
|
|
|
2019-01-27 20:12:00 +01:00
|
|
|
|
if (!info)
|
|
|
|
|
return print_help
|
|
|
|
|
("name [--clear]\n\n"
|
|
|
|
|
"Set the name field of an OpenPGP card. With --clear the stored\n"
|
|
|
|
|
"name is cleared off the card.", APP_TYPE_OPENPGP, APP_TYPE_NKS, 0);
|
2019-01-22 09:07:24 +01:00
|
|
|
|
|
2019-01-27 20:12:00 +01:00
|
|
|
|
if (info->apptype != APP_TYPE_OPENPGP)
|
|
|
|
|
{
|
|
|
|
|
log_info ("Note: This is an OpenPGP only command.\n");
|
|
|
|
|
return gpg_error (GPG_ERR_NOT_SUPPORTED);
|
|
|
|
|
}
|
2019-01-22 09:07:24 +01:00
|
|
|
|
|
2019-01-27 20:12:00 +01:00
|
|
|
|
again:
|
|
|
|
|
if (!strcmp (argstr, "--clear"))
|
|
|
|
|
isoname = xstrdup (" "); /* No real way to clear; set to space instead. */
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
surname = ask_one_name (_("Cardholder's surname: "));
|
|
|
|
|
givenname = ask_one_name (_("Cardholder's given name: "));
|
|
|
|
|
if (!surname || !givenname || (!*surname && !*givenname))
|
|
|
|
|
{
|
|
|
|
|
xfree (surname);
|
|
|
|
|
xfree (givenname);
|
|
|
|
|
return gpg_error (GPG_ERR_CANCELED);
|
2019-01-22 09:07:24 +01:00
|
|
|
|
}
|
|
|
|
|
|
2019-01-27 20:12:00 +01:00
|
|
|
|
isoname = xstrconcat (surname, "<<", givenname, NULL);
|
|
|
|
|
xfree (surname);
|
|
|
|
|
xfree (givenname);
|
|
|
|
|
for (p=isoname; *p; p++)
|
|
|
|
|
if (*p == ' ')
|
|
|
|
|
*p = '<';
|
|
|
|
|
|
|
|
|
|
if (strlen (isoname) > 39 )
|
2019-01-22 09:07:24 +01:00
|
|
|
|
{
|
2019-01-27 20:12:00 +01:00
|
|
|
|
log_info (_("Error: Combined name too long "
|
|
|
|
|
"(limit is %d characters).\n"), 39);
|
|
|
|
|
xfree (isoname);
|
|
|
|
|
goto again;
|
|
|
|
|
}
|
|
|
|
|
}
|
2019-01-22 09:07:24 +01:00
|
|
|
|
|
2019-01-27 20:12:00 +01:00
|
|
|
|
err = scd_setattr ("DISP-NAME", isoname, strlen (isoname));
|
2019-01-22 09:07:24 +01:00
|
|
|
|
|
2019-01-27 20:12:00 +01:00
|
|
|
|
xfree (isoname);
|
|
|
|
|
return err;
|
|
|
|
|
}
|
2019-01-22 09:07:24 +01:00
|
|
|
|
|
2019-01-27 20:12:00 +01:00
|
|
|
|
|
|
|
|
|
static gpg_error_t
|
|
|
|
|
cmd_url (card_info_t info, const char *argstr)
|
|
|
|
|
{
|
|
|
|
|
gpg_error_t err;
|
|
|
|
|
char *url;
|
|
|
|
|
|
|
|
|
|
if (!info)
|
|
|
|
|
return print_help
|
|
|
|
|
("URL [--clear]\n\n"
|
|
|
|
|
"Set the URL data object. That data object can be used by\n"
|
|
|
|
|
"the FETCH command to retrieve the full public key. The\n"
|
|
|
|
|
"option --clear deletes the content of that data object.",
|
|
|
|
|
APP_TYPE_OPENPGP, 0);
|
|
|
|
|
|
|
|
|
|
if (info->apptype != APP_TYPE_OPENPGP)
|
|
|
|
|
{
|
|
|
|
|
log_info ("Note: This is an OpenPGP only command.\n");
|
|
|
|
|
return gpg_error (GPG_ERR_NOT_SUPPORTED);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (!strcmp (argstr, "--clear"))
|
|
|
|
|
url = xstrdup (" "); /* No real way to clear; set to space instead. */
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
url = tty_get (_("URL to retrieve public key: "));
|
|
|
|
|
trim_spaces (url);
|
|
|
|
|
tty_kill_prompt ();
|
|
|
|
|
if (!*url || *url == CONTROL_D)
|
|
|
|
|
{
|
|
|
|
|
err = gpg_error (GPG_ERR_CANCELED);
|
|
|
|
|
goto leave;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
err = scd_setattr ("PUBKEY-URL", url, strlen (url));
|
|
|
|
|
|
|
|
|
|
leave:
|
|
|
|
|
xfree (url);
|
|
|
|
|
return err;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Fetch the key from the URL given on the card or try to get it from
|
|
|
|
|
* the default keyserver. */
|
|
|
|
|
static gpg_error_t
|
|
|
|
|
cmd_fetch (card_info_t info)
|
|
|
|
|
{
|
|
|
|
|
gpg_error_t err;
|
2019-01-29 08:48:53 +01:00
|
|
|
|
key_info_t kinfo;
|
2019-01-27 20:12:00 +01:00
|
|
|
|
|
|
|
|
|
if (!info)
|
|
|
|
|
return print_help
|
|
|
|
|
("FETCH\n\n"
|
|
|
|
|
"Retrieve a key using the URL data object or if that is missing\n"
|
|
|
|
|
"using the fingerprint.", APP_TYPE_OPENPGP, 0);
|
|
|
|
|
|
|
|
|
|
if (info->pubkey_url && *info->pubkey_url)
|
|
|
|
|
{
|
|
|
|
|
/* strlist_t sl = NULL; */
|
|
|
|
|
|
|
|
|
|
/* add_to_strlist (&sl, info.pubkey_url); */
|
|
|
|
|
/* err = keyserver_fetch (ctrl, sl, KEYORG_URL); */
|
|
|
|
|
/* free_strlist (sl); */
|
|
|
|
|
err = gpg_error (GPG_ERR_NOT_IMPLEMENTED); /* FIXME */
|
|
|
|
|
}
|
2019-01-29 08:48:53 +01:00
|
|
|
|
else if ((kinfo = find_kinfo (info, "OPENPGP.1")) && kinfo->fprlen)
|
2019-01-27 20:12:00 +01:00
|
|
|
|
{
|
|
|
|
|
/* rc = keyserver_import_fprint (ctrl, info.fpr1, info.fpr1len, */
|
|
|
|
|
/* opt.keyserver, 0); */
|
|
|
|
|
err = gpg_error (GPG_ERR_NOT_IMPLEMENTED); /* FIXME */
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
err = gpg_error (GPG_ERR_NO_DATA);
|
|
|
|
|
|
|
|
|
|
return err;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
static gpg_error_t
|
|
|
|
|
cmd_login (card_info_t info, char *argstr)
|
|
|
|
|
{
|
|
|
|
|
gpg_error_t err;
|
|
|
|
|
char *data;
|
|
|
|
|
size_t datalen;
|
|
|
|
|
|
|
|
|
|
if (!info)
|
|
|
|
|
return print_help
|
|
|
|
|
("LOGIN [--clear] [< FILE]\n\n"
|
|
|
|
|
"Set the login data object. If FILE is given the data is\n"
|
|
|
|
|
"is read from that file. This allows for binary data.\n"
|
|
|
|
|
"The option --clear deletes the login data.",
|
|
|
|
|
APP_TYPE_OPENPGP, 0);
|
|
|
|
|
|
|
|
|
|
if (!strcmp (argstr, "--clear"))
|
|
|
|
|
{
|
|
|
|
|
data = xstrdup (" "); /* kludge. */
|
|
|
|
|
datalen = 1;
|
|
|
|
|
}
|
|
|
|
|
else if (*argstr == '<') /* Read it from a file */
|
|
|
|
|
{
|
|
|
|
|
for (argstr++; spacep (argstr); argstr++)
|
|
|
|
|
;
|
|
|
|
|
err = get_data_from_file (argstr, &data, &datalen);
|
|
|
|
|
if (err)
|
|
|
|
|
goto leave;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
data = tty_get (_("Login data (account name): "));
|
|
|
|
|
trim_spaces (data);
|
|
|
|
|
tty_kill_prompt ();
|
|
|
|
|
if (!*data || *data == CONTROL_D)
|
|
|
|
|
{
|
|
|
|
|
err = gpg_error (GPG_ERR_CANCELED);
|
|
|
|
|
goto leave;
|
|
|
|
|
}
|
|
|
|
|
datalen = strlen (data);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
err = scd_setattr ("LOGIN-DATA", data, datalen);
|
|
|
|
|
|
|
|
|
|
leave:
|
|
|
|
|
xfree (data);
|
|
|
|
|
return err;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
static gpg_error_t
|
|
|
|
|
cmd_lang (card_info_t info, const char *argstr)
|
|
|
|
|
{
|
|
|
|
|
gpg_error_t err;
|
|
|
|
|
char *data, *p;
|
|
|
|
|
|
|
|
|
|
if (!info)
|
|
|
|
|
return print_help
|
|
|
|
|
("LANG [--clear]\n\n"
|
|
|
|
|
"Change the language info for the card. This info can be used\n"
|
|
|
|
|
"by applications for a personalized greeting. Up to 4 two-digit\n"
|
|
|
|
|
"language identifiers can be entered as a preference. The option\n"
|
|
|
|
|
"--clear removes all identifiers. GnuPG does not use this info.",
|
|
|
|
|
APP_TYPE_OPENPGP, 0);
|
|
|
|
|
|
|
|
|
|
if (!strcmp (argstr, "--clear"))
|
|
|
|
|
data = xstrdup (" "); /* Note that we need two spaces here. */
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
again:
|
|
|
|
|
data = tty_get (_("Language preferences: "));
|
|
|
|
|
trim_spaces (data);
|
|
|
|
|
tty_kill_prompt ();
|
|
|
|
|
if (!*data || *data == CONTROL_D)
|
|
|
|
|
{
|
|
|
|
|
err = gpg_error (GPG_ERR_CANCELED);
|
|
|
|
|
goto leave;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (strlen (data) > 8 || (strlen (data) & 1))
|
|
|
|
|
{
|
|
|
|
|
log_info (_("Error: invalid length of preference string.\n"));
|
|
|
|
|
xfree (data);
|
|
|
|
|
goto again;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
for (p=data; *p && *p >= 'a' && *p <= 'z'; p++)
|
|
|
|
|
;
|
|
|
|
|
if (*p)
|
|
|
|
|
{
|
|
|
|
|
log_info (_("Error: invalid characters in preference string.\n"));
|
|
|
|
|
xfree (data);
|
|
|
|
|
goto again;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
err = scd_setattr ("DISP-LANG", data, strlen (data));
|
|
|
|
|
|
|
|
|
|
leave:
|
|
|
|
|
xfree (data);
|
|
|
|
|
return err;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
static gpg_error_t
|
|
|
|
|
cmd_salut (card_info_t info, const char *argstr)
|
|
|
|
|
{
|
|
|
|
|
gpg_error_t err;
|
|
|
|
|
char *data = NULL;
|
|
|
|
|
const char *str;
|
|
|
|
|
|
|
|
|
|
if (!info)
|
|
|
|
|
return print_help
|
|
|
|
|
("SALUT [--clear]\n\n"
|
|
|
|
|
"Change the salutation info for the card. This info can be used\n"
|
|
|
|
|
"by applications for a personalized greeting. The option --clear\n"
|
|
|
|
|
"removes this data object. GnuPG does not use this info.",
|
|
|
|
|
APP_TYPE_OPENPGP, 0);
|
|
|
|
|
|
|
|
|
|
again:
|
|
|
|
|
if (!strcmp (argstr, "--clear"))
|
|
|
|
|
str = "9";
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
data = tty_get (_("Salutation (M = Mr., F = Mrs., or space): "));
|
|
|
|
|
trim_spaces (data);
|
|
|
|
|
tty_kill_prompt ();
|
|
|
|
|
if (*data == CONTROL_D)
|
|
|
|
|
{
|
|
|
|
|
err = gpg_error (GPG_ERR_CANCELED);
|
|
|
|
|
goto leave;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (!*data)
|
|
|
|
|
str = "9";
|
|
|
|
|
else if ((*data == 'M' || *data == 'm') && !data[1])
|
|
|
|
|
str = "1";
|
|
|
|
|
else if ((*data == 'F' || *data == 'f') && !data[1])
|
|
|
|
|
str = "2";
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
tty_printf (_("Error: invalid response.\n"));
|
|
|
|
|
xfree (data);
|
|
|
|
|
goto again;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
err = scd_setattr ("DISP-SEX", str, 1);
|
|
|
|
|
leave:
|
|
|
|
|
xfree (data);
|
|
|
|
|
return err;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
static gpg_error_t
|
|
|
|
|
cmd_cafpr (card_info_t info, char *argstr)
|
|
|
|
|
{
|
|
|
|
|
gpg_error_t err;
|
|
|
|
|
char *data = NULL;
|
|
|
|
|
const char *s;
|
|
|
|
|
int i, c;
|
|
|
|
|
unsigned char fpr[32];
|
|
|
|
|
int fprlen;
|
|
|
|
|
int fprno;
|
|
|
|
|
int opt_clear = 0;
|
|
|
|
|
|
|
|
|
|
if (!info)
|
|
|
|
|
return print_help
|
|
|
|
|
("CAFPR [--clear] N\n\n"
|
|
|
|
|
"Change the CA fingerprint number N. N must be in the\n"
|
|
|
|
|
"range 1 to 3. The option --clear clears the specified\n"
|
|
|
|
|
"CA fingerprint N or all of them if N is 0 or not given.",
|
|
|
|
|
APP_TYPE_OPENPGP, 0);
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
opt_clear = has_leading_option (argstr, "--clear");
|
|
|
|
|
argstr = skip_options (argstr);
|
|
|
|
|
|
|
|
|
|
if (digitp (argstr))
|
|
|
|
|
{
|
|
|
|
|
fprno = atoi (argstr);
|
|
|
|
|
while (digitp (argstr))
|
|
|
|
|
argstr++;
|
|
|
|
|
while (spacep (argstr))
|
|
|
|
|
argstr++;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
fprno = 0;
|
|
|
|
|
|
|
|
|
|
if (opt_clear && !fprno)
|
|
|
|
|
; /* Okay: clear all fprs. */
|
|
|
|
|
else if (fprno < 1 || fprno > 3)
|
|
|
|
|
{
|
|
|
|
|
err = gpg_error (GPG_ERR_INV_ARG);
|
|
|
|
|
goto leave;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
again:
|
|
|
|
|
if (opt_clear)
|
|
|
|
|
{
|
|
|
|
|
memset (fpr, 0, 20);
|
|
|
|
|
fprlen = 20;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
xfree (data);
|
|
|
|
|
data = tty_get (_("CA fingerprint: "));
|
|
|
|
|
trim_spaces (data);
|
|
|
|
|
tty_kill_prompt ();
|
|
|
|
|
if (!*data || *data == CONTROL_D)
|
|
|
|
|
{
|
|
|
|
|
err = gpg_error (GPG_ERR_CANCELED);
|
|
|
|
|
goto leave;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
for (i=0, s=data; i < sizeof fpr && *s; )
|
|
|
|
|
{
|
|
|
|
|
while (spacep(s))
|
|
|
|
|
s++;
|
|
|
|
|
if (*s == ':')
|
|
|
|
|
s++;
|
|
|
|
|
while (spacep(s))
|
|
|
|
|
s++;
|
|
|
|
|
c = hextobyte (s);
|
|
|
|
|
if (c == -1)
|
|
|
|
|
break;
|
|
|
|
|
fpr[i++] = c;
|
|
|
|
|
s += 2;
|
|
|
|
|
}
|
|
|
|
|
fprlen = i;
|
|
|
|
|
if ((fprlen != 20 && fprlen != 32) || *s)
|
|
|
|
|
{
|
|
|
|
|
log_error (_("Error: invalid formatted fingerprint.\n"));
|
|
|
|
|
goto again;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (!fprno)
|
|
|
|
|
{
|
|
|
|
|
log_assert (opt_clear);
|
|
|
|
|
err = scd_setattr ("CA-FPR-1", fpr, fprlen);
|
|
|
|
|
if (!err)
|
|
|
|
|
err = scd_setattr ("CA-FPR-2", fpr, fprlen);
|
|
|
|
|
if (!err)
|
|
|
|
|
err = scd_setattr ("CA-FPR-3", fpr, fprlen);
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
err = scd_setattr (fprno==1?"CA-FPR-1":
|
|
|
|
|
fprno==2?"CA-FPR-2":
|
|
|
|
|
fprno==3?"CA-FPR-3":"x", fpr, fprlen);
|
|
|
|
|
|
|
|
|
|
leave:
|
|
|
|
|
xfree (data);
|
|
|
|
|
return err;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
static gpg_error_t
|
|
|
|
|
cmd_privatedo (card_info_t info, char *argstr)
|
|
|
|
|
{
|
|
|
|
|
gpg_error_t err;
|
|
|
|
|
int opt_clear;
|
|
|
|
|
char *do_name = NULL;
|
|
|
|
|
char *data = NULL;
|
|
|
|
|
size_t datalen;
|
|
|
|
|
int do_no;
|
|
|
|
|
|
|
|
|
|
if (!info)
|
|
|
|
|
return print_help
|
|
|
|
|
("PRIVATEDO [--clear] N [< FILE]\n\n"
|
|
|
|
|
"Change the private data object N. N must be in the\n"
|
|
|
|
|
"range 1 to 4. If FILE is given the data is is read\n"
|
|
|
|
|
"from that file. The option --clear clears the data.",
|
|
|
|
|
APP_TYPE_OPENPGP, 0);
|
|
|
|
|
|
|
|
|
|
opt_clear = has_leading_option (argstr, "--clear");
|
|
|
|
|
argstr = skip_options (argstr);
|
|
|
|
|
|
|
|
|
|
if (digitp (argstr))
|
|
|
|
|
{
|
|
|
|
|
do_no = atoi (argstr);
|
|
|
|
|
while (digitp (argstr))
|
|
|
|
|
argstr++;
|
|
|
|
|
while (spacep (argstr))
|
|
|
|
|
argstr++;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
do_no = 0;
|
|
|
|
|
|
|
|
|
|
if (do_no < 1 || do_no > 4)
|
|
|
|
|
{
|
|
|
|
|
err = gpg_error (GPG_ERR_INV_ARG);
|
|
|
|
|
goto leave;
|
|
|
|
|
}
|
|
|
|
|
do_name = xasprintf ("PRIVATE-DO-%d", do_no);
|
|
|
|
|
|
|
|
|
|
if (opt_clear)
|
|
|
|
|
{
|
|
|
|
|
data = xstrdup (" ");
|
|
|
|
|
datalen = 1;
|
|
|
|
|
}
|
|
|
|
|
else if (*argstr == '<') /* Read it from a file */
|
|
|
|
|
{
|
|
|
|
|
for (argstr++; spacep (argstr); argstr++)
|
|
|
|
|
;
|
|
|
|
|
err = get_data_from_file (argstr, &data, &datalen);
|
|
|
|
|
if (err)
|
|
|
|
|
goto leave;
|
|
|
|
|
}
|
|
|
|
|
else if (*argstr)
|
|
|
|
|
{
|
|
|
|
|
err = gpg_error (GPG_ERR_INV_ARG);
|
|
|
|
|
goto leave;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
data = tty_get (_("Private DO data: "));
|
|
|
|
|
trim_spaces (data);
|
|
|
|
|
tty_kill_prompt ();
|
|
|
|
|
datalen = strlen (data);
|
|
|
|
|
if (!*data || *data == CONTROL_D)
|
|
|
|
|
{
|
|
|
|
|
err = gpg_error (GPG_ERR_CANCELED);
|
|
|
|
|
goto leave;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
err = scd_setattr (do_name, data, datalen);
|
|
|
|
|
|
|
|
|
|
leave:
|
|
|
|
|
xfree (do_name);
|
|
|
|
|
xfree (data);
|
|
|
|
|
return err;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
static gpg_error_t
|
|
|
|
|
cmd_writecert (card_info_t info, char *argstr)
|
|
|
|
|
{
|
|
|
|
|
gpg_error_t err;
|
|
|
|
|
int opt_clear;
|
|
|
|
|
int do_no;
|
|
|
|
|
char *data = NULL;
|
|
|
|
|
size_t datalen;
|
|
|
|
|
|
|
|
|
|
if (!info)
|
|
|
|
|
return print_help
|
|
|
|
|
("WRITECERT [--clear] 3 < FILE\n\n"
|
|
|
|
|
"Write a certificate for key 3. Unless --clear is given\n"
|
|
|
|
|
"the file argement is mandatory. The option --clear removes\n"
|
|
|
|
|
"the certificate from the card.",
|
|
|
|
|
APP_TYPE_OPENPGP, 0);
|
|
|
|
|
|
|
|
|
|
opt_clear = has_leading_option (argstr, "--clear");
|
|
|
|
|
argstr = skip_options (argstr);
|
|
|
|
|
|
|
|
|
|
if (digitp (argstr))
|
|
|
|
|
{
|
|
|
|
|
do_no = atoi (argstr);
|
|
|
|
|
while (digitp (argstr))
|
|
|
|
|
argstr++;
|
|
|
|
|
while (spacep (argstr))
|
|
|
|
|
argstr++;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
do_no = 0;
|
|
|
|
|
|
|
|
|
|
if (do_no != 3)
|
|
|
|
|
{
|
|
|
|
|
err = gpg_error (GPG_ERR_INV_ARG);
|
|
|
|
|
goto leave;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (opt_clear)
|
|
|
|
|
{
|
|
|
|
|
data = xstrdup (" ");
|
|
|
|
|
datalen = 1;
|
|
|
|
|
}
|
|
|
|
|
else if (*argstr == '<') /* Read it from a file */
|
|
|
|
|
{
|
|
|
|
|
for (argstr++; spacep (argstr); argstr++)
|
|
|
|
|
;
|
|
|
|
|
err = get_data_from_file (argstr, &data, &datalen);
|
|
|
|
|
if (err)
|
|
|
|
|
goto leave;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
err = gpg_error (GPG_ERR_INV_ARG);
|
|
|
|
|
goto leave;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
err = scd_writecert ("OPENPGP.3", data, datalen);
|
|
|
|
|
|
|
|
|
|
leave:
|
|
|
|
|
xfree (data);
|
|
|
|
|
return err;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
static gpg_error_t
|
|
|
|
|
cmd_readcert (card_info_t info, char *argstr)
|
|
|
|
|
{
|
|
|
|
|
gpg_error_t err;
|
|
|
|
|
int do_no;
|
|
|
|
|
void *data = NULL;
|
|
|
|
|
size_t datalen;
|
|
|
|
|
const char *fname;
|
|
|
|
|
|
|
|
|
|
if (!info)
|
|
|
|
|
return print_help
|
|
|
|
|
("READCERT 3 > FILE\n\n"
|
|
|
|
|
"Read the certificate for key 3 and store it in FILE.",
|
|
|
|
|
APP_TYPE_OPENPGP, 0);
|
|
|
|
|
|
|
|
|
|
argstr = skip_options (argstr);
|
|
|
|
|
|
|
|
|
|
if (digitp (argstr))
|
|
|
|
|
{
|
|
|
|
|
do_no = atoi (argstr);
|
|
|
|
|
while (digitp (argstr))
|
|
|
|
|
argstr++;
|
|
|
|
|
while (spacep (argstr))
|
|
|
|
|
argstr++;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
do_no = 0;
|
|
|
|
|
|
|
|
|
|
if (do_no != 3)
|
|
|
|
|
{
|
|
|
|
|
err = gpg_error (GPG_ERR_INV_ARG);
|
|
|
|
|
goto leave;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (*argstr == '>') /* Read it from a file */
|
|
|
|
|
{
|
|
|
|
|
for (argstr++; spacep (argstr); argstr++)
|
|
|
|
|
;
|
|
|
|
|
fname = argstr;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
err = gpg_error (GPG_ERR_INV_ARG);
|
|
|
|
|
goto leave;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
err = scd_readcert ("OPENPGP.3", &data, &datalen);
|
|
|
|
|
if (err)
|
|
|
|
|
goto leave;
|
|
|
|
|
|
|
|
|
|
err = put_data_to_file (fname, data, datalen);
|
|
|
|
|
|
|
|
|
|
leave:
|
|
|
|
|
xfree (data);
|
|
|
|
|
return err;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
static gpg_error_t
|
|
|
|
|
cmd_forcesig (card_info_t info)
|
|
|
|
|
{
|
|
|
|
|
gpg_error_t err;
|
|
|
|
|
int newstate;
|
|
|
|
|
|
|
|
|
|
if (!info)
|
|
|
|
|
return print_help
|
|
|
|
|
("FORCESIG\n\n"
|
|
|
|
|
"Toggle the forcesig flag of an OpenPGP card.",
|
|
|
|
|
APP_TYPE_OPENPGP, 0);
|
|
|
|
|
|
|
|
|
|
if (info->apptype != APP_TYPE_OPENPGP)
|
|
|
|
|
{
|
|
|
|
|
log_info ("Note: This is an OpenPGP only command.\n");
|
|
|
|
|
return gpg_error (GPG_ERR_NOT_SUPPORTED);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
newstate = !info->chv1_cached;
|
|
|
|
|
|
|
|
|
|
err = scd_setattr ("CHV-STATUS-1", newstate? "\x01":"", 1);
|
|
|
|
|
if (err)
|
|
|
|
|
goto leave;
|
|
|
|
|
|
|
|
|
|
/* Read it back to be sure we have the right toggle state the next
|
|
|
|
|
* time. */
|
|
|
|
|
err = scd_getattr ("CHV-STATUS", info);
|
|
|
|
|
|
|
|
|
|
leave:
|
|
|
|
|
return err;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Helper for cmd_generate. Noe that either 0 or 1 is stored at
|
|
|
|
|
* FORCED_CHV1. */
|
|
|
|
|
static gpg_error_t
|
|
|
|
|
check_pin_for_key_operation (card_info_t info, int *forced_chv1)
|
|
|
|
|
{
|
|
|
|
|
gpg_error_t err = 0;
|
|
|
|
|
|
|
|
|
|
*forced_chv1 = !info->chv1_cached;
|
|
|
|
|
if (*forced_chv1)
|
|
|
|
|
{ /* Switch off the forced mode so that during key generation we
|
|
|
|
|
* don't get bothered with PIN queries for each self-signature. */
|
|
|
|
|
err = scd_setattr ("CHV-STATUS-1", "\x01", 1);
|
|
|
|
|
if (err)
|
|
|
|
|
{
|
|
|
|
|
log_error ("error clearing forced signature PIN flag: %s\n",
|
|
|
|
|
gpg_strerror (err));
|
|
|
|
|
*forced_chv1 = -1; /* Not changed. */
|
|
|
|
|
goto leave;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Check the PIN now, so that we won't get asked later for each
|
|
|
|
|
* binding signature. */
|
|
|
|
|
err = scd_checkpin (info->serialno);
|
|
|
|
|
if (err)
|
|
|
|
|
log_error ("error checking the PIN: %s\n", gpg_strerror (err));
|
|
|
|
|
|
|
|
|
|
leave:
|
|
|
|
|
return err;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Helper for cmd_generate. */
|
|
|
|
|
static void
|
|
|
|
|
restore_forced_chv1 (int *forced_chv1)
|
|
|
|
|
{
|
|
|
|
|
gpg_error_t err;
|
|
|
|
|
|
|
|
|
|
/* Note the possible values stored at FORCED_CHV1:
|
|
|
|
|
* 0 - forcesig was not enabled.
|
|
|
|
|
* 1 - forcesig was enabled - enable it again.
|
|
|
|
|
* -1 - We have not changed anything. */
|
|
|
|
|
if (*forced_chv1 == 1)
|
|
|
|
|
{ /* Switch back to forced state. */
|
|
|
|
|
err = scd_setattr ("CHV-STATUS-1", "", 1);
|
|
|
|
|
if (err)
|
|
|
|
|
log_error ("error setting forced signature PIN flag: %s\n",
|
|
|
|
|
gpg_strerror (err));
|
|
|
|
|
*forced_chv1 = 0;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
static gpg_error_t
|
|
|
|
|
cmd_generate (card_info_t info)
|
|
|
|
|
{
|
|
|
|
|
gpg_error_t err;
|
|
|
|
|
int forced_chv1 = -1;
|
|
|
|
|
int want_backup;
|
|
|
|
|
char *answer = NULL;
|
2019-01-29 08:48:53 +01:00
|
|
|
|
key_info_t kinfo1, kinfo2, kinfo3;
|
2019-01-27 20:12:00 +01:00
|
|
|
|
|
|
|
|
|
if (!info)
|
|
|
|
|
return print_help
|
|
|
|
|
("GENERATE\n\n"
|
|
|
|
|
"Menu to generate a new keys.",
|
|
|
|
|
APP_TYPE_OPENPGP, 0);
|
|
|
|
|
|
|
|
|
|
if (info->apptype != APP_TYPE_OPENPGP)
|
|
|
|
|
{
|
|
|
|
|
log_info ("Note: This is an OpenPGP only command.\n");
|
|
|
|
|
return gpg_error (GPG_ERR_NOT_SUPPORTED);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (info->extcap.ki)
|
|
|
|
|
{
|
|
|
|
|
xfree (answer);
|
|
|
|
|
answer = tty_get (_("Make off-card backup of encryption key? (Y/n) "));
|
|
|
|
|
want_backup = answer_is_yes_no_default (answer, 1/*(default to Yes)*/);
|
|
|
|
|
tty_kill_prompt ();
|
|
|
|
|
if (*answer == CONTROL_D)
|
|
|
|
|
{
|
|
|
|
|
err = gpg_error (GPG_ERR_CANCELED);
|
|
|
|
|
goto leave;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
want_backup = 0;
|
|
|
|
|
|
2019-01-29 08:48:53 +01:00
|
|
|
|
|
|
|
|
|
kinfo1 = find_kinfo (info, "OPENPGP.1");
|
|
|
|
|
kinfo2 = find_kinfo (info, "OPENPGP.2");
|
|
|
|
|
kinfo3 = find_kinfo (info, "OPENPGP.3");
|
|
|
|
|
|
|
|
|
|
if ((kinfo1 && kinfo1->fprlen && !mem_is_zero (kinfo1->fpr,kinfo1->fprlen))
|
|
|
|
|
|| (kinfo2 && kinfo2->fprlen && !mem_is_zero (kinfo2->fpr,kinfo2->fprlen))
|
|
|
|
|
|| (kinfo3 && kinfo3->fprlen && !mem_is_zero (kinfo3->fpr,kinfo3->fprlen))
|
|
|
|
|
)
|
2019-01-27 20:12:00 +01:00
|
|
|
|
{
|
|
|
|
|
tty_printf ("\n");
|
|
|
|
|
log_info (_("Note: keys are already stored on the card!\n"));
|
|
|
|
|
tty_printf ("\n");
|
|
|
|
|
answer = tty_get (_("Replace existing keys? (y/N) "));
|
|
|
|
|
tty_kill_prompt ();
|
|
|
|
|
if (*answer == CONTROL_D)
|
|
|
|
|
{
|
|
|
|
|
err = gpg_error (GPG_ERR_CANCELED);
|
|
|
|
|
goto leave;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (!answer_is_yes_no_default (answer, 0/*(default to No)*/))
|
|
|
|
|
{
|
|
|
|
|
err = gpg_error (GPG_ERR_CANCELED);
|
|
|
|
|
goto leave;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* If no displayed name has been set, we assume that this is a fresh
|
|
|
|
|
* card and print a hint about the default PINs. */
|
|
|
|
|
if (!info->disp_name || !*info->disp_name)
|
|
|
|
|
{
|
|
|
|
|
tty_printf ("\n");
|
|
|
|
|
tty_printf (_("Please note that the factory settings of the PINs are\n"
|
|
|
|
|
" PIN = '%s' Admin PIN = '%s'\n"
|
|
|
|
|
"You should change them using the command --change-pin\n"),
|
|
|
|
|
OPENPGP_USER_PIN_DEFAULT, OPENPGP_ADMIN_PIN_DEFAULT);
|
|
|
|
|
tty_printf ("\n");
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
err = check_pin_for_key_operation (info, &forced_chv1);
|
|
|
|
|
if (err)
|
|
|
|
|
goto leave;
|
|
|
|
|
|
|
|
|
|
/* FIXME: We need to divert to a function which spwans gpg which
|
|
|
|
|
* will then create the key. This also requires new features in
|
|
|
|
|
* gpg. We might also first create the keys on the card and then
|
|
|
|
|
* tell gpg to use them to create the OpenPGP keyblock. */
|
|
|
|
|
/* generate_keypair (ctrl, 1, NULL, info.serialno, want_backup); */
|
|
|
|
|
err = gpg_error (GPG_ERR_NOT_IMPLEMENTED);
|
|
|
|
|
|
|
|
|
|
leave:
|
|
|
|
|
restore_forced_chv1 (&forced_chv1);
|
|
|
|
|
xfree (answer);
|
|
|
|
|
return err;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Sub-menu to change a PIN. The presented options may depend on the
|
|
|
|
|
* the ALLOW_ADMIN flag. */
|
|
|
|
|
static gpg_error_t
|
|
|
|
|
cmd_passwd (card_info_t info, int allow_admin)
|
|
|
|
|
{
|
|
|
|
|
gpg_error_t err;
|
|
|
|
|
char *answer = NULL;
|
|
|
|
|
|
|
|
|
|
if (!info)
|
|
|
|
|
return print_help
|
|
|
|
|
("PASSWD\n\n"
|
|
|
|
|
"Menu to change or unblock the PINs. Note that the\n"
|
|
|
|
|
"presented menu options depend on the type of card\n"
|
|
|
|
|
"and whether the admin mode is enabled.",
|
|
|
|
|
0);
|
|
|
|
|
|
|
|
|
|
/* Convenience message because we did this in gpg --card-edit too. */
|
|
|
|
|
if (info->apptype == APP_TYPE_OPENPGP)
|
|
|
|
|
log_info (_("OpenPGP card no. %s detected\n"),
|
|
|
|
|
info->dispserialno? info->dispserialno : info->serialno);
|
|
|
|
|
|
|
|
|
|
if (!allow_admin)
|
|
|
|
|
{
|
|
|
|
|
err = scd_change_pin (1);
|
|
|
|
|
if (err)
|
|
|
|
|
goto leave;
|
|
|
|
|
log_info ("PIN changed.\n");
|
|
|
|
|
}
|
|
|
|
|
else if (info->apptype == APP_TYPE_OPENPGP)
|
|
|
|
|
{
|
|
|
|
|
for (;;)
|
|
|
|
|
{
|
|
|
|
|
tty_printf ("\n");
|
|
|
|
|
tty_printf ("1 - change PIN\n"
|
|
|
|
|
"2 - unblock and set new PIN\n"
|
|
|
|
|
"3 - change Admin PIN\n"
|
|
|
|
|
"4 - set the Reset Code\n"
|
|
|
|
|
"Q - quit\n");
|
|
|
|
|
tty_printf ("\n");
|
|
|
|
|
|
|
|
|
|
err = 0;
|
|
|
|
|
xfree (answer);
|
|
|
|
|
answer = tty_get (_("Your selection? "));
|
|
|
|
|
tty_kill_prompt ();
|
|
|
|
|
if (*answer == CONTROL_D)
|
|
|
|
|
break; /* Quit. */
|
|
|
|
|
if (strlen (answer) != 1)
|
|
|
|
|
continue;
|
|
|
|
|
if (*answer == 'q' || *answer == 'Q')
|
|
|
|
|
break; /* Quit. */
|
|
|
|
|
|
|
|
|
|
if (*answer == '1')
|
|
|
|
|
{
|
|
|
|
|
/* Change PIN (same as the direct thing in non-admin mode). */
|
|
|
|
|
err = scd_change_pin (1);
|
|
|
|
|
if (err)
|
|
|
|
|
log_error ("Error changing the PIN: %s\n", gpg_strerror (err));
|
|
|
|
|
else
|
|
|
|
|
log_info ("PIN changed.\n");
|
|
|
|
|
}
|
|
|
|
|
else if (*answer == '2')
|
|
|
|
|
{
|
|
|
|
|
/* Unblock PIN by setting a new PIN. */
|
|
|
|
|
err = scd_change_pin (101);
|
|
|
|
|
if (err)
|
|
|
|
|
log_error ("Error unblocking the PIN: %s\n", gpg_strerror(err));
|
|
|
|
|
else
|
|
|
|
|
log_info ("PIN unblocked and new PIN set.\n");
|
|
|
|
|
}
|
|
|
|
|
else if (*answer == '3')
|
|
|
|
|
{
|
|
|
|
|
/* Change Admin PIN. */
|
|
|
|
|
err = scd_change_pin (3);
|
|
|
|
|
if (err)
|
|
|
|
|
log_error ("Error changing the PIN: %s\n", gpg_strerror (err));
|
|
|
|
|
else
|
|
|
|
|
log_info ("PIN changed.\n");
|
|
|
|
|
}
|
|
|
|
|
else if (*answer == '4')
|
|
|
|
|
{
|
|
|
|
|
/* Set a new Reset Code. */
|
|
|
|
|
err = scd_change_pin (102);
|
|
|
|
|
if (err)
|
|
|
|
|
log_error ("Error setting the Reset Code: %s\n",
|
|
|
|
|
gpg_strerror (err));
|
|
|
|
|
else
|
|
|
|
|
log_info ("Reset Code set.\n");
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
} /*end for loop*/
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
log_info ("Admin related passwd options not yet supported for '%s'\n",
|
|
|
|
|
app_type_string (info->apptype));
|
|
|
|
|
err = gpg_error (GPG_ERR_NOT_SUPPORTED);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
leave:
|
|
|
|
|
xfree (answer);
|
|
|
|
|
return err;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
static gpg_error_t
|
|
|
|
|
cmd_unblock (card_info_t info)
|
|
|
|
|
{
|
|
|
|
|
gpg_error_t err = 0;
|
|
|
|
|
|
|
|
|
|
if (!info)
|
|
|
|
|
return print_help
|
|
|
|
|
("UNBLOCK\n\n"
|
|
|
|
|
"Unblock a PIN using a PUK or Reset Code. Note that OpenPGP\n"
|
|
|
|
|
"cards prior to version 2 can't use this; instead the PASSWD\n"
|
|
|
|
|
"command can be used to set a new PIN.",
|
|
|
|
|
0);
|
|
|
|
|
|
|
|
|
|
if (info->apptype == APP_TYPE_OPENPGP)
|
|
|
|
|
log_info (_("OpenPGP card no. %s detected\n"),
|
|
|
|
|
info->dispserialno? info->dispserialno : info->serialno);
|
|
|
|
|
|
|
|
|
|
if (info->apptype == APP_TYPE_OPENPGP && !info->is_v2)
|
|
|
|
|
log_error (_("This command is only available for version 2 cards\n"));
|
2019-01-29 09:30:15 +01:00
|
|
|
|
else if (info->apptype == APP_TYPE_OPENPGP && !info->chvinfo[1])
|
2019-01-27 20:12:00 +01:00
|
|
|
|
log_error (_("Reset Code not or not anymore available\n"));
|
|
|
|
|
else if (info->apptype == APP_TYPE_OPENPGP)
|
|
|
|
|
{
|
|
|
|
|
err = scd_change_pin (2);
|
|
|
|
|
if (!err)
|
|
|
|
|
log_info ("PIN changed.\n");
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
log_info ("Unblocking not yet supported for '%s'\n",
|
|
|
|
|
app_type_string (info->apptype));
|
|
|
|
|
|
|
|
|
|
return err;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Direct sending of an hex encoded APDU with error printing. */
|
|
|
|
|
static gpg_error_t
|
|
|
|
|
send_apdu (const char *hexapdu, const char *desc, unsigned int ignore)
|
|
|
|
|
{
|
|
|
|
|
gpg_error_t err;
|
|
|
|
|
unsigned int sw;
|
|
|
|
|
|
|
|
|
|
err = scd_apdu (hexapdu, &sw);
|
|
|
|
|
if (err)
|
|
|
|
|
log_error ("sending card command %s failed: %s\n", desc,
|
|
|
|
|
gpg_strerror (err));
|
|
|
|
|
else if (!hexapdu || !strcmp (hexapdu, "undefined"))
|
|
|
|
|
;
|
|
|
|
|
else if (ignore == 0xffff)
|
|
|
|
|
; /* Ignore all status words. */
|
|
|
|
|
else if (sw != 0x9000)
|
|
|
|
|
{
|
|
|
|
|
switch (sw)
|
|
|
|
|
{
|
|
|
|
|
case 0x6285: err = gpg_error (GPG_ERR_OBJ_TERM_STATE); break;
|
|
|
|
|
case 0x6982: err = gpg_error (GPG_ERR_BAD_PIN); break;
|
|
|
|
|
case 0x6985: err = gpg_error (GPG_ERR_USE_CONDITIONS); break;
|
|
|
|
|
default: err = gpg_error (GPG_ERR_CARD);
|
|
|
|
|
}
|
|
|
|
|
if (!(ignore && ignore == sw))
|
|
|
|
|
log_error ("card command %s failed: %s (0x%04x)\n", desc,
|
|
|
|
|
gpg_strerror (err), sw);
|
|
|
|
|
}
|
|
|
|
|
return err;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Note: On successful execution a redisplay should be scheduled. If
|
|
|
|
|
* this function fails the card may be in an unknown state. */
|
|
|
|
|
static gpg_error_t
|
|
|
|
|
cmd_factoryreset (card_info_t info)
|
|
|
|
|
{
|
|
|
|
|
gpg_error_t err;
|
|
|
|
|
char *answer = NULL;
|
|
|
|
|
int termstate = 0;
|
|
|
|
|
int any_apdu = 0;
|
2019-01-29 13:28:10 +01:00
|
|
|
|
int is_yubikey = 0;
|
2019-01-27 20:12:00 +01:00
|
|
|
|
int i;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if (!info)
|
|
|
|
|
return print_help
|
|
|
|
|
("FACTORY-RESET\n\n"
|
2019-01-29 13:28:10 +01:00
|
|
|
|
"Do a complete reset of some OpenPGP and PIV cards. This\n"
|
|
|
|
|
"deletes all data and keys and resets the PINs to their default.\n"
|
|
|
|
|
"This is mainly used by developers with scratch cards. Don't\n"
|
|
|
|
|
"worry, you need to confirm before the command proceeds.",
|
|
|
|
|
APP_TYPE_OPENPGP, APP_TYPE_PIV, 0);
|
|
|
|
|
|
|
|
|
|
/* We support the factory reset for most OpenPGP cards and Yubikeys
|
|
|
|
|
* with the PIV application. */
|
|
|
|
|
if (info->apptype == APP_TYPE_OPENPGP)
|
|
|
|
|
;
|
|
|
|
|
else if (info->apptype == APP_TYPE_PIV
|
|
|
|
|
&& info->cardtype && !strcmp (info->cardtype, "yubikey"))
|
|
|
|
|
is_yubikey = 1;
|
|
|
|
|
else
|
2019-01-27 20:12:00 +01:00
|
|
|
|
|
2019-01-29 13:28:10 +01:00
|
|
|
|
return gpg_error (GPG_ERR_NOT_SUPPORTED);
|
2019-01-27 20:12:00 +01:00
|
|
|
|
|
2019-01-29 13:28:10 +01:00
|
|
|
|
/* For an OpenPGP card the code below basically does the same what
|
|
|
|
|
* this gpg-connect-agent script does:
|
2019-01-27 20:12:00 +01:00
|
|
|
|
*
|
|
|
|
|
* scd reset
|
|
|
|
|
* scd serialno undefined
|
|
|
|
|
* scd apdu 00 A4 04 00 06 D2 76 00 01 24 01
|
|
|
|
|
* scd apdu 00 20 00 81 08 40 40 40 40 40 40 40 40
|
|
|
|
|
* scd apdu 00 20 00 81 08 40 40 40 40 40 40 40 40
|
|
|
|
|
* scd apdu 00 20 00 81 08 40 40 40 40 40 40 40 40
|
|
|
|
|
* scd apdu 00 20 00 81 08 40 40 40 40 40 40 40 40
|
|
|
|
|
* scd apdu 00 20 00 83 08 40 40 40 40 40 40 40 40
|
|
|
|
|
* scd apdu 00 20 00 83 08 40 40 40 40 40 40 40 40
|
|
|
|
|
* scd apdu 00 20 00 83 08 40 40 40 40 40 40 40 40
|
|
|
|
|
* scd apdu 00 20 00 83 08 40 40 40 40 40 40 40 40
|
|
|
|
|
* scd apdu 00 e6 00 00
|
|
|
|
|
* scd apdu 00 44 00 00
|
|
|
|
|
* scd reset
|
|
|
|
|
* /echo Card has been reset to factory defaults
|
|
|
|
|
*
|
2019-01-29 13:28:10 +01:00
|
|
|
|
* For a PIV application on a Yubikey it merely issues the Yubikey
|
|
|
|
|
* specific resset command.
|
2019-01-27 20:12:00 +01:00
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
err = scd_learn (info);
|
|
|
|
|
if (gpg_err_code (err) == GPG_ERR_OBJ_TERM_STATE
|
|
|
|
|
&& gpg_err_source (err) == GPG_ERR_SOURCE_SCD)
|
|
|
|
|
termstate = 1;
|
|
|
|
|
else if (err)
|
|
|
|
|
{
|
|
|
|
|
log_error (_("OpenPGP card not available: %s\n"), gpg_strerror (err));
|
|
|
|
|
goto leave;
|
|
|
|
|
}
|
|
|
|
|
|
2019-01-29 13:28:10 +01:00
|
|
|
|
if (!termstate || is_yubikey)
|
2019-01-27 20:12:00 +01:00
|
|
|
|
{
|
2019-01-29 13:28:10 +01:00
|
|
|
|
if (is_yubikey)
|
|
|
|
|
log_info (_("Yubikey no. %s with PIV application detected\n"),
|
|
|
|
|
info->dispserialno? info->dispserialno : info->serialno);
|
|
|
|
|
else
|
2019-01-27 20:12:00 +01:00
|
|
|
|
{
|
2019-01-29 13:28:10 +01:00
|
|
|
|
log_info (_("OpenPGP card no. %s detected\n"),
|
|
|
|
|
info->dispserialno? info->dispserialno : info->serialno);
|
|
|
|
|
if (!(info->status_indicator == 3 || info->status_indicator == 5))
|
|
|
|
|
{
|
|
|
|
|
/* Note: We won't see status-indicator 3 here because it
|
|
|
|
|
* is not possible to select a card application in
|
|
|
|
|
* termination state. */
|
|
|
|
|
log_error (_("This command is not supported by this card\n"));
|
|
|
|
|
err = gpg_error (GPG_ERR_NOT_SUPPORTED);
|
|
|
|
|
goto leave;
|
|
|
|
|
}
|
2019-01-27 20:12:00 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
tty_printf ("\n");
|
|
|
|
|
log_info
|
|
|
|
|
(_("Note: This command destroys all keys stored on the card!\n"));
|
|
|
|
|
tty_printf ("\n");
|
|
|
|
|
xfree (answer);
|
|
|
|
|
answer = tty_get (_("Continue? (y/N) "));
|
|
|
|
|
tty_kill_prompt ();
|
|
|
|
|
trim_spaces (answer);
|
|
|
|
|
if (*answer == CONTROL_D
|
|
|
|
|
|| !answer_is_yes_no_default (answer, 0/*(default to no)*/))
|
|
|
|
|
{
|
|
|
|
|
err = gpg_error (GPG_ERR_CANCELED);
|
|
|
|
|
goto leave;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
xfree (answer);
|
|
|
|
|
answer = tty_get (_("Really do a factory reset? (enter \"yes\") "));
|
|
|
|
|
tty_kill_prompt ();
|
|
|
|
|
trim_spaces (answer);
|
|
|
|
|
if (strcmp (answer, "yes") && strcmp (answer,_("yes")))
|
|
|
|
|
{
|
|
|
|
|
err = gpg_error (GPG_ERR_CANCELED);
|
|
|
|
|
goto leave;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2019-01-29 13:28:10 +01:00
|
|
|
|
if (is_yubikey)
|
|
|
|
|
{
|
|
|
|
|
/* The PIV application si already selected, we only need to
|
|
|
|
|
* send the special reset APDU after having blocked PIN and
|
|
|
|
|
* PUK. Note that blocking the PUK is done using the
|
|
|
|
|
* unblock PIN command. */
|
|
|
|
|
any_apdu = 1;
|
|
|
|
|
for (i=0; i < 5; i++)
|
|
|
|
|
send_apdu ("0020008008FFFFFFFFFFFFFFFF", "VERIFY", 0xffff);
|
|
|
|
|
for (i=0; i < 5; i++)
|
|
|
|
|
send_apdu ("002C008010FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF",
|
|
|
|
|
"RESET RETRY COUNTER", 0xffff);
|
|
|
|
|
err = send_apdu ("00FB000001FF", "YUBIKEY RESET", 0);
|
|
|
|
|
if (err)
|
|
|
|
|
goto leave;
|
|
|
|
|
}
|
|
|
|
|
else /* OpenPGP card. */
|
|
|
|
|
{
|
|
|
|
|
any_apdu = 1;
|
|
|
|
|
/* We need to select a card application before we can send APDUs
|
|
|
|
|
* to the card without scdaemon doing anything on its own. */
|
|
|
|
|
err = send_apdu (NULL, "RESET", 0);
|
|
|
|
|
if (err)
|
|
|
|
|
goto leave;
|
|
|
|
|
err = send_apdu ("undefined", "dummy select ", 0);
|
|
|
|
|
if (err)
|
|
|
|
|
goto leave;
|
|
|
|
|
/* Select the OpenPGP application. */
|
|
|
|
|
err = send_apdu ("00A4040006D27600012401", "SELECT AID", 0);
|
|
|
|
|
if (err)
|
|
|
|
|
goto leave;
|
|
|
|
|
|
|
|
|
|
/* Do some dummy verifies with wrong PINs to set the retry
|
|
|
|
|
* counter to zero. We can't easily use the card version 2.1
|
|
|
|
|
* feature of presenting the admin PIN to allow the terminate
|
|
|
|
|
* command because there is no machinery in scdaemon to catch
|
|
|
|
|
* the verify command and ask for the PIN when the "APDU"
|
|
|
|
|
* command is used.
|
|
|
|
|
* Here, the length of dummy wrong PIN is 32-byte, also
|
|
|
|
|
* supporting authentication with KDF DO. */
|
|
|
|
|
for (i=0; i < 4; i++)
|
|
|
|
|
send_apdu ("0020008120"
|
|
|
|
|
"40404040404040404040404040404040"
|
|
|
|
|
"40404040404040404040404040404040", "VERIFY", 0xffff);
|
|
|
|
|
for (i=0; i < 4; i++)
|
|
|
|
|
send_apdu ("0020008320"
|
|
|
|
|
"40404040404040404040404040404040"
|
|
|
|
|
"40404040404040404040404040404040", "VERIFY", 0xffff);
|
|
|
|
|
|
|
|
|
|
/* Send terminate datafile command. */
|
|
|
|
|
err = send_apdu ("00e60000", "TERMINATE DF", 0x6985);
|
|
|
|
|
if (err)
|
|
|
|
|
goto leave;
|
|
|
|
|
}
|
|
|
|
|
}
|
2019-01-27 20:12:00 +01:00
|
|
|
|
|
2019-01-29 13:28:10 +01:00
|
|
|
|
if (!is_yubikey)
|
|
|
|
|
{
|
|
|
|
|
any_apdu = 1;
|
|
|
|
|
/* Send activate datafile command. This is used without
|
|
|
|
|
* confirmation if the card is already in termination state. */
|
|
|
|
|
err = send_apdu ("00440000", "ACTIVATE DF", 0);
|
2019-01-27 20:12:00 +01:00
|
|
|
|
if (err)
|
|
|
|
|
goto leave;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Finally we reset the card reader once more. */
|
|
|
|
|
err = send_apdu (NULL, "RESET", 0);
|
|
|
|
|
if (err)
|
|
|
|
|
goto leave;
|
|
|
|
|
|
|
|
|
|
/* Then, connect the card again (answer used as a dummy). */
|
|
|
|
|
xfree (answer); answer = NULL;
|
|
|
|
|
err = scd_serialno (&answer, NULL);
|
|
|
|
|
|
|
|
|
|
leave:
|
2019-01-29 13:28:10 +01:00
|
|
|
|
if (err && any_apdu && !is_yubikey)
|
2019-01-27 20:12:00 +01:00
|
|
|
|
{
|
|
|
|
|
log_info ("Due to an error the card might be in an inconsistent state\n"
|
|
|
|
|
"You should run the LIST command to check this.\n");
|
|
|
|
|
/* FIXME: We need a better solution in the case that the card is
|
|
|
|
|
* in a termination state, i.e. the card was removed before the
|
|
|
|
|
* activate was sent. The best solution I found with v2.1
|
|
|
|
|
* Zeitcontrol card was to kill scdaemon and the issue this
|
|
|
|
|
* sequence with gpg-connect-agent:
|
|
|
|
|
* scd reset
|
|
|
|
|
* scd serialno undefined
|
|
|
|
|
* scd apdu 00A4040006D27600012401 (returns error)
|
|
|
|
|
* scd apdu 00440000
|
|
|
|
|
* Then kill scdaemon again and issue:
|
|
|
|
|
* scd reset
|
|
|
|
|
* scd serialno openpgp
|
|
|
|
|
*/
|
|
|
|
|
}
|
|
|
|
|
xfree (answer);
|
|
|
|
|
return err;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Generate KDF data. This is a helper for cmd_kdfsetup. */
|
|
|
|
|
static gpg_error_t
|
|
|
|
|
gen_kdf_data (unsigned char *data, int single_salt)
|
|
|
|
|
{
|
|
|
|
|
gpg_error_t err;
|
|
|
|
|
const unsigned char h0[] = { 0x81, 0x01, 0x03,
|
|
|
|
|
0x82, 0x01, 0x08,
|
|
|
|
|
0x83, 0x04 };
|
|
|
|
|
const unsigned char h1[] = { 0x84, 0x08 };
|
|
|
|
|
const unsigned char h2[] = { 0x85, 0x08 };
|
|
|
|
|
const unsigned char h3[] = { 0x86, 0x08 };
|
|
|
|
|
const unsigned char h4[] = { 0x87, 0x20 };
|
|
|
|
|
const unsigned char h5[] = { 0x88, 0x20 };
|
|
|
|
|
unsigned char *p, *salt_user, *salt_admin;
|
|
|
|
|
unsigned char s2k_char;
|
|
|
|
|
unsigned int iterations;
|
|
|
|
|
unsigned char count_4byte[4];
|
|
|
|
|
|
|
|
|
|
p = data;
|
|
|
|
|
|
|
|
|
|
s2k_char = encode_s2k_iterations (agent_get_s2k_count ());
|
|
|
|
|
iterations = S2K_DECODE_COUNT (s2k_char);
|
|
|
|
|
count_4byte[0] = (iterations >> 24) & 0xff;
|
|
|
|
|
count_4byte[1] = (iterations >> 16) & 0xff;
|
|
|
|
|
count_4byte[2] = (iterations >> 8) & 0xff;
|
|
|
|
|
count_4byte[3] = (iterations & 0xff);
|
|
|
|
|
|
|
|
|
|
memcpy (p, h0, sizeof h0);
|
|
|
|
|
p += sizeof h0;
|
|
|
|
|
memcpy (p, count_4byte, sizeof count_4byte);
|
|
|
|
|
p += sizeof count_4byte;
|
|
|
|
|
memcpy (p, h1, sizeof h1);
|
|
|
|
|
salt_user = (p += sizeof h1);
|
|
|
|
|
gcry_randomize (p, 8, GCRY_STRONG_RANDOM);
|
|
|
|
|
p += 8;
|
|
|
|
|
|
|
|
|
|
if (single_salt)
|
|
|
|
|
salt_admin = salt_user;
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
memcpy (p, h2, sizeof h2);
|
|
|
|
|
p += sizeof h2;
|
|
|
|
|
gcry_randomize (p, 8, GCRY_STRONG_RANDOM);
|
|
|
|
|
p += 8;
|
|
|
|
|
memcpy (p, h3, sizeof h3);
|
|
|
|
|
salt_admin = (p += sizeof h3);
|
|
|
|
|
gcry_randomize (p, 8, GCRY_STRONG_RANDOM);
|
|
|
|
|
p += 8;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
memcpy (p, h4, sizeof h4);
|
|
|
|
|
p += sizeof h4;
|
|
|
|
|
err = gcry_kdf_derive (OPENPGP_USER_PIN_DEFAULT,
|
|
|
|
|
strlen (OPENPGP_USER_PIN_DEFAULT),
|
|
|
|
|
GCRY_KDF_ITERSALTED_S2K, GCRY_MD_SHA256,
|
|
|
|
|
salt_user, 8, iterations, 32, p);
|
|
|
|
|
p += 32;
|
|
|
|
|
if (!err)
|
|
|
|
|
{
|
|
|
|
|
memcpy (p, h5, sizeof h5);
|
|
|
|
|
p += sizeof h5;
|
|
|
|
|
err = gcry_kdf_derive (OPENPGP_ADMIN_PIN_DEFAULT,
|
|
|
|
|
strlen (OPENPGP_ADMIN_PIN_DEFAULT),
|
|
|
|
|
GCRY_KDF_ITERSALTED_S2K, GCRY_MD_SHA256,
|
|
|
|
|
salt_admin, 8, iterations, 32, p);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return err;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
static gpg_error_t
|
|
|
|
|
cmd_kdfsetup (card_info_t info, char *argstr)
|
|
|
|
|
{
|
|
|
|
|
gpg_error_t err;
|
|
|
|
|
unsigned char kdf_data[OPENPGP_KDF_DATA_LENGTH_MAX];
|
|
|
|
|
int single = (*argstr != 0);
|
|
|
|
|
|
|
|
|
|
if (!info)
|
|
|
|
|
return print_help
|
|
|
|
|
("KDF-SETUP\n\n"
|
|
|
|
|
"Prepare the OpenPGP card KDF feature for this card.",
|
|
|
|
|
APP_TYPE_OPENPGP, 0);
|
|
|
|
|
|
|
|
|
|
if (info->apptype != APP_TYPE_OPENPGP)
|
|
|
|
|
{
|
|
|
|
|
log_info ("Note: This is an OpenPGP only command.\n");
|
|
|
|
|
return gpg_error (GPG_ERR_NOT_SUPPORTED);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (!info->extcap.kdf)
|
|
|
|
|
{
|
|
|
|
|
log_error (_("This command is not supported by this card\n"));
|
|
|
|
|
err = gpg_error (GPG_ERR_NOT_SUPPORTED);
|
|
|
|
|
goto leave;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
err = gen_kdf_data (kdf_data, single);
|
|
|
|
|
if (err)
|
|
|
|
|
goto leave;
|
|
|
|
|
|
|
|
|
|
err = scd_setattr ("KDF", kdf_data,
|
|
|
|
|
single ? OPENPGP_KDF_DATA_LENGTH_MIN
|
|
|
|
|
/* */ : OPENPGP_KDF_DATA_LENGTH_MAX);
|
|
|
|
|
if (err)
|
|
|
|
|
goto leave;
|
|
|
|
|
|
|
|
|
|
err = scd_getattr ("KDF", info);
|
|
|
|
|
|
|
|
|
|
leave:
|
|
|
|
|
return err;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
show_keysize_warning (void)
|
|
|
|
|
{
|
|
|
|
|
static int shown;
|
|
|
|
|
|
|
|
|
|
if (shown)
|
|
|
|
|
return;
|
|
|
|
|
shown = 1;
|
|
|
|
|
tty_printf
|
|
|
|
|
(_("Note: There is no guarantee that the card supports the requested\n"
|
|
|
|
|
" key type or size. If the key generation does not succeed,\n"
|
|
|
|
|
" please check the documentation of your card to see which\n"
|
|
|
|
|
" key types and sizes are supported.\n")
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Ask for the size of a card key. NBITS is the current size
|
|
|
|
|
* configured for the card. Returns 0 on success and stored the
|
|
|
|
|
* chosen key size at R_KEYSIZE; 0 is stored to indicate that the
|
|
|
|
|
* default size shall be used. */
|
|
|
|
|
static gpg_error_t
|
|
|
|
|
ask_card_rsa_keysize (unsigned int nbits, unsigned int *r_keysize)
|
|
|
|
|
{
|
|
|
|
|
unsigned int min_nbits = 1024;
|
|
|
|
|
unsigned int max_nbits = 4096;
|
|
|
|
|
char*answer;
|
|
|
|
|
unsigned int req_nbits;
|
|
|
|
|
|
|
|
|
|
for (;;)
|
|
|
|
|
{
|
|
|
|
|
answer = tty_getf (_("What keysize do you want? (%u) "), nbits);
|
|
|
|
|
trim_spaces (answer);
|
|
|
|
|
tty_kill_prompt ();
|
|
|
|
|
if (*answer == CONTROL_D)
|
|
|
|
|
{
|
|
|
|
|
xfree (answer);
|
|
|
|
|
return gpg_error (GPG_ERR_CANCELED);
|
|
|
|
|
}
|
|
|
|
|
req_nbits = *answer? atoi (answer): nbits;
|
|
|
|
|
xfree (answer);
|
|
|
|
|
|
|
|
|
|
if (req_nbits != nbits && (req_nbits % 32) )
|
|
|
|
|
{
|
|
|
|
|
req_nbits = ((req_nbits + 31) / 32) * 32;
|
|
|
|
|
tty_printf (_("rounded up to %u bits\n"), req_nbits);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (req_nbits == nbits)
|
|
|
|
|
{
|
|
|
|
|
/* Use default. */
|
|
|
|
|
*r_keysize = 0;
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (req_nbits < min_nbits || req_nbits > max_nbits)
|
|
|
|
|
{
|
|
|
|
|
tty_printf (_("%s keysizes must be in the range %u-%u\n"),
|
|
|
|
|
"RSA", min_nbits, max_nbits);
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
*r_keysize = req_nbits;
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Ask for the key attribute of a card key. CURRENT is the current
|
|
|
|
|
* attribute configured for the card. KEYNO is the number of the key
|
|
|
|
|
* used to select the prompt. Stores NULL at result to use the
|
|
|
|
|
* default attribute or stores the selected attribute structure at
|
|
|
|
|
* RESULT. On error an error code is returned. */
|
|
|
|
|
static gpg_error_t
|
|
|
|
|
ask_card_keyattr (int keyno, const struct key_attr *current,
|
|
|
|
|
struct key_attr **result)
|
|
|
|
|
{
|
|
|
|
|
gpg_error_t err;
|
|
|
|
|
struct key_attr *key_attr = NULL;
|
|
|
|
|
char *answer = NULL;
|
|
|
|
|
int selection;
|
|
|
|
|
|
|
|
|
|
*result = NULL;
|
|
|
|
|
|
|
|
|
|
key_attr = xcalloc (1, sizeof *key_attr);
|
|
|
|
|
|
|
|
|
|
tty_printf (_("Changing card key attribute for: "));
|
|
|
|
|
if (keyno == 0)
|
|
|
|
|
tty_printf (_("Signature key\n"));
|
|
|
|
|
else if (keyno == 1)
|
|
|
|
|
tty_printf (_("Encryption key\n"));
|
|
|
|
|
else
|
|
|
|
|
tty_printf (_("Authentication key\n"));
|
|
|
|
|
|
|
|
|
|
tty_printf (_("Please select what kind of key you want:\n"));
|
|
|
|
|
tty_printf (_(" (%d) RSA\n"), 1 );
|
|
|
|
|
tty_printf (_(" (%d) ECC\n"), 2 );
|
|
|
|
|
|
|
|
|
|
for (;;)
|
|
|
|
|
{
|
|
|
|
|
xfree (answer);
|
|
|
|
|
answer = tty_get (_("Your selection? "));
|
|
|
|
|
trim_spaces (answer);
|
|
|
|
|
tty_kill_prompt ();
|
|
|
|
|
if (!*answer || *answer == CONTROL_D)
|
|
|
|
|
{
|
|
|
|
|
err = gpg_error (GPG_ERR_CANCELED);
|
|
|
|
|
goto leave;
|
|
|
|
|
}
|
|
|
|
|
selection = *answer? atoi (answer) : 0;
|
|
|
|
|
|
|
|
|
|
if (selection == 1 || selection == 2)
|
|
|
|
|
break;
|
|
|
|
|
else
|
|
|
|
|
tty_printf (_("Invalid selection.\n"));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if (selection == 1)
|
|
|
|
|
{
|
|
|
|
|
unsigned int nbits, result_nbits;
|
|
|
|
|
|
|
|
|
|
if (current->algo == PUBKEY_ALGO_RSA)
|
|
|
|
|
nbits = current->nbits;
|
|
|
|
|
else
|
|
|
|
|
nbits = 2048;
|
|
|
|
|
|
|
|
|
|
err = ask_card_rsa_keysize (nbits, &result_nbits);
|
|
|
|
|
if (err)
|
|
|
|
|
goto leave;
|
|
|
|
|
if (result_nbits == 0)
|
|
|
|
|
{
|
|
|
|
|
if (current->algo == PUBKEY_ALGO_RSA)
|
|
|
|
|
{
|
|
|
|
|
xfree (key_attr);
|
|
|
|
|
key_attr = NULL;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
result_nbits = nbits;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (key_attr)
|
|
|
|
|
{
|
|
|
|
|
key_attr->algo = PUBKEY_ALGO_RSA;
|
|
|
|
|
key_attr->nbits = result_nbits;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
else if (selection == 2)
|
|
|
|
|
{
|
|
|
|
|
const char *curve;
|
|
|
|
|
/* const char *oid_str; */
|
|
|
|
|
int algo;
|
|
|
|
|
|
|
|
|
|
if (current->algo == PUBKEY_ALGO_RSA)
|
|
|
|
|
{
|
|
|
|
|
if (keyno == 1) /* Encryption key */
|
|
|
|
|
algo = PUBKEY_ALGO_ECDH;
|
|
|
|
|
else /* Signature key or Authentication key */
|
|
|
|
|
algo = PUBKEY_ALGO_ECDSA;
|
|
|
|
|
curve = NULL;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
algo = current->algo;
|
|
|
|
|
curve = current->curve;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
err = GPG_ERR_NOT_IMPLEMENTED;
|
|
|
|
|
goto leave;
|
|
|
|
|
/* FIXME: We need to mve the ask_cure code out to common or
|
|
|
|
|
* provide another sultion. */
|
|
|
|
|
/* curve = ask_curve (&algo, NULL, curve); */
|
|
|
|
|
/* if (curve) */
|
|
|
|
|
/* { */
|
|
|
|
|
/* key_attr->algo = algo; */
|
|
|
|
|
/* oid_str = openpgp_curve_to_oid (curve, NULL); */
|
|
|
|
|
/* key_attr->curve = openpgp_oid_to_curve (oid_str, 0); */
|
|
|
|
|
/* } */
|
|
|
|
|
/* else */
|
|
|
|
|
/* { */
|
|
|
|
|
/* xfree (key_attr); */
|
|
|
|
|
/* key_attr = NULL; */
|
|
|
|
|
/* } */
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
err = gpg_error (GPG_ERR_BUG);
|
|
|
|
|
goto leave;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Tell the user what we are going to do. */
|
|
|
|
|
if (key_attr->algo == PUBKEY_ALGO_RSA)
|
|
|
|
|
{
|
|
|
|
|
tty_printf (_("The card will now be re-configured"
|
|
|
|
|
" to generate a key of %u bits\n"), key_attr->nbits);
|
|
|
|
|
}
|
|
|
|
|
else if (key_attr->algo == PUBKEY_ALGO_ECDH
|
|
|
|
|
|| key_attr->algo == PUBKEY_ALGO_ECDSA
|
|
|
|
|
|| key_attr->algo == PUBKEY_ALGO_EDDSA)
|
|
|
|
|
{
|
|
|
|
|
tty_printf (_("The card will now be re-configured"
|
|
|
|
|
" to generate a key of type: %s\n"), key_attr->curve);
|
|
|
|
|
}
|
|
|
|
|
show_keysize_warning ();
|
|
|
|
|
|
|
|
|
|
*result = key_attr;
|
|
|
|
|
key_attr = NULL;
|
|
|
|
|
|
|
|
|
|
leave:
|
|
|
|
|
xfree (key_attr);
|
|
|
|
|
xfree (answer);
|
|
|
|
|
return err;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Change the key attribute of key KEYNO (0..2) and show an error
|
|
|
|
|
* message if that fails. */
|
|
|
|
|
static gpg_error_t
|
|
|
|
|
do_change_keyattr (int keyno, const struct key_attr *key_attr)
|
|
|
|
|
{
|
|
|
|
|
gpg_error_t err = 0;
|
|
|
|
|
char args[100];
|
|
|
|
|
|
|
|
|
|
if (key_attr->algo == PUBKEY_ALGO_RSA)
|
|
|
|
|
snprintf (args, sizeof args, "--force %d 1 rsa%u", keyno+1,
|
|
|
|
|
key_attr->nbits);
|
|
|
|
|
else if (key_attr->algo == PUBKEY_ALGO_ECDH
|
|
|
|
|
|| key_attr->algo == PUBKEY_ALGO_ECDSA
|
|
|
|
|
|| key_attr->algo == PUBKEY_ALGO_EDDSA)
|
|
|
|
|
snprintf (args, sizeof args, "--force %d %d %s",
|
|
|
|
|
keyno+1, key_attr->algo, key_attr->curve);
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
/* FIXME: Above we use opnepgp algo names but in the error
|
|
|
|
|
* message we use the gcrypt names. We should settle for a
|
|
|
|
|
* consistent solution. */
|
|
|
|
|
log_error (_("public key algorithm %d (%s) is not supported\n"),
|
|
|
|
|
key_attr->algo, gcry_pk_algo_name (key_attr->algo));
|
|
|
|
|
err = gpg_error (GPG_ERR_PUBKEY_ALGO);
|
|
|
|
|
goto leave;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
err = scd_setattr ("KEY-ATTR", args, strlen (args));
|
|
|
|
|
if (err)
|
|
|
|
|
log_error (_("error changing key attribute for key %d: %s\n"),
|
|
|
|
|
keyno+1, gpg_strerror (err));
|
|
|
|
|
leave:
|
|
|
|
|
return err;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
static gpg_error_t
|
|
|
|
|
cmd_keyattr (card_info_t info, char *argstr)
|
|
|
|
|
{
|
|
|
|
|
gpg_error_t err = 0;
|
|
|
|
|
int keyno;
|
|
|
|
|
struct key_attr *key_attr = NULL;
|
|
|
|
|
|
|
|
|
|
(void)argstr;
|
|
|
|
|
|
|
|
|
|
if (!info)
|
|
|
|
|
return print_help
|
|
|
|
|
("KEY-ATTR\n\n"
|
|
|
|
|
"Menu to change the key attributes of an OpenPGP card.",
|
|
|
|
|
APP_TYPE_OPENPGP, 0);
|
|
|
|
|
|
|
|
|
|
if (info->apptype != APP_TYPE_OPENPGP)
|
|
|
|
|
{
|
|
|
|
|
log_info ("Note: This is an OpenPGP only command.\n");
|
|
|
|
|
return gpg_error (GPG_ERR_NOT_SUPPORTED);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (!(info->is_v2 && info->extcap.aac))
|
|
|
|
|
{
|
|
|
|
|
log_error (_("This command is not supported by this card\n"));
|
|
|
|
|
err = gpg_error (GPG_ERR_NOT_SUPPORTED);
|
|
|
|
|
goto leave;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
for (keyno = 0; keyno < DIM (info->key_attr); keyno++)
|
|
|
|
|
{
|
|
|
|
|
xfree (key_attr);
|
|
|
|
|
key_attr = NULL;
|
|
|
|
|
err = ask_card_keyattr (keyno, &info->key_attr[keyno], &key_attr);
|
|
|
|
|
if (err)
|
|
|
|
|
goto leave;
|
|
|
|
|
|
|
|
|
|
err = do_change_keyattr (keyno, key_attr);
|
|
|
|
|
if (err)
|
|
|
|
|
{
|
|
|
|
|
/* Error: Better read the default key attribute again. */
|
|
|
|
|
log_debug ("FIXME\n");
|
|
|
|
|
/* Ask again for this key. */
|
|
|
|
|
keyno--;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
leave:
|
|
|
|
|
xfree (key_attr);
|
|
|
|
|
return err;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
static gpg_error_t
|
|
|
|
|
cmd_uif (card_info_t info, char *argstr)
|
|
|
|
|
{
|
|
|
|
|
gpg_error_t err;
|
|
|
|
|
int keyno;
|
|
|
|
|
|
|
|
|
|
if (!info)
|
|
|
|
|
return print_help
|
|
|
|
|
("UIF N [on|off|permanent]\n\n"
|
|
|
|
|
"Change the User Interaction Flag. N must in the range 1 to 3.",
|
|
|
|
|
APP_TYPE_OPENPGP, APP_TYPE_PIV, 0);
|
|
|
|
|
|
|
|
|
|
argstr = skip_options (argstr);
|
|
|
|
|
|
|
|
|
|
if (digitp (argstr))
|
|
|
|
|
{
|
|
|
|
|
keyno = atoi (argstr);
|
|
|
|
|
while (digitp (argstr))
|
|
|
|
|
argstr++;
|
|
|
|
|
while (spacep (argstr))
|
|
|
|
|
argstr++;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
keyno = 0;
|
|
|
|
|
|
|
|
|
|
if (keyno < 1 || keyno > 3)
|
|
|
|
|
{
|
|
|
|
|
err = gpg_error (GPG_ERR_INV_ARG);
|
|
|
|
|
goto leave;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
err = GPG_ERR_NOT_IMPLEMENTED;
|
|
|
|
|
|
|
|
|
|
leave:
|
|
|
|
|
return err;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Data used by the command parser. This needs to be outside of the
|
|
|
|
|
* function scope to allow readline based command completion. */
|
|
|
|
|
enum cmdids
|
|
|
|
|
{
|
|
|
|
|
cmdNOP = 0,
|
|
|
|
|
cmdQUIT, cmdADMIN, cmdHELP, cmdLIST, cmdRESET, cmdVERIFY,
|
|
|
|
|
cmdNAME, cmdURL, cmdFETCH, cmdLOGIN, cmdLANG, cmdSALUT, cmdCAFPR,
|
|
|
|
|
cmdFORCESIG, cmdGENERATE, cmdPASSWD, cmdPRIVATEDO, cmdWRITECERT,
|
|
|
|
|
cmdREADCERT, cmdUNBLOCK, cmdFACTORYRESET, cmdKDFSETUP,
|
|
|
|
|
cmdKEYATTR, cmdUIF,
|
|
|
|
|
cmdINVCMD
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
static struct
|
|
|
|
|
{
|
|
|
|
|
const char *name;
|
|
|
|
|
enum cmdids id;
|
|
|
|
|
int admin_only;
|
|
|
|
|
const char *desc;
|
|
|
|
|
} cmds[] = {
|
|
|
|
|
{ "quit" , cmdQUIT , 0, N_("quit this menu")},
|
|
|
|
|
{ "q" , cmdQUIT , 0, NULL },
|
|
|
|
|
{ "admin" , cmdADMIN , 0, N_("show admin commands")},
|
|
|
|
|
{ "help" , cmdHELP , 0, N_("show this help")},
|
|
|
|
|
{ "?" , cmdHELP , 0, NULL },
|
|
|
|
|
{ "list" , cmdLIST , 0, N_("list all available data")},
|
|
|
|
|
{ "l" , cmdLIST , 0, NULL },
|
|
|
|
|
{ "name" , cmdNAME , 1, N_("change card holder's name")},
|
|
|
|
|
{ "url" , cmdURL , 1, N_("change URL to retrieve key")},
|
|
|
|
|
{ "fetch" , cmdFETCH , 0, N_("fetch the key specified in the card URL")},
|
|
|
|
|
{ "login" , cmdLOGIN , 1, N_("change the login name")},
|
|
|
|
|
{ "lang" , cmdLANG , 1, N_("change the language preferences")},
|
|
|
|
|
{ "salutation",cmdSALUT, 1, N_("change card holder's salutation")},
|
|
|
|
|
{ "salut" , cmdSALUT, 1, NULL },
|
|
|
|
|
{ "cafpr" , cmdCAFPR , 1, N_("change a CA fingerprint")},
|
|
|
|
|
{ "forcesig", cmdFORCESIG, 1, N_("toggle the signature force PIN flag")},
|
|
|
|
|
{ "generate", cmdGENERATE, 1, N_("generate new keys")},
|
|
|
|
|
{ "passwd" , cmdPASSWD, 0, N_("menu to change or unblock the PIN")},
|
|
|
|
|
{ "verify" , cmdVERIFY, 0, N_("verify the PIN and list all data")},
|
|
|
|
|
{ "unblock" , cmdUNBLOCK,0, N_("unblock the PIN using a Reset Code")},
|
|
|
|
|
{ "factory-reset", cmdFACTORYRESET, 1, N_("destroy all keys and data")},
|
|
|
|
|
{ "kdf-setup", cmdKDFSETUP, 1, N_("setup KDF for PIN authentication")},
|
|
|
|
|
{ "key-attr", cmdKEYATTR, 1, N_("change the key attribute")},
|
|
|
|
|
{ "uif", cmdUIF, 1, N_("change the User Interaction Flag")},
|
|
|
|
|
/* Note, that we do not announce these command yet. */
|
|
|
|
|
{ "privatedo", cmdPRIVATEDO, 0, N_("change a private data object")},
|
|
|
|
|
{ "readcert", cmdREADCERT, 0, N_("read a certificate from a data object")},
|
|
|
|
|
{ "writecert", cmdWRITECERT, 1, N_("store a certificate to a data object")},
|
|
|
|
|
{ NULL, cmdINVCMD, 0, NULL }
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* The main loop. */
|
|
|
|
|
static void
|
|
|
|
|
interactive_loop (void)
|
|
|
|
|
{
|
|
|
|
|
gpg_error_t err;
|
|
|
|
|
char *answer = NULL; /* The input line. */
|
|
|
|
|
enum cmdids cmd = cmdNOP; /* The command. */
|
|
|
|
|
int cmd_admin_only; /* The command is an admin only command. */
|
|
|
|
|
char *argstr; /* The argument as a string. */
|
|
|
|
|
int redisplay = 1; /* Whether to redisplay the main info. */
|
|
|
|
|
int allow_admin = 0; /* Whether admin commands are allowed. */
|
|
|
|
|
char *help_arg = NULL; /* Argument of the HELP command. */
|
|
|
|
|
struct card_info_s info_buffer;
|
|
|
|
|
card_info_t info = &info_buffer;
|
|
|
|
|
char *p;
|
|
|
|
|
int i;
|
|
|
|
|
|
|
|
|
|
/* In the interactive mode we do not want to print the program prefix. */
|
|
|
|
|
log_set_prefix (NULL, 0);
|
|
|
|
|
|
|
|
|
|
for (;;)
|
|
|
|
|
{
|
|
|
|
|
if (help_arg)
|
|
|
|
|
{
|
|
|
|
|
/* Clear info to indicate helpmode */
|
|
|
|
|
info = NULL;
|
|
|
|
|
}
|
|
|
|
|
else if (!info)
|
|
|
|
|
{
|
|
|
|
|
/* Get out of help. */
|
|
|
|
|
info = &info_buffer;
|
|
|
|
|
help_arg = NULL;
|
|
|
|
|
redisplay = 0;
|
|
|
|
|
}
|
|
|
|
|
else if (redisplay)
|
|
|
|
|
{
|
|
|
|
|
err = scd_learn (info);
|
|
|
|
|
if (err)
|
|
|
|
|
{
|
|
|
|
|
log_error ("Error reading card: %s\n", gpg_strerror (err));
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
list_card (info);
|
|
|
|
|
tty_printf("\n");
|
|
|
|
|
redisplay = 0;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (!info)
|
|
|
|
|
{
|
|
|
|
|
/* Copy the pending help arg into our answer. Noe that
|
|
|
|
|
* help_arg points into answer. */
|
|
|
|
|
p = xstrdup (help_arg);
|
|
|
|
|
help_arg = NULL;
|
|
|
|
|
xfree (answer);
|
|
|
|
|
answer = p;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
do
|
|
|
|
|
{
|
|
|
|
|
xfree (answer);
|
|
|
|
|
tty_enable_completion (command_completion);
|
|
|
|
|
answer = tty_get (_("gpg/card> "));
|
|
|
|
|
tty_kill_prompt();
|
|
|
|
|
tty_disable_completion ();
|
|
|
|
|
trim_spaces(answer);
|
|
|
|
|
}
|
|
|
|
|
while ( *answer == '#' );
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
argstr = NULL;
|
|
|
|
|
cmd_admin_only = 0;
|
|
|
|
|
if (!*answer)
|
|
|
|
|
cmd = cmdLIST; /* We default to the list command */
|
|
|
|
|
else if (*answer == CONTROL_D)
|
|
|
|
|
cmd = cmdQUIT;
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
if ((argstr = strchr (answer,' ')))
|
|
|
|
|
{
|
|
|
|
|
*argstr++ = 0;
|
|
|
|
|
trim_spaces (answer);
|
|
|
|
|
trim_spaces (argstr);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
for (i=0; cmds[i].name; i++ )
|
|
|
|
|
if (!ascii_strcasecmp (answer, cmds[i].name ))
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
cmd = cmds[i].id;
|
|
|
|
|
cmd_admin_only = cmds[i].admin_only;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Make sure we have valid strings for the args. They are
|
|
|
|
|
* allowed to be modifed and must thus point to a buffer. */
|
|
|
|
|
if (!argstr)
|
|
|
|
|
argstr = answer + strlen (answer);
|
|
|
|
|
|
|
|
|
|
if (!(cmd == cmdNOP || cmd == cmdQUIT || cmd == cmdHELP))
|
|
|
|
|
{
|
|
|
|
|
/* If redisplay is set we know that there was an error reading
|
|
|
|
|
* the card. In this case we force a LIST command to retry. */
|
|
|
|
|
if (!info)
|
|
|
|
|
; /* In help mode. */
|
|
|
|
|
else if (redisplay)
|
|
|
|
|
{
|
|
|
|
|
cmd = cmdLIST;
|
|
|
|
|
cmd_admin_only = 0;
|
|
|
|
|
}
|
|
|
|
|
else if (!info->serialno)
|
|
|
|
|
{
|
|
|
|
|
/* Without a serial number most commands won't work.
|
|
|
|
|
* Catch it here. */
|
|
|
|
|
tty_printf ("\n");
|
|
|
|
|
tty_printf ("Serial number missing\n");
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
else if (!allow_admin && cmd_admin_only)
|
|
|
|
|
{
|
|
|
|
|
tty_printf ("\n");
|
|
|
|
|
tty_printf (_("Admin-only command\n"));
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
err = 0;
|
|
|
|
|
switch (cmd)
|
|
|
|
|
{
|
|
|
|
|
case cmdNOP:
|
|
|
|
|
if (!info)
|
|
|
|
|
print_help ("NOP\n\n"
|
|
|
|
|
"Dummy command.", 0);
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case cmdQUIT:
|
|
|
|
|
if (!info)
|
|
|
|
|
print_help ("QUIT\n\n"
|
|
|
|
|
"Leave this tool.", 0);
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
tty_printf ("\n");
|
|
|
|
|
goto leave;
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case cmdHELP:
|
|
|
|
|
if (!info)
|
|
|
|
|
print_help ("HELP [command]\n\n"
|
|
|
|
|
"Show all commands. With an argument show help\n"
|
|
|
|
|
"for that command.", 0);
|
|
|
|
|
else if (*argstr)
|
|
|
|
|
help_arg = argstr; /* Trigger help for a command. */
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
tty_printf
|
|
|
|
|
("List of commands (\"help <command>\" for details):\n");
|
|
|
|
|
for (i=0; cmds[i].name; i++ )
|
|
|
|
|
if(cmds[i].desc
|
|
|
|
|
&& (!cmds[i].admin_only
|
|
|
|
|
|| (cmds[i].admin_only && allow_admin)))
|
|
|
|
|
tty_printf("%-14s %s\n", cmds[i].name, _(cmds[i].desc) );
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case cmdLIST:
|
|
|
|
|
if (!info)
|
|
|
|
|
print_help ("LIST\n\n"
|
|
|
|
|
"Show content of the card.", 0);
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
/* Actual work is done by the redisplay code block. */
|
|
|
|
|
redisplay = 1;
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case cmdRESET:
|
|
|
|
|
if (!info)
|
|
|
|
|
print_help ("RESET\n\n"
|
|
|
|
|
"Send a RESET to the card daemon.", 0);
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
err = scd_apdu (NULL, NULL);
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case cmdADMIN:
|
|
|
|
|
if ( !strcmp (argstr, "on") )
|
2019-01-22 09:07:24 +01:00
|
|
|
|
allow_admin = 1;
|
2019-01-27 20:12:00 +01:00
|
|
|
|
else if ( !strcmp (argstr, "off") )
|
2019-01-22 09:07:24 +01:00
|
|
|
|
allow_admin = 0;
|
2019-01-27 20:12:00 +01:00
|
|
|
|
else if ( !strcmp (argstr, "verify") )
|
2019-01-22 09:07:24 +01:00
|
|
|
|
{
|
|
|
|
|
/* Force verification of the Admin Command. However,
|
|
|
|
|
this is only done if the retry counter is at initial
|
|
|
|
|
state. */
|
|
|
|
|
/* FIXME: Must depend on the type of the card. */
|
|
|
|
|
/* char *tmp = xmalloc (strlen (serialnobuf) + 6 + 1); */
|
|
|
|
|
/* strcpy (stpcpy (tmp, serialnobuf), "[CHV3]"); */
|
|
|
|
|
/* allow_admin = !agent_scd_checkpin (tmp); */
|
|
|
|
|
/* xfree (tmp); */
|
|
|
|
|
}
|
|
|
|
|
else /* Toggle. */
|
|
|
|
|
allow_admin=!allow_admin;
|
|
|
|
|
if(allow_admin)
|
|
|
|
|
tty_printf(_("Admin commands are allowed\n"));
|
|
|
|
|
else
|
|
|
|
|
tty_printf(_("Admin commands are not allowed\n"));
|
|
|
|
|
break;
|
|
|
|
|
|
2019-01-27 20:12:00 +01:00
|
|
|
|
case cmdVERIFY:
|
|
|
|
|
err = cmd_verify (info, argstr);
|
|
|
|
|
if (!err)
|
|
|
|
|
redisplay = 1;
|
|
|
|
|
break;
|
|
|
|
|
case cmdNAME: err = cmd_name (info, argstr); break;
|
|
|
|
|
case cmdURL: err = cmd_url (info, argstr); break;
|
|
|
|
|
case cmdFETCH: err = cmd_fetch (info); break;
|
|
|
|
|
case cmdLOGIN: err = cmd_login (info, argstr); break;
|
|
|
|
|
case cmdLANG: err = cmd_lang (info, argstr); break;
|
|
|
|
|
case cmdSALUT: err = cmd_salut (info, argstr); break;
|
|
|
|
|
case cmdCAFPR: err = cmd_cafpr (info, argstr); break;
|
|
|
|
|
case cmdPRIVATEDO: err = cmd_privatedo (info, argstr); break;
|
|
|
|
|
case cmdWRITECERT: err = cmd_writecert (info, argstr); break;
|
|
|
|
|
case cmdREADCERT: err = cmd_readcert (info, argstr); break;
|
|
|
|
|
case cmdFORCESIG: err = cmd_forcesig (info); break;
|
|
|
|
|
case cmdGENERATE: err = cmd_generate (info); break;
|
|
|
|
|
case cmdPASSWD: err = cmd_passwd (info, allow_admin); break;
|
|
|
|
|
case cmdUNBLOCK: err = cmd_unblock (info); break;
|
|
|
|
|
case cmdFACTORYRESET:
|
|
|
|
|
err = cmd_factoryreset (info);
|
|
|
|
|
if (!err)
|
|
|
|
|
redisplay = 1;
|
|
|
|
|
break;
|
|
|
|
|
case cmdKDFSETUP: err = cmd_kdfsetup (info, argstr); break;
|
|
|
|
|
case cmdKEYATTR: err = cmd_keyattr (info, argstr); break;
|
|
|
|
|
case cmdUIF: err = cmd_uif (info, argstr); break;
|
2019-01-22 09:07:24 +01:00
|
|
|
|
|
|
|
|
|
case cmdINVCMD:
|
|
|
|
|
default:
|
|
|
|
|
tty_printf ("\n");
|
|
|
|
|
tty_printf (_("Invalid command (try \"help\")\n"));
|
|
|
|
|
break;
|
|
|
|
|
} /* End command switch. */
|
2019-01-27 20:12:00 +01:00
|
|
|
|
|
|
|
|
|
if (gpg_err_code (err) == GPG_ERR_CANCELED)
|
|
|
|
|
tty_fprintf (NULL, "\n");
|
|
|
|
|
else if (err)
|
|
|
|
|
{
|
|
|
|
|
const char *s = "?";
|
|
|
|
|
for (i=0; cmds[i].name; i++ )
|
|
|
|
|
if (cmd == cmds[i].id)
|
|
|
|
|
{
|
|
|
|
|
s = cmds[i].name;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
log_error ("Command '%s' failed: %s\n", s, gpg_strerror (err));
|
|
|
|
|
}
|
|
|
|
|
|
2019-01-22 09:07:24 +01:00
|
|
|
|
} /* End of main menu loop. */
|
|
|
|
|
|
|
|
|
|
leave:
|
2019-01-27 20:12:00 +01:00
|
|
|
|
release_card_info (info);
|
2019-01-22 09:07:24 +01:00
|
|
|
|
xfree (answer);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#ifdef HAVE_LIBREADLINE
|
|
|
|
|
/* Helper function for readline's command completion. */
|
|
|
|
|
static char *
|
|
|
|
|
command_generator (const char *text, int state)
|
|
|
|
|
{
|
|
|
|
|
static int list_index, len;
|
|
|
|
|
const char *name;
|
|
|
|
|
|
|
|
|
|
/* If this is a new word to complete, initialize now. This includes
|
|
|
|
|
* saving the length of TEXT for efficiency, and initializing the
|
|
|
|
|
index variable to 0. */
|
|
|
|
|
if (!state)
|
|
|
|
|
{
|
|
|
|
|
list_index = 0;
|
|
|
|
|
len = strlen(text);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Return the next partial match */
|
|
|
|
|
while ((name = cmds[list_index].name))
|
|
|
|
|
{
|
|
|
|
|
/* Only complete commands that have help text. */
|
|
|
|
|
if (cmds[list_index++].desc && !strncmp (name, text, len))
|
|
|
|
|
return strdup(name);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return NULL;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Second helper function for readline's command completion. */
|
|
|
|
|
static char **
|
|
|
|
|
command_completion (const char *text, int start, int end)
|
|
|
|
|
{
|
|
|
|
|
(void)end;
|
|
|
|
|
|
|
|
|
|
/* If we are at the start of a line, we try and command-complete.
|
|
|
|
|
* If not, just do nothing for now. */
|
|
|
|
|
if (!start)
|
|
|
|
|
return rl_completion_matches (text, command_generator);
|
|
|
|
|
|
|
|
|
|
rl_attempted_completion_over = 1;
|
|
|
|
|
|
|
|
|
|
return NULL;
|
|
|
|
|
}
|
|
|
|
|
#endif /*HAVE_LIBREADLINE*/
|