Bugfix: free incomplete containers when JSON parsing fails
related to #2755
This commit is contained in:
24
src/con.c
24
src/con.c
@ -73,6 +73,30 @@ Con *con_new(Con *parent, i3Window *window) {
|
||||
return new;
|
||||
}
|
||||
|
||||
/*
|
||||
* Frees the specified container.
|
||||
*
|
||||
*/
|
||||
void con_free(Con *con) {
|
||||
free(con->name);
|
||||
FREE(con->deco_render_params);
|
||||
TAILQ_REMOVE(&all_cons, con, all_cons);
|
||||
while (!TAILQ_EMPTY(&(con->swallow_head))) {
|
||||
Match *match = TAILQ_FIRST(&(con->swallow_head));
|
||||
TAILQ_REMOVE(&(con->swallow_head), match, matches);
|
||||
match_free(match);
|
||||
free(match);
|
||||
}
|
||||
while (!TAILQ_EMPTY(&(con->marks_head))) {
|
||||
mark_t *mark = TAILQ_FIRST(&(con->marks_head));
|
||||
TAILQ_REMOVE(&(con->marks_head), mark, marks);
|
||||
FREE(mark->name);
|
||||
FREE(mark);
|
||||
}
|
||||
free(con);
|
||||
DLOG("con %p freed\n", con);
|
||||
}
|
||||
|
||||
static void _con_attach(Con *con, Con *parent, Con *previous, bool ignore_focus) {
|
||||
con->parent = parent;
|
||||
Con *loop;
|
||||
|
@ -18,6 +18,7 @@
|
||||
/* TODO: refactor the whole parsing thing */
|
||||
|
||||
static char *last_key;
|
||||
static int incomplete;
|
||||
static Con *json_node;
|
||||
static Con *to_focus;
|
||||
static bool parsing_swallows;
|
||||
@ -68,6 +69,9 @@ static int json_start_map(void *ctx) {
|
||||
json_node->name = NULL;
|
||||
json_node->parent = parent;
|
||||
}
|
||||
/* json_node is incomplete and should be removed if parsing fails */
|
||||
incomplete++;
|
||||
DLOG("incomplete = %d\n", incomplete);
|
||||
}
|
||||
}
|
||||
return 1;
|
||||
@ -166,6 +170,8 @@ static int json_end_map(void *ctx) {
|
||||
LOG("Creating window\n");
|
||||
x_con_init(json_node);
|
||||
json_node = json_node->parent;
|
||||
incomplete--;
|
||||
DLOG("incomplete = %d\n", incomplete);
|
||||
}
|
||||
|
||||
if (parsing_swallows && swallow_is_empty) {
|
||||
@ -634,6 +640,7 @@ void tree_append_json(Con *con, const char *filename, char **errormsg) {
|
||||
yajl_status stat;
|
||||
json_node = con;
|
||||
to_focus = NULL;
|
||||
incomplete = 0;
|
||||
parsing_swallows = false;
|
||||
parsing_rect = false;
|
||||
parsing_deco_rect = false;
|
||||
@ -649,6 +656,12 @@ void tree_append_json(Con *con, const char *filename, char **errormsg) {
|
||||
if (errormsg != NULL)
|
||||
*errormsg = sstrdup((const char *)str);
|
||||
yajl_free_error(hand, str);
|
||||
while (incomplete-- > 0) {
|
||||
Con *parent = json_node->parent;
|
||||
DLOG("freeing incomplete container %p\n", json_node);
|
||||
con_free(json_node);
|
||||
json_node = parent;
|
||||
}
|
||||
}
|
||||
|
||||
/* In case not all containers were restored, we need to fix the
|
||||
|
17
src/tree.c
17
src/tree.c
@ -320,22 +320,7 @@ bool tree_close_internal(Con *con, kill_window_t kill_window, bool dont_kill_par
|
||||
DLOG("parent container killed\n");
|
||||
}
|
||||
|
||||
free(con->name);
|
||||
FREE(con->deco_render_params);
|
||||
TAILQ_REMOVE(&all_cons, con, all_cons);
|
||||
while (!TAILQ_EMPTY(&(con->swallow_head))) {
|
||||
Match *match = TAILQ_FIRST(&(con->swallow_head));
|
||||
TAILQ_REMOVE(&(con->swallow_head), match, matches);
|
||||
match_free(match);
|
||||
free(match);
|
||||
}
|
||||
while (!TAILQ_EMPTY(&(con->marks_head))) {
|
||||
mark_t *mark = TAILQ_FIRST(&(con->marks_head));
|
||||
TAILQ_REMOVE(&(con->marks_head), mark, marks);
|
||||
FREE(mark->name);
|
||||
FREE(mark);
|
||||
}
|
||||
free(con);
|
||||
con_free(con);
|
||||
|
||||
/* in the case of floating windows, we already focused another container
|
||||
* when closing the parent, so we can exit now. */
|
||||
|
Reference in New Issue
Block a user