mirror of
https://github.com/neovim/neovim.git
synced 2025-10-02 16:08:36 +00:00
vim-patch:8.2.1473: items in a list given to :const can still be modified
Problem: Items in a list given to :const can still be modified.
Solution: Work like ":lockvar! name" but don't lock referenced items.
Make locking a blob work.
021bda5671
This commit is contained in:
@@ -2373,6 +2373,9 @@ static void set_var_lval(lval_T *lp, char_u *endp, typval_T *rettv,
|
||||
EMSG2(_(e_letwrong), op);
|
||||
return;
|
||||
}
|
||||
if (var_check_lock(lp->ll_blob->bv_lock, lp->ll_name, TV_CSTRING)) {
|
||||
return;
|
||||
}
|
||||
|
||||
if (lp->ll_range && rettv->v_type == VAR_BLOB) {
|
||||
if (lp->ll_empty2) {
|
||||
@@ -3082,7 +3085,7 @@ static int do_lock_var(lval_T *lp, char_u *name_end FUNC_ATTR_UNUSED,
|
||||
} else {
|
||||
di->di_flags &= ~DI_FLAGS_LOCK;
|
||||
}
|
||||
tv_item_lock(&di->di_tv, deep, lock);
|
||||
tv_item_lock(&di->di_tv, deep, lock, false);
|
||||
}
|
||||
}
|
||||
} else if (lp->ll_range) {
|
||||
@@ -3090,16 +3093,16 @@ static int do_lock_var(lval_T *lp, char_u *name_end FUNC_ATTR_UNUSED,
|
||||
|
||||
// (un)lock a range of List items.
|
||||
while (li != NULL && (lp->ll_empty2 || lp->ll_n2 >= lp->ll_n1)) {
|
||||
tv_item_lock(TV_LIST_ITEM_TV(li), deep, lock);
|
||||
tv_item_lock(TV_LIST_ITEM_TV(li), deep, lock, false);
|
||||
li = TV_LIST_ITEM_NEXT(lp->ll_list, li);
|
||||
lp->ll_n1++;
|
||||
}
|
||||
} else if (lp->ll_list != NULL) {
|
||||
// (un)lock a List item.
|
||||
tv_item_lock(TV_LIST_ITEM_TV(lp->ll_li), deep, lock);
|
||||
tv_item_lock(TV_LIST_ITEM_TV(lp->ll_li), deep, lock, false);
|
||||
} else {
|
||||
// (un)lock a Dictionary item.
|
||||
tv_item_lock(&lp->ll_di->di_tv, deep, lock);
|
||||
tv_item_lock(&lp->ll_di->di_tv, deep, lock, false);
|
||||
}
|
||||
|
||||
return ret;
|
||||
@@ -9536,7 +9539,10 @@ static void set_var_const(const char *name, const size_t name_len,
|
||||
}
|
||||
|
||||
if (is_const) {
|
||||
tv_item_lock(&v->di_tv, 1, true);
|
||||
// Like :lockvar! name: lock the value and what it contains, but only
|
||||
// if the reference count is up to one. That locks only literal
|
||||
// values.
|
||||
tv_item_lock(&v->di_tv, DICT_MAXNEST, true, true);
|
||||
}
|
||||
}
|
||||
|
||||
|
@@ -2659,7 +2659,10 @@ void tv_copy(const typval_T *const from, typval_T *const to)
|
||||
/// @param[out] tv Item to (un)lock.
|
||||
/// @param[in] deep Levels to (un)lock, -1 to (un)lock everything.
|
||||
/// @param[in] lock True if it is needed to lock an item, false to unlock.
|
||||
void tv_item_lock(typval_T *const tv, const int deep, const bool lock)
|
||||
/// @param[in] check_refcount If true, do not lock a list or dict with a
|
||||
/// reference count larger than 1.
|
||||
void tv_item_lock(typval_T *const tv, const int deep, const bool lock,
|
||||
const bool check_refcount)
|
||||
FUNC_ATTR_NONNULL_ALL
|
||||
{
|
||||
// TODO(ZyX-I): Make this not recursive
|
||||
@@ -2688,19 +2691,19 @@ void tv_item_lock(typval_T *const tv, const int deep, const bool lock)
|
||||
switch (tv->v_type) {
|
||||
case VAR_BLOB: {
|
||||
blob_T *const b = tv->vval.v_blob;
|
||||
if (b != NULL) {
|
||||
if (b != NULL && !(check_refcount && b->bv_refcount > 1)) {
|
||||
CHANGE_LOCK(lock, b->bv_lock);
|
||||
}
|
||||
break;
|
||||
}
|
||||
case VAR_LIST: {
|
||||
list_T *const l = tv->vval.v_list;
|
||||
if (l != NULL) {
|
||||
if (l != NULL && !(check_refcount && l->lv_refcount > 1)) {
|
||||
CHANGE_LOCK(lock, l->lv_lock);
|
||||
if (deep < 0 || deep > 1) {
|
||||
// Recursive: lock/unlock the items the List contains.
|
||||
TV_LIST_ITER(l, li, {
|
||||
tv_item_lock(TV_LIST_ITEM_TV(li), deep - 1, lock);
|
||||
tv_item_lock(TV_LIST_ITEM_TV(li), deep - 1, lock, check_refcount);
|
||||
});
|
||||
}
|
||||
}
|
||||
@@ -2708,12 +2711,12 @@ void tv_item_lock(typval_T *const tv, const int deep, const bool lock)
|
||||
}
|
||||
case VAR_DICT: {
|
||||
dict_T *const d = tv->vval.v_dict;
|
||||
if (d != NULL) {
|
||||
if (d != NULL && !(check_refcount && d->dv_refcount > 1)) {
|
||||
CHANGE_LOCK(lock, d->dv_lock);
|
||||
if (deep < 0 || deep > 1) {
|
||||
// recursive: lock/unlock the items the List contains
|
||||
TV_DICT_ITER(d, di, {
|
||||
tv_item_lock(&di->di_tv, deep - 1, lock);
|
||||
tv_item_lock(&di->di_tv, deep - 1, lock, check_refcount);
|
||||
});
|
||||
}
|
||||
}
|
||||
|
@@ -244,18 +244,33 @@ func Test_const_with_eval_name()
|
||||
call assert_fails('const {s2} = "bar"', 'E995:')
|
||||
endfunc
|
||||
|
||||
func Test_lock_depth_is_1()
|
||||
const l = [1, 2, 3]
|
||||
const d = {'foo': 10}
|
||||
|
||||
" Modify list - setting item is OK, adding/removing items not
|
||||
let l[0] = 42
|
||||
func Test_lock_depth_is_2()
|
||||
" Modify list - error when changing item or adding/removing items
|
||||
const l = [1, 2, [3, 4]]
|
||||
call assert_fails('let l[0] = 42', 'E741:')
|
||||
call assert_fails('let l[2][0] = 42', 'E741:')
|
||||
call assert_fails('call add(l, 4)', 'E741:')
|
||||
call assert_fails('unlet l[1]', 'E741:')
|
||||
|
||||
" Modify dict - changing item is OK, adding/removing items not
|
||||
let d['foo'] = 'hello'
|
||||
let d.foo = 44
|
||||
" Modify blob - error when changing
|
||||
const b = 0z001122
|
||||
call assert_fails('let b[0] = 42', 'E741:')
|
||||
|
||||
" Modify dict - error when changing item or adding/removing items
|
||||
const d = {'foo': 10}
|
||||
call assert_fails("let d['foo'] = 'hello'", 'E741:')
|
||||
call assert_fails("let d.foo = 'hello'", 'E741:')
|
||||
call assert_fails("let d['bar'] = 'hello'", 'E741:')
|
||||
call assert_fails("unlet d['foo']", 'E741:')
|
||||
|
||||
" Modifying list or dict item contents is OK.
|
||||
let lvar = ['a', 'b']
|
||||
let bvar = 0z1122
|
||||
const l2 = [0, lvar, bvar]
|
||||
let l2[1][0] = 'c'
|
||||
let l2[2][1] = 0x33
|
||||
call assert_equal([0, ['c', 'b'], 0z1133], l2)
|
||||
|
||||
const d2 = #{a: 0, b: lvar, c: 4}
|
||||
let d2.b[1] = 'd'
|
||||
endfunc
|
||||
|
Reference in New Issue
Block a user