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 ef392614aa)
This commit is contained in:
Jiri Denemark 2012-06-11 14:24:06 +02:00 committed by Cole Robinson
parent 5badf8c44b
commit 4779cf0ff5

View File

@ -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);
VIR_DEBUG("Keeping unfinished non-blocking call %p in the queue", call);
call->haveThread = false;
virCondSignal(&call->cond);
return true;
}
return 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;