summaryrefslogtreecommitdiffstats
path: root/src/eval.c
diff options
context:
space:
mode:
authorBram Moolenaar <Bram@vim.org>2021-06-27 15:04:05 +0200
committerBram Moolenaar <Bram@vim.org>2021-06-27 15:04:05 +0200
commite65081d1b591f16dc6e380a830d87565c5eb7b03 (patch)
treeea8c2e725d7dcd81f3b0f83dfab33ee1936ee692 /src/eval.c
parent65aee0b714e809b9f19862f3decd35055ed4de10 (diff)
patch 8.2.3064: Vim9: in script cannot set item in uninitialized listv8.2.3064
Problem: Vim9: in script cannot set item in uninitialized list. Solution: When a list is NULL allocate an empty one. (closes #8461)
Diffstat (limited to 'src/eval.c')
-rw-r--r--src/eval.c33
1 files changed, 25 insertions, 8 deletions
diff --git a/src/eval.c b/src/eval.c
index 7704d46e98..7990c23f00 100644
--- a/src/eval.c
+++ b/src/eval.c
@@ -932,15 +932,22 @@ get_lval(
semsg(_(e_dot_can_only_be_used_on_dictionary_str), name);
return NULL;
}
- if (!(lp->ll_tv->v_type == VAR_LIST && lp->ll_tv->vval.v_list != NULL)
- && !(lp->ll_tv->v_type == VAR_DICT)
- && !(lp->ll_tv->v_type == VAR_BLOB
- && lp->ll_tv->vval.v_blob != NULL))
+ if (lp->ll_tv->v_type != VAR_LIST
+ && lp->ll_tv->v_type != VAR_DICT
+ && lp->ll_tv->v_type != VAR_BLOB)
{
if (!quiet)
emsg(_("E689: Can only index a List, Dictionary or Blob"));
return NULL;
}
+
+ // a NULL list/blob works like an empty list/blob, allocate one now.
+ if (lp->ll_tv->v_type == VAR_LIST && lp->ll_tv->vval.v_list == NULL)
+ rettv_list_alloc(lp->ll_tv);
+ else if (lp->ll_tv->v_type == VAR_BLOB
+ && lp->ll_tv->vval.v_blob == NULL)
+ rettv_blob_alloc(lp->ll_tv);
+
if (lp->ll_range)
{
if (!quiet)
@@ -1201,10 +1208,20 @@ get_lval(
lp->ll_li = list_find_index(lp->ll_list, &lp->ll_n1);
if (lp->ll_li == NULL)
{
- clear_tv(&var2);
- if (!quiet)
- semsg(_(e_listidx), lp->ll_n1);
- return NULL;
+ // Vim9: Allow for adding an item at the end.
+ if (in_vim9script() && lp->ll_n1 == lp->ll_list->lv_len
+ && lp->ll_list->lv_lock == 0)
+ {
+ list_append_number(lp->ll_list, 0);
+ lp->ll_li = list_find_index(lp->ll_list, &lp->ll_n1);
+ }
+ if (lp->ll_li == NULL)
+ {
+ clear_tv(&var2);
+ if (!quiet)
+ semsg(_(e_listidx), lp->ll_n1);
+ return NULL;
+ }
}
if (lp->ll_valtype != NULL)