summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorBram Moolenaar <Bram@vim.org>2020-07-05 16:42:13 +0200
committerBram Moolenaar <Bram@vim.org>2020-07-05 16:42:13 +0200
commit435d89789ef4dd329938edbe17c646db9f0ea772 (patch)
treec378f22e1c3b758c17b03c4939fcfb54be4fcb19
parentfce82b3aa7dd87f9e15a4c12eda2c65de285d99a (diff)
downloadvim-git-435d89789ef4dd329938edbe17c646db9f0ea772.tar.gz
patch 8.2.1134: Vim9: getting a list member may not workv8.2.1134
Problem: Vim9: getting a list member may not work. Solution: Clear the list only after copying the item. (closes #6393)
-rw-r--r--src/testdir/test_vim9_expr.vim5
-rw-r--r--src/version.c2
-rw-r--r--src/vim9execute.c9
3 files changed, 14 insertions, 2 deletions
diff --git a/src/testdir/test_vim9_expr.vim b/src/testdir/test_vim9_expr.vim
index 3bf578059..aef181512 100644
--- a/src/testdir/test_vim9_expr.vim
+++ b/src/testdir/test_vim9_expr.vim
@@ -1141,6 +1141,11 @@ def Test_expr_member()
call CheckDefExecFailure(["let d: dict<number>", "d = g:list_empty"], 'E1029: Expected dict but got list')
enddef
+def Test_expr_index()
+ # getting the one member should clear the list only after getting the item
+ assert_equal('bbb', ['aaa', 'bbb', 'ccc'][1])
+enddef
+
def Test_expr_member_vim9script()
let lines =<< trim END
vim9script
diff --git a/src/version.c b/src/version.c
index 6a2295191..8774b2a8d 100644
--- a/src/version.c
+++ b/src/version.c
@@ -755,6 +755,8 @@ static char *(features[]) =
static int included_patches[] =
{ /* Add new patch number below this line */
/**/
+ 1134,
+/**/
1133,
/**/
1132,
diff --git a/src/vim9execute.c b/src/vim9execute.c
index bcacaabaa..47e072e54 100644
--- a/src/vim9execute.c
+++ b/src/vim9execute.c
@@ -2085,6 +2085,7 @@ call_def_function(
list_T *list;
varnumber_T n;
listitem_T *li;
+ typval_T temp_tv;
// list index: list is at stack-2, index at stack-1
tv = STACK_TV_BOT(-2);
@@ -2109,8 +2110,12 @@ call_def_function(
goto failed;
}
--ectx.ec_stack.ga_len;
- clear_tv(STACK_TV_BOT(-1));
- copy_tv(&li->li_tv, STACK_TV_BOT(-1));
+ // Clear the list after getting the item, to avoid that it
+ // make the item invalid.
+ tv = STACK_TV_BOT(-1);
+ temp_tv = *tv;
+ copy_tv(&li->li_tv, tv);
+ clear_tv(&temp_tv);
}
break;