Cygwin: FIFO: allow take_ownership to be interrupted
Use cygwait in take_ownership to allow interruption while waiting to become owner. Return the cygwait return value or a suitable value to indicate an error. raw_read now checks the return value and acts accordingly.
This commit is contained in:
@@ -1489,7 +1489,7 @@ public:
|
|||||||
void owner_lock () { shmem->owner_lock (); }
|
void owner_lock () { shmem->owner_lock (); }
|
||||||
void owner_unlock () { shmem->owner_unlock (); }
|
void owner_unlock () { shmem->owner_unlock (); }
|
||||||
|
|
||||||
void take_ownership ();
|
DWORD take_ownership ();
|
||||||
void reading_lock () { shmem->reading_lock (); }
|
void reading_lock () { shmem->reading_lock (); }
|
||||||
void reading_unlock () { shmem->reading_unlock (); }
|
void reading_unlock () { shmem->reading_unlock (); }
|
||||||
|
|
||||||
|
@@ -1175,15 +1175,16 @@ fhandler_fifo::raw_write (const void *ptr, size_t len)
|
|||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Called from raw_read and select.cc:peek_fifo. */
|
/* Called from raw_read and select.cc:peek_fifo. Return WAIT_OBJECT_0
|
||||||
void
|
on success. */
|
||||||
|
DWORD
|
||||||
fhandler_fifo::take_ownership ()
|
fhandler_fifo::take_ownership ()
|
||||||
{
|
{
|
||||||
owner_lock ();
|
owner_lock ();
|
||||||
if (get_owner () == me)
|
if (get_owner () == me)
|
||||||
{
|
{
|
||||||
owner_unlock ();
|
owner_unlock ();
|
||||||
return;
|
return WAIT_OBJECT_0;
|
||||||
}
|
}
|
||||||
set_pending_owner (me);
|
set_pending_owner (me);
|
||||||
/* Wake up my fifo_reader_thread. */
|
/* Wake up my fifo_reader_thread. */
|
||||||
@@ -1192,8 +1193,19 @@ fhandler_fifo::take_ownership ()
|
|||||||
/* Wake up owner's fifo_reader_thread. */
|
/* Wake up owner's fifo_reader_thread. */
|
||||||
SetEvent (update_needed_evt);
|
SetEvent (update_needed_evt);
|
||||||
owner_unlock ();
|
owner_unlock ();
|
||||||
/* The reader threads should now do the transfer. */
|
/* The reader threads should now do the transfer. */
|
||||||
WaitForSingleObject (owner_found_evt, INFINITE);
|
DWORD waitret = cygwait (owner_found_evt, cw_cancel | cw_sig_eintr);
|
||||||
|
owner_lock ();
|
||||||
|
if (waitret == WAIT_OBJECT_0
|
||||||
|
&& (get_owner () != me || get_pending_owner ()))
|
||||||
|
{
|
||||||
|
/* Something went wrong. Return WAIT_TIMEOUT, which can't be
|
||||||
|
returned by the above cygwait call. */
|
||||||
|
set_pending_owner (null_fr_id);
|
||||||
|
waitret = WAIT_TIMEOUT;
|
||||||
|
}
|
||||||
|
owner_unlock ();
|
||||||
|
return waitret;
|
||||||
}
|
}
|
||||||
|
|
||||||
void __reg3
|
void __reg3
|
||||||
@@ -1206,7 +1218,37 @@ fhandler_fifo::raw_read (void *in_ptr, size_t& len)
|
|||||||
{
|
{
|
||||||
/* No one else can take ownership while we hold the reading_lock. */
|
/* No one else can take ownership while we hold the reading_lock. */
|
||||||
reading_lock ();
|
reading_lock ();
|
||||||
take_ownership ();
|
switch (take_ownership ())
|
||||||
|
{
|
||||||
|
case WAIT_OBJECT_0:
|
||||||
|
break;
|
||||||
|
case WAIT_SIGNALED:
|
||||||
|
if (_my_tls.call_signal_handler ())
|
||||||
|
{
|
||||||
|
reading_unlock ();
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
set_errno (EINTR);
|
||||||
|
reading_unlock ();
|
||||||
|
goto errout;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case WAIT_CANCELED:
|
||||||
|
reading_unlock ();
|
||||||
|
pthread::static_cancel_self ();
|
||||||
|
break;
|
||||||
|
case WAIT_TIMEOUT:
|
||||||
|
reading_unlock ();
|
||||||
|
debug_printf ("take_ownership returned an unexpected result; retry");
|
||||||
|
continue;
|
||||||
|
default:
|
||||||
|
reading_unlock ();
|
||||||
|
debug_printf ("unknown error while trying to take ownership, %E");
|
||||||
|
goto errout;
|
||||||
|
}
|
||||||
|
|
||||||
/* Poll the connected clients for input. Make two passes. On
|
/* Poll the connected clients for input. Make two passes. On
|
||||||
the first pass, just try to read from the client from which
|
the first pass, just try to read from the client from which
|
||||||
we last read successfully. This should minimize
|
we last read successfully. This should minimize
|
||||||
|
@@ -867,7 +867,16 @@ peek_fifo (select_record *s, bool from_select)
|
|||||||
}
|
}
|
||||||
|
|
||||||
fh->reading_lock ();
|
fh->reading_lock ();
|
||||||
fh->take_ownership ();
|
if (fh->take_ownership () != WAIT_OBJECT_0)
|
||||||
|
{
|
||||||
|
select_printf ("%s, unable to take ownership", fh->get_name ());
|
||||||
|
fh->reading_unlock ();
|
||||||
|
gotone += s->read_ready = true;
|
||||||
|
if (s->except_selected)
|
||||||
|
gotone += s->except_ready = true;
|
||||||
|
goto out;
|
||||||
|
}
|
||||||
|
|
||||||
fh->fifo_client_lock ();
|
fh->fifo_client_lock ();
|
||||||
int nconnected = 0;
|
int nconnected = 0;
|
||||||
for (int i = 0; i < fh->get_nhandlers (); i++)
|
for (int i = 0; i < fh->get_nhandlers (); i++)
|
||||||
|
Reference in New Issue
Block a user