[Spice-devel] [PATCH v2 3/4] worker: avoid blocking loop

Jonathon Jongsma jjongsma at redhat.com
Wed Feb 3 16:50:14 CET 2016


Acked-by: Jonathon Jongsma <jjongsma at redhat.com>

On Wed, 2016-02-03 at 10:48 +0000, Frediano Ziglio wrote:
> Make sure we process commands after we can send data to client.
> If during processing we detected that there was too much data in the
> clients queues the processing of commands just stop till the next
> iteration.
> However if all data are pushed in a single iteration of the loop
> and commands were already processed there was the (remote) possibility
> that the loop hangs till a separate event (like a screen resize on
> client window) arrive.
> I manage to reproduce and after half an hour no events arrived.
> This patch detect that processing was stuck and now we can process new
> commands and force a new iteration.
> 
> Signed-off-by: Frediano Ziglio <fziglio at redhat.com>
> Acked-by: Jonathon Jongsma <jjongsma at redhat.com>
> ---
>  server/red-worker.c | 14 ++++++++++++++
>  1 file changed, 14 insertions(+)
> 
> diff --git a/server/red-worker.c b/server/red-worker.c
> index 031350a..45a597f 100644
> --- a/server/red-worker.c
> +++ b/server/red-worker.c
> @@ -69,6 +69,7 @@ struct RedWorker {
>  
>      DisplayChannel *display_channel;
>      uint32_t display_poll_tries;
> +    gboolean was_blocked;
>  
>      CursorChannel *cursor_channel;
>      uint32_t cursor_poll_tries;
> @@ -196,6 +197,7 @@ static int red_process_cursor(RedWorker *worker, int
> *ring_is_empty)
>          }
>          n++;
>      }
> +    worker->was_blocked = TRUE;
>      return n;
>  }
>  
> @@ -318,9 +320,16 @@ static int red_process_display(RedWorker *worker, int
> *ring_is_empty)
>              return n;
>          }
>      }
> +    worker->was_blocked = TRUE;
>      return n;
>  }
>  
> +static bool red_process_is_blocked(RedWorker *worker)
> +{
> +    return red_channel_max_pipe_size(RED_CHANNEL(worker->cursor_channel)) >
> MAX_PIPE_SIZE ||
> +           red_channel_max_pipe_size(RED_CHANNEL(worker->display_channel)) >
> MAX_PIPE_SIZE;
> +}
> +
>  static void red_disconnect_display(RedWorker *worker)
>  {
>      spice_warning("update timeout");
> @@ -1416,6 +1425,10 @@ static gboolean worker_source_prepare(GSource *source,
> gint *p_timeout)
>      if (*p_timeout == 0)
>          return TRUE;
>  
> +    if (worker->was_blocked && !red_process_is_blocked(worker)) {
> +        return TRUE;
> +    }
> +
>      return FALSE;
>  }
>  
> @@ -1445,6 +1458,7 @@ static gboolean worker_source_dispatch(GSource *source,
> GSourceFunc callback,
>      stream_timeout(display);
>  
>      worker->event_timeout = INF_EVENT_WAIT;
> +    worker->was_blocked = FALSE;
>      red_process_cursor(worker, &ring_is_empty);
>      red_process_display(worker, &ring_is_empty);
>  


More information about the Spice-devel mailing list