mirror of
https://github.com/vim/vim.git
synced 2025-09-24 03:44:06 -04:00
patch 9.1.0257: Vim9: :call may not find imported class members
Problem: Vim9: :call may not find imported class members (mityu) Solution: Set the typval of an imported lval variable correctly (Yegappan Lakshmanan) fixes: #14334 closes: #14386 Signed-off-by: Yegappan Lakshmanan <yegappan@yahoo.com> Signed-off-by: Christian Brabandt <cb@256bit.org>
This commit is contained in:
committed by
Christian Brabandt
parent
78d742ab88
commit
f1750ca0c2
112
src/eval.c
112
src/eval.c
@@ -1145,6 +1145,91 @@ get_lval_check_access(
|
||||
return OK;
|
||||
}
|
||||
|
||||
/*
|
||||
* Get lval information for a variable imported from script "imp_sid". On
|
||||
* success, updates "lp" with the variable name, type, script ID and typval.
|
||||
* The variable name starts at or after "p".
|
||||
* If "rettv" is not NULL it points to the value to be assigned. This used to
|
||||
* match the rhs and lhs types.
|
||||
* Returns a pointer to the character after the variable name if the imported
|
||||
* variable is valid and writable.
|
||||
* Returns NULL if the variable is not exported or typval is not found or the
|
||||
* rhs type doesn't match the lhs type or the variable is not writable.
|
||||
*/
|
||||
static char_u *
|
||||
get_lval_imported(
|
||||
lval_T *lp,
|
||||
typval_T *rettv,
|
||||
scid_T imp_sid,
|
||||
char_u *p,
|
||||
dictitem_T **dip,
|
||||
int fne_flags,
|
||||
int vim9script)
|
||||
{
|
||||
ufunc_T *ufunc;
|
||||
type_T *type = NULL;
|
||||
int cc;
|
||||
int rc = FAIL;
|
||||
|
||||
p = skipwhite(p);
|
||||
|
||||
import_check_sourced_sid(&imp_sid);
|
||||
lp->ll_sid = imp_sid;
|
||||
lp->ll_name = p;
|
||||
p = find_name_end(lp->ll_name, NULL, NULL, fne_flags);
|
||||
lp->ll_name_end = p;
|
||||
|
||||
// check the item is exported
|
||||
cc = *p;
|
||||
*p = NUL;
|
||||
if (find_exported(imp_sid, lp->ll_name, &ufunc, &type, NULL, NULL,
|
||||
TRUE) == -1)
|
||||
goto failed;
|
||||
|
||||
if (vim9script && type != NULL)
|
||||
{
|
||||
where_T where = WHERE_INIT;
|
||||
|
||||
// In a vim9 script, do type check and make sure the variable is
|
||||
// writable.
|
||||
if (check_typval_type(type, rettv, where) == FAIL)
|
||||
goto failed;
|
||||
}
|
||||
|
||||
// Get the typval for the exported item
|
||||
hashtab_T *ht = &SCRIPT_VARS(imp_sid);
|
||||
if (ht == NULL)
|
||||
goto failed;
|
||||
|
||||
dictitem_T *di = find_var_in_ht(ht, 0, lp->ll_name, TRUE);
|
||||
if (di == NULL)
|
||||
// variable is not found
|
||||
goto success;
|
||||
|
||||
*dip = di;
|
||||
|
||||
// Check whether the variable is writable.
|
||||
svar_T *sv = find_typval_in_script(&di->di_tv, imp_sid, FALSE);
|
||||
if (sv != NULL && sv->sv_const != 0)
|
||||
{
|
||||
semsg(_(e_cannot_change_readonly_variable_str), lp->ll_name);
|
||||
goto failed;
|
||||
}
|
||||
|
||||
// check whether variable is locked
|
||||
if (value_check_lock(di->di_tv.v_lock, lp->ll_name, FALSE))
|
||||
goto failed;
|
||||
|
||||
lp->ll_tv = &di->di_tv;
|
||||
|
||||
success:
|
||||
rc = OK;
|
||||
|
||||
failed:
|
||||
*p = cc;
|
||||
return rc == OK ? p : NULL;
|
||||
}
|
||||
|
||||
/*
|
||||
* Get an lval: variable, Dict item or List item that can be assigned a value
|
||||
* to: "name", "na{me}", "name[expr]", "name[expr:expr]", "name[expr][expr]",
|
||||
@@ -1177,7 +1262,7 @@ get_lval(
|
||||
char_u *p;
|
||||
char_u *expr_start, *expr_end;
|
||||
int cc;
|
||||
dictitem_T *v;
|
||||
dictitem_T *v = NULL;
|
||||
typval_T var1;
|
||||
typval_T var2;
|
||||
int empty1 = FALSE;
|
||||
@@ -1311,28 +1396,13 @@ get_lval(
|
||||
if (*p == '.')
|
||||
{
|
||||
imported_T *import = find_imported(lp->ll_name, p - lp->ll_name, TRUE);
|
||||
|
||||
if (import != NULL)
|
||||
{
|
||||
ufunc_T *ufunc;
|
||||
type_T *type;
|
||||
|
||||
import_check_sourced_sid(&import->imp_sid);
|
||||
lp->ll_sid = import->imp_sid;
|
||||
lp->ll_name = skipwhite(p + 1);
|
||||
p = find_name_end(lp->ll_name, NULL, NULL, fne_flags);
|
||||
lp->ll_name_end = p;
|
||||
|
||||
// check the item is exported
|
||||
cc = *p;
|
||||
*p = NUL;
|
||||
if (find_exported(import->imp_sid, lp->ll_name, &ufunc, &type,
|
||||
NULL, NULL, TRUE) == -1)
|
||||
{
|
||||
*p = cc;
|
||||
p++; // skip '.'
|
||||
p = get_lval_imported(lp, rettv, import->imp_sid, p, &v,
|
||||
fne_flags, vim9script);
|
||||
if (p == NULL)
|
||||
return NULL;
|
||||
}
|
||||
*p = cc;
|
||||
}
|
||||
}
|
||||
|
||||
@@ -1352,7 +1422,7 @@ get_lval(
|
||||
lp->ll_tv = lval_root->lr_tv;
|
||||
v = NULL;
|
||||
}
|
||||
else
|
||||
else if (lp->ll_tv == NULL)
|
||||
{
|
||||
cc = *p;
|
||||
*p = NUL;
|
||||
|
Reference in New Issue
Block a user