summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorBram Moolenaar <Bram@vim.org>2020-04-07 22:45:00 +0200
committerBram Moolenaar <Bram@vim.org>2020-04-07 22:45:00 +0200
commit49cf7cc8d2df1509cbe23647166c6cc954d59513 (patch)
tree99aa1fd77604c8b0a1f0dc325d529cbae4fdbb11
parent0b76b42d0a09fb6f1ed79cfc153da4edd6154c89 (diff)
patch 8.2.0529: Vim9: function argument with default not checkedv8.2.0529
Problem: Vim9: function argument with default not checked. Solution: Check type of argument with default value.
-rw-r--r--src/testdir/test_vim9_func.vim1
-rw-r--r--src/userfunc.c4
-rw-r--r--src/version.c2
-rw-r--r--src/vim9compile.c26
4 files changed, 28 insertions, 5 deletions
diff --git a/src/testdir/test_vim9_func.vim b/src/testdir/test_vim9_func.vim
index 558f23cb47..6f9aa0cca4 100644
--- a/src/testdir/test_vim9_func.vim
+++ b/src/testdir/test_vim9_func.vim
@@ -128,6 +128,7 @@ def Test_call_def_varargs()
assert_equal('one,foo', MyDefVarargs('one'))
assert_equal('one,two', MyDefVarargs('one', 'two'))
assert_equal('one,two,three', MyDefVarargs('one', 'two', 'three'))
+ call CheckDefFailure(['MyDefVarargs("one", 22)'], 'E1013: argument 2: type mismatch, expected string but got number')
enddef
" Only varargs
diff --git a/src/userfunc.c b/src/userfunc.c
index 69a57c28d1..316eac6aa0 100644
--- a/src/userfunc.c
+++ b/src/userfunc.c
@@ -3045,8 +3045,8 @@ ex_function(exarg_T *eap)
{
p = ((char_u **)argtypes.ga_data)[i];
if (p == NULL)
- // todo: get type from default value
- type = &t_any;
+ // will get the type from the default value
+ type = &t_unknown;
else
type = parse_type(&p, &fp->uf_type_list);
if (type == NULL)
diff --git a/src/version.c b/src/version.c
index 3518aecab8..a3c54cb85c 100644
--- a/src/version.c
+++ b/src/version.c
@@ -739,6 +739,8 @@ static char *(features[]) =
static int included_patches[] =
{ /* Add new patch number below this line */
/**/
+ 529,
+/**/
528,
/**/
527,
diff --git a/src/vim9compile.c b/src/vim9compile.c
index fb0adb3b42..a8a7647468 100644
--- a/src/vim9compile.c
+++ b/src/vim9compile.c
@@ -5548,6 +5548,7 @@ compile_def_function(ufunc_T *ufunc, int set_return_type)
if (ufunc->uf_def_args.ga_len > 0)
{
int count = ufunc->uf_def_args.ga_len;
+ int first_def_arg = ufunc->uf_args.ga_len - count;
int i;
char_u *arg;
int off = STACK_FRAME_SIZE + (ufunc->uf_va_name != NULL ? 1 : 0);
@@ -5561,11 +5562,30 @@ compile_def_function(ufunc_T *ufunc, int set_return_type)
goto erret;
for (i = 0; i < count; ++i)
{
+ garray_T *stack = &cctx.ctx_type_stack;
+ type_T *val_type;
+ int arg_idx = first_def_arg + i;
+
ufunc->uf_def_arg_idx[i] = instr->ga_len;
arg = ((char_u **)(ufunc->uf_def_args.ga_data))[i];
- if (compile_expr1(&arg, &cctx) == FAIL
- || generate_STORE(&cctx, ISN_STORE,
- i - count - off, NULL) == FAIL)
+ if (compile_expr1(&arg, &cctx) == FAIL)
+ goto erret;
+
+ // If no type specified use the type of the default value.
+ // Otherwise check that the default value type matches the
+ // specified type.
+ val_type = ((type_T **)stack->ga_data)[stack->ga_len - 1];
+ if (ufunc->uf_arg_types[arg_idx] == &t_unknown)
+ ufunc->uf_arg_types[arg_idx] = val_type;
+ else if (check_type(ufunc->uf_arg_types[i], val_type, FALSE)
+ == FAIL)
+ {
+ arg_type_mismatch(ufunc->uf_arg_types[arg_idx], val_type,
+ arg_idx + 1);
+ goto erret;
+ }
+
+ if (generate_STORE(&cctx, ISN_STORE, i - count - off, NULL) == FAIL)
goto erret;
}