1
0
mirror of git://git.gnupg.org/gnupg.git synced 2025-01-20 14:37:08 +01:00

scd:ccid:spr532: Extend abort_cmd for initialization time.

* scd/ccid-driver.c (abort_cmd): Add INIT argument to support
synchronize until success, even ignoring timeout.
(bulk_in): Normal use case of abort_cmd.
(ccid_vendor_specific_init): Initial use case of abort_cmd.

--

Another backport to stabilize SCM SPR332/SPR532 card reader.

GnuPG-bug-id: 5297
Backport-master-commit: a9aa30ed2c2c399c2baa6a5aa2624d8fdee6286f
Signed-off-by: NIIBE Yutaka <gniibe@fsij.org>
This commit is contained in:
NIIBE Yutaka 2020-12-09 12:54:43 +09:00
parent af2fd9f0af
commit 8e941e19b0

View File

@ -278,7 +278,7 @@ static int bulk_out (ccid_driver_t handle, unsigned char *msg, size_t msglen,
static int bulk_in (ccid_driver_t handle, unsigned char *buffer, size_t length, static int bulk_in (ccid_driver_t handle, unsigned char *buffer, size_t length,
size_t *nread, int expected_type, int seqno, int timeout, size_t *nread, int expected_type, int seqno, int timeout,
int no_debug); int no_debug);
static int abort_cmd (ccid_driver_t handle, int seqno); static int abort_cmd (ccid_driver_t handle, int seqno, int init);
static int send_escape_cmd (ccid_driver_t handle, const unsigned char *data, static int send_escape_cmd (ccid_driver_t handle, const unsigned char *data,
size_t datalen, unsigned char *result, size_t datalen, unsigned char *result,
size_t resultmax, size_t *resultlen); size_t resultmax, size_t *resultlen);
@ -1288,7 +1288,7 @@ ccid_vendor_specific_init (ccid_driver_t handle)
* It seems that SEQ may be out of sync between host and the card reader, * It seems that SEQ may be out of sync between host and the card reader,
* and SET_INTERFACE doesn't reset it. Make sure it works at the init. * and SET_INTERFACE doesn't reset it. Make sure it works at the init.
*/ */
abort_cmd (handle, 0); abort_cmd (handle, 0, 1);
} }
if (r != 0 && r != CCID_DRIVER_ERR_CARD_INACTIVE if (r != 0 && r != CCID_DRIVER_ERR_CARD_INACTIVE
@ -1902,6 +1902,8 @@ do_close_reader (ccid_driver_t handle)
libusb_free_transfer (handle->transfer); libusb_free_transfer (handle->transfer);
handle->transfer = NULL; handle->transfer = NULL;
} }
DEBUGOUT ("libusb_release_interface and libusb_close\n");
libusb_release_interface (handle->idev, handle->ifc_no); libusb_release_interface (handle->idev, handle->ifc_no);
--ccid_usb_thread_is_alive; --ccid_usb_thread_is_alive;
libusb_close (handle->idev); libusb_close (handle->idev);
@ -2095,7 +2097,7 @@ bulk_in (ccid_driver_t handle, unsigned char *buffer, size_t length,
if (msglen < 10) if (msglen < 10)
{ {
DEBUGOUT_1 ("bulk-in msg too short (%u)\n", (unsigned int)msglen); DEBUGOUT_1 ("bulk-in msg too short (%u)\n", (unsigned int)msglen);
abort_cmd (handle, seqno); abort_cmd (handle, seqno, 0);
return CCID_DRIVER_ERR_INV_VALUE; return CCID_DRIVER_ERR_INV_VALUE;
} }
if (buffer[5] != 0) if (buffer[5] != 0)
@ -2141,7 +2143,7 @@ bulk_in (ccid_driver_t handle, unsigned char *buffer, size_t length,
if (buffer[0] != expected_type && buffer[0] != RDR_to_PC_SlotStatus) if (buffer[0] != expected_type && buffer[0] != RDR_to_PC_SlotStatus)
{ {
DEBUGOUT_1 ("unexpected bulk-in msg type (%02x)\n", buffer[0]); DEBUGOUT_1 ("unexpected bulk-in msg type (%02x)\n", buffer[0]);
abort_cmd (handle, seqno); abort_cmd (handle, seqno, 0);
return CCID_DRIVER_ERR_INV_VALUE; return CCID_DRIVER_ERR_INV_VALUE;
} }
@ -2207,7 +2209,7 @@ bulk_in (ccid_driver_t handle, unsigned char *buffer, size_t length,
/* Send an abort sequence and wait until everything settled. */ /* Send an abort sequence and wait until everything settled. */
static int static int
abort_cmd (ccid_driver_t handle, int seqno) abort_cmd (ccid_driver_t handle, int seqno, int init)
{ {
int rc; int rc;
unsigned char dummybuf[8]; unsigned char dummybuf[8];
@ -2236,6 +2238,7 @@ abort_cmd (ccid_driver_t handle, int seqno)
if (rc) if (rc)
{ {
DEBUGOUT_1 ("usb_control_msg error: %s\n", libusb_error_name (rc)); DEBUGOUT_1 ("usb_control_msg error: %s\n", libusb_error_name (rc));
if (!init)
return map_libusb_error (rc); return map_libusb_error (rc);
} }
@ -2262,7 +2265,7 @@ abort_cmd (ccid_driver_t handle, int seqno)
#endif #endif
rc = libusb_bulk_transfer (handle->idev, handle->ep_bulk_out, rc = libusb_bulk_transfer (handle->idev, handle->ep_bulk_out,
msg, msglen, &transferred, msg, msglen, &transferred,
5000 /* ms timeout */); init? 100: 5000 /* ms timeout */);
#ifdef USE_NPTH #ifdef USE_NPTH
npth_protect (); npth_protect ();
#endif #endif
@ -2280,7 +2283,7 @@ abort_cmd (ccid_driver_t handle, int seqno)
#endif #endif
rc = libusb_bulk_transfer (handle->idev, handle->ep_bulk_in, rc = libusb_bulk_transfer (handle->idev, handle->ep_bulk_in,
msg, sizeof msg, &msglen, msg, sizeof msg, &msglen,
5000 /*ms timeout*/); init? 100: 5000 /*ms timeout*/);
#ifdef USE_NPTH #ifdef USE_NPTH
npth_protect (); npth_protect ();
#endif #endif
@ -2288,6 +2291,9 @@ abort_cmd (ccid_driver_t handle, int seqno)
{ {
DEBUGOUT_1 ("usb_bulk_read error in abort_cmd: %s\n", DEBUGOUT_1 ("usb_bulk_read error in abort_cmd: %s\n",
libusb_error_name (rc)); libusb_error_name (rc));
if (init && rc == LIBUSB_ERROR_TIMEOUT)
continue;
else
return map_libusb_error (rc); return map_libusb_error (rc);
} }
@ -2308,7 +2314,8 @@ abort_cmd (ccid_driver_t handle, int seqno)
if (CCID_COMMAND_FAILED (msg)) if (CCID_COMMAND_FAILED (msg))
print_command_failed (msg); print_command_failed (msg);
} }
while (msg[0] != RDR_to_PC_SlotStatus && msg[5] != 0 && msg[6] != seqno); while (rc == LIBUSB_ERROR_TIMEOUT
|| (msg[0] != RDR_to_PC_SlotStatus && msg[5] != 0 && msg[6] != seqno));
handle->seqno = ((seqno + 1) & 0xff); handle->seqno = ((seqno + 1) & 0xff);
DEBUGOUT ("sending abort sequence succeeded\n"); DEBUGOUT ("sending abort sequence succeeded\n");