mirror of
git://git.gnupg.org/gnupg.git
synced 2025-01-17 14:07:03 +01:00
scd: Fix resetting and closing of the reader.
* scd/command.c (update_card_removed): Do no act on an invalid VRDR. (do_reset): Ignore apdu_reset error codes for no and inactive card. Close the reader before setting the slot to -1. (update_reader_status_file): Notify the application before closing the reader. -- With this change the scd now works as it did in the past. In particular there is no more endless loop trying to open the reader by the update_reader_status_file ticker function. That bug basically blocked all card operations until the scdaemon was killed.
This commit is contained in:
parent
07ea8c56b5
commit
2d91febbd8
@ -180,25 +180,6 @@ initialize_module_command (void)
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
/* Update the CARD_REMOVED element of all sessions using the virtual
|
|
||||||
reader given by VRDR to VALUE. */
|
|
||||||
static void
|
|
||||||
update_card_removed (int vrdr, int value)
|
|
||||||
{
|
|
||||||
struct server_local_s *sl;
|
|
||||||
|
|
||||||
for (sl=session_list; sl; sl = sl->next_session)
|
|
||||||
if (sl->ctrl_backlink
|
|
||||||
&& sl->ctrl_backlink->server_local->vreader_idx == vrdr)
|
|
||||||
{
|
|
||||||
sl->card_removed = value;
|
|
||||||
}
|
|
||||||
/* Let the card application layer know about the removal. */
|
|
||||||
if (value)
|
|
||||||
application_notify_card_reset (vrdr);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/* Helper to return the slot number for a given virtual reader index
|
/* Helper to return the slot number for a given virtual reader index
|
||||||
VRDR. In case on an error -1 is returned. */
|
VRDR. In case on an error -1 is returned. */
|
||||||
static int
|
static int
|
||||||
@ -212,6 +193,28 @@ vreader_slot (int vrdr)
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/* Update the CARD_REMOVED element of all sessions using the virtual
|
||||||
|
reader given by VRDR to VALUE. */
|
||||||
|
static void
|
||||||
|
update_card_removed (int vrdr, int value)
|
||||||
|
{
|
||||||
|
struct server_local_s *sl;
|
||||||
|
|
||||||
|
if (vrdr == -1)
|
||||||
|
return;
|
||||||
|
|
||||||
|
for (sl=session_list; sl; sl = sl->next_session)
|
||||||
|
if (sl->ctrl_backlink
|
||||||
|
&& sl->ctrl_backlink->server_local->vreader_idx == vrdr)
|
||||||
|
{
|
||||||
|
sl->card_removed = value;
|
||||||
|
}
|
||||||
|
/* Let the card application layer know about the removal. */
|
||||||
|
if (value)
|
||||||
|
application_notify_card_reset (vreader_slot (vrdr));
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
/* Check whether the option NAME appears in LINE. Returns 1 or 0. */
|
/* Check whether the option NAME appears in LINE. Returns 1 or 0. */
|
||||||
static int
|
static int
|
||||||
has_option (const char *line, const char *name)
|
has_option (const char *line, const char *name)
|
||||||
@ -298,6 +301,7 @@ static void
|
|||||||
do_reset (ctrl_t ctrl, int send_reset)
|
do_reset (ctrl_t ctrl, int send_reset)
|
||||||
{
|
{
|
||||||
int vrdr = ctrl->server_local->vreader_idx;
|
int vrdr = ctrl->server_local->vreader_idx;
|
||||||
|
int slot;
|
||||||
|
|
||||||
if (!(vrdr == -1 || (vrdr >= 0 && vrdr < DIM(vreader_table))))
|
if (!(vrdr == -1 || (vrdr >= 0 && vrdr < DIM(vreader_table))))
|
||||||
BUG ();
|
BUG ();
|
||||||
@ -324,13 +328,22 @@ do_reset (ctrl_t ctrl, int send_reset)
|
|||||||
|
|
||||||
/* If we want a real reset for the card, send the reset APDU and
|
/* If we want a real reset for the card, send the reset APDU and
|
||||||
tell the application layer about it. */
|
tell the application layer about it. */
|
||||||
if (vrdr != -1 && send_reset && !IS_LOCKED (ctrl) )
|
slot = vreader_slot (vrdr);
|
||||||
|
if (slot != -1 && send_reset && !IS_LOCKED (ctrl) )
|
||||||
{
|
{
|
||||||
if (apdu_reset (vreader_table[vrdr].slot))
|
application_notify_card_reset (slot);
|
||||||
|
switch (apdu_reset (slot))
|
||||||
{
|
{
|
||||||
vreader_table[vrdr].valid = 0;
|
case 0:
|
||||||
|
break;
|
||||||
|
case SW_HOST_NO_CARD:
|
||||||
|
case SW_HOST_CARD_INACTIVE:
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
apdu_close_reader (slot);
|
||||||
|
vreader_table[vrdr].slot = slot = -1;
|
||||||
|
break;
|
||||||
}
|
}
|
||||||
application_notify_card_reset (vrdr);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/* If we hold a lock, unlock now. */
|
/* If we hold a lock, unlock now. */
|
||||||
@ -1696,11 +1709,7 @@ cmd_getinfo (assuan_context_t ctx, char *line)
|
|||||||
BUG ();
|
BUG ();
|
||||||
|
|
||||||
vr = &vreader_table[vrdr];
|
vr = &vreader_table[vrdr];
|
||||||
|
if (vr->valid && vr->any && (vr->status & 1))
|
||||||
if (!vr->valid)
|
|
||||||
BUG ();
|
|
||||||
|
|
||||||
if (vr->any && (vr->status & 1))
|
|
||||||
flag = 'u';
|
flag = 'u';
|
||||||
}
|
}
|
||||||
rc = assuan_send_data (ctx, &flag, 1);
|
rc = assuan_send_data (ctx, &flag, 1);
|
||||||
@ -2248,9 +2257,9 @@ update_reader_status_file (int set_card_removed_flag)
|
|||||||
if (sw_apdu == SW_HOST_NO_READER)
|
if (sw_apdu == SW_HOST_NO_READER)
|
||||||
{
|
{
|
||||||
/* Most likely the _reader_ has been unplugged. */
|
/* Most likely the _reader_ has been unplugged. */
|
||||||
|
application_notify_card_reset (vr->slot);
|
||||||
apdu_close_reader (vr->slot);
|
apdu_close_reader (vr->slot);
|
||||||
vr->slot = -1;
|
vr->slot = -1;
|
||||||
vr->valid = 0;
|
|
||||||
status = 0;
|
status = 0;
|
||||||
changed = vr->changed;
|
changed = vr->changed;
|
||||||
}
|
}
|
||||||
|
Loading…
x
Reference in New Issue
Block a user