/* Whether a redraw is needed for appending a line to a buffer. */
static int channel_need_redraw = FALSE;
+/* Whether we are inside channel_parse_messages() or another situation where it
+ * is safe to invoke callbacks. */
+static int safe_to_invoke_callback = 0;
#ifdef WIN32
static int
{
if (!in_free_unref_items)
{
- channel_free_contents(channel);
- channel_free_channel(channel);
+ if (safe_to_invoke_callback == 0)
+ {
+ channel->ch_to_be_freed = TRUE;
+ }
+ else
+ {
+ channel_free_contents(channel);
+ channel_free_channel(channel);
+ }
}
}
int did_free = FALSE;
channel_T *ch;
+ /* This is invoked from the garbage collector, which only runs at a safe
+ * point. */
+ ++safe_to_invoke_callback;
+
for (ch = first_channel; ch != NULL; ch = ch->ch_next)
if (!channel_still_useful(ch)
&& (ch->ch_copyID & mask) != (copyID & mask))
channel_free_contents(ch);
did_free = TRUE;
}
+
+ --safe_to_invoke_callback;
return did_free;
}
typval_T rettv;
int dummy;
+ if (safe_to_invoke_callback == 0)
+ EMSG("INTERNAL: Invoking callback when it is not safe");
+
argv[0].v_type = VAR_CHANNEL;
argv[0].vval.v_channel = channel;
int r;
int part = PART_SOCK;
+ ++safe_to_invoke_callback;
+
/* Only do this message when another message was given, otherwise we get
* lots of them. */
if (did_log_msg)
channel = first_channel;
continue;
}
+ if (channel->ch_to_be_freed)
+ {
+ channel_free(channel);
+ /* channel has been freed, start over */
+ channel = first_channel;
+ continue;
+ }
if (channel->ch_refcount == 0 && !channel_still_useful(channel))
{
/* channel is no longer useful, free it */
redraw_after_callback();
}
+ --safe_to_invoke_callback;
+
return ret;
}
int ch_to_be_closed; /* When TRUE reading or writing failed and
* the channel must be closed when it's safe
* to invoke callbacks. */
+ int ch_to_be_freed; /* When TRUE channel must be freed when it's
+ * safe to invoke callbacks. */
int ch_error; /* When TRUE an error was reported. Avoids
* giving pages full of error messages when
* the other side has exited, only mention the