1
0
mirror of https://git.tartarus.org/simon/putty.git synced 2025-01-10 09:58:01 +00:00

Make GTK idle and quit function setup idempotent.

I found last week that when a local proxy process terminated
unexpectedly, Unix PuTTY went into a tight loop calling quit
functions, because if idle_toplevel_callback_func is called from
inside a subsidiary gtk_main then it will schedule a quit function and
_not_ disable itself, so that that quit function keeps being
rescheduled on subsequent calls.

To fix, I've tried to make the whole handling of idle and quit
functions more sensibly robust: we keep our own boolean flag
indicating whether each of our functions has already been scheduled
with GTK, and if so, we don't schedule the same one again. Also, when
idle_toplevel_callback_func schedules a quit function, it should
unschedule itself since it's now done everything it can until a
gtk_main instance quits.

[originally from svn r10100]
This commit is contained in:
Simon Tatham 2013-11-30 18:04:57 +00:00
parent 7a53bd65d7
commit df1eee6027

View File

@ -94,6 +94,7 @@ struct gui_data {
int mouseptr_visible;
int busy_status;
guint toplevel_callback_idle_id;
int idle_fn_scheduled, quit_fn_scheduled;
int alt_keycode;
int alt_digits;
char *wintitle;
@ -1403,6 +1404,8 @@ static gint quit_toplevel_callback_func(gpointer data)
notify_toplevel_callback(inst);
inst->quit_fn_scheduled = FALSE;
return 0;
}
@ -1414,16 +1417,36 @@ static gint idle_toplevel_callback_func(gpointer data)
/*
* We don't run the callbacks if we're in the middle of a
* subsidiary gtk_main. Instead, ask for a callback when we
* get back out of the subsidiary main loop, so we can
* reschedule ourself then.
* get back out of the subsidiary main loop (if we haven't
* already arranged one), so we can reschedule ourself then.
*/
if (!inst->quit_fn_scheduled) {
gtk_quit_add(2, quit_toplevel_callback_func, inst);
inst->quit_fn_scheduled = TRUE;
}
/*
* And unschedule this idle function, since we've now done
* everything we can until the innermost gtk_main has quit and
* can reschedule us with a chance of actually taking action.
*/
if (inst->idle_fn_scheduled) { /* double-check, just in case */
gtk_idle_remove(inst->toplevel_callback_idle_id);
inst->idle_fn_scheduled = FALSE;
}
} else {
run_toplevel_callbacks();
}
if (!toplevel_callback_pending())
/*
* If we've emptied our toplevel callback queue, unschedule
* ourself. Otherwise, leave ourselves pending so we'll be called
* again to deal with more callbacks after another round of the
* event loop.
*/
if (!toplevel_callback_pending() && inst->idle_fn_scheduled) {
gtk_idle_remove(inst->toplevel_callback_idle_id);
inst->idle_fn_scheduled = FALSE;
}
return TRUE;
}
@ -1432,8 +1455,11 @@ static void notify_toplevel_callback(void *frontend)
{
struct gui_data *inst = (struct gui_data *)frontend;
if (!inst->idle_fn_scheduled) {
inst->toplevel_callback_idle_id =
gtk_idle_add(idle_toplevel_callback_func, inst);
inst->idle_fn_scheduled = TRUE;
}
}
static gint timer_trigger(gpointer data)
@ -3604,6 +3630,8 @@ int pt_main(int argc, char **argv)
inst->busy_status = BUSY_NOT;
inst->conf = conf_new();
inst->wintitle = inst->icontitle = NULL;
inst->quit_fn_scheduled = FALSE;
inst->idle_fn_scheduled = FALSE;
/* defer any child exit handling until we're ready to deal with
* it */