patch 8.2.4973: Vim9: type error for list unpack mentions argument
Problem: Vim9: type error for list unpack mentions argument. Solution: Mention variable. (close #10435)
This commit is contained in:
		| @ -9,9 +9,10 @@ vartype_T operator_type(type_T *type1, type_T *type2); | |||||||
| int generate_two_op(cctx_T *cctx, char_u *op); | int generate_two_op(cctx_T *cctx, char_u *op); | ||||||
| int check_compare_types(exprtype_T type, typval_T *tv1, typval_T *tv2); | int check_compare_types(exprtype_T type, typval_T *tv1, typval_T *tv2); | ||||||
| int generate_COMPARE(cctx_T *cctx, exprtype_T exprtype, int ic); | int generate_COMPARE(cctx_T *cctx, exprtype_T exprtype, int ic); | ||||||
|  | int generate_CONCAT(cctx_T *cctx, int count); | ||||||
| int generate_2BOOL(cctx_T *cctx, int invert, int offset); | int generate_2BOOL(cctx_T *cctx, int invert, int offset); | ||||||
| int generate_COND2BOOL(cctx_T *cctx); | int generate_COND2BOOL(cctx_T *cctx); | ||||||
| int generate_TYPECHECK(cctx_T *cctx, type_T *expected, int offset, int argidx); | int generate_TYPECHECK(cctx_T *cctx, type_T *expected, int offset, int is_var, int argidx); | ||||||
| int generate_SETTYPE(cctx_T *cctx, type_T *expected); | int generate_SETTYPE(cctx_T *cctx, type_T *expected); | ||||||
| int generate_tv_PUSH(cctx_T *cctx, typval_T *tv); | int generate_tv_PUSH(cctx_T *cctx, typval_T *tv); | ||||||
| int generate_PUSHNR(cctx_T *cctx, varnumber_T number); | int generate_PUSHNR(cctx_T *cctx, varnumber_T number); | ||||||
| @ -62,7 +63,6 @@ int generate_LEGACY_EVAL(cctx_T *cctx, char_u *line); | |||||||
| int generate_EXECCONCAT(cctx_T *cctx, int count); | int generate_EXECCONCAT(cctx_T *cctx, int count); | ||||||
| int generate_RANGE(cctx_T *cctx, char_u *range); | int generate_RANGE(cctx_T *cctx, char_u *range); | ||||||
| int generate_UNPACK(cctx_T *cctx, int var_count, int semicolon); | int generate_UNPACK(cctx_T *cctx, int var_count, int semicolon); | ||||||
| int generate_CONCAT(cctx_T *cctx, int count); |  | ||||||
| int generate_cmdmods(cctx_T *cctx, cmdmod_T *cmod); | int generate_cmdmods(cctx_T *cctx, cmdmod_T *cmod); | ||||||
| int generate_undo_cmdmods(cctx_T *cctx); | int generate_undo_cmdmods(cctx_T *cctx); | ||||||
| int generate_store_var(cctx_T *cctx, assign_dest_T dest, int opt_flags, int vimvaridx, int scriptvar_idx, int scriptvar_sid, type_T *type, char_u *name); | int generate_store_var(cctx_T *cctx, assign_dest_T dest, int opt_flags, int vimvaridx, int scriptvar_idx, int scriptvar_sid, type_T *type, char_u *name); | ||||||
|  | |||||||
| @ -581,10 +581,10 @@ def Test_disassemble_list_assign() | |||||||
|         '\d CHECKTYPE list<any> stack\[-1\]\_s*' .. |         '\d CHECKTYPE list<any> stack\[-1\]\_s*' .. | ||||||
|         '\d CHECKLEN >= 2\_s*' .. |         '\d CHECKLEN >= 2\_s*' .. | ||||||
|         '\d\+ ITEM 0\_s*' .. |         '\d\+ ITEM 0\_s*' .. | ||||||
|         '\d\+ CHECKTYPE string stack\[-1\] arg 1\_s*' .. |         '\d\+ CHECKTYPE string stack\[-1\] var 1\_s*' .. | ||||||
|         '\d\+ STORE $0\_s*' .. |         '\d\+ STORE $0\_s*' .. | ||||||
|         '\d\+ ITEM 1\_s*' .. |         '\d\+ ITEM 1\_s*' .. | ||||||
|         '\d\+ CHECKTYPE string stack\[-1\] arg 2\_s*' .. |         '\d\+ CHECKTYPE string stack\[-1\] var 2\_s*' .. | ||||||
|         '\d\+ STORE $1\_s*' .. |         '\d\+ STORE $1\_s*' .. | ||||||
|         '\d\+ SLICE 2\_s*' .. |         '\d\+ SLICE 2\_s*' .. | ||||||
|         '\d\+ STORE $2\_s*' .. |         '\d\+ STORE $2\_s*' .. | ||||||
|  | |||||||
| @ -2302,7 +2302,7 @@ def Test_for_loop_fails() | |||||||
|         echo k v |         echo k v | ||||||
|       endfor |       endfor | ||||||
|   END |   END | ||||||
|   v9.CheckDefExecAndScriptFailure(lines, ['E1013: Argument 1: type mismatch, expected job but got string', 'E1012: Type mismatch; expected job but got string'], 2) |   v9.CheckDefExecAndScriptFailure(lines, ['E1163: Variable 1: type mismatch, expected job but got string', 'E1012: Type mismatch; expected job but got string'], 2) | ||||||
|  |  | ||||||
|   lines =<< trim END |   lines =<< trim END | ||||||
|       var i = 0 |       var i = 0 | ||||||
|  | |||||||
| @ -746,6 +746,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 */ | ||||||
|  | /**/ | ||||||
|  |     4973, | ||||||
| /**/ | /**/ | ||||||
|     4972, |     4972, | ||||||
| /**/ | /**/ | ||||||
|  | |||||||
| @ -296,6 +296,7 @@ typedef struct { | |||||||
|     type_T	*ct_type; |     type_T	*ct_type; | ||||||
|     int8_T	ct_off;		// offset in stack, -1 is bottom |     int8_T	ct_off;		// offset in stack, -1 is bottom | ||||||
|     int8_T	ct_arg_idx;	// argument index or zero |     int8_T	ct_arg_idx;	// argument index or zero | ||||||
|  |     int8_T	ct_is_var;	// when TRUE checking variable instead of arg | ||||||
| } checktype_T; | } checktype_T; | ||||||
|  |  | ||||||
| // arguments to ISN_STORENR | // arguments to ISN_STORENR | ||||||
|  | |||||||
| @ -412,7 +412,8 @@ need_type_where( | |||||||
|     // If the actual type can be the expected type add a runtime check. |     // If the actual type can be the expected type add a runtime check. | ||||||
|     if (!actual_is_const && ret == MAYBE && use_typecheck(actual, expected)) |     if (!actual_is_const && ret == MAYBE && use_typecheck(actual, expected)) | ||||||
|     { |     { | ||||||
| 	generate_TYPECHECK(cctx, expected, offset, where.wt_index); | 	generate_TYPECHECK(cctx, expected, offset, | ||||||
|  | 					    where.wt_variable, where.wt_index); | ||||||
| 	return OK; | 	return OK; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  | |||||||
| @ -4652,14 +4652,17 @@ exec_instructions(ectx_T *ectx) | |||||||
| 	    case ISN_CHECKTYPE: | 	    case ISN_CHECKTYPE: | ||||||
| 		{ | 		{ | ||||||
| 		    checktype_T *ct = &iptr->isn_arg.type; | 		    checktype_T *ct = &iptr->isn_arg.type; | ||||||
|  | 		    int		save_wt_variable = ectx->ec_where.wt_variable; | ||||||
|  |  | ||||||
| 		    tv = STACK_TV_BOT((int)ct->ct_off); | 		    tv = STACK_TV_BOT((int)ct->ct_off); | ||||||
| 		    SOURCING_LNUM = iptr->isn_lnum; | 		    SOURCING_LNUM = iptr->isn_lnum; | ||||||
| 		    if (!ectx->ec_where.wt_variable) | 		    if (!ectx->ec_where.wt_variable) | ||||||
| 			ectx->ec_where.wt_index = ct->ct_arg_idx; | 			ectx->ec_where.wt_index = ct->ct_arg_idx; | ||||||
|  | 		    ectx->ec_where.wt_variable = ct->ct_is_var; | ||||||
| 		    if (check_typval_type(ct->ct_type, tv, ectx->ec_where) | 		    if (check_typval_type(ct->ct_type, tv, ectx->ec_where) | ||||||
| 								       == FAIL) | 								       == FAIL) | ||||||
| 			goto on_error; | 			goto on_error; | ||||||
|  | 		    ectx->ec_where.wt_variable = save_wt_variable; | ||||||
| 		    if (!ectx->ec_where.wt_variable) | 		    if (!ectx->ec_where.wt_variable) | ||||||
| 			ectx->ec_where.wt_index = 0; | 			ectx->ec_where.wt_index = 0; | ||||||
|  |  | ||||||
| @ -6114,18 +6117,19 @@ list_instructions(char *pfx, isn_T *instr, int instr_count, ufunc_T *ufunc) | |||||||
|  |  | ||||||
| 	    case ISN_CHECKTYPE: | 	    case ISN_CHECKTYPE: | ||||||
| 		  { | 		  { | ||||||
| 		      checktype_T *ct = &iptr->isn_arg.type; | 		      checktype_T   *ct = &iptr->isn_arg.type; | ||||||
| 		      char *tofree; | 		      char	    *tofree; | ||||||
|  |  | ||||||
| 		      if (ct->ct_arg_idx == 0) | 		      if (ct->ct_arg_idx == 0) | ||||||
| 			  smsg("%s%4d CHECKTYPE %s stack[%d]", pfx, current, | 			  smsg("%s%4d CHECKTYPE %s stack[%d]", pfx, current, | ||||||
| 					  type_name(ct->ct_type, &tofree), | 					  type_name(ct->ct_type, &tofree), | ||||||
| 					  (int)ct->ct_off); | 					  (int)ct->ct_off); | ||||||
| 		      else | 		      else | ||||||
| 			  smsg("%s%4d CHECKTYPE %s stack[%d] arg %d", | 			  smsg("%s%4d CHECKTYPE %s stack[%d] %s %d", | ||||||
| 					  pfx, current, | 					  pfx, current, | ||||||
| 					  type_name(ct->ct_type, &tofree), | 					  type_name(ct->ct_type, &tofree), | ||||||
| 					  (int)ct->ct_off, | 					  (int)ct->ct_off, | ||||||
|  | 					  ct->ct_is_var ? "var": "arg", | ||||||
| 					  (int)ct->ct_arg_idx); | 					  (int)ct->ct_arg_idx); | ||||||
| 		      vim_free(tofree); | 		      vim_free(tofree); | ||||||
| 		      break; | 		      break; | ||||||
|  | |||||||
| @ -542,6 +542,7 @@ generate_TYPECHECK( | |||||||
| 	cctx_T	    *cctx, | 	cctx_T	    *cctx, | ||||||
| 	type_T	    *expected, | 	type_T	    *expected, | ||||||
| 	int	    offset, | 	int	    offset, | ||||||
|  | 	int	    is_var, | ||||||
| 	int	    argidx) | 	int	    argidx) | ||||||
| { | { | ||||||
|     isn_T	*isn; |     isn_T	*isn; | ||||||
| @ -551,6 +552,7 @@ generate_TYPECHECK( | |||||||
| 	return FAIL; | 	return FAIL; | ||||||
|     isn->isn_arg.type.ct_type = alloc_type(expected); |     isn->isn_arg.type.ct_type = alloc_type(expected); | ||||||
|     isn->isn_arg.type.ct_off = (int8_T)offset; |     isn->isn_arg.type.ct_off = (int8_T)offset; | ||||||
|  |     isn->isn_arg.type.ct_is_var = is_var; | ||||||
|     isn->isn_arg.type.ct_arg_idx = (int8_T)argidx; |     isn->isn_arg.type.ct_arg_idx = (int8_T)argidx; | ||||||
|  |  | ||||||
|     // type becomes expected |     // type becomes expected | ||||||
| @ -1437,7 +1439,7 @@ generate_BCALL(cctx_T *cctx, int func_idx, int argcount, int method_call) | |||||||
|     if (maptype != NULL && maptype[0].type_decl->tt_member != NULL |     if (maptype != NULL && maptype[0].type_decl->tt_member != NULL | ||||||
| 				  && maptype[0].type_decl->tt_member != &t_any) | 				  && maptype[0].type_decl->tt_member != &t_any) | ||||||
| 	// Check that map() didn't change the item types. | 	// Check that map() didn't change the item types. | ||||||
| 	generate_TYPECHECK(cctx, maptype[0].type_decl, -1, 1); | 	generate_TYPECHECK(cctx, maptype[0].type_decl, -1, FALSE, 1); | ||||||
|  |  | ||||||
|     return OK; |     return OK; | ||||||
| } | } | ||||||
|  | |||||||
		Reference in New Issue
	
	Block a user