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

scd: Finish DEVINFO --watch command on input close.

* scd/app.c (card_list_signal): Use pipe on POSIX system, event on
Windows.
(card_list_wait): Detect input change as well as card list event
change.
(app_send_devinfo): Finish the command on input close.
(initialize_module_command): Initialize pipe or event.

--

GnuPG-bug-id: 7151
Signed-off-by: NIIBE Yutaka <gniibe@fsij.org>
(cherry picked from commit 36d8cffc6cd2838e7cb439c566fdd2b3dd076c15)
This commit is contained in:
NIIBE Yutaka 2024-06-25 15:29:30 +09:00 committed by Werner Koch
parent fc732131a1
commit 6996e5f6ff
No known key found for this signature in database
GPG Key ID: E3FDFF218E45B72B

View File

@ -49,7 +49,12 @@ struct mrsw_lock
int num_readers_active; int num_readers_active;
int num_writers_waiting; int num_writers_waiting;
int writer_active; int writer_active;
npth_cond_t notify_cond; #ifdef HAVE_W32_SYSTEM
HANDLE events[2];
HANDLE the_event;
#else
int notify_pipe[2];
#endif
}; };
/* MRSW lock to protect the list of cards. /* MRSW lock to protect the list of cards.
@ -366,17 +371,79 @@ card_list_w_unlock (void)
static void static void
card_list_signal (void) card_list_signal (void)
{ {
npth_cond_broadcast (&card_list_lock.notify_cond); #ifdef HAVE_W32_SYSTEM
if (SetEvent (card_list_lock.the_event) == 0)
log_error ("SetEvent for card_list_signal failed: %s\n",
w32_strerror (-1));
#else
write (card_list_lock.notify_pipe[1], "", 1);
#endif
} }
static void /* Wait for some card list event or input shutdown.
card_list_wait (void) Return 1 on input shutdown, return 0 otherwise. */
static int
card_list_wait (ctrl_t ctrl)
{ {
gnupg_fd_t fd = ctrl->thread_startup.fd;
int nfd;
fd_set fdset;
#ifdef HAVE_W32_SYSTEM
unsigned int events_set;
#endif
int ret;
npth_mutex_lock (&card_list_lock.lock); npth_mutex_lock (&card_list_lock.lock);
card_list_lock.writer_active--; card_list_lock.writer_active--;
npth_cond_broadcast (&card_list_lock.cond); npth_cond_broadcast (&card_list_lock.cond);
npth_cond_wait (&card_list_lock.notify_cond, &card_list_lock.lock); npth_mutex_unlock (&card_list_lock.lock);
while (1)
{
FD_ZERO (&fdset);
FD_SET (FD2INT (fd), &fdset);
nfd = FD2NUM (fd);
#ifdef HAVE_W32_SYSTEM
ret = npth_eselect (nfd+1, &fdset, NULL, NULL, NULL,
card_list_lock.events, &events_set);
#else
FD_SET (card_list_lock.notify_pipe[0], &fdset);
if (nfd < card_list_lock.notify_pipe[0])
nfd = card_list_lock.notify_pipe[0];
ret = npth_pselect (nfd+1, &fdset, NULL, NULL, NULL,
npth_sigev_sigmask ());
#endif
if (ret <= 0)
continue;
if (FD_ISSET (fd, &fdset))
{
ret = 1;
break;
}
#ifdef HAVE_W32_SYSTEM
if (events_set & 1)
{
ret = 0;
break;
}
#else
if (FD_ISSET (card_list_lock.notify_pipe[0], &fdset))
{
char buf[256];
read (card_list_lock.notify_pipe[0], buf, sizeof buf);
ret = 0;
break;
}
#endif
}
npth_mutex_lock (&card_list_lock.lock);
card_list_lock.num_writers_waiting++; card_list_lock.num_writers_waiting++;
while (card_list_lock.num_readers_active while (card_list_lock.num_readers_active
@ -386,6 +453,7 @@ card_list_wait (void)
card_list_lock.writer_active++; card_list_lock.writer_active++;
npth_mutex_unlock (&card_list_lock.lock); npth_mutex_unlock (&card_list_lock.lock);
return ret;
} }
@ -451,7 +519,8 @@ app_send_devinfo (ctrl_t ctrl, int keep_looping)
if (keep_looping == 0) if (keep_looping == 0)
break; break;
card_list_wait (); if (card_list_wait (ctrl))
break;
} }
card_list_w_unlock (); card_list_w_unlock ();
@ -2530,6 +2599,9 @@ gpg_error_t
initialize_module_command (void) initialize_module_command (void)
{ {
gpg_error_t err; gpg_error_t err;
#ifndef HAVE_W32_SYSTEM
int ret;
#endif
card_list_lock.num_readers_active = 0; card_list_lock.num_readers_active = 0;
card_list_lock.num_writers_waiting = 0; card_list_lock.num_writers_waiting = 0;
@ -2550,13 +2622,17 @@ initialize_module_command (void)
return err; return err;
} }
err = npth_cond_init (&card_list_lock.notify_cond, NULL); #ifdef HAVE_W32_SYSTEM
if (err) scd_init_event (&card_list_lock.the_event, card_list_lock.events);
#else
ret = gnupg_create_pipe (card_list_lock.notify_pipe);
if (ret)
{ {
err = gpg_error_from_syserror (); err = gpg_error_from_syserror ();
log_error ("npth_cond_init failed: %s\n", gpg_strerror (err)); log_error ("pipe creation failed: %s\n", gpg_strerror (ret));
return err; return err;
} }
#endif
return apdu_init (); return apdu_init ();
} }