patch 8.2.1435: Vim9: always converting to string for ".." leads to mistakes

Problem:    Vim9: always converting to string for ".." leads to mistakes.
Solution:   Only automatically convert simple types.
This commit is contained in:
Bram Moolenaar
2020-08-12 21:34:49 +02:00
parent fd77748df2
commit 418f1df547
10 changed files with 157 additions and 12 deletions

View File

@ -426,11 +426,14 @@ The boolean operators "||" and "&&" do not change the value: >
2 && 0 == 0 2 && 0 == 0
[] && 2 == [] [] && 2 == []
When using `..` for string concatenation the arguments are always converted to When using `..` for string concatenation arguments of simple types are always
string. > converted to string. >
'hello ' .. 123 == 'hello 123' 'hello ' .. 123 == 'hello 123'
'hello ' .. v:true == 'hello true' 'hello ' .. v:true == 'hello true'
Simple types are string, float, special and bool. For other types |string()|
can be used.
In Vim9 script one can use "true" for v:true and "false" for v:false. In Vim9 script one can use "true" for v:true and "false" for v:false.
@ -805,6 +808,9 @@ actually needed. A recommended mechanism:
... ...
< This goes in .../import/someother.vim. < This goes in .../import/someother.vim.
When compiling a `:def` function and a function in an autoload script is
encountered, the script is not loaded until the `:def` function is called.
Import in legacy Vim script ~ Import in legacy Vim script ~

View File

