aboutsummaryrefslogtreecommitdiff
path: root/src
diff options
context:
space:
mode:
authorZyX <kp-pav@yandex.ru>2017-02-24 05:09:34 +0300
committerZyX <kp-pav@yandex.ru>2017-02-24 05:09:34 +0300
commitfe30d8ccef17fff23676b8670dfec86444e2cb32 (patch)
tree66fbbfc3636704d7eb2f3716319489b5924efbfc /src
parentcd8f07cb757d116e91dfde71dfa9f7070a07db6c (diff)
downloadrneovim-fe30d8ccef17fff23676b8670dfec86444e2cb32.tar.gz
rneovim-fe30d8ccef17fff23676b8670dfec86444e2cb32.tar.bz2
rneovim-fe30d8ccef17fff23676b8670dfec86444e2cb32.zip
memory: Free buffers after freeing variables
Avoids use-after-free crashes when compiling with -DEXITFREE.
Diffstat (limited to 'src')
-rw-r--r--src/nvim/memory.c26
1 files changed, 14 insertions, 12 deletions
diff --git a/src/nvim/memory.c b/src/nvim/memory.c
index 51e0dd926e..4861299956 100644
--- a/src/nvim/memory.c
+++ b/src/nvim/memory.c
@@ -625,18 +625,6 @@ void free_all_mem(void)
/* Destroy all windows. Must come before freeing buffers. */
win_free_all();
- /* Free all buffers. Reset 'autochdir' to avoid accessing things that
- * were freed already. */
- p_acd = false;
- for (buf = firstbuf; buf != NULL; ) {
- bufref_T bufref;
- set_bufref(&bufref, buf);
- nextbuf = buf->b_next;
- close_buffer(NULL, buf, DOBUF_WIPE, false);
- // Didn't work, try next one.
- buf = bufref_valid(&bufref) ? nextbuf : firstbuf;
- }
-
free_cmdline_buf();
/* Clear registers. */
@@ -660,6 +648,20 @@ void free_all_mem(void)
eval_clear();
+ // Free all buffers. Reset 'autochdir' to avoid accessing things that
+ // were freed already.
+ // Must be after eval_clear to avoid it trying to access b:changedtick after
+ // freeing it.
+ p_acd = false;
+ for (buf = firstbuf; buf != NULL; ) {
+ bufref_T bufref;
+ set_bufref(&bufref, buf);
+ nextbuf = buf->b_next;
+ close_buffer(NULL, buf, DOBUF_WIPE, false);
+ // Didn't work, try next one.
+ buf = bufref_valid(&bufref) ? nextbuf : firstbuf;
+ }
+
/* screenlines (can't display anything now!) */
free_screenlines();