1
0
mirror of git://git.gnupg.org/gnupg.git synced 2025-01-18 14:17:03 +01:00

agent: Fix ask_for_card to allow a key on multiple cards.

* agent/divert-scd.c (ask_for_card): Don't use SERIALNO to select
card, but use KEYGRIP.

GnuPG-bug-id: 4695
Signed-off-by: NIIBE Yutaka <gniibe@fsij.org>
This commit is contained in:
NIIBE Yutaka 2019-09-05 17:18:39 +09:00
parent fed9c93e05
commit 9f39e0167d

View File

@ -36,128 +36,81 @@ static gpg_error_t
ask_for_card (ctrl_t ctrl, const unsigned char *shadow_info, ask_for_card (ctrl_t ctrl, const unsigned char *shadow_info,
const unsigned char *grip, char **r_kid) const unsigned char *grip, char **r_kid)
{ {
int i;
char *serialno; char *serialno;
int no_card = 0;
char *desc; char *desc;
char *want_sn, *want_kid, *want_sn_disp; char *want_sn;
int len; int len;
struct card_key_info_s *keyinfo;
gpg_error_t err; gpg_error_t err;
char hexgrip[41]; char hexgrip[41];
*r_kid = NULL; *r_kid = NULL;
bin2hex (grip, 20, hexgrip);
/* Scan device(s), and check if key for GRIP is available. */ err = parse_shadow_info (shadow_info, &want_sn, NULL, NULL);
err = agent_card_serialno (ctrl, &serialno, NULL);
if (!err)
{
xfree (serialno);
bin2hex (grip, 20, hexgrip);
err = agent_card_keyinfo (ctrl, hexgrip, &keyinfo);
if (!err)
{
/* Key for GRIP found, use it directly. */
agent_card_free_keyinfo (keyinfo);
if ((*r_kid = xtrystrdup (hexgrip)))
return 0;
else
return gpg_error_from_syserror ();
}
}
err = parse_shadow_info (shadow_info, &want_sn, &want_kid, NULL);
if (err) if (err)
return err; return err;
want_sn_disp = xtrystrdup (want_sn);
if (!want_sn_disp)
{
err = gpg_error_from_syserror ();
xfree (want_sn);
xfree (want_kid);
return err;
}
len = strlen (want_sn_disp); len = strlen (want_sn);
if (len == 32 && !strncmp (want_sn_disp, "D27600012401", 12)) if (len == 32 && !strncmp (want_sn, "D27600012401", 12))
{ {
/* This is an OpenPGP card - reformat */ /* This is an OpenPGP card - reformat */
memmove (want_sn_disp, want_sn_disp+16, 4); memmove (want_sn, want_sn+16, 4);
want_sn_disp[4] = ' '; want_sn[4] = ' ';
memmove (want_sn_disp+5, want_sn_disp+20, 8); memmove (want_sn+5, want_sn+20, 8);
want_sn_disp[13] = 0; want_sn[13] = 0;
} }
else if (len == 20 && want_sn_disp[19] == '0') else if (len == 20 && want_sn[19] == '0')
{ {
/* We assume that a 20 byte serial number is a standard one /* We assume that a 20 byte serial number is a standard one
* which has the property to have a zero in the last nibble (Due * which has the property to have a zero in the last nibble (Due
* to BCD representation). We don't display this '0' because it * to BCD representation). We don't display this '0' because it
* may confuse the user. */ * may confuse the user. */
want_sn_disp[19] = 0; want_sn[19] = 0;
} }
for (;;) for (;;)
{ {
err = agent_card_serialno (ctrl, &serialno, want_sn); /* Scan device(s), and check if key for GRIP is available. */
err = agent_card_serialno (ctrl, &serialno, NULL);
if (!err) if (!err)
{ {
log_debug ("detected card with S/N %s\n", serialno); struct card_key_info_s *keyinfo;
i = strcmp (serialno, want_sn);
xfree (serialno); xfree (serialno);
serialno = NULL; err = agent_card_keyinfo (ctrl, hexgrip, &keyinfo);
if (!i) if (!err)
{ {
xfree (want_sn_disp); /* Key for GRIP found, use it directly. */
agent_card_free_keyinfo (keyinfo);
xfree (want_sn); xfree (want_sn);
*r_kid = want_kid; if ((*r_kid = xtrystrdup (hexgrip)))
return 0; /* yes, we have the correct card */ return 0;
else
return gpg_error_from_syserror ();
} }
} }
else if (gpg_err_code (err) == GPG_ERR_ENODEV)
if (asprintf (&desc,
"%s:%%0A%%0A"
" %s",
L_("Please insert the card with serial number"),
want_sn) < 0)
{ {
log_debug ("no device present\n"); err = out_of_core ();
err = 0;
no_card = 1;
}
else if (gpg_err_code (err) == GPG_ERR_CARD_NOT_PRESENT)
{
log_debug ("no card present\n");
err = 0;
no_card = 2;
} }
else else
{ {
log_error ("error accessing card: %s\n", gpg_strerror (err)); err = agent_get_confirmation (ctrl, desc, NULL, NULL, 0);
if (ctrl->pinentry_mode == PINENTRY_MODE_LOOPBACK &&
gpg_err_code (err) == GPG_ERR_NO_PIN_ENTRY)
err = gpg_error (GPG_ERR_CARD_NOT_PRESENT);
xfree (desc);
} }
if (!err)
{
if (asprintf (&desc,
"%s:%%0A%%0A"
" %s",
no_card
? L_("Please insert the card with serial number")
: L_("Please remove the current card and "
"insert the one with serial number"),
want_sn_disp) < 0)
{
err = out_of_core ();
}
else
{
err = agent_get_confirmation (ctrl, desc, NULL, NULL, 0);
if (ctrl->pinentry_mode == PINENTRY_MODE_LOOPBACK &&
gpg_err_code (err) == GPG_ERR_NO_PIN_ENTRY)
err = gpg_error (GPG_ERR_CARD_NOT_PRESENT);
xfree (desc);
}
}
if (err) if (err)
{ {
xfree (want_sn_disp);
xfree (want_sn); xfree (want_sn);
xfree (want_kid);
return err; return err;
} }
} }