From: Anthony Liguori <anthony@codemonkey.ws>
To: Michael Roth <mdroth@linux.vnet.ibm.com>
Cc: aliguori@linux.vnet.ibm.com, lcapitulino@redhat.com,
agl@linux.vnet.ibm.com, qemu-devel@nongnu.org,
Jes.Sorensen@redhat.com
Subject: Re: [Qemu-devel] [PATCH v4][ 1/7] guest agent: worker thread class
Date: Tue, 07 Jun 2011 14:11:58 -0500 [thread overview]
Message-ID: <4DEE77FE.8090500@codemonkey.ws> (raw)
In-Reply-To: <1307141286-9392-2-git-send-email-mdroth@linux.vnet.ibm.com>
On 06/03/2011 05:48 PM, Michael Roth wrote:
> Signed-off-by: Michael Roth<mdroth@linux.vnet.ibm.com>
> ---
> qga/guest-agent-worker.c | 179 ++++++++++++++++++++++++++++++++++++++++++++++
> 1 files changed, 179 insertions(+), 0 deletions(-)
> create mode 100644 qga/guest-agent-worker.c
>
> diff --git a/qga/guest-agent-worker.c b/qga/guest-agent-worker.c
> new file mode 100644
> index 0000000..e5fc845
> --- /dev/null
> +++ b/qga/guest-agent-worker.c
> @@ -0,0 +1,179 @@
> +/*
> + * QEMU Guest Agent worker thread interfaces
> + *
> + * Copyright IBM Corp. 2011
> + *
> + * Authors:
> + * Michael Roth<mdroth@linux.vnet.ibm.com>
> + *
> + * This work is licensed under the terms of the GNU GPL, version 2 or later.
> + * See the COPYING file in the top-level directory.
> + */
> +#include<glib.h>
> +#include<stdlib.h>
> +#include<stdio.h>
> +#include<stdbool.h>
> +#include<pthread.h>
> +#include<errno.h>
> +#include<string.h>
> +#include "error.h"
> +#include "qga/guest-agent-core.h"
> +
> +struct GAWorker {
> + pthread_t thread;
> + ga_worker_func execute;
> + pthread_mutex_t input_mutex;
> + pthread_cond_t input_avail_cond;
> + void *input;
> + bool input_avail;
> + pthread_mutex_t output_mutex;
> + pthread_cond_t output_avail_cond;
> + void *output;
> + Error *output_error;
> + bool output_avail;
> +};
> +
> +static void *worker_run(void *worker_p)
> +{
> + GAWorker *worker = worker_p;
> + Error *err = NULL;
> + void *input = NULL, *output = NULL;
> +
> + while (1) {
> + /* wait for input */
> + pthread_mutex_lock(&worker->input_mutex);
It's more typical to push the lock() and unlock to the outside of the
loop. That makes it safer in the event that someone adds a continue or
break.
Regards,
Anthony Liguori
> + while (!worker->input_avail) {
> + pthread_cond_wait(&worker->input_avail_cond,&worker->input_mutex);
> + }
> + input = worker->input;
> + worker->input = NULL;
> + worker->input_avail = false;
> + pthread_mutex_unlock(&worker->input_mutex);
> +
> + /* process input. input points to shared data, so if we ever add
> + * asynchronous dispatch, we'll need to copy the input instead
> + */
> + worker->execute(input,&output,&err);
> +
> + /* signal waiters */
> + pthread_mutex_lock(&worker->output_mutex);
> + worker->output = output;
> + worker->output_error = err;
> + worker->output_avail = true;
> + pthread_cond_signal(&worker->output_avail_cond);
> + pthread_mutex_unlock(&worker->output_mutex);
> + }
> +
> + return NULL;
> +}
> +
> +static void ga_worker_set_input(GAWorker *worker, void *input)
> +{
> + pthread_mutex_lock(&worker->input_mutex);
> +
> + /* provide input for thread, and signal it */
> + worker->input = input;
> + worker->input_avail = true;
> + pthread_cond_signal(&worker->input_avail_cond);
> +
> + pthread_mutex_unlock(&worker->input_mutex);
> +}
> +
> +static bool ga_worker_get_output(GAWorker *worker, void **output, int timeout)
> +{
> + struct timespec ts;
> + GTimeVal tv;
> + bool timed_out = false;
> + int ret;
> +
> + pthread_mutex_lock(&worker->output_mutex);
> +
> + while (!worker->output_avail) {
> + if (timeout> 0) {
> + g_get_current_time(&tv);
> + g_time_val_add(&tv, timeout * 1000);
> + ts.tv_sec = tv.tv_sec;
> + ts.tv_nsec = tv.tv_usec * 1000;
> + ret = pthread_cond_timedwait(&worker->output_avail_cond,
> +&worker->output_mutex,&ts);
> + if (ret == ETIMEDOUT) {
> + timed_out = true;
> + goto out;
> + }
> + } else {
> + ret = pthread_cond_wait(&worker->output_avail_cond,
> +&worker->output_mutex);
> + }
> + }
> +
> + /* handle output from thread */
> + worker->output_avail = false;
> + *output = worker->output;
> + worker->output = NULL;
> +
> +out:
> + pthread_mutex_unlock(&worker->output_mutex);
> + return timed_out;
> +}
> +
> +bool ga_worker_dispatch(GAWorker *worker, void *input, void *output,
> + int timeout, Error **errp)
> +{
> + ga_worker_set_input(worker, input);
> + return ga_worker_get_output(worker, output, timeout);
> +}
> +
> +static void ga_worker_start(GAWorker *worker)
> +{
> + int ret;
> +
> + pthread_cond_init(&worker->input_avail_cond, NULL);
> + pthread_cond_init(&worker->output_avail_cond, NULL);
> + pthread_mutex_init(&worker->input_mutex, NULL);
> + pthread_mutex_init(&worker->output_mutex, NULL);
> + worker->output_avail = false;
> + worker->input_avail = false;
> +
> + ret = pthread_create(&worker->thread, NULL, worker_run, worker);
> + if (ret == -1) {
> + g_error("error: %s", strerror(errno));
> + }
> +}
> +
> +static void ga_worker_stop(GAWorker *worker)
> +{
> + int ret;
> + void *status;
> +
> + ret = pthread_cancel(worker->thread);
> + if (ret == -1) {
> + g_error("pthread_cancel() failed: %s", strerror(errno));
> + }
> +
> + ret = pthread_join(worker->thread,&status);
> + if (ret) {
> + g_error("pthread_join() failed: %s", strerror(ret));
> + }
> +
> + pthread_mutex_destroy(&worker->input_mutex);
> + pthread_mutex_destroy(&worker->output_mutex);
> + pthread_cond_destroy(&worker->input_avail_cond);
> + pthread_cond_destroy(&worker->input_avail_cond);
> +}
> +
> +GAWorker *ga_worker_new(ga_worker_func func)
> +{
> + GAWorker *worker = g_malloc0(sizeof(GAWorker));
> +
> + g_assert(func);
> + worker->execute = func;
> + ga_worker_start(worker);
> +
> + return worker;
> +}
> +
> +void ga_worker_cleanup(GAWorker *worker)
> +{
> + ga_worker_stop(worker);
> + g_free(worker);
> +}
next prev parent reply other threads:[~2011-06-07 19:12 UTC|newest]
Thread overview: 11+ messages / expand[flat|nested] mbox.gz Atom feed top
2011-06-03 22:47 [Qemu-devel] [QAPI+QGA 3/3] QEMU Guest Agent (virtagent) v4 Michael Roth
2011-06-03 22:48 ` [Qemu-devel] [PATCH v4][ 1/7] guest agent: worker thread class Michael Roth
2011-06-07 19:11 ` Anthony Liguori [this message]
2011-06-03 22:48 ` [Qemu-devel] [PATCH v4][ 2/7] guest agent: command state class Michael Roth
2011-06-03 22:48 ` [Qemu-devel] [PATCH v4][ 3/7] guest agent: qemu-ga daemon Michael Roth
2011-06-03 22:48 ` [Qemu-devel] [PATCH v4][ 4/7] guest agent: add error class for QERR_QGA_LOGGING_FAILED Michael Roth
2011-06-03 22:48 ` [Qemu-devel] [PATCH v4][ 5/7] guest agent: add guest agent RPCs/commands Michael Roth
2011-06-04 20:08 ` Andi Kleen
2011-06-04 22:29 ` Anthony Liguori
2011-06-03 22:48 ` [Qemu-devel] [PATCH v4][ 6/7] guest agent: add guest agent commands schema file Michael Roth
2011-06-03 22:48 ` [Qemu-devel] [PATCH v4][ 7/7] guest agent: Makefile, build qemu-ga Michael Roth
Reply instructions:
You may reply publicly to this message via plain-text email
using any one of the following methods:
* Save the following mbox file, import it into your mail client,
and reply-to-all from there: mbox
Avoid top-posting and favor interleaved quoting:
https://en.wikipedia.org/wiki/Posting_style#Interleaved_style
* Reply using the --to, --cc, and --in-reply-to
switches of git-send-email(1):
git send-email \
--in-reply-to=4DEE77FE.8090500@codemonkey.ws \
--to=anthony@codemonkey.ws \
--cc=Jes.Sorensen@redhat.com \
--cc=agl@linux.vnet.ibm.com \
--cc=aliguori@linux.vnet.ibm.com \
--cc=lcapitulino@redhat.com \
--cc=mdroth@linux.vnet.ibm.com \
--cc=qemu-devel@nongnu.org \
/path/to/YOUR_REPLY
https://kernel.org/pub/software/scm/git/docs/git-send-email.html
* If your mail client supports setting the In-Reply-To header
via mailto: links, try the mailto: link
Be sure your reply has a Subject: header at the top and a blank line
before the message body.
This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox;
as well as URLs for NNTP newsgroup(s).