mirror of
https://github.com/neovim/neovim.git
synced 2025-09-12 22:38:16 +00:00
vim-patch:8.1.0897: can modify a:000 when using a reference (#14902)
Problem: Can modify a:000 when using a reference.
Solution: Make check for locked variable stricter. (Ozaki Kiichi,
closes vim/vim#3930)
05c00c038b
This commit is contained in:
@@ -2340,10 +2340,8 @@ static void set_var_lval(lval_T *lp, char_u *endp, typval_T *rettv,
|
||||
if (get_var_tv((const char *)lp->ll_name, (int)STRLEN(lp->ll_name),
|
||||
&tv, &di, true, false) == OK) {
|
||||
if ((di == NULL
|
||||
|| (!var_check_ro(di->di_flags, (const char *)lp->ll_name,
|
||||
TV_CSTRING)
|
||||
&& !tv_check_lock(di->di_tv.v_lock, (const char *)lp->ll_name,
|
||||
TV_CSTRING)))
|
||||
|| (!var_check_ro(di->di_flags, lp->ll_name, TV_CSTRING)
|
||||
&& !tv_check_lock(&di->di_tv, lp->ll_name, TV_CSTRING)))
|
||||
&& eexe_mod_op(&tv, rettv, op) == OK) {
|
||||
set_var(lp->ll_name, lp->ll_name_len, &tv, false);
|
||||
}
|
||||
@@ -2353,10 +2351,10 @@ static void set_var_lval(lval_T *lp, char_u *endp, typval_T *rettv,
|
||||
set_var_const(lp->ll_name, lp->ll_name_len, rettv, copy, is_const);
|
||||
}
|
||||
*endp = cc;
|
||||
} else if (tv_check_lock(lp->ll_newkey == NULL
|
||||
? lp->ll_tv->v_lock
|
||||
: lp->ll_tv->vval.v_dict->dv_lock,
|
||||
(const char *)lp->ll_name, TV_CSTRING)) {
|
||||
} else if (var_check_lock(lp->ll_newkey == NULL
|
||||
? lp->ll_tv->v_lock
|
||||
: lp->ll_tv->vval.v_dict->dv_lock,
|
||||
lp->ll_name, TV_CSTRING)) {
|
||||
} else if (lp->ll_range) {
|
||||
listitem_T *ll_li = lp->ll_li;
|
||||
int ll_n1 = lp->ll_n1;
|
||||
@@ -2369,9 +2367,8 @@ static void set_var_lval(lval_T *lp, char_u *endp, typval_T *rettv,
|
||||
// Check whether any of the list items is locked
|
||||
for (ri = tv_list_first(rettv->vval.v_list);
|
||||
ri != NULL && ll_li != NULL; ) {
|
||||
if (tv_check_lock(TV_LIST_ITEM_TV(ll_li)->v_lock,
|
||||
(const char *)lp->ll_name,
|
||||
TV_CSTRING)) {
|
||||
if (var_check_lock(TV_LIST_ITEM_TV(ll_li)->v_lock, lp->ll_name,
|
||||
TV_CSTRING)) {
|
||||
return;
|
||||
}
|
||||
ri = TV_LIST_ITEM_NEXT(rettv->vval.v_list, ri);
|
||||
@@ -2795,13 +2792,13 @@ static int do_unlet_var(lval_T *lp, char_u *name_end, exarg_T *eap,
|
||||
} else if ((lp->ll_list != NULL
|
||||
// ll_list is not NULL when lvalue is not in a list, NULL lists
|
||||
// yield E689.
|
||||
&& tv_check_lock(tv_list_locked(lp->ll_list),
|
||||
(const char *)lp->ll_name,
|
||||
lp->ll_name_len))
|
||||
&& var_check_lock(tv_list_locked(lp->ll_list),
|
||||
lp->ll_name,
|
||||
lp->ll_name_len))
|
||||
|| (lp->ll_dict != NULL
|
||||
&& tv_check_lock(lp->ll_dict->dv_lock,
|
||||
(const char *)lp->ll_name,
|
||||
lp->ll_name_len))) {
|
||||
&& var_check_lock(lp->ll_dict->dv_lock,
|
||||
lp->ll_name,
|
||||
lp->ll_name_len))) {
|
||||
return FAIL;
|
||||
} else if (lp->ll_range) {
|
||||
assert(lp->ll_list != NULL);
|
||||
@@ -2810,9 +2807,9 @@ static int do_unlet_var(lval_T *lp, char_u *name_end, exarg_T *eap,
|
||||
listitem_T *last_li = first_li;
|
||||
for (;;) {
|
||||
listitem_T *const li = TV_LIST_ITEM_NEXT(lp->ll_list, lp->ll_li);
|
||||
if (tv_check_lock(TV_LIST_ITEM_TV(lp->ll_li)->v_lock,
|
||||
(const char *)lp->ll_name,
|
||||
lp->ll_name_len)) {
|
||||
if (var_check_lock(TV_LIST_ITEM_TV(lp->ll_li)->v_lock,
|
||||
lp->ll_name,
|
||||
lp->ll_name_len)) {
|
||||
return false;
|
||||
}
|
||||
lp->ll_li = li;
|
||||
@@ -2897,11 +2894,11 @@ int do_unlet(const char *const name, const size_t name_len, const bool forceit)
|
||||
dictitem_T *const di = TV_DICT_HI2DI(hi);
|
||||
if (var_check_fixed(di->di_flags, (const char *)name, TV_CSTRING)
|
||||
|| var_check_ro(di->di_flags, (const char *)name, TV_CSTRING)
|
||||
|| tv_check_lock(d->dv_lock, (const char *)name, TV_CSTRING)) {
|
||||
|| var_check_lock(d->dv_lock, name, TV_CSTRING)) {
|
||||
return FAIL;
|
||||
}
|
||||
|
||||
if (tv_check_lock(d->dv_lock, (const char *)name, TV_CSTRING)) {
|
||||
if (var_check_lock(d->dv_lock, name, TV_CSTRING)) {
|
||||
return FAIL;
|
||||
}
|
||||
|
||||
@@ -5962,14 +5959,14 @@ void filter_map(typval_T *argvars, typval_T *rettv, int map)
|
||||
if (argvars[0].v_type == VAR_LIST) {
|
||||
tv_copy(&argvars[0], rettv);
|
||||
if ((l = argvars[0].vval.v_list) == NULL
|
||||
|| (!map && tv_check_lock(tv_list_locked(l), arg_errmsg,
|
||||
TV_TRANSLATE))) {
|
||||
|| (!map
|
||||
&& var_check_lock(tv_list_locked(l), arg_errmsg, TV_TRANSLATE))) {
|
||||
return;
|
||||
}
|
||||
} else if (argvars[0].v_type == VAR_DICT) {
|
||||
tv_copy(&argvars[0], rettv);
|
||||
if ((d = argvars[0].vval.v_dict) == NULL
|
||||
|| (!map && tv_check_lock(d->dv_lock, arg_errmsg, TV_TRANSLATE))) {
|
||||
|| (!map && var_check_lock(d->dv_lock, arg_errmsg, TV_TRANSLATE))) {
|
||||
return;
|
||||
}
|
||||
} else {
|
||||
@@ -6002,7 +5999,7 @@ void filter_map(typval_T *argvars, typval_T *rettv, int map)
|
||||
|
||||
di = TV_DICT_HI2DI(hi);
|
||||
if (map
|
||||
&& (tv_check_lock(di->di_tv.v_lock, arg_errmsg, TV_TRANSLATE)
|
||||
&& (var_check_lock(di->di_tv.v_lock, arg_errmsg, TV_TRANSLATE)
|
||||
|| var_check_ro(di->di_flags, arg_errmsg, TV_TRANSLATE))) {
|
||||
break;
|
||||
}
|
||||
@@ -6029,8 +6026,8 @@ void filter_map(typval_T *argvars, typval_T *rettv, int map)
|
||||
|
||||
for (listitem_T *li = tv_list_first(l); li != NULL;) {
|
||||
if (map
|
||||
&& tv_check_lock(TV_LIST_ITEM_TV(li)->v_lock, arg_errmsg,
|
||||
TV_TRANSLATE)) {
|
||||
&& var_check_lock(TV_LIST_ITEM_TV(li)->v_lock, arg_errmsg,
|
||||
TV_TRANSLATE)) {
|
||||
break;
|
||||
}
|
||||
vimvars[VV_KEY].vv_nr = idx;
|
||||
@@ -8947,7 +8944,7 @@ static void set_var_const(const char *name, const size_t name_len,
|
||||
|
||||
// existing variable, need to clear the value
|
||||
if (var_check_ro(v->di_flags, name, name_len)
|
||||
|| tv_check_lock(v->di_tv.v_lock, name, name_len)) {
|
||||
|| var_check_lock(v->di_tv.v_lock, name, name_len)) {
|
||||
return;
|
||||
}
|
||||
|
||||
|
Reference in New Issue
Block a user