mirror of
https://gitlab.com/libvirt/libvirt.git
synced 2025-01-20 03:25:18 +00:00
client rpc: Just queue non-blocking call if another thread has the buck
As non-blocking calls are no longer dropped, we don't really need to care that much about their fate and wait for the thread with the buck to process them. If another thread has the buck, we can just push a non-blocking call to the queue and be done with it. (cherry picked from commit ef392614aa585d168e618ffaf2c83545d81f0113)
This commit is contained in:
parent
5badf8c44b
commit
4779cf0ff5
@ -1188,20 +1188,11 @@ static bool virNetClientIOEventLoopRemoveDone(virNetClientCallPtr call,
|
||||
}
|
||||
|
||||
|
||||
static bool
|
||||
virNetClientIOEventLoopDetachNonBlocking(virNetClientCallPtr call,
|
||||
void *opaque)
|
||||
static void
|
||||
virNetClientIODetachNonBlocking(virNetClientCallPtr call)
|
||||
{
|
||||
virNetClientCallPtr thiscall = opaque;
|
||||
|
||||
if (call != thiscall && call->nonBlock && call->haveThread) {
|
||||
VIR_DEBUG("Waking up sleep %p", call);
|
||||
call->haveThread = false;
|
||||
virCondSignal(&call->cond);
|
||||
return true;
|
||||
}
|
||||
|
||||
return false;
|
||||
VIR_DEBUG("Keeping unfinished non-blocking call %p in the queue", call);
|
||||
call->haveThread = false;
|
||||
}
|
||||
|
||||
|
||||
@ -1250,13 +1241,6 @@ virNetClientIOEventLoopPassTheBuck(virNetClientPtr client,
|
||||
}
|
||||
|
||||
|
||||
static bool
|
||||
virNetClientIOEventLoopWantNonBlock(virNetClientCallPtr call,
|
||||
void *opaque ATTRIBUTE_UNUSED)
|
||||
{
|
||||
return call->nonBlock && call->haveThread;
|
||||
}
|
||||
|
||||
/*
|
||||
* Process all calls pending dispatch/receive until we
|
||||
* get a reply to our own call. Then quit and pass the buck
|
||||
@ -1287,12 +1271,8 @@ static int virNetClientIOEventLoop(virNetClientPtr client,
|
||||
if (virNetSocketHasCachedData(client->sock) || client->wantClose)
|
||||
timeout = 0;
|
||||
|
||||
/* If there are any non-blocking calls with an associated thread
|
||||
* in the queue, then we don't want to sleep in poll()
|
||||
*/
|
||||
if (virNetClientCallMatchPredicate(client->waitDispatch,
|
||||
virNetClientIOEventLoopWantNonBlock,
|
||||
NULL))
|
||||
/* If we are non-blocking, then we don't want to sleep in poll() */
|
||||
if (thiscall->nonBlock)
|
||||
timeout = 0;
|
||||
|
||||
fds[0].events = fds[0].revents = 0;
|
||||
@ -1359,19 +1339,6 @@ static int virNetClientIOEventLoop(virNetClientPtr client,
|
||||
_("read on wakeup fd failed"));
|
||||
goto error;
|
||||
}
|
||||
|
||||
/* If we were woken up because a new non-blocking call was queued,
|
||||
* we need to re-poll to check if we can send it. To be precise, we
|
||||
* will re-poll even if a blocking call arrived when unhandled
|
||||
* non-blocking calls are still in the queue. But this can't hurt.
|
||||
*/
|
||||
if (virNetClientCallMatchPredicate(client->waitDispatch,
|
||||
virNetClientIOEventLoopWantNonBlock,
|
||||
NULL)) {
|
||||
VIR_DEBUG("The queue contains new non-blocking call(s);"
|
||||
" repolling");
|
||||
continue;
|
||||
}
|
||||
}
|
||||
|
||||
if (ret < 0) {
|
||||
@ -1400,13 +1367,6 @@ static int virNetClientIOEventLoop(virNetClientPtr client,
|
||||
virNetClientIOEventLoopRemoveDone,
|
||||
thiscall);
|
||||
|
||||
/* Iterate through waiting calls and wake up and detach threads
|
||||
* attached to non-blocking calls.
|
||||
*/
|
||||
virNetClientCallMatchPredicate(client->waitDispatch,
|
||||
virNetClientIOEventLoopDetachNonBlocking,
|
||||
thiscall);
|
||||
|
||||
/* Now see if *we* are done */
|
||||
if (thiscall->mode == VIR_NET_CLIENT_MODE_COMPLETE) {
|
||||
virNetClientCallRemove(&client->waitDispatch, thiscall);
|
||||
@ -1416,7 +1376,7 @@ static int virNetClientIOEventLoop(virNetClientPtr client,
|
||||
|
||||
/* We're not done, but we're non-blocking; keep the call queued */
|
||||
if (thiscall->nonBlock) {
|
||||
thiscall->haveThread = false;
|
||||
virNetClientIODetachNonBlocking(thiscall);
|
||||
virNetClientIOEventLoopPassTheBuck(client, thiscall);
|
||||
return 1;
|
||||
}
|
||||
@ -1550,6 +1510,14 @@ static int virNetClientIO(virNetClientPtr client,
|
||||
return -1;
|
||||
}
|
||||
|
||||
/* If we are non-blocking, detach the thread and keep the call in the
|
||||
* queue. */
|
||||
if (thiscall->nonBlock) {
|
||||
virNetClientIODetachNonBlocking(thiscall);
|
||||
rv = 1;
|
||||
goto cleanup;
|
||||
}
|
||||
|
||||
VIR_DEBUG("Going to sleep head=%p call=%p",
|
||||
client->waitDispatch, thiscall);
|
||||
/* Go to sleep while other thread is working... */
|
||||
@ -1567,7 +1535,6 @@ static int virNetClientIO(virNetClientPtr client,
|
||||
* 2. Other thread is all done, and it is our turn to
|
||||
* be the dispatcher to finish waiting for
|
||||
* our reply
|
||||
* 3. I/O was expected to block
|
||||
*/
|
||||
if (thiscall->mode == VIR_NET_CLIENT_MODE_COMPLETE) {
|
||||
rv = 2;
|
||||
@ -1579,17 +1546,6 @@ static int virNetClientIO(virNetClientPtr client,
|
||||
goto cleanup;
|
||||
}
|
||||
|
||||
/* If we're non-blocking, we were either queued (and detached) or the
|
||||
* call was not sent because of an error.
|
||||
*/
|
||||
if (thiscall->nonBlock) {
|
||||
if (!thiscall->haveThread)
|
||||
rv = 1; /* In progress */
|
||||
else
|
||||
rv = 0; /* none at all */
|
||||
goto cleanup;
|
||||
}
|
||||
|
||||
/* Grr, someone passed the buck onto us ... */
|
||||
} else {
|
||||
client->haveTheBuck = true;
|
||||
|
Loading…
x
Reference in New Issue
Block a user