mirror of
https://github.com/vim/vim.git
synced 2025-09-23 03:43:49 -04:00
patch 8.2.2672: Vim9: cannot use :lockvar and :unlockvar in compiled script
Problem: Vim9: cannot use :lockvar and :unlockvar in compiled script. Solution: Implement locking support.
This commit is contained in:
@@ -391,3 +391,5 @@ EXTERN char e_misplaced_command_modifier[]
|
|||||||
INIT(= N_("E1176: Misplaced command modifier"));
|
INIT(= N_("E1176: Misplaced command modifier"));
|
||||||
EXTERN char e_for_loop_on_str_not_supported[]
|
EXTERN char e_for_loop_on_str_not_supported[]
|
||||||
INIT(= N_("E1177: For loop on %s not supported"));
|
INIT(= N_("E1177: For loop on %s not supported"));
|
||||||
|
EXTERN char e_cannot_lock_unlock_local_variable[]
|
||||||
|
INIT(= N_("E1178: Cannot lock or unlock a local variable"));
|
||||||
|
@@ -1135,4 +1135,42 @@ def Test_windo_missing_endif()
|
|||||||
CheckDefExecFailure(lines, 'E171:', 1)
|
CheckDefExecFailure(lines, 'E171:', 1)
|
||||||
enddef
|
enddef
|
||||||
|
|
||||||
|
let s:theList = [1, 2, 3]
|
||||||
|
|
||||||
|
def Test_lockvar()
|
||||||
|
s:theList[1] = 22
|
||||||
|
assert_equal([1, 22, 3], s:theList)
|
||||||
|
lockvar s:theList
|
||||||
|
assert_fails('theList[1] = 77', 'E741:')
|
||||||
|
unlockvar s:theList
|
||||||
|
s:theList[1] = 44
|
||||||
|
assert_equal([1, 44, 3], s:theList)
|
||||||
|
|
||||||
|
var lines =<< trim END
|
||||||
|
vim9script
|
||||||
|
var theList = [1, 2, 3]
|
||||||
|
def SetList()
|
||||||
|
theList[1] = 22
|
||||||
|
assert_equal([1, 22, 3], theList)
|
||||||
|
lockvar theList
|
||||||
|
theList[1] = 77
|
||||||
|
enddef
|
||||||
|
SetList()
|
||||||
|
END
|
||||||
|
CheckScriptFailure(lines, 'E1119', 4)
|
||||||
|
|
||||||
|
lines =<< trim END
|
||||||
|
var theList = [1, 2, 3]
|
||||||
|
lockvar theList
|
||||||
|
END
|
||||||
|
CheckDefFailure(lines, 'E1178', 2)
|
||||||
|
|
||||||
|
lines =<< trim END
|
||||||
|
var theList = [1, 2, 3]
|
||||||
|
unlockvar theList
|
||||||
|
END
|
||||||
|
CheckDefFailure(lines, 'E1178', 2)
|
||||||
|
enddef
|
||||||
|
|
||||||
|
|
||||||
" vim: ts=8 sw=2 sts=2 expandtab tw=80 fdm=marker
|
" vim: ts=8 sw=2 sts=2 expandtab tw=80 fdm=marker
|
||||||
|
@@ -750,6 +750,8 @@ static char *(features[]) =
|
|||||||
|
|
||||||
static int included_patches[] =
|
static int included_patches[] =
|
||||||
{ /* Add new patch number below this line */
|
{ /* Add new patch number below this line */
|
||||||
|
/**/
|
||||||
|
2672,
|
||||||
/**/
|
/**/
|
||||||
2671,
|
2671,
|
||||||
/**/
|
/**/
|
||||||
|
@@ -6707,6 +6707,58 @@ compile_unlet(
|
|||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Callback passed to ex_unletlock().
|
||||||
|
*/
|
||||||
|
static int
|
||||||
|
compile_lock_unlock(
|
||||||
|
lval_T *lvp,
|
||||||
|
char_u *name_end,
|
||||||
|
exarg_T *eap,
|
||||||
|
int deep UNUSED,
|
||||||
|
void *coookie)
|
||||||
|
{
|
||||||
|
cctx_T *cctx = coookie;
|
||||||
|
int cc = *name_end;
|
||||||
|
char_u *p = lvp->ll_name;
|
||||||
|
int ret = OK;
|
||||||
|
size_t len;
|
||||||
|
char_u *buf;
|
||||||
|
|
||||||
|
if (cctx->ctx_skip == SKIP_YES)
|
||||||
|
return OK;
|
||||||
|
|
||||||
|
// Cannot use :lockvar and :unlockvar on local variables.
|
||||||
|
if (p[1] != ':')
|
||||||
|
{
|
||||||
|
char_u *end = skip_var_one(p, FALSE);
|
||||||
|
|
||||||
|
if (lookup_local(p, end - p, NULL, cctx) == OK)
|
||||||
|
{
|
||||||
|
emsg(_(e_cannot_lock_unlock_local_variable));
|
||||||
|
return FAIL;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Checking is done at runtime.
|
||||||
|
*name_end = NUL;
|
||||||
|
len = name_end - p + 20;
|
||||||
|
buf = alloc(len);
|
||||||
|
if (buf == NULL)
|
||||||
|
ret = FAIL;
|
||||||
|
else
|
||||||
|
{
|
||||||
|
vim_snprintf((char *)buf, len, "%s %s",
|
||||||
|
eap->cmdidx == CMD_lockvar ? "lockvar" : "unlockvar",
|
||||||
|
p);
|
||||||
|
ret = generate_EXEC(cctx, buf);
|
||||||
|
|
||||||
|
vim_free(buf);
|
||||||
|
*name_end = cc;
|
||||||
|
}
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* compile "unlet var", "lock var" and "unlock var"
|
* compile "unlet var", "lock var" and "unlock var"
|
||||||
* "arg" points to "var".
|
* "arg" points to "var".
|
||||||
@@ -6714,16 +6766,9 @@ compile_unlet(
|
|||||||
static char_u *
|
static char_u *
|
||||||
compile_unletlock(char_u *arg, exarg_T *eap, cctx_T *cctx)
|
compile_unletlock(char_u *arg, exarg_T *eap, cctx_T *cctx)
|
||||||
{
|
{
|
||||||
char_u *p = arg;
|
ex_unletlock(eap, arg, 0, GLV_NO_AUTOLOAD | GLV_COMPILING,
|
||||||
|
eap->cmdidx == CMD_unlet ? compile_unlet : compile_lock_unlock,
|
||||||
if (eap->cmdidx != CMD_unlet)
|
cctx);
|
||||||
{
|
|
||||||
emsg("Sorry, :lock and unlock not implemented yet");
|
|
||||||
return NULL;
|
|
||||||
}
|
|
||||||
|
|
||||||
ex_unletlock(eap, p, 0, GLV_NO_AUTOLOAD | GLV_COMPILING,
|
|
||||||
compile_unlet, cctx);
|
|
||||||
return eap->nextcmd == NULL ? (char_u *)"" : eap->nextcmd;
|
return eap->nextcmd == NULL ? (char_u *)"" : eap->nextcmd;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Reference in New Issue
Block a user