@ -2712,7 +2712,7 @@ eval5(char_u **arg, typval_T *rettv, evalarg_T *evalarg)
return FAIL; return FAIL;
} }
*arg = skipwhite_and_linebreak(*arg + oplen, evalarg); *arg = skipwhite_and_linebreak(*arg + oplen, evalarg);
if (eval6(arg, &var2, evalarg, op == '.') == FAIL) if (eval6(arg, &var2, evalarg, !in_vim9script() && op == '.') == FAIL)
{ {
clear_tv(rettv); clear_tv(rettv);
return FAIL; return FAIL;
@ -2727,8 +2727,22 @@ eval5(char_u **arg, typval_T *rettv, evalarg_T *evalarg)
{ {
char_u buf1[NUMBUFLEN], buf2[NUMBUFLEN]; char_u buf1[NUMBUFLEN], buf2[NUMBUFLEN];
char_u *s1 = tv_get_string_buf(rettv, buf1); char_u *s1 = tv_get_string_buf(rettv, buf1);
char_u *s2 = tv_get_string_buf_chk(&var2, buf2); char_u *s2 = NULL;
if (in_vim9script() && (var2.v_type == VAR_VOID
|| var2.v_type == VAR_CHANNEL
|| var2.v_type == VAR_JOB))
emsg(_(e_inval_string));
#ifdef FEAT_FLOAT
else if (var2.v_type == VAR_FLOAT)
{
vim_snprintf((char *)buf2, NUMBUFLEN, "%g",
var2.vval.v_float);
s2 = buf2;
}
#endif
else
s2 = tv_get_string_buf_chk(&var2, buf2);
if (s2 == NULL) // type error ? if (s2 == NULL) // type error ?
{ {
clear_tv(rettv); clear_tv(rettv);

View File

@ -1046,7 +1046,7 @@ static funcentry_T global_functions[] =
{"test_settime", 1, 1, FEARG_1, ret_void, f_test_settime}, {"test_settime", 1, 1, FEARG_1, ret_void, f_test_settime},
{"test_srand_seed", 0, 1, FEARG_1, ret_void, f_test_srand_seed}, {"test_srand_seed", 0, 1, FEARG_1, ret_void, f_test_srand_seed},
{"test_unknown", 0, 0, 0, ret_any, f_test_unknown}, {"test_unknown", 0, 0, 0, ret_any, f_test_unknown},
{"test_void", 0, 0, 0, ret_any, f_test_void}, {"test_void", 0, 0, 0, ret_void, f_test_void},
{"timer_info", 0, 1, FEARG_1, ret_list_dict_any, TIMER_FUNC(f_timer_info)}, {"timer_info", 0, 1, FEARG_1, ret_list_dict_any, TIMER_FUNC(f_timer_info)},
{"timer_pause", 2, 2, FEARG_1, ret_void, TIMER_FUNC(f_timer_pause)}, {"timer_pause", 2, 2, FEARG_1, ret_void, TIMER_FUNC(f_timer_pause)},
{"timer_start", 2, 3, FEARG_1, ret_number, TIMER_FUNC(f_timer_start)}, {"timer_start", 2, 3, FEARG_1, ret_number, TIMER_FUNC(f_timer_start)},

View File

@ -1,4 +1,5 @@
/* vim9execute.c */ /* vim9execute.c */
void to_string_error(vartype_T vartype);
int call_def_function(ufunc_T *ufunc, int argc_arg, typval_T *argv, partial_T *partial, typval_T *rettv); int call_def_function(ufunc_T *ufunc, int argc_arg, typval_T *argv, partial_T *partial, typval_T *rettv);
void ex_disassemble(exarg_T *eap); void ex_disassemble(exarg_T *eap);
int tv2bool(typval_T *tv); int tv2bool(typval_T *tv);

View File

@ -708,7 +708,7 @@ def Test_disassemble_lambda()
'return "X" .. a .. "X"\_s*' .. 'return "X" .. a .. "X"\_s*' ..
'\d PUSHS "X"\_s*' .. '\d PUSHS "X"\_s*' ..
'\d LOAD arg\[-1\]\_s*' .. '\d LOAD arg\[-1\]\_s*' ..
'\d 2STRING stack\[-1\]\_s*' .. '\d 2STRING_ANY stack\[-1\]\_s*' ..
'\d CONCAT\_s*' .. '\d CONCAT\_s*' ..
'\d PUSHS "X"\_s*' .. '\d PUSHS "X"\_s*' ..
'\d CONCAT\_s*' .. '\d CONCAT\_s*' ..
@ -964,7 +964,7 @@ def Test_disassemble_concat()
'let res = g:aa .. "bb".*' .. 'let res = g:aa .. "bb".*' ..
'\d LOADG g:aa.*' .. '\d LOADG g:aa.*' ..
'\d PUSHS "bb".*' .. '\d PUSHS "bb".*' ..
'\d 2STRING stack\[-2].*' .. '\d 2STRING_ANY stack\[-2].*' ..
'\d CONCAT.*' .. '\d CONCAT.*' ..
'\d STORE $.*', '\d STORE $.*',
instr) instr)

View File

@ -921,6 +921,14 @@ def Test_expr5()
assert_equal('123 hello', 123 .. ' hello') assert_equal('123 hello', 123 .. ' hello')
assert_equal('123456', 123 .. 456) assert_equal('123456', 123 .. 456)
assert_equal('av:true', 'a' .. true)
assert_equal('av:false', 'a' .. false)
assert_equal('av:null', 'a' .. v:null)
assert_equal('av:none', 'a' .. v:none)
if has('float')
assert_equal('a0.123', 'a' .. 0.123)
endif
assert_equal([1, 2, 3, 4], [1, 2] + [3, 4]) assert_equal([1, 2, 3, 4], [1, 2] + [3, 4])
assert_equal(0z11223344, 0z1122 + 0z3344) assert_equal(0z11223344, 0z1122 + 0z3344)
assert_equal(0z112201ab, 0z1122 assert_equal(0z112201ab, 0z1122
@ -1013,6 +1021,56 @@ def Test_expr5_vim9script()
echo 'a'.. 'b' echo 'a'.. 'b'
END END
CheckScriptFailure(lines, 'E1004:') CheckScriptFailure(lines, 'E1004:')
# check valid string concatenation
lines =<< trim END
vim9script
assert_equal('one123', 'one' .. 123)
assert_equal('onev:true', 'one' .. true)
assert_equal('onev:null', 'one' .. v:null)
assert_equal('onev:none', 'one' .. v:none)
if has('float')
assert_equal('a0.123', 'a' .. 0.123)
endif
END
CheckScriptSuccess(lines)
# check invalid string concatenation
lines =<< trim END
vim9script
echo 'a' .. [1]
END
CheckScriptFailure(lines, 'E730:')
lines =<< trim END
vim9script
echo 'a' .. #{a: 1}
END
CheckScriptFailure(lines, 'E731:')
lines =<< trim END
vim9script
echo 'a' .. test_void()
END
CheckScriptFailure(lines, 'E908:')
lines =<< trim END
vim9script
echo 'a' .. 0z33
END
CheckScriptFailure(lines, 'E976:')
lines =<< trim END
vim9script
echo 'a' .. function('len')
END
CheckScriptFailure(lines, 'E729:')
lines =<< trim END
vim9script
echo 'a' .. test_null_job()
END
CheckScriptFailure(lines, 'E908:')
lines =<< trim END
vim9script
echo 'a' .. test_null_channel()
END
CheckScriptFailure(lines, 'E908:')
enddef enddef
def Test_expr5_float() def Test_expr5_float()
@ -1063,6 +1121,15 @@ func Test_expr5_fails()
call CheckDefFailure(["let x = [3] + 0z1122"], 'E1051') call CheckDefFailure(["let x = [3] + 0z1122"], 'E1051')
call CheckDefFailure(["let x = 'asdf' + 0z1122"], 'E1051') call CheckDefFailure(["let x = 'asdf' + 0z1122"], 'E1051')
call CheckDefFailure(["let x = 6 + xxx"], 'E1001') call CheckDefFailure(["let x = 6 + xxx"], 'E1001')
call CheckDefFailure(["let x = 'a' .. [1]"], 'E1105')
call CheckDefFailure(["let x = 'a' .. #{a: 1}"], 'E1105')
call CheckDefFailure(["let x = 'a' .. test_void()"], 'E1105')
call CheckDefFailure(["let x = 'a' .. 0z32"], 'E1105')
call CheckDefFailure(["let x = 'a' .. function('len')"], 'E1105')
call CheckDefFailure(["let x = 'a' .. function('len', ['a'])"], 'E1105')
call CheckDefFailure(["let x = 'a' .. test_null_job()"], 'E1105')
call CheckDefFailure(["let x = 'a' .. test_null_channel()"], 'E1105')
endfunc endfunc
" test multiply, divide, modulo " test multiply, divide, modulo

View File

@ -754,6 +754,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 */
/**/
1435,
/**/ /**/
1434, 1434,
/**/ /**/

View File

@ -93,7 +93,7 @@ typedef enum {
ISN_CATCH, // drop v:exception ISN_CATCH, // drop v:exception
ISN_ENDTRY, // take entry off from ec_trystack ISN_ENDTRY, // take entry off from ec_trystack
// moreexpression operations // more expression operations
ISN_ADDLIST, ISN_ADDLIST,
ISN_ADDBLOB, ISN_ADDBLOB,
@ -124,6 +124,7 @@ typedef enum {
ISN_STRINGMEMBER, // dict.member using isn_arg.string ISN_STRINGMEMBER, // dict.member using isn_arg.string
ISN_2BOOL, // convert value to bool, invert if isn_arg.number != 0 ISN_2BOOL, // convert value to bool, invert if isn_arg.number != 0
ISN_2STRING, // convert value to string at isn_arg.number on stack ISN_2STRING, // convert value to string at isn_arg.number on stack
ISN_2STRING_ANY, // like ISN_2STRING but check type
ISN_NEGATENR, // apply "-" to number ISN_NEGATENR, // apply "-" to number
ISN_CHECKNR, // check value can be used as a number ISN_CHECKNR, // check value can be used as a number

View File

@ -374,19 +374,49 @@ generate_instr_type(cctx_T *cctx, isntype_T isn_type, type_T *type)
/* /*
* If type at "offset" isn't already VAR_STRING then generate ISN_2STRING. * If type at "offset" isn't already VAR_STRING then generate ISN_2STRING.
* But only for simple types.
*/ */
static int static int
may_generate_2STRING(int offset, cctx_T *cctx) may_generate_2STRING(int offset, cctx_T *cctx)
{ {
isn_T *isn; isn_T *isn;
isntype_T isntype = ISN_2STRING;
garray_T *stack = &cctx->ctx_type_stack; garray_T *stack = &cctx->ctx_type_stack;
type_T **type = ((type_T **)stack->ga_data) + stack->ga_len + offset; type_T **type = ((type_T **)stack->ga_data) + stack->ga_len + offset;
if ((*type)->tt_type == VAR_STRING) switch ((*type)->tt_type)
return OK; {
*type = &t_string; // nothing to be done
case VAR_STRING: return OK;
if ((isn = generate_instr(cctx, ISN_2STRING)) == NULL) // conversion possible
case VAR_SPECIAL:
case VAR_BOOL:
case VAR_NUMBER:
case VAR_FLOAT:
break;
// conversion possible (with runtime check)
case VAR_ANY:
case VAR_UNKNOWN:
isntype = ISN_2STRING_ANY;
break;
// conversion not possible
case VAR_VOID:
case VAR_BLOB:
case VAR_FUNC:
case VAR_PARTIAL:
case VAR_LIST:
case VAR_DICT:
case VAR_JOB:
case VAR_CHANNEL:
to_string_error((*type)->tt_type);
return FAIL;
}
*type = &t_string;
if ((isn = generate_instr(cctx, isntype)) == NULL)
return FAIL; return FAIL;
isn->isn_arg.number = offset; isn->isn_arg.number = offset;
@ -7005,6 +7035,7 @@ delete_instr(isn_T *isn)
case ISN_2BOOL: case ISN_2BOOL:
case ISN_2STRING: case ISN_2STRING:
case ISN_2STRING_ANY:
case ISN_ADDBLOB: case ISN_ADDBLOB:
case ISN_ADDLIST: case ISN_ADDLIST:
case ISN_BCALL: case ISN_BCALL:

View File

@ -72,6 +72,12 @@ typedef struct {
// Get pointer to item relative to the bottom of the stack, -1 is the last one. // Get pointer to item relative to the bottom of the stack, -1 is the last one.
#define STACK_TV_BOT(idx) (((typval_T *)ectx->ec_stack.ga_data) + ectx->ec_stack.ga_len + idx) #define STACK_TV_BOT(idx) (((typval_T *)ectx->ec_stack.ga_data) + ectx->ec_stack.ga_len + idx)
void
to_string_error(vartype_T vartype)
{
semsg(_("E1105: Cannot convert %s to string"), vartype_name(vartype));
}
/* /*
* Return the number of arguments, including optional arguments and any vararg. * Return the number of arguments, including optional arguments and any vararg.
*/ */
@ -2476,12 +2482,26 @@ call_def_function(
break; break;
case ISN_2STRING: case ISN_2STRING:
case ISN_2STRING_ANY:
{ {
char_u *str; char_u *str;
tv = STACK_TV_BOT(iptr->isn_arg.number); tv = STACK_TV_BOT(iptr->isn_arg.number);
if (tv->v_type != VAR_STRING) if (tv->v_type != VAR_STRING)
{ {
if (iptr->isn_type == ISN_2STRING_ANY)
{
switch (tv->v_type)
{
case VAR_SPECIAL:
case VAR_BOOL:
case VAR_NUMBER:
case VAR_FLOAT:
case VAR_BLOB: break;
default: to_string_error(tv->v_type);
goto on_error;
}
}
str = typval_tostring(tv); str = typval_tostring(tv);
clear_tv(tv); clear_tv(tv);
tv->v_type = VAR_STRING; tv->v_type = VAR_STRING;
@ -3127,6 +3147,9 @@ ex_disassemble(exarg_T *eap)
case ISN_2STRING: smsg("%4d 2STRING stack[%lld]", current, case ISN_2STRING: smsg("%4d 2STRING stack[%lld]", current,
(long long)(iptr->isn_arg.number)); (long long)(iptr->isn_arg.number));
break; break;
case ISN_2STRING_ANY: smsg("%4d 2STRING_ANY stack[%lld]", current,
(long long)(iptr->isn_arg.number));
break;
case ISN_SHUFFLE: smsg("%4d SHUFFLE %d up %d", current, case ISN_SHUFFLE: smsg("%4d SHUFFLE %d up %d", current,
iptr->isn_arg.shuffle.shfl_item, iptr->isn_arg.shuffle.shfl_item,