xref: /vim-8.2.3635/src/vim9execute.c (revision e0807ea4)
18a7d6542SBram Moolenaar /* vi:set ts=8 sts=4 sw=4 noet:
28a7d6542SBram Moolenaar  *
38a7d6542SBram Moolenaar  * VIM - Vi IMproved	by Bram Moolenaar
48a7d6542SBram Moolenaar  *
58a7d6542SBram Moolenaar  * Do ":help uganda"  in Vim to read copying and usage conditions.
68a7d6542SBram Moolenaar  * Do ":help credits" in Vim to see a list of people who contributed.
78a7d6542SBram Moolenaar  * See README.txt for an overview of the Vim source code.
88a7d6542SBram Moolenaar  */
98a7d6542SBram Moolenaar 
108a7d6542SBram Moolenaar /*
118a7d6542SBram Moolenaar  * vim9execute.c: execute Vim9 script instructions
128a7d6542SBram Moolenaar  */
138a7d6542SBram Moolenaar 
148a7d6542SBram Moolenaar #define USING_FLOAT_STUFF
158a7d6542SBram Moolenaar #include "vim.h"
168a7d6542SBram Moolenaar 
178a7d6542SBram Moolenaar #if defined(FEAT_EVAL) || defined(PROTO)
188a7d6542SBram Moolenaar 
198a7d6542SBram Moolenaar #ifdef VMS
208a7d6542SBram Moolenaar # include <float.h>
218a7d6542SBram Moolenaar #endif
228a7d6542SBram Moolenaar 
238a7d6542SBram Moolenaar #include "vim9.h"
248a7d6542SBram Moolenaar 
258a7d6542SBram Moolenaar // Structure put on ec_trystack when ISN_TRY is encountered.
268a7d6542SBram Moolenaar typedef struct {
278a7d6542SBram Moolenaar     int	    tcd_frame;		// ec_frame when ISN_TRY was encountered
288a7d6542SBram Moolenaar     int	    tcd_catch_idx;	// instruction of the first catch
298a7d6542SBram Moolenaar     int	    tcd_finally_idx;	// instruction of the finally block
308a7d6542SBram Moolenaar     int	    tcd_caught;		// catch block entered
318a7d6542SBram Moolenaar     int	    tcd_return;		// when TRUE return from end of :finally
328a7d6542SBram Moolenaar } trycmd_T;
338a7d6542SBram Moolenaar 
348a7d6542SBram Moolenaar 
358a7d6542SBram Moolenaar // A stack is used to store:
368a7d6542SBram Moolenaar // - arguments passed to a :def function
378a7d6542SBram Moolenaar // - info about the calling function, to use when returning
388a7d6542SBram Moolenaar // - local variables
398a7d6542SBram Moolenaar // - temporary values
408a7d6542SBram Moolenaar //
418a7d6542SBram Moolenaar // In detail (FP == Frame Pointer):
428a7d6542SBram Moolenaar //	  arg1		first argument from caller (if present)
438a7d6542SBram Moolenaar //	  arg2		second argument from caller (if present)
448a7d6542SBram Moolenaar //	  extra_arg1	any missing optional argument default value
458a7d6542SBram Moolenaar // FP ->  cur_func	calling function
468a7d6542SBram Moolenaar //        current	previous instruction pointer
478a7d6542SBram Moolenaar //        frame_ptr	previous Frame Pointer
488a7d6542SBram Moolenaar //        var1		space for local variable
498a7d6542SBram Moolenaar //        var2		space for local variable
508a7d6542SBram Moolenaar //        ....		fixed space for max. number of local variables
518a7d6542SBram Moolenaar //        temp		temporary values
528a7d6542SBram Moolenaar //        ....		flexible space for temporary values (can grow big)
538a7d6542SBram Moolenaar 
548a7d6542SBram Moolenaar /*
558a7d6542SBram Moolenaar  * Execution context.
568a7d6542SBram Moolenaar  */
578a7d6542SBram Moolenaar typedef struct {
588a7d6542SBram Moolenaar     garray_T	ec_stack;	// stack of typval_T values
598a7d6542SBram Moolenaar     int		ec_frame;	// index in ec_stack: context of ec_dfunc_idx
608a7d6542SBram Moolenaar 
618a7d6542SBram Moolenaar     garray_T	ec_trystack;	// stack of trycmd_T values
628a7d6542SBram Moolenaar     int		ec_in_catch;	// when TRUE in catch or finally block
638a7d6542SBram Moolenaar 
648a7d6542SBram Moolenaar     int		ec_dfunc_idx;	// current function index
658a7d6542SBram Moolenaar     isn_T	*ec_instr;	// array with instructions
668a7d6542SBram Moolenaar     int		ec_iidx;	// index in ec_instr: instruction to execute
678a7d6542SBram Moolenaar } ectx_T;
688a7d6542SBram Moolenaar 
698a7d6542SBram Moolenaar // Get pointer to item relative to the bottom of the stack, -1 is the last one.
708a7d6542SBram Moolenaar #define STACK_TV_BOT(idx) (((typval_T *)ectx->ec_stack.ga_data) + ectx->ec_stack.ga_len + idx)
718a7d6542SBram Moolenaar 
728a7d6542SBram Moolenaar /*
73170fcfcfSBram Moolenaar  * Return the number of arguments, including optional arguments and any vararg.
748a7d6542SBram Moolenaar  */
758a7d6542SBram Moolenaar     static int
768a7d6542SBram Moolenaar ufunc_argcount(ufunc_T *ufunc)
778a7d6542SBram Moolenaar {
788a7d6542SBram Moolenaar     return ufunc->uf_args.ga_len + (ufunc->uf_va_name != NULL ? 1 : 0);
798a7d6542SBram Moolenaar }
808a7d6542SBram Moolenaar 
818a7d6542SBram Moolenaar /*
82170fcfcfSBram Moolenaar  * Set the instruction index, depending on omitted arguments, where the default
83170fcfcfSBram Moolenaar  * values are to be computed.  If all optional arguments are present, start
84170fcfcfSBram Moolenaar  * with the function body.
85170fcfcfSBram Moolenaar  * The expression evaluation is at the start of the instructions:
86170fcfcfSBram Moolenaar  *  0 ->  EVAL default1
87170fcfcfSBram Moolenaar  *	       STORE arg[-2]
88170fcfcfSBram Moolenaar  *  1 ->  EVAL default2
89170fcfcfSBram Moolenaar  *	       STORE arg[-1]
90170fcfcfSBram Moolenaar  *  2 ->  function body
91170fcfcfSBram Moolenaar  */
92170fcfcfSBram Moolenaar     static void
93170fcfcfSBram Moolenaar init_instr_idx(ufunc_T *ufunc, int argcount, ectx_T *ectx)
94170fcfcfSBram Moolenaar {
95170fcfcfSBram Moolenaar     if (ufunc->uf_def_args.ga_len == 0)
96170fcfcfSBram Moolenaar 	ectx->ec_iidx = 0;
97170fcfcfSBram Moolenaar     else
98170fcfcfSBram Moolenaar     {
99170fcfcfSBram Moolenaar 	int	defcount = ufunc->uf_args.ga_len - argcount;
100170fcfcfSBram Moolenaar 
101170fcfcfSBram Moolenaar 	// If there is a varargs argument defcount can be negative, no defaults
102170fcfcfSBram Moolenaar 	// to evaluate then.
103170fcfcfSBram Moolenaar 	if (defcount < 0)
104170fcfcfSBram Moolenaar 	    defcount = 0;
105170fcfcfSBram Moolenaar 	ectx->ec_iidx = ufunc->uf_def_arg_idx[
106170fcfcfSBram Moolenaar 					 ufunc->uf_def_args.ga_len - defcount];
107170fcfcfSBram Moolenaar     }
108170fcfcfSBram Moolenaar }
109170fcfcfSBram Moolenaar 
110170fcfcfSBram Moolenaar /*
1118a7d6542SBram Moolenaar  * Call compiled function "cdf_idx" from compiled code.
1128a7d6542SBram Moolenaar  *
1138a7d6542SBram Moolenaar  * Stack has:
1148a7d6542SBram Moolenaar  * - current arguments (already there)
1158a7d6542SBram Moolenaar  * - omitted optional argument (default values) added here
1168a7d6542SBram Moolenaar  * - stack frame:
1178a7d6542SBram Moolenaar  *	- pointer to calling function
1188a7d6542SBram Moolenaar  *	- Index of next instruction in calling function
1198a7d6542SBram Moolenaar  *	- previous frame pointer
1208a7d6542SBram Moolenaar  * - reserved space for local variables
1218a7d6542SBram Moolenaar  */
1228a7d6542SBram Moolenaar     static int
1238a7d6542SBram Moolenaar call_dfunc(int cdf_idx, int argcount, ectx_T *ectx)
1248a7d6542SBram Moolenaar {
1258a7d6542SBram Moolenaar     dfunc_T *dfunc = ((dfunc_T *)def_functions.ga_data) + cdf_idx;
1268a7d6542SBram Moolenaar     ufunc_T *ufunc = dfunc->df_ufunc;
1278a7d6542SBram Moolenaar     int	    optcount = ufunc_argcount(ufunc) - argcount;
1288a7d6542SBram Moolenaar     int	    idx;
1298a7d6542SBram Moolenaar 
1308a7d6542SBram Moolenaar     if (dfunc->df_deleted)
1318a7d6542SBram Moolenaar     {
1328a7d6542SBram Moolenaar 	emsg_funcname(e_func_deleted, ufunc->uf_name);
1338a7d6542SBram Moolenaar 	return FAIL;
1348a7d6542SBram Moolenaar     }
1358a7d6542SBram Moolenaar 
1368a7d6542SBram Moolenaar     if (ga_grow(&ectx->ec_stack, optcount + 3 + dfunc->df_varcount) == FAIL)
1378a7d6542SBram Moolenaar 	return FAIL;
1388a7d6542SBram Moolenaar 
1398a7d6542SBram Moolenaar     if (optcount < 0)
1408a7d6542SBram Moolenaar     {
1418a7d6542SBram Moolenaar 	emsg("argument count wrong?");
1428a7d6542SBram Moolenaar 	return FAIL;
1438a7d6542SBram Moolenaar     }
144170fcfcfSBram Moolenaar 
145170fcfcfSBram Moolenaar     // Reserve space for omitted optional arguments, filled in soon.
146170fcfcfSBram Moolenaar     // Also any empty varargs argument.
147170fcfcfSBram Moolenaar     ectx->ec_stack.ga_len += optcount;
1488a7d6542SBram Moolenaar 
1498a7d6542SBram Moolenaar     // Store current execution state in stack frame for ISN_RETURN.
1508a7d6542SBram Moolenaar     // TODO: If the actual number of arguments doesn't match what the called
1518a7d6542SBram Moolenaar     // function expects things go bad.
1528a7d6542SBram Moolenaar     STACK_TV_BOT(0)->vval.v_number = ectx->ec_dfunc_idx;
1538a7d6542SBram Moolenaar     STACK_TV_BOT(1)->vval.v_number = ectx->ec_iidx;
1548a7d6542SBram Moolenaar     STACK_TV_BOT(2)->vval.v_number = ectx->ec_frame;
1558a7d6542SBram Moolenaar     ectx->ec_frame = ectx->ec_stack.ga_len;
1568a7d6542SBram Moolenaar 
1578a7d6542SBram Moolenaar     // Initialize local variables
1588a7d6542SBram Moolenaar     for (idx = 0; idx < dfunc->df_varcount; ++idx)
1598a7d6542SBram Moolenaar 	STACK_TV_BOT(STACK_FRAME_SIZE + idx)->v_type = VAR_UNKNOWN;
1608a7d6542SBram Moolenaar     ectx->ec_stack.ga_len += STACK_FRAME_SIZE + dfunc->df_varcount;
1618a7d6542SBram Moolenaar 
1628a7d6542SBram Moolenaar     // Set execution state to the start of the called function.
1638a7d6542SBram Moolenaar     ectx->ec_dfunc_idx = cdf_idx;
1648a7d6542SBram Moolenaar     ectx->ec_instr = dfunc->df_instr;
1658a7d6542SBram Moolenaar     estack_push_ufunc(ETYPE_UFUNC, dfunc->df_ufunc, 1);
166170fcfcfSBram Moolenaar 
167170fcfcfSBram Moolenaar     // Decide where to start execution, handles optional arguments.
168170fcfcfSBram Moolenaar     init_instr_idx(ufunc, argcount, ectx);
1698a7d6542SBram Moolenaar 
1708a7d6542SBram Moolenaar     return OK;
1718a7d6542SBram Moolenaar }
1728a7d6542SBram Moolenaar 
1738a7d6542SBram Moolenaar // Get pointer to item in the stack.
1748a7d6542SBram Moolenaar #define STACK_TV(idx) (((typval_T *)ectx->ec_stack.ga_data) + idx)
1758a7d6542SBram Moolenaar 
1768a7d6542SBram Moolenaar /*
1778a7d6542SBram Moolenaar  * Return from the current function.
1788a7d6542SBram Moolenaar  */
1798a7d6542SBram Moolenaar     static void
1808a7d6542SBram Moolenaar func_return(ectx_T *ectx)
1818a7d6542SBram Moolenaar {
1828a7d6542SBram Moolenaar     int		idx;
1838a7d6542SBram Moolenaar     dfunc_T	*dfunc;
184170fcfcfSBram Moolenaar     int		top;
1858a7d6542SBram Moolenaar 
1868a7d6542SBram Moolenaar     // execution context goes one level up
1878a7d6542SBram Moolenaar     estack_pop();
1888a7d6542SBram Moolenaar 
1898a7d6542SBram Moolenaar     // Clear the local variables and temporary values, but not
1908a7d6542SBram Moolenaar     // the return value.
1918a7d6542SBram Moolenaar     for (idx = ectx->ec_frame + STACK_FRAME_SIZE;
1928a7d6542SBram Moolenaar 					idx < ectx->ec_stack.ga_len - 1; ++idx)
1938a7d6542SBram Moolenaar 	clear_tv(STACK_TV(idx));
194170fcfcfSBram Moolenaar 
195170fcfcfSBram Moolenaar     // Clear the arguments.
1968a7d6542SBram Moolenaar     dfunc = ((dfunc_T *)def_functions.ga_data) + ectx->ec_dfunc_idx;
197170fcfcfSBram Moolenaar     top = ectx->ec_frame - ufunc_argcount(dfunc->df_ufunc);
198170fcfcfSBram Moolenaar     for (idx = top; idx < ectx->ec_frame; ++idx)
199170fcfcfSBram Moolenaar 	clear_tv(STACK_TV(idx));
200170fcfcfSBram Moolenaar 
201170fcfcfSBram Moolenaar     // Restore the previous frame.
2028a7d6542SBram Moolenaar     ectx->ec_dfunc_idx = STACK_TV(ectx->ec_frame)->vval.v_number;
2038a7d6542SBram Moolenaar     ectx->ec_iidx = STACK_TV(ectx->ec_frame + 1)->vval.v_number;
2048a7d6542SBram Moolenaar     ectx->ec_frame = STACK_TV(ectx->ec_frame + 2)->vval.v_number;
2058a7d6542SBram Moolenaar     dfunc = ((dfunc_T *)def_functions.ga_data) + ectx->ec_dfunc_idx;
2068a7d6542SBram Moolenaar     ectx->ec_instr = dfunc->df_instr;
207170fcfcfSBram Moolenaar 
208170fcfcfSBram Moolenaar     // Reset the stack to the position before the call, move the return value
209170fcfcfSBram Moolenaar     // to the top of the stack.
210170fcfcfSBram Moolenaar     idx = ectx->ec_stack.ga_len - 1;
211170fcfcfSBram Moolenaar     ectx->ec_stack.ga_len = top + 1;
212170fcfcfSBram Moolenaar     *STACK_TV_BOT(-1) = *STACK_TV(idx);
2138a7d6542SBram Moolenaar }
2148a7d6542SBram Moolenaar 
2158a7d6542SBram Moolenaar #undef STACK_TV
2168a7d6542SBram Moolenaar 
2178a7d6542SBram Moolenaar /*
2188a7d6542SBram Moolenaar  * Prepare arguments and rettv for calling a builtin or user function.
2198a7d6542SBram Moolenaar  */
2208a7d6542SBram Moolenaar     static int
2218a7d6542SBram Moolenaar call_prepare(int argcount, typval_T *argvars, ectx_T *ectx)
2228a7d6542SBram Moolenaar {
2238a7d6542SBram Moolenaar     int		idx;
2248a7d6542SBram Moolenaar     typval_T	*tv;
2258a7d6542SBram Moolenaar 
2268a7d6542SBram Moolenaar     // Move arguments from bottom of the stack to argvars[] and add terminator.
2278a7d6542SBram Moolenaar     for (idx = 0; idx < argcount; ++idx)
2288a7d6542SBram Moolenaar 	argvars[idx] = *STACK_TV_BOT(idx - argcount);
2298a7d6542SBram Moolenaar     argvars[argcount].v_type = VAR_UNKNOWN;
2308a7d6542SBram Moolenaar 
2318a7d6542SBram Moolenaar     // Result replaces the arguments on the stack.
2328a7d6542SBram Moolenaar     if (argcount > 0)
2338a7d6542SBram Moolenaar 	ectx->ec_stack.ga_len -= argcount - 1;
2348a7d6542SBram Moolenaar     else if (ga_grow(&ectx->ec_stack, 1) == FAIL)
2358a7d6542SBram Moolenaar 	return FAIL;
2368a7d6542SBram Moolenaar     else
2378a7d6542SBram Moolenaar 	++ectx->ec_stack.ga_len;
2388a7d6542SBram Moolenaar 
2398a7d6542SBram Moolenaar     // Default return value is zero.
2408a7d6542SBram Moolenaar     tv = STACK_TV_BOT(-1);
2418a7d6542SBram Moolenaar     tv->v_type = VAR_NUMBER;
2428a7d6542SBram Moolenaar     tv->vval.v_number = 0;
2438a7d6542SBram Moolenaar 
2448a7d6542SBram Moolenaar     return OK;
2458a7d6542SBram Moolenaar }
2468a7d6542SBram Moolenaar 
2478a7d6542SBram Moolenaar /*
2488a7d6542SBram Moolenaar  * Call a builtin function by index.
2498a7d6542SBram Moolenaar  */
2508a7d6542SBram Moolenaar     static int
2518a7d6542SBram Moolenaar call_bfunc(int func_idx, int argcount, ectx_T *ectx)
2528a7d6542SBram Moolenaar {
2538a7d6542SBram Moolenaar     typval_T	argvars[MAX_FUNC_ARGS];
2548a7d6542SBram Moolenaar     int		idx;
2558a7d6542SBram Moolenaar 
2568a7d6542SBram Moolenaar     if (call_prepare(argcount, argvars, ectx) == FAIL)
2578a7d6542SBram Moolenaar 	return FAIL;
2588a7d6542SBram Moolenaar 
2598a7d6542SBram Moolenaar     // Call the builtin function.
2608a7d6542SBram Moolenaar     call_internal_func_by_idx(func_idx, argvars, STACK_TV_BOT(-1));
2618a7d6542SBram Moolenaar 
2628a7d6542SBram Moolenaar     // Clear the arguments.
2638a7d6542SBram Moolenaar     for (idx = 0; idx < argcount; ++idx)
2648a7d6542SBram Moolenaar 	clear_tv(&argvars[idx]);
2658a7d6542SBram Moolenaar     return OK;
2668a7d6542SBram Moolenaar }
2678a7d6542SBram Moolenaar 
2688a7d6542SBram Moolenaar /*
2698a7d6542SBram Moolenaar  * Execute a user defined function.
2708a7d6542SBram Moolenaar  */
2718a7d6542SBram Moolenaar     static int
2728a7d6542SBram Moolenaar call_ufunc(ufunc_T *ufunc, int argcount, ectx_T *ectx)
2738a7d6542SBram Moolenaar {
2748a7d6542SBram Moolenaar     typval_T	argvars[MAX_FUNC_ARGS];
2758a7d6542SBram Moolenaar     funcexe_T   funcexe;
2768a7d6542SBram Moolenaar     int		error;
2778a7d6542SBram Moolenaar     int		idx;
2788a7d6542SBram Moolenaar 
2798a7d6542SBram Moolenaar     if (ufunc->uf_dfunc_idx >= 0)
2808a7d6542SBram Moolenaar 	// The function has been compiled, can call it quickly.
2818a7d6542SBram Moolenaar 	return call_dfunc(ufunc->uf_dfunc_idx, argcount, ectx);
2828a7d6542SBram Moolenaar 
2838a7d6542SBram Moolenaar     if (call_prepare(argcount, argvars, ectx) == FAIL)
2848a7d6542SBram Moolenaar 	return FAIL;
2858a7d6542SBram Moolenaar     vim_memset(&funcexe, 0, sizeof(funcexe));
2868a7d6542SBram Moolenaar     funcexe.evaluate = TRUE;
2878a7d6542SBram Moolenaar 
2888a7d6542SBram Moolenaar     // Call the user function.  Result goes in last position on the stack.
2898a7d6542SBram Moolenaar     // TODO: add selfdict if there is one
2908a7d6542SBram Moolenaar     error = call_user_func_check(ufunc, argcount, argvars,
2918a7d6542SBram Moolenaar 					     STACK_TV_BOT(-1), &funcexe, NULL);
2928a7d6542SBram Moolenaar 
2938a7d6542SBram Moolenaar     // Clear the arguments.
2948a7d6542SBram Moolenaar     for (idx = 0; idx < argcount; ++idx)
2958a7d6542SBram Moolenaar 	clear_tv(&argvars[idx]);
2968a7d6542SBram Moolenaar 
2978a7d6542SBram Moolenaar     if (error != FCERR_NONE)
2988a7d6542SBram Moolenaar     {
2998a7d6542SBram Moolenaar 	user_func_error(error, ufunc->uf_name);
3008a7d6542SBram Moolenaar 	return FAIL;
3018a7d6542SBram Moolenaar     }
3028a7d6542SBram Moolenaar     return OK;
3038a7d6542SBram Moolenaar }
3048a7d6542SBram Moolenaar 
3058a7d6542SBram Moolenaar /*
3068a7d6542SBram Moolenaar  * Execute a function by "name".
3078a7d6542SBram Moolenaar  * This can be a builtin function or a user function.
3088a7d6542SBram Moolenaar  * Returns FAIL if not found without an error message.
3098a7d6542SBram Moolenaar  */
3108a7d6542SBram Moolenaar     static int
3118a7d6542SBram Moolenaar call_by_name(char_u *name, int argcount, ectx_T *ectx)
3128a7d6542SBram Moolenaar {
3138a7d6542SBram Moolenaar     ufunc_T *ufunc;
3148a7d6542SBram Moolenaar 
3158a7d6542SBram Moolenaar     if (builtin_function(name, -1))
3168a7d6542SBram Moolenaar     {
3178a7d6542SBram Moolenaar 	int func_idx = find_internal_func(name);
3188a7d6542SBram Moolenaar 
3198a7d6542SBram Moolenaar 	if (func_idx < 0)
3208a7d6542SBram Moolenaar 	    return FAIL;
3218a7d6542SBram Moolenaar 	if (check_internal_func(func_idx, argcount) == FAIL)
3228a7d6542SBram Moolenaar 	    return FAIL;
3238a7d6542SBram Moolenaar 	return call_bfunc(func_idx, argcount, ectx);
3248a7d6542SBram Moolenaar     }
3258a7d6542SBram Moolenaar 
3268a7d6542SBram Moolenaar     ufunc = find_func(name, NULL);
3278a7d6542SBram Moolenaar     if (ufunc != NULL)
3288a7d6542SBram Moolenaar 	return call_ufunc(ufunc, argcount, ectx);
3298a7d6542SBram Moolenaar 
3308a7d6542SBram Moolenaar     return FAIL;
3318a7d6542SBram Moolenaar }
3328a7d6542SBram Moolenaar 
3338a7d6542SBram Moolenaar     static int
3348a7d6542SBram Moolenaar call_partial(typval_T *tv, int argcount, ectx_T *ectx)
3358a7d6542SBram Moolenaar {
3368a7d6542SBram Moolenaar     char_u	*name;
3378a7d6542SBram Moolenaar     int		called_emsg_before = called_emsg;
3388a7d6542SBram Moolenaar 
3398a7d6542SBram Moolenaar     if (tv->v_type == VAR_PARTIAL)
3408a7d6542SBram Moolenaar     {
3418a7d6542SBram Moolenaar 	partial_T *pt = tv->vval.v_partial;
3428a7d6542SBram Moolenaar 
3438a7d6542SBram Moolenaar 	if (pt->pt_func != NULL)
3448a7d6542SBram Moolenaar 	    return call_ufunc(pt->pt_func, argcount, ectx);
3458a7d6542SBram Moolenaar 	name = pt->pt_name;
3468a7d6542SBram Moolenaar     }
3478a7d6542SBram Moolenaar     else
3488a7d6542SBram Moolenaar 	name = tv->vval.v_string;
3498a7d6542SBram Moolenaar     if (call_by_name(name, argcount, ectx) == FAIL)
3508a7d6542SBram Moolenaar     {
3518a7d6542SBram Moolenaar 	if (called_emsg == called_emsg_before)
3528a7d6542SBram Moolenaar 	    semsg(_(e_unknownfunc), name);
3538a7d6542SBram Moolenaar 	return FAIL;
3548a7d6542SBram Moolenaar     }
3558a7d6542SBram Moolenaar     return OK;
3568a7d6542SBram Moolenaar }
3578a7d6542SBram Moolenaar 
3588a7d6542SBram Moolenaar /*
3590bbf722aSBram Moolenaar  * Store "tv" in variable "name".
3600bbf722aSBram Moolenaar  * This is for s: and g: variables.
3610bbf722aSBram Moolenaar  */
3620bbf722aSBram Moolenaar     static void
3630bbf722aSBram Moolenaar store_var(char_u *name, typval_T *tv)
3640bbf722aSBram Moolenaar {
3650bbf722aSBram Moolenaar     funccal_entry_T entry;
3660bbf722aSBram Moolenaar 
3670bbf722aSBram Moolenaar     save_funccal(&entry);
3680bbf722aSBram Moolenaar     set_var_const(name, NULL, tv, FALSE, 0);
3690bbf722aSBram Moolenaar     restore_funccal();
3700bbf722aSBram Moolenaar }
3710bbf722aSBram Moolenaar 
3720bbf722aSBram Moolenaar /*
3738a7d6542SBram Moolenaar  * Execute a function by "name".
3748a7d6542SBram Moolenaar  * This can be a builtin function, user function or a funcref.
3758a7d6542SBram Moolenaar  */
3768a7d6542SBram Moolenaar     static int
3778a7d6542SBram Moolenaar call_eval_func(char_u *name, int argcount, ectx_T *ectx)
3788a7d6542SBram Moolenaar {
3798a7d6542SBram Moolenaar     int		called_emsg_before = called_emsg;
3808a7d6542SBram Moolenaar 
3818a7d6542SBram Moolenaar     if (call_by_name(name, argcount, ectx) == FAIL
3828a7d6542SBram Moolenaar 					  && called_emsg == called_emsg_before)
3838a7d6542SBram Moolenaar     {
3848a7d6542SBram Moolenaar 	// "name" may be a variable that is a funcref or partial
3858a7d6542SBram Moolenaar 	//    if find variable
3868a7d6542SBram Moolenaar 	//      call_partial()
3878a7d6542SBram Moolenaar 	//    else
3888a7d6542SBram Moolenaar 	//      semsg(_(e_unknownfunc), name);
3898a7d6542SBram Moolenaar 	emsg("call_eval_func(partial) not implemented yet");
3908a7d6542SBram Moolenaar 	return FAIL;
3918a7d6542SBram Moolenaar     }
3928a7d6542SBram Moolenaar     return OK;
3938a7d6542SBram Moolenaar }
3948a7d6542SBram Moolenaar 
3958a7d6542SBram Moolenaar /*
3968a7d6542SBram Moolenaar  * Call a "def" function from old Vim script.
3978a7d6542SBram Moolenaar  * Return OK or FAIL.
3988a7d6542SBram Moolenaar  */
3998a7d6542SBram Moolenaar     int
4008a7d6542SBram Moolenaar call_def_function(
4018a7d6542SBram Moolenaar     ufunc_T	*ufunc,
4028a7d6542SBram Moolenaar     int		argc,		// nr of arguments
4038a7d6542SBram Moolenaar     typval_T	*argv,		// arguments
4048a7d6542SBram Moolenaar     typval_T	*rettv)		// return value
4058a7d6542SBram Moolenaar {
4068a7d6542SBram Moolenaar     ectx_T	ectx;		// execution context
4078a7d6542SBram Moolenaar     int		initial_frame_ptr;
4088a7d6542SBram Moolenaar     typval_T	*tv;
4098a7d6542SBram Moolenaar     int		idx;
4108a7d6542SBram Moolenaar     int		ret = FAIL;
4118a7d6542SBram Moolenaar     dfunc_T	*dfunc;
412170fcfcfSBram Moolenaar     int		defcount = ufunc->uf_args.ga_len - argc;
4138a7d6542SBram Moolenaar 
4148a7d6542SBram Moolenaar // Get pointer to item in the stack.
4158a7d6542SBram Moolenaar #define STACK_TV(idx) (((typval_T *)ectx.ec_stack.ga_data) + idx)
4168a7d6542SBram Moolenaar 
4178a7d6542SBram Moolenaar // Get pointer to item at the bottom of the stack, -1 is the bottom.
4188a7d6542SBram Moolenaar #undef STACK_TV_BOT
4198a7d6542SBram Moolenaar #define STACK_TV_BOT(idx) (((typval_T *)ectx.ec_stack.ga_data) + ectx.ec_stack.ga_len + idx)
4208a7d6542SBram Moolenaar 
4218a7d6542SBram Moolenaar // Get pointer to local variable on the stack.
4228a7d6542SBram Moolenaar #define STACK_TV_VAR(idx) (((typval_T *)ectx.ec_stack.ga_data) + ectx.ec_frame + STACK_FRAME_SIZE + idx)
4238a7d6542SBram Moolenaar 
4248a7d6542SBram Moolenaar     vim_memset(&ectx, 0, sizeof(ectx));
4258a7d6542SBram Moolenaar     ga_init2(&ectx.ec_stack, sizeof(typval_T), 500);
4268a7d6542SBram Moolenaar     if (ga_grow(&ectx.ec_stack, 20) == FAIL)
4278a7d6542SBram Moolenaar 	goto failed;
4288a7d6542SBram Moolenaar     ectx.ec_dfunc_idx = ufunc->uf_dfunc_idx;
4298a7d6542SBram Moolenaar 
4308a7d6542SBram Moolenaar     ga_init2(&ectx.ec_trystack, sizeof(trycmd_T), 10);
4318a7d6542SBram Moolenaar 
4328a7d6542SBram Moolenaar     // Put arguments on the stack.
4338a7d6542SBram Moolenaar     for (idx = 0; idx < argc; ++idx)
4348a7d6542SBram Moolenaar     {
4358a7d6542SBram Moolenaar 	copy_tv(&argv[idx], STACK_TV_BOT(0));
4368a7d6542SBram Moolenaar 	++ectx.ec_stack.ga_len;
4378a7d6542SBram Moolenaar     }
438170fcfcfSBram Moolenaar     // Make space for omitted arguments, will store default value below.
439170fcfcfSBram Moolenaar     if (defcount > 0)
440170fcfcfSBram Moolenaar 	for (idx = 0; idx < defcount; ++idx)
441170fcfcfSBram Moolenaar 	{
442170fcfcfSBram Moolenaar 	    STACK_TV_BOT(0)->v_type = VAR_UNKNOWN;
443170fcfcfSBram Moolenaar 	    ++ectx.ec_stack.ga_len;
444170fcfcfSBram Moolenaar 	}
4458a7d6542SBram Moolenaar 
4468a7d6542SBram Moolenaar     // Frame pointer points to just after arguments.
4478a7d6542SBram Moolenaar     ectx.ec_frame = ectx.ec_stack.ga_len;
4488a7d6542SBram Moolenaar     initial_frame_ptr = ectx.ec_frame;
4498a7d6542SBram Moolenaar 
4508a7d6542SBram Moolenaar     // dummy frame entries
4518a7d6542SBram Moolenaar     for (idx = 0; idx < STACK_FRAME_SIZE; ++idx)
4528a7d6542SBram Moolenaar     {
4538a7d6542SBram Moolenaar 	STACK_TV(ectx.ec_stack.ga_len)->v_type = VAR_UNKNOWN;
4548a7d6542SBram Moolenaar 	++ectx.ec_stack.ga_len;
4558a7d6542SBram Moolenaar     }
4568a7d6542SBram Moolenaar 
4578a7d6542SBram Moolenaar     // Reserve space for local variables.
4588a7d6542SBram Moolenaar     dfunc = ((dfunc_T *)def_functions.ga_data) + ufunc->uf_dfunc_idx;
4598a7d6542SBram Moolenaar     for (idx = 0; idx < dfunc->df_varcount; ++idx)
4608a7d6542SBram Moolenaar 	STACK_TV_VAR(idx)->v_type = VAR_UNKNOWN;
4618a7d6542SBram Moolenaar     ectx.ec_stack.ga_len += dfunc->df_varcount;
4628a7d6542SBram Moolenaar 
4638a7d6542SBram Moolenaar     ectx.ec_instr = dfunc->df_instr;
464170fcfcfSBram Moolenaar 
465170fcfcfSBram Moolenaar     // Decide where to start execution, handles optional arguments.
466170fcfcfSBram Moolenaar     init_instr_idx(ufunc, argc, &ectx);
467170fcfcfSBram Moolenaar 
4688a7d6542SBram Moolenaar     for (;;)
4698a7d6542SBram Moolenaar     {
4708a7d6542SBram Moolenaar 	isn_T	    *iptr;
4718a7d6542SBram Moolenaar 	trycmd_T    *trycmd = NULL;
4728a7d6542SBram Moolenaar 
4738a7d6542SBram Moolenaar 	if (did_throw && !ectx.ec_in_catch)
4748a7d6542SBram Moolenaar 	{
4758a7d6542SBram Moolenaar 	    garray_T	*trystack = &ectx.ec_trystack;
4768a7d6542SBram Moolenaar 
4778a7d6542SBram Moolenaar 	    // An exception jumps to the first catch, finally, or returns from
4788a7d6542SBram Moolenaar 	    // the current function.
4798a7d6542SBram Moolenaar 	    if (trystack->ga_len > 0)
4808a7d6542SBram Moolenaar 		trycmd = ((trycmd_T *)trystack->ga_data) + trystack->ga_len - 1;
4818a7d6542SBram Moolenaar 	    if (trycmd != NULL && trycmd->tcd_frame == ectx.ec_frame)
4828a7d6542SBram Moolenaar 	    {
4838a7d6542SBram Moolenaar 		// jump to ":catch" or ":finally"
4848a7d6542SBram Moolenaar 		ectx.ec_in_catch = TRUE;
4858a7d6542SBram Moolenaar 		ectx.ec_iidx = trycmd->tcd_catch_idx;
4868a7d6542SBram Moolenaar 	    }
4878a7d6542SBram Moolenaar 	    else
4888a7d6542SBram Moolenaar 	    {
4898a7d6542SBram Moolenaar 		// not inside try or need to return from current functions.
4908a7d6542SBram Moolenaar 		if (ectx.ec_frame == initial_frame_ptr)
4918a7d6542SBram Moolenaar 		{
4928a7d6542SBram Moolenaar 		    // At the toplevel we are done.  Push a dummy return value.
4938a7d6542SBram Moolenaar 		    if (ga_grow(&ectx.ec_stack, 1) == FAIL)
4948a7d6542SBram Moolenaar 			goto failed;
4958a7d6542SBram Moolenaar 		    tv = STACK_TV_BOT(0);
4968a7d6542SBram Moolenaar 		    tv->v_type = VAR_NUMBER;
4978a7d6542SBram Moolenaar 		    tv->vval.v_number = 0;
4988a7d6542SBram Moolenaar 		    ++ectx.ec_stack.ga_len;
499257cc5eeSBram Moolenaar 		    need_rethrow = TRUE;
5008a7d6542SBram Moolenaar 		    goto done;
5018a7d6542SBram Moolenaar 		}
5028a7d6542SBram Moolenaar 
5038a7d6542SBram Moolenaar 		func_return(&ectx);
5048a7d6542SBram Moolenaar 	    }
5058a7d6542SBram Moolenaar 	    continue;
5068a7d6542SBram Moolenaar 	}
5078a7d6542SBram Moolenaar 
5088a7d6542SBram Moolenaar 	iptr = &ectx.ec_instr[ectx.ec_iidx++];
5098a7d6542SBram Moolenaar 	switch (iptr->isn_type)
5108a7d6542SBram Moolenaar 	{
5118a7d6542SBram Moolenaar 	    // execute Ex command line
5128a7d6542SBram Moolenaar 	    case ISN_EXEC:
5138a7d6542SBram Moolenaar 		do_cmdline_cmd(iptr->isn_arg.string);
5148a7d6542SBram Moolenaar 		break;
5158a7d6542SBram Moolenaar 
5168a7d6542SBram Moolenaar 	    // execute :echo {string} ...
5178a7d6542SBram Moolenaar 	    case ISN_ECHO:
5188a7d6542SBram Moolenaar 		{
5198a7d6542SBram Moolenaar 		    int count = iptr->isn_arg.echo.echo_count;
5208a7d6542SBram Moolenaar 		    int	atstart = TRUE;
5218a7d6542SBram Moolenaar 		    int needclr = TRUE;
5228a7d6542SBram Moolenaar 
5238a7d6542SBram Moolenaar 		    for (idx = 0; idx < count; ++idx)
5248a7d6542SBram Moolenaar 		    {
5258a7d6542SBram Moolenaar 			tv = STACK_TV_BOT(idx - count);
5268a7d6542SBram Moolenaar 			echo_one(tv, iptr->isn_arg.echo.echo_with_white,
5278a7d6542SBram Moolenaar 							   &atstart, &needclr);
5288a7d6542SBram Moolenaar 			clear_tv(tv);
5298a7d6542SBram Moolenaar 		    }
530*e0807ea4SBram Moolenaar 		    if (needclr)
531*e0807ea4SBram Moolenaar 			msg_clr_eos();
5328a7d6542SBram Moolenaar 		    ectx.ec_stack.ga_len -= count;
5338a7d6542SBram Moolenaar 		}
5348a7d6542SBram Moolenaar 		break;
5358a7d6542SBram Moolenaar 
5368a7d6542SBram Moolenaar 	    // load local variable or argument
5378a7d6542SBram Moolenaar 	    case ISN_LOAD:
5388a7d6542SBram Moolenaar 		if (ga_grow(&ectx.ec_stack, 1) == FAIL)
5398a7d6542SBram Moolenaar 		    goto failed;
5408a7d6542SBram Moolenaar 		copy_tv(STACK_TV_VAR(iptr->isn_arg.number), STACK_TV_BOT(0));
5418a7d6542SBram Moolenaar 		++ectx.ec_stack.ga_len;
5428a7d6542SBram Moolenaar 		break;
5438a7d6542SBram Moolenaar 
5448a7d6542SBram Moolenaar 	    // load v: variable
5458a7d6542SBram Moolenaar 	    case ISN_LOADV:
5468a7d6542SBram Moolenaar 		if (ga_grow(&ectx.ec_stack, 1) == FAIL)
5478a7d6542SBram Moolenaar 		    goto failed;
5488a7d6542SBram Moolenaar 		copy_tv(get_vim_var_tv(iptr->isn_arg.number), STACK_TV_BOT(0));
5498a7d6542SBram Moolenaar 		++ectx.ec_stack.ga_len;
5508a7d6542SBram Moolenaar 		break;
5518a7d6542SBram Moolenaar 
552b283a8a6SBram Moolenaar 	    // load s: variable in Vim9 script
5538a7d6542SBram Moolenaar 	    case ISN_LOADSCRIPT:
5548a7d6542SBram Moolenaar 		{
5558a7d6542SBram Moolenaar 		    scriptitem_T *si =
55621b9e977SBram Moolenaar 				  SCRIPT_ITEM(iptr->isn_arg.script.script_sid);
5578a7d6542SBram Moolenaar 		    svar_T	 *sv;
5588a7d6542SBram Moolenaar 
5598a7d6542SBram Moolenaar 		    sv = ((svar_T *)si->sn_var_vals.ga_data)
5608a7d6542SBram Moolenaar 					     + iptr->isn_arg.script.script_idx;
5618a7d6542SBram Moolenaar 		    if (ga_grow(&ectx.ec_stack, 1) == FAIL)
5628a7d6542SBram Moolenaar 			goto failed;
5638a7d6542SBram Moolenaar 		    copy_tv(sv->sv_tv, STACK_TV_BOT(0));
5648a7d6542SBram Moolenaar 		    ++ectx.ec_stack.ga_len;
5658a7d6542SBram Moolenaar 		}
5668a7d6542SBram Moolenaar 		break;
5678a7d6542SBram Moolenaar 
5688a7d6542SBram Moolenaar 	    // load s: variable in old script
5698a7d6542SBram Moolenaar 	    case ISN_LOADS:
5708a7d6542SBram Moolenaar 		{
571b283a8a6SBram Moolenaar 		    hashtab_T	*ht = &SCRIPT_VARS(
572b283a8a6SBram Moolenaar 					       iptr->isn_arg.loadstore.ls_sid);
573b283a8a6SBram Moolenaar 		    char_u	*name = iptr->isn_arg.loadstore.ls_name;
5748a7d6542SBram Moolenaar 		    dictitem_T	*di = find_var_in_ht(ht, 0, name, TRUE);
5750bbf722aSBram Moolenaar 
5768a7d6542SBram Moolenaar 		    if (di == NULL)
5778a7d6542SBram Moolenaar 		    {
578b283a8a6SBram Moolenaar 			semsg(_(e_undefvar), name);
5798a7d6542SBram Moolenaar 			goto failed;
5808a7d6542SBram Moolenaar 		    }
5818a7d6542SBram Moolenaar 		    else
5828a7d6542SBram Moolenaar 		    {
5838a7d6542SBram Moolenaar 			if (ga_grow(&ectx.ec_stack, 1) == FAIL)
5848a7d6542SBram Moolenaar 			    goto failed;
5858a7d6542SBram Moolenaar 			copy_tv(&di->di_tv, STACK_TV_BOT(0));
5868a7d6542SBram Moolenaar 			++ectx.ec_stack.ga_len;
5878a7d6542SBram Moolenaar 		    }
5888a7d6542SBram Moolenaar 		}
5898a7d6542SBram Moolenaar 		break;
5908a7d6542SBram Moolenaar 
5918a7d6542SBram Moolenaar 	    // load g: variable
5928a7d6542SBram Moolenaar 	    case ISN_LOADG:
5938a7d6542SBram Moolenaar 		{
5940bbf722aSBram Moolenaar 		    dictitem_T *di = find_var_in_ht(get_globvar_ht(), 0,
5958a7d6542SBram Moolenaar 						   iptr->isn_arg.string, TRUE);
5960bbf722aSBram Moolenaar 
5978a7d6542SBram Moolenaar 		    if (di == NULL)
5988a7d6542SBram Moolenaar 		    {
5998a7d6542SBram Moolenaar 			semsg(_("E121: Undefined variable: g:%s"),
6008a7d6542SBram Moolenaar 							 iptr->isn_arg.string);
6018a7d6542SBram Moolenaar 			goto failed;
6028a7d6542SBram Moolenaar 		    }
6038a7d6542SBram Moolenaar 		    else
6048a7d6542SBram Moolenaar 		    {
6058a7d6542SBram Moolenaar 			if (ga_grow(&ectx.ec_stack, 1) == FAIL)
6068a7d6542SBram Moolenaar 			    goto failed;
6078a7d6542SBram Moolenaar 			copy_tv(&di->di_tv, STACK_TV_BOT(0));
6088a7d6542SBram Moolenaar 			++ectx.ec_stack.ga_len;
6098a7d6542SBram Moolenaar 		    }
6108a7d6542SBram Moolenaar 		}
6118a7d6542SBram Moolenaar 		break;
6128a7d6542SBram Moolenaar 
6138a7d6542SBram Moolenaar 	    // load &option
6148a7d6542SBram Moolenaar 	    case ISN_LOADOPT:
6158a7d6542SBram Moolenaar 		{
6168a7d6542SBram Moolenaar 		    typval_T	optval;
6178a7d6542SBram Moolenaar 		    char_u	*name = iptr->isn_arg.string;
6188a7d6542SBram Moolenaar 
6198a7d6542SBram Moolenaar 		    if (ga_grow(&ectx.ec_stack, 1) == FAIL)
6208a7d6542SBram Moolenaar 			goto failed;
62158ceca5cSBram Moolenaar 		    if (get_option_tv(&name, &optval, TRUE) == FAIL)
62258ceca5cSBram Moolenaar 			goto failed;
6238a7d6542SBram Moolenaar 		    *STACK_TV_BOT(0) = optval;
6248a7d6542SBram Moolenaar 		    ++ectx.ec_stack.ga_len;
6258a7d6542SBram Moolenaar 		}
6268a7d6542SBram Moolenaar 		break;
6278a7d6542SBram Moolenaar 
6288a7d6542SBram Moolenaar 	    // load $ENV
6298a7d6542SBram Moolenaar 	    case ISN_LOADENV:
6308a7d6542SBram Moolenaar 		{
6318a7d6542SBram Moolenaar 		    typval_T	optval;
6328a7d6542SBram Moolenaar 		    char_u	*name = iptr->isn_arg.string;
6338a7d6542SBram Moolenaar 
6348a7d6542SBram Moolenaar 		    if (ga_grow(&ectx.ec_stack, 1) == FAIL)
6358a7d6542SBram Moolenaar 			goto failed;
6360bbf722aSBram Moolenaar 		    // name is always valid, checked when compiling
6370bbf722aSBram Moolenaar 		    (void)get_env_tv(&name, &optval, TRUE);
6388a7d6542SBram Moolenaar 		    *STACK_TV_BOT(0) = optval;
6398a7d6542SBram Moolenaar 		    ++ectx.ec_stack.ga_len;
6408a7d6542SBram Moolenaar 		}
6418a7d6542SBram Moolenaar 		break;
6428a7d6542SBram Moolenaar 
6438a7d6542SBram Moolenaar 	    // load @register
6448a7d6542SBram Moolenaar 	    case ISN_LOADREG:
6458a7d6542SBram Moolenaar 		if (ga_grow(&ectx.ec_stack, 1) == FAIL)
6468a7d6542SBram Moolenaar 		    goto failed;
6478a7d6542SBram Moolenaar 		tv = STACK_TV_BOT(0);
6488a7d6542SBram Moolenaar 		tv->v_type = VAR_STRING;
6498a7d6542SBram Moolenaar 		tv->vval.v_string = get_reg_contents(
6508a7d6542SBram Moolenaar 					  iptr->isn_arg.number, GREG_EXPR_SRC);
6518a7d6542SBram Moolenaar 		++ectx.ec_stack.ga_len;
6528a7d6542SBram Moolenaar 		break;
6538a7d6542SBram Moolenaar 
6548a7d6542SBram Moolenaar 	    // store local variable
6558a7d6542SBram Moolenaar 	    case ISN_STORE:
6568a7d6542SBram Moolenaar 		--ectx.ec_stack.ga_len;
6578a7d6542SBram Moolenaar 		tv = STACK_TV_VAR(iptr->isn_arg.number);
6588a7d6542SBram Moolenaar 		clear_tv(tv);
6598a7d6542SBram Moolenaar 		*tv = *STACK_TV_BOT(0);
6608a7d6542SBram Moolenaar 		break;
6618a7d6542SBram Moolenaar 
662b283a8a6SBram Moolenaar 	    // store s: variable in old script
663b283a8a6SBram Moolenaar 	    case ISN_STORES:
664b283a8a6SBram Moolenaar 		{
665b283a8a6SBram Moolenaar 		    hashtab_T	*ht = &SCRIPT_VARS(
666b283a8a6SBram Moolenaar 					       iptr->isn_arg.loadstore.ls_sid);
667b283a8a6SBram Moolenaar 		    char_u	*name = iptr->isn_arg.loadstore.ls_name;
6680bbf722aSBram Moolenaar 		    dictitem_T	*di = find_var_in_ht(ht, 0, name + 2, TRUE);
669b283a8a6SBram Moolenaar 
670b283a8a6SBram Moolenaar 		    --ectx.ec_stack.ga_len;
6710bbf722aSBram Moolenaar 		    if (di == NULL)
6720bbf722aSBram Moolenaar 			store_var(iptr->isn_arg.string, STACK_TV_BOT(0));
6730bbf722aSBram Moolenaar 		    else
6740bbf722aSBram Moolenaar 		    {
675b283a8a6SBram Moolenaar 			clear_tv(&di->di_tv);
676b283a8a6SBram Moolenaar 			di->di_tv = *STACK_TV_BOT(0);
677b283a8a6SBram Moolenaar 		    }
6780bbf722aSBram Moolenaar 		}
679b283a8a6SBram Moolenaar 		break;
680b283a8a6SBram Moolenaar 
681b283a8a6SBram Moolenaar 	    // store script-local variable in Vim9 script
6828a7d6542SBram Moolenaar 	    case ISN_STORESCRIPT:
6838a7d6542SBram Moolenaar 		{
68421b9e977SBram Moolenaar 		    scriptitem_T *si = SCRIPT_ITEM(
6858a7d6542SBram Moolenaar 					      iptr->isn_arg.script.script_sid);
6868a7d6542SBram Moolenaar 		    svar_T	 *sv = ((svar_T *)si->sn_var_vals.ga_data)
6878a7d6542SBram Moolenaar 					     + iptr->isn_arg.script.script_idx;
6888a7d6542SBram Moolenaar 
6898a7d6542SBram Moolenaar 		    --ectx.ec_stack.ga_len;
6908a7d6542SBram Moolenaar 		    clear_tv(sv->sv_tv);
6918a7d6542SBram Moolenaar 		    *sv->sv_tv = *STACK_TV_BOT(0);
6928a7d6542SBram Moolenaar 		}
6938a7d6542SBram Moolenaar 		break;
6948a7d6542SBram Moolenaar 
6958a7d6542SBram Moolenaar 	    // store option
6968a7d6542SBram Moolenaar 	    case ISN_STOREOPT:
6978a7d6542SBram Moolenaar 		{
6988a7d6542SBram Moolenaar 		    long	n = 0;
6998a7d6542SBram Moolenaar 		    char_u	*s = NULL;
7008a7d6542SBram Moolenaar 		    char	*msg;
7018a7d6542SBram Moolenaar 
7028a7d6542SBram Moolenaar 		    --ectx.ec_stack.ga_len;
7038a7d6542SBram Moolenaar 		    tv = STACK_TV_BOT(0);
7048a7d6542SBram Moolenaar 		    if (tv->v_type == VAR_STRING)
70597a2af39SBram Moolenaar 		    {
7068a7d6542SBram Moolenaar 			s = tv->vval.v_string;
70797a2af39SBram Moolenaar 			if (s == NULL)
70897a2af39SBram Moolenaar 			    s = (char_u *)"";
70997a2af39SBram Moolenaar 		    }
7108a7d6542SBram Moolenaar 		    else if (tv->v_type == VAR_NUMBER)
7118a7d6542SBram Moolenaar 			n = tv->vval.v_number;
7128a7d6542SBram Moolenaar 		    else
7138a7d6542SBram Moolenaar 		    {
7148a7d6542SBram Moolenaar 			emsg(_("E1051: Expected string or number"));
7158a7d6542SBram Moolenaar 			goto failed;
7168a7d6542SBram Moolenaar 		    }
7178a7d6542SBram Moolenaar 		    msg = set_option_value(iptr->isn_arg.storeopt.so_name,
7188a7d6542SBram Moolenaar 					n, s, iptr->isn_arg.storeopt.so_flags);
7198a7d6542SBram Moolenaar 		    if (msg != NULL)
7208a7d6542SBram Moolenaar 		    {
7218a7d6542SBram Moolenaar 			emsg(_(msg));
7228a7d6542SBram Moolenaar 			goto failed;
7238a7d6542SBram Moolenaar 		    }
7248a7d6542SBram Moolenaar 		    clear_tv(tv);
7258a7d6542SBram Moolenaar 		}
7268a7d6542SBram Moolenaar 		break;
7278a7d6542SBram Moolenaar 
728b283a8a6SBram Moolenaar 	    // store $ENV
729b283a8a6SBram Moolenaar 	    case ISN_STOREENV:
730b283a8a6SBram Moolenaar 		--ectx.ec_stack.ga_len;
731b283a8a6SBram Moolenaar 		vim_setenv_ext(iptr->isn_arg.string,
732b283a8a6SBram Moolenaar 					       tv_get_string(STACK_TV_BOT(0)));
733b283a8a6SBram Moolenaar 		break;
734b283a8a6SBram Moolenaar 
735b283a8a6SBram Moolenaar 	    // store @r
736b283a8a6SBram Moolenaar 	    case ISN_STOREREG:
737b283a8a6SBram Moolenaar 		{
738b283a8a6SBram Moolenaar 		    int	reg = iptr->isn_arg.number;
739b283a8a6SBram Moolenaar 
740b283a8a6SBram Moolenaar 		    --ectx.ec_stack.ga_len;
741401d9ffbSBram Moolenaar 		    tv = STACK_TV_BOT(0);
742b283a8a6SBram Moolenaar 		    write_reg_contents(reg == '@' ? '"' : reg,
743401d9ffbSBram Moolenaar 						 tv_get_string(tv), -1, FALSE);
744401d9ffbSBram Moolenaar 		    clear_tv(tv);
745b283a8a6SBram Moolenaar 		}
746b283a8a6SBram Moolenaar 		break;
747b283a8a6SBram Moolenaar 
748b283a8a6SBram Moolenaar 	    // store v: variable
749b283a8a6SBram Moolenaar 	    case ISN_STOREV:
750b283a8a6SBram Moolenaar 		--ectx.ec_stack.ga_len;
751b283a8a6SBram Moolenaar 		if (set_vim_var_tv(iptr->isn_arg.number, STACK_TV_BOT(0))
752b283a8a6SBram Moolenaar 								       == FAIL)
753b283a8a6SBram Moolenaar 		    goto failed;
754b283a8a6SBram Moolenaar 		break;
755b283a8a6SBram Moolenaar 
7568a7d6542SBram Moolenaar 	    // store g: variable
7578a7d6542SBram Moolenaar 	    case ISN_STOREG:
7588a7d6542SBram Moolenaar 		{
7598a7d6542SBram Moolenaar 		    dictitem_T *di;
7608a7d6542SBram Moolenaar 
7618a7d6542SBram Moolenaar 		    --ectx.ec_stack.ga_len;
7628a7d6542SBram Moolenaar 		    di = find_var_in_ht(get_globvar_ht(), 0,
763401d9ffbSBram Moolenaar 					       iptr->isn_arg.string + 2, TRUE);
7648a7d6542SBram Moolenaar 		    if (di == NULL)
7650bbf722aSBram Moolenaar 			store_var(iptr->isn_arg.string, STACK_TV_BOT(0));
7668a7d6542SBram Moolenaar 		    else
7678a7d6542SBram Moolenaar 		    {
7688a7d6542SBram Moolenaar 			clear_tv(&di->di_tv);
7698a7d6542SBram Moolenaar 			di->di_tv = *STACK_TV_BOT(0);
7708a7d6542SBram Moolenaar 		    }
7718a7d6542SBram Moolenaar 		}
7728a7d6542SBram Moolenaar 		break;
7738a7d6542SBram Moolenaar 
7748a7d6542SBram Moolenaar 	    // store number in local variable
7758a7d6542SBram Moolenaar 	    case ISN_STORENR:
7768a7d6542SBram Moolenaar 		tv = STACK_TV_VAR(iptr->isn_arg.storenr.str_idx);
7778a7d6542SBram Moolenaar 		clear_tv(tv);
7788a7d6542SBram Moolenaar 		tv->v_type = VAR_NUMBER;
7798a7d6542SBram Moolenaar 		tv->vval.v_number = iptr->isn_arg.storenr.str_val;
7808a7d6542SBram Moolenaar 		break;
7818a7d6542SBram Moolenaar 
7828a7d6542SBram Moolenaar 	    // push constant
7838a7d6542SBram Moolenaar 	    case ISN_PUSHNR:
7848a7d6542SBram Moolenaar 	    case ISN_PUSHBOOL:
7858a7d6542SBram Moolenaar 	    case ISN_PUSHSPEC:
7868a7d6542SBram Moolenaar 	    case ISN_PUSHF:
7878a7d6542SBram Moolenaar 	    case ISN_PUSHS:
7888a7d6542SBram Moolenaar 	    case ISN_PUSHBLOB:
7898a7d6542SBram Moolenaar 		if (ga_grow(&ectx.ec_stack, 1) == FAIL)
7908a7d6542SBram Moolenaar 		    goto failed;
7918a7d6542SBram Moolenaar 		tv = STACK_TV_BOT(0);
7928a7d6542SBram Moolenaar 		++ectx.ec_stack.ga_len;
7938a7d6542SBram Moolenaar 		switch (iptr->isn_type)
7948a7d6542SBram Moolenaar 		{
7958a7d6542SBram Moolenaar 		    case ISN_PUSHNR:
7968a7d6542SBram Moolenaar 			tv->v_type = VAR_NUMBER;
7978a7d6542SBram Moolenaar 			tv->vval.v_number = iptr->isn_arg.number;
7988a7d6542SBram Moolenaar 			break;
7998a7d6542SBram Moolenaar 		    case ISN_PUSHBOOL:
8008a7d6542SBram Moolenaar 			tv->v_type = VAR_BOOL;
8018a7d6542SBram Moolenaar 			tv->vval.v_number = iptr->isn_arg.number;
8028a7d6542SBram Moolenaar 			break;
8038a7d6542SBram Moolenaar 		    case ISN_PUSHSPEC:
8048a7d6542SBram Moolenaar 			tv->v_type = VAR_SPECIAL;
8058a7d6542SBram Moolenaar 			tv->vval.v_number = iptr->isn_arg.number;
8068a7d6542SBram Moolenaar 			break;
8078a7d6542SBram Moolenaar #ifdef FEAT_FLOAT
8088a7d6542SBram Moolenaar 		    case ISN_PUSHF:
8098a7d6542SBram Moolenaar 			tv->v_type = VAR_FLOAT;
8108a7d6542SBram Moolenaar 			tv->vval.v_float = iptr->isn_arg.fnumber;
8118a7d6542SBram Moolenaar 			break;
8128a7d6542SBram Moolenaar #endif
8138a7d6542SBram Moolenaar 		    case ISN_PUSHBLOB:
8148a7d6542SBram Moolenaar 			blob_copy(iptr->isn_arg.blob, tv);
8158a7d6542SBram Moolenaar 			break;
8168a7d6542SBram Moolenaar 		    default:
8178a7d6542SBram Moolenaar 			tv->v_type = VAR_STRING;
8188a7d6542SBram Moolenaar 			tv->vval.v_string = vim_strsave(iptr->isn_arg.string);
8198a7d6542SBram Moolenaar 		}
8208a7d6542SBram Moolenaar 		break;
8218a7d6542SBram Moolenaar 
8228a7d6542SBram Moolenaar 	    // create a list from items on the stack; uses a single allocation
8238a7d6542SBram Moolenaar 	    // for the list header and the items
8248a7d6542SBram Moolenaar 	    case ISN_NEWLIST:
8258a7d6542SBram Moolenaar 		{
8268a7d6542SBram Moolenaar 		    int	    count = iptr->isn_arg.number;
8278a7d6542SBram Moolenaar 		    list_T  *list = list_alloc_with_items(count);
8288a7d6542SBram Moolenaar 
8298a7d6542SBram Moolenaar 		    if (list == NULL)
8308a7d6542SBram Moolenaar 			goto failed;
8318a7d6542SBram Moolenaar 		    for (idx = 0; idx < count; ++idx)
8328a7d6542SBram Moolenaar 			list_set_item(list, idx, STACK_TV_BOT(idx - count));
8338a7d6542SBram Moolenaar 
8348a7d6542SBram Moolenaar 		    if (count > 0)
8358a7d6542SBram Moolenaar 			ectx.ec_stack.ga_len -= count - 1;
8368a7d6542SBram Moolenaar 		    else if (ga_grow(&ectx.ec_stack, 1) == FAIL)
8378a7d6542SBram Moolenaar 			goto failed;
8388a7d6542SBram Moolenaar 		    else
8398a7d6542SBram Moolenaar 			++ectx.ec_stack.ga_len;
8408a7d6542SBram Moolenaar 		    tv = STACK_TV_BOT(-1);
8418a7d6542SBram Moolenaar 		    tv->v_type = VAR_LIST;
8428a7d6542SBram Moolenaar 		    tv->vval.v_list = list;
8438a7d6542SBram Moolenaar 		    ++list->lv_refcount;
8448a7d6542SBram Moolenaar 		}
8458a7d6542SBram Moolenaar 		break;
8468a7d6542SBram Moolenaar 
8478a7d6542SBram Moolenaar 	    // create a dict from items on the stack
8488a7d6542SBram Moolenaar 	    case ISN_NEWDICT:
8498a7d6542SBram Moolenaar 		{
8508a7d6542SBram Moolenaar 		    int	    count = iptr->isn_arg.number;
8518a7d6542SBram Moolenaar 		    dict_T  *dict = dict_alloc();
8528a7d6542SBram Moolenaar 		    dictitem_T *item;
8538a7d6542SBram Moolenaar 
8548a7d6542SBram Moolenaar 		    if (dict == NULL)
8558a7d6542SBram Moolenaar 			goto failed;
8568a7d6542SBram Moolenaar 		    for (idx = 0; idx < count; ++idx)
8578a7d6542SBram Moolenaar 		    {
8588a7d6542SBram Moolenaar 			// check key type is VAR_STRING
8598a7d6542SBram Moolenaar 			tv = STACK_TV_BOT(2 * (idx - count));
8608a7d6542SBram Moolenaar 			item = dictitem_alloc(tv->vval.v_string);
8618a7d6542SBram Moolenaar 			clear_tv(tv);
8628a7d6542SBram Moolenaar 			if (item == NULL)
8638a7d6542SBram Moolenaar 			    goto failed;
8648a7d6542SBram Moolenaar 			item->di_tv = *STACK_TV_BOT(2 * (idx - count) + 1);
8658a7d6542SBram Moolenaar 			item->di_tv.v_lock = 0;
8668a7d6542SBram Moolenaar 			if (dict_add(dict, item) == FAIL)
8678a7d6542SBram Moolenaar 			    goto failed;
8688a7d6542SBram Moolenaar 		    }
8698a7d6542SBram Moolenaar 
8708a7d6542SBram Moolenaar 		    if (count > 0)
8718a7d6542SBram Moolenaar 			ectx.ec_stack.ga_len -= 2 * count - 1;
8728a7d6542SBram Moolenaar 		    else if (ga_grow(&ectx.ec_stack, 1) == FAIL)
8738a7d6542SBram Moolenaar 			goto failed;
8748a7d6542SBram Moolenaar 		    else
8758a7d6542SBram Moolenaar 			++ectx.ec_stack.ga_len;
8768a7d6542SBram Moolenaar 		    tv = STACK_TV_BOT(-1);
8778a7d6542SBram Moolenaar 		    tv->v_type = VAR_DICT;
8788a7d6542SBram Moolenaar 		    tv->vval.v_dict = dict;
8798a7d6542SBram Moolenaar 		    ++dict->dv_refcount;
8808a7d6542SBram Moolenaar 		}
8818a7d6542SBram Moolenaar 		break;
8828a7d6542SBram Moolenaar 
8838a7d6542SBram Moolenaar 	    // call a :def function
8848a7d6542SBram Moolenaar 	    case ISN_DCALL:
8858a7d6542SBram Moolenaar 		if (call_dfunc(iptr->isn_arg.dfunc.cdf_idx,
8868a7d6542SBram Moolenaar 			      iptr->isn_arg.dfunc.cdf_argcount,
8878a7d6542SBram Moolenaar 			      &ectx) == FAIL)
8888a7d6542SBram Moolenaar 		    goto failed;
8898a7d6542SBram Moolenaar 		break;
8908a7d6542SBram Moolenaar 
8918a7d6542SBram Moolenaar 	    // call a builtin function
8928a7d6542SBram Moolenaar 	    case ISN_BCALL:
8938a7d6542SBram Moolenaar 		SOURCING_LNUM = iptr->isn_lnum;
8948a7d6542SBram Moolenaar 		if (call_bfunc(iptr->isn_arg.bfunc.cbf_idx,
8958a7d6542SBram Moolenaar 			      iptr->isn_arg.bfunc.cbf_argcount,
8968a7d6542SBram Moolenaar 			      &ectx) == FAIL)
8978a7d6542SBram Moolenaar 		    goto failed;
8988a7d6542SBram Moolenaar 		break;
8998a7d6542SBram Moolenaar 
9008a7d6542SBram Moolenaar 	    // call a funcref or partial
9018a7d6542SBram Moolenaar 	    case ISN_PCALL:
9028a7d6542SBram Moolenaar 		{
9038a7d6542SBram Moolenaar 		    cpfunc_T	*pfunc = &iptr->isn_arg.pfunc;
9048a7d6542SBram Moolenaar 		    int		r;
9058a7d6542SBram Moolenaar 		    typval_T	partial;
9068a7d6542SBram Moolenaar 
9078a7d6542SBram Moolenaar 		    SOURCING_LNUM = iptr->isn_lnum;
9088a7d6542SBram Moolenaar 		    if (pfunc->cpf_top)
9098a7d6542SBram Moolenaar 		    {
9108a7d6542SBram Moolenaar 			// funcref is above the arguments
9118a7d6542SBram Moolenaar 			tv = STACK_TV_BOT(-pfunc->cpf_argcount - 1);
9128a7d6542SBram Moolenaar 		    }
9138a7d6542SBram Moolenaar 		    else
9148a7d6542SBram Moolenaar 		    {
9158a7d6542SBram Moolenaar 			// Get the funcref from the stack.
9168a7d6542SBram Moolenaar 			--ectx.ec_stack.ga_len;
9178a7d6542SBram Moolenaar 			partial = *STACK_TV_BOT(0);
9188a7d6542SBram Moolenaar 			tv = &partial;
9198a7d6542SBram Moolenaar 		    }
9208a7d6542SBram Moolenaar 		    r = call_partial(tv, pfunc->cpf_argcount, &ectx);
9218a7d6542SBram Moolenaar 		    if (tv == &partial)
9228a7d6542SBram Moolenaar 			clear_tv(&partial);
9238a7d6542SBram Moolenaar 		    if (r == FAIL)
9248a7d6542SBram Moolenaar 			goto failed;
9258a7d6542SBram Moolenaar 
9268a7d6542SBram Moolenaar 		    if (pfunc->cpf_top)
9278a7d6542SBram Moolenaar 		    {
9288a7d6542SBram Moolenaar 			// Get the funcref from the stack, overwrite with the
9298a7d6542SBram Moolenaar 			// return value.
9308a7d6542SBram Moolenaar 			clear_tv(tv);
9318a7d6542SBram Moolenaar 			--ectx.ec_stack.ga_len;
9328a7d6542SBram Moolenaar 			*STACK_TV_BOT(-1) = *STACK_TV_BOT(0);
9338a7d6542SBram Moolenaar 		    }
9348a7d6542SBram Moolenaar 		}
9358a7d6542SBram Moolenaar 		break;
9368a7d6542SBram Moolenaar 
9378a7d6542SBram Moolenaar 	    // call a user defined function or funcref/partial
9388a7d6542SBram Moolenaar 	    case ISN_UCALL:
9398a7d6542SBram Moolenaar 		{
9408a7d6542SBram Moolenaar 		    cufunc_T	*cufunc = &iptr->isn_arg.ufunc;
9418a7d6542SBram Moolenaar 
9428a7d6542SBram Moolenaar 		    SOURCING_LNUM = iptr->isn_lnum;
9438a7d6542SBram Moolenaar 		    if (call_eval_func(cufunc->cuf_name,
9448a7d6542SBram Moolenaar 					  cufunc->cuf_argcount, &ectx) == FAIL)
9458a7d6542SBram Moolenaar 			goto failed;
9468a7d6542SBram Moolenaar 		}
9478a7d6542SBram Moolenaar 		break;
9488a7d6542SBram Moolenaar 
9498a7d6542SBram Moolenaar 	    // return from a :def function call
9508a7d6542SBram Moolenaar 	    case ISN_RETURN:
9518a7d6542SBram Moolenaar 		{
9528cbd6dfcSBram Moolenaar 		    garray_T	*trystack = &ectx.ec_trystack;
9538cbd6dfcSBram Moolenaar 
9548cbd6dfcSBram Moolenaar 		    if (trystack->ga_len > 0)
9558cbd6dfcSBram Moolenaar 			trycmd = ((trycmd_T *)trystack->ga_data)
9568cbd6dfcSBram Moolenaar 							+ trystack->ga_len - 1;
9578a7d6542SBram Moolenaar 		    if (trycmd != NULL && trycmd->tcd_frame == ectx.ec_frame
9588a7d6542SBram Moolenaar 			    && trycmd->tcd_finally_idx != 0)
9598a7d6542SBram Moolenaar 		    {
9608a7d6542SBram Moolenaar 			// jump to ":finally"
9618a7d6542SBram Moolenaar 			ectx.ec_iidx = trycmd->tcd_finally_idx;
9628a7d6542SBram Moolenaar 			trycmd->tcd_return = TRUE;
9638a7d6542SBram Moolenaar 		    }
9648a7d6542SBram Moolenaar 		    else
9658a7d6542SBram Moolenaar 		    {
9668a7d6542SBram Moolenaar 			// Restore previous function. If the frame pointer
9678a7d6542SBram Moolenaar 			// is zero then there is none and we are done.
9688a7d6542SBram Moolenaar 			if (ectx.ec_frame == initial_frame_ptr)
9698a7d6542SBram Moolenaar 			    goto done;
9708a7d6542SBram Moolenaar 
9718a7d6542SBram Moolenaar 			func_return(&ectx);
9728a7d6542SBram Moolenaar 		    }
9738a7d6542SBram Moolenaar 		}
9748a7d6542SBram Moolenaar 		break;
9758a7d6542SBram Moolenaar 
9768a7d6542SBram Moolenaar 	    // push a function reference to a compiled function
9778a7d6542SBram Moolenaar 	    case ISN_FUNCREF:
9788a7d6542SBram Moolenaar 		{
9798a7d6542SBram Moolenaar 		    partial_T   *pt = NULL;
9808a7d6542SBram Moolenaar 
9818a7d6542SBram Moolenaar 		    pt = ALLOC_CLEAR_ONE(partial_T);
9828a7d6542SBram Moolenaar 		    if (pt == NULL)
9838a7d6542SBram Moolenaar 			goto failed;
9848a7d6542SBram Moolenaar 		    dfunc = ((dfunc_T *)def_functions.ga_data)
9858a7d6542SBram Moolenaar 							+ iptr->isn_arg.number;
9868a7d6542SBram Moolenaar 		    pt->pt_func = dfunc->df_ufunc;
9878a7d6542SBram Moolenaar 		    pt->pt_refcount = 1;
9888a7d6542SBram Moolenaar 		    ++dfunc->df_ufunc->uf_refcount;
9898a7d6542SBram Moolenaar 
9908a7d6542SBram Moolenaar 		    if (ga_grow(&ectx.ec_stack, 1) == FAIL)
9918a7d6542SBram Moolenaar 			goto failed;
9928a7d6542SBram Moolenaar 		    tv = STACK_TV_BOT(0);
9938a7d6542SBram Moolenaar 		    ++ectx.ec_stack.ga_len;
9948a7d6542SBram Moolenaar 		    tv->vval.v_partial = pt;
9958a7d6542SBram Moolenaar 		    tv->v_type = VAR_PARTIAL;
9968a7d6542SBram Moolenaar 		}
9978a7d6542SBram Moolenaar 		break;
9988a7d6542SBram Moolenaar 
9998a7d6542SBram Moolenaar 	    // jump if a condition is met
10008a7d6542SBram Moolenaar 	    case ISN_JUMP:
10018a7d6542SBram Moolenaar 		{
10028a7d6542SBram Moolenaar 		    jumpwhen_T	when = iptr->isn_arg.jump.jump_when;
10038a7d6542SBram Moolenaar 		    int		jump = TRUE;
10048a7d6542SBram Moolenaar 
10058a7d6542SBram Moolenaar 		    if (when != JUMP_ALWAYS)
10068a7d6542SBram Moolenaar 		    {
10078a7d6542SBram Moolenaar 			tv = STACK_TV_BOT(-1);
10088a7d6542SBram Moolenaar 			jump = tv2bool(tv);
10098a7d6542SBram Moolenaar 			if (when == JUMP_IF_FALSE
10108a7d6542SBram Moolenaar 					     || when == JUMP_AND_KEEP_IF_FALSE)
10118a7d6542SBram Moolenaar 			    jump = !jump;
1012777770fbSBram Moolenaar 			if (when == JUMP_IF_FALSE || !jump)
10138a7d6542SBram Moolenaar 			{
10148a7d6542SBram Moolenaar 			    // drop the value from the stack
10158a7d6542SBram Moolenaar 			    clear_tv(tv);
10168a7d6542SBram Moolenaar 			    --ectx.ec_stack.ga_len;
10178a7d6542SBram Moolenaar 			}
10188a7d6542SBram Moolenaar 		    }
10198a7d6542SBram Moolenaar 		    if (jump)
10208a7d6542SBram Moolenaar 			ectx.ec_iidx = iptr->isn_arg.jump.jump_where;
10218a7d6542SBram Moolenaar 		}
10228a7d6542SBram Moolenaar 		break;
10238a7d6542SBram Moolenaar 
10248a7d6542SBram Moolenaar 	    // top of a for loop
10258a7d6542SBram Moolenaar 	    case ISN_FOR:
10268a7d6542SBram Moolenaar 		{
10278a7d6542SBram Moolenaar 		    list_T	*list = STACK_TV_BOT(-1)->vval.v_list;
10288a7d6542SBram Moolenaar 		    typval_T	*idxtv =
10298a7d6542SBram Moolenaar 				   STACK_TV_VAR(iptr->isn_arg.forloop.for_idx);
10308a7d6542SBram Moolenaar 
10318a7d6542SBram Moolenaar 		    // push the next item from the list
10328a7d6542SBram Moolenaar 		    if (ga_grow(&ectx.ec_stack, 1) == FAIL)
10338a7d6542SBram Moolenaar 			goto failed;
10348a7d6542SBram Moolenaar 		    if (++idxtv->vval.v_number >= list->lv_len)
10358a7d6542SBram Moolenaar 			// past the end of the list, jump to "endfor"
10368a7d6542SBram Moolenaar 			ectx.ec_iidx = iptr->isn_arg.forloop.for_end;
10378a7d6542SBram Moolenaar 		    else if (list->lv_first == &range_list_item)
10388a7d6542SBram Moolenaar 		    {
10398a7d6542SBram Moolenaar 			// non-materialized range() list
10408a7d6542SBram Moolenaar 			tv = STACK_TV_BOT(0);
10418a7d6542SBram Moolenaar 			tv->v_type = VAR_NUMBER;
10428a7d6542SBram Moolenaar 			tv->vval.v_number = list_find_nr(
10438a7d6542SBram Moolenaar 					     list, idxtv->vval.v_number, NULL);
10448a7d6542SBram Moolenaar 			++ectx.ec_stack.ga_len;
10458a7d6542SBram Moolenaar 		    }
10468a7d6542SBram Moolenaar 		    else
10478a7d6542SBram Moolenaar 		    {
10488a7d6542SBram Moolenaar 			listitem_T *li = list_find(list, idxtv->vval.v_number);
10498a7d6542SBram Moolenaar 
10508a7d6542SBram Moolenaar 			if (li == NULL)
10518a7d6542SBram Moolenaar 			    goto failed;
10528a7d6542SBram Moolenaar 			copy_tv(&li->li_tv, STACK_TV_BOT(0));
10538a7d6542SBram Moolenaar 			++ectx.ec_stack.ga_len;
10548a7d6542SBram Moolenaar 		    }
10558a7d6542SBram Moolenaar 		}
10568a7d6542SBram Moolenaar 		break;
10578a7d6542SBram Moolenaar 
10588a7d6542SBram Moolenaar 	    // start of ":try" block
10598a7d6542SBram Moolenaar 	    case ISN_TRY:
10608a7d6542SBram Moolenaar 		{
10618a7d6542SBram Moolenaar 		    if (ga_grow(&ectx.ec_trystack, 1) == FAIL)
10628a7d6542SBram Moolenaar 			goto failed;
10638a7d6542SBram Moolenaar 		    trycmd = ((trycmd_T *)ectx.ec_trystack.ga_data)
10648a7d6542SBram Moolenaar 						     + ectx.ec_trystack.ga_len;
10658a7d6542SBram Moolenaar 		    ++ectx.ec_trystack.ga_len;
10668a7d6542SBram Moolenaar 		    ++trylevel;
10678a7d6542SBram Moolenaar 		    trycmd->tcd_frame = ectx.ec_frame;
10688a7d6542SBram Moolenaar 		    trycmd->tcd_catch_idx = iptr->isn_arg.try.try_catch;
10698a7d6542SBram Moolenaar 		    trycmd->tcd_finally_idx = iptr->isn_arg.try.try_finally;
1070f575adffSBram Moolenaar 		    trycmd->tcd_caught = FALSE;
10718a7d6542SBram Moolenaar 		}
10728a7d6542SBram Moolenaar 		break;
10738a7d6542SBram Moolenaar 
10748a7d6542SBram Moolenaar 	    case ISN_PUSHEXC:
10758a7d6542SBram Moolenaar 		if (current_exception == NULL)
10768a7d6542SBram Moolenaar 		{
10778a7d6542SBram Moolenaar 		    iemsg("Evaluating catch while current_exception is NULL");
10788a7d6542SBram Moolenaar 		    goto failed;
10798a7d6542SBram Moolenaar 		}
10808a7d6542SBram Moolenaar 		if (ga_grow(&ectx.ec_stack, 1) == FAIL)
10818a7d6542SBram Moolenaar 		    goto failed;
10828a7d6542SBram Moolenaar 		tv = STACK_TV_BOT(0);
10838a7d6542SBram Moolenaar 		++ectx.ec_stack.ga_len;
10848a7d6542SBram Moolenaar 		tv->v_type = VAR_STRING;
10858a7d6542SBram Moolenaar 		tv->vval.v_string = vim_strsave(
10868a7d6542SBram Moolenaar 					   (char_u *)current_exception->value);
10878a7d6542SBram Moolenaar 		break;
10888a7d6542SBram Moolenaar 
10898a7d6542SBram Moolenaar 	    case ISN_CATCH:
10908a7d6542SBram Moolenaar 		{
10918a7d6542SBram Moolenaar 		    garray_T	*trystack = &ectx.ec_trystack;
10928a7d6542SBram Moolenaar 
10938a7d6542SBram Moolenaar 		    if (trystack->ga_len > 0)
10948a7d6542SBram Moolenaar 		    {
10958a7d6542SBram Moolenaar 			trycmd = ((trycmd_T *)trystack->ga_data)
10968a7d6542SBram Moolenaar 							+ trystack->ga_len - 1;
10978a7d6542SBram Moolenaar 			trycmd->tcd_caught = TRUE;
10988a7d6542SBram Moolenaar 		    }
10998a7d6542SBram Moolenaar 		    did_emsg = got_int = did_throw = FALSE;
11008a7d6542SBram Moolenaar 		    catch_exception(current_exception);
11018a7d6542SBram Moolenaar 		}
11028a7d6542SBram Moolenaar 		break;
11038a7d6542SBram Moolenaar 
11048a7d6542SBram Moolenaar 	    // end of ":try" block
11058a7d6542SBram Moolenaar 	    case ISN_ENDTRY:
11068a7d6542SBram Moolenaar 		{
11078a7d6542SBram Moolenaar 		    garray_T	*trystack = &ectx.ec_trystack;
11088a7d6542SBram Moolenaar 
11098a7d6542SBram Moolenaar 		    if (trystack->ga_len > 0)
11108a7d6542SBram Moolenaar 		    {
11118a7d6542SBram Moolenaar 			--trystack->ga_len;
11128a7d6542SBram Moolenaar 			--trylevel;
11138a7d6542SBram Moolenaar 			trycmd = ((trycmd_T *)trystack->ga_data)
11148a7d6542SBram Moolenaar 							    + trystack->ga_len;
1115f575adffSBram Moolenaar 			if (trycmd->tcd_caught && current_exception != NULL)
11168a7d6542SBram Moolenaar 			{
11178a7d6542SBram Moolenaar 			    // discard the exception
11188a7d6542SBram Moolenaar 			    if (caught_stack == current_exception)
11198a7d6542SBram Moolenaar 				caught_stack = caught_stack->caught;
11208a7d6542SBram Moolenaar 			    discard_current_exception();
11218a7d6542SBram Moolenaar 			}
11228a7d6542SBram Moolenaar 
11238a7d6542SBram Moolenaar 			if (trycmd->tcd_return)
11248a7d6542SBram Moolenaar 			{
11258a7d6542SBram Moolenaar 			    // Restore previous function. If the frame pointer
11268a7d6542SBram Moolenaar 			    // is zero then there is none and we are done.
11278a7d6542SBram Moolenaar 			    if (ectx.ec_frame == initial_frame_ptr)
11288a7d6542SBram Moolenaar 				goto done;
11298a7d6542SBram Moolenaar 
11308a7d6542SBram Moolenaar 			    func_return(&ectx);
11318a7d6542SBram Moolenaar 			}
11328a7d6542SBram Moolenaar 		    }
11338a7d6542SBram Moolenaar 		}
11348a7d6542SBram Moolenaar 		break;
11358a7d6542SBram Moolenaar 
11368a7d6542SBram Moolenaar 	    case ISN_THROW:
11378a7d6542SBram Moolenaar 		--ectx.ec_stack.ga_len;
11388a7d6542SBram Moolenaar 		tv = STACK_TV_BOT(0);
11398a7d6542SBram Moolenaar 		if (throw_exception(tv->vval.v_string, ET_USER, NULL) == FAIL)
11408a7d6542SBram Moolenaar 		{
11418a7d6542SBram Moolenaar 		    vim_free(tv->vval.v_string);
11428a7d6542SBram Moolenaar 		    goto failed;
11438a7d6542SBram Moolenaar 		}
11448a7d6542SBram Moolenaar 		did_throw = TRUE;
11458a7d6542SBram Moolenaar 		break;
11468a7d6542SBram Moolenaar 
11478a7d6542SBram Moolenaar 	    // compare with special values
11488a7d6542SBram Moolenaar 	    case ISN_COMPAREBOOL:
11498a7d6542SBram Moolenaar 	    case ISN_COMPARESPECIAL:
11508a7d6542SBram Moolenaar 		{
11518a7d6542SBram Moolenaar 		    typval_T	*tv1 = STACK_TV_BOT(-2);
11528a7d6542SBram Moolenaar 		    typval_T	*tv2 = STACK_TV_BOT(-1);
11538a7d6542SBram Moolenaar 		    varnumber_T arg1 = tv1->vval.v_number;
11548a7d6542SBram Moolenaar 		    varnumber_T arg2 = tv2->vval.v_number;
11558a7d6542SBram Moolenaar 		    int		res;
11568a7d6542SBram Moolenaar 
11578a7d6542SBram Moolenaar 		    switch (iptr->isn_arg.op.op_type)
11588a7d6542SBram Moolenaar 		    {
11598a7d6542SBram Moolenaar 			case EXPR_EQUAL: res = arg1 == arg2; break;
11608a7d6542SBram Moolenaar 			case EXPR_NEQUAL: res = arg1 != arg2; break;
11618a7d6542SBram Moolenaar 			default: res = 0; break;
11628a7d6542SBram Moolenaar 		    }
11638a7d6542SBram Moolenaar 
11648a7d6542SBram Moolenaar 		    --ectx.ec_stack.ga_len;
11658a7d6542SBram Moolenaar 		    tv1->v_type = VAR_BOOL;
11668a7d6542SBram Moolenaar 		    tv1->vval.v_number = res ? VVAL_TRUE : VVAL_FALSE;
11678a7d6542SBram Moolenaar 		}
11688a7d6542SBram Moolenaar 		break;
11698a7d6542SBram Moolenaar 
11708a7d6542SBram Moolenaar 	    // Operation with two number arguments
11718a7d6542SBram Moolenaar 	    case ISN_OPNR:
11728a7d6542SBram Moolenaar 	    case ISN_COMPARENR:
11738a7d6542SBram Moolenaar 		{
11748a7d6542SBram Moolenaar 		    typval_T	*tv1 = STACK_TV_BOT(-2);
11758a7d6542SBram Moolenaar 		    typval_T	*tv2 = STACK_TV_BOT(-1);
11768a7d6542SBram Moolenaar 		    varnumber_T arg1 = tv1->vval.v_number;
11778a7d6542SBram Moolenaar 		    varnumber_T arg2 = tv2->vval.v_number;
11788a7d6542SBram Moolenaar 		    varnumber_T res;
11798a7d6542SBram Moolenaar 
11808a7d6542SBram Moolenaar 		    switch (iptr->isn_arg.op.op_type)
11818a7d6542SBram Moolenaar 		    {
11828a7d6542SBram Moolenaar 			case EXPR_MULT: res = arg1 * arg2; break;
11838a7d6542SBram Moolenaar 			case EXPR_DIV: res = arg1 / arg2; break;
11848a7d6542SBram Moolenaar 			case EXPR_REM: res = arg1 % arg2; break;
11858a7d6542SBram Moolenaar 			case EXPR_SUB: res = arg1 - arg2; break;
11868a7d6542SBram Moolenaar 			case EXPR_ADD: res = arg1 + arg2; break;
11878a7d6542SBram Moolenaar 
11888a7d6542SBram Moolenaar 			case EXPR_EQUAL: res = arg1 == arg2; break;
11898a7d6542SBram Moolenaar 			case EXPR_NEQUAL: res = arg1 != arg2; break;
11908a7d6542SBram Moolenaar 			case EXPR_GREATER: res = arg1 > arg2; break;
11918a7d6542SBram Moolenaar 			case EXPR_GEQUAL: res = arg1 >= arg2; break;
11928a7d6542SBram Moolenaar 			case EXPR_SMALLER: res = arg1 < arg2; break;
11938a7d6542SBram Moolenaar 			case EXPR_SEQUAL: res = arg1 <= arg2; break;
11948a7d6542SBram Moolenaar 			default: res = 0; break;
11958a7d6542SBram Moolenaar 		    }
11968a7d6542SBram Moolenaar 
11978a7d6542SBram Moolenaar 		    --ectx.ec_stack.ga_len;
11988a7d6542SBram Moolenaar 		    if (iptr->isn_type == ISN_COMPARENR)
11998a7d6542SBram Moolenaar 		    {
12008a7d6542SBram Moolenaar 			tv1->v_type = VAR_BOOL;
12018a7d6542SBram Moolenaar 			tv1->vval.v_number = res ? VVAL_TRUE : VVAL_FALSE;
12028a7d6542SBram Moolenaar 		    }
12038a7d6542SBram Moolenaar 		    else
12048a7d6542SBram Moolenaar 			tv1->vval.v_number = res;
12058a7d6542SBram Moolenaar 		}
12068a7d6542SBram Moolenaar 		break;
12078a7d6542SBram Moolenaar 
12088a7d6542SBram Moolenaar 	    // Computation with two float arguments
12098a7d6542SBram Moolenaar 	    case ISN_OPFLOAT:
12108a7d6542SBram Moolenaar 	    case ISN_COMPAREFLOAT:
1211a5d5953dSBram Moolenaar #ifdef FEAT_FLOAT
12128a7d6542SBram Moolenaar 		{
12138a7d6542SBram Moolenaar 		    typval_T	*tv1 = STACK_TV_BOT(-2);
12148a7d6542SBram Moolenaar 		    typval_T	*tv2 = STACK_TV_BOT(-1);
12158a7d6542SBram Moolenaar 		    float_T	arg1 = tv1->vval.v_float;
12168a7d6542SBram Moolenaar 		    float_T	arg2 = tv2->vval.v_float;
12178a7d6542SBram Moolenaar 		    float_T	res = 0;
12188a7d6542SBram Moolenaar 		    int		cmp = FALSE;
12198a7d6542SBram Moolenaar 
12208a7d6542SBram Moolenaar 		    switch (iptr->isn_arg.op.op_type)
12218a7d6542SBram Moolenaar 		    {
12228a7d6542SBram Moolenaar 			case EXPR_MULT: res = arg1 * arg2; break;
12238a7d6542SBram Moolenaar 			case EXPR_DIV: res = arg1 / arg2; break;
12248a7d6542SBram Moolenaar 			case EXPR_SUB: res = arg1 - arg2; break;
12258a7d6542SBram Moolenaar 			case EXPR_ADD: res = arg1 + arg2; break;
12268a7d6542SBram Moolenaar 
12278a7d6542SBram Moolenaar 			case EXPR_EQUAL: cmp = arg1 == arg2; break;
12288a7d6542SBram Moolenaar 			case EXPR_NEQUAL: cmp = arg1 != arg2; break;
12298a7d6542SBram Moolenaar 			case EXPR_GREATER: cmp = arg1 > arg2; break;
12308a7d6542SBram Moolenaar 			case EXPR_GEQUAL: cmp = arg1 >= arg2; break;
12318a7d6542SBram Moolenaar 			case EXPR_SMALLER: cmp = arg1 < arg2; break;
12328a7d6542SBram Moolenaar 			case EXPR_SEQUAL: cmp = arg1 <= arg2; break;
12338a7d6542SBram Moolenaar 			default: cmp = 0; break;
12348a7d6542SBram Moolenaar 		    }
12358a7d6542SBram Moolenaar 		    --ectx.ec_stack.ga_len;
12368a7d6542SBram Moolenaar 		    if (iptr->isn_type == ISN_COMPAREFLOAT)
12378a7d6542SBram Moolenaar 		    {
12388a7d6542SBram Moolenaar 			tv1->v_type = VAR_BOOL;
12398a7d6542SBram Moolenaar 			tv1->vval.v_number = cmp ? VVAL_TRUE : VVAL_FALSE;
12408a7d6542SBram Moolenaar 		    }
12418a7d6542SBram Moolenaar 		    else
12428a7d6542SBram Moolenaar 			tv1->vval.v_float = res;
12438a7d6542SBram Moolenaar 		}
1244a5d5953dSBram Moolenaar #endif
12458a7d6542SBram Moolenaar 		break;
12468a7d6542SBram Moolenaar 
12478a7d6542SBram Moolenaar 	    case ISN_COMPARELIST:
12488a7d6542SBram Moolenaar 		{
12498a7d6542SBram Moolenaar 		    typval_T	*tv1 = STACK_TV_BOT(-2);
12508a7d6542SBram Moolenaar 		    typval_T	*tv2 = STACK_TV_BOT(-1);
12518a7d6542SBram Moolenaar 		    list_T	*arg1 = tv1->vval.v_list;
12528a7d6542SBram Moolenaar 		    list_T	*arg2 = tv2->vval.v_list;
12538a7d6542SBram Moolenaar 		    int		cmp = FALSE;
12548a7d6542SBram Moolenaar 		    int		ic = iptr->isn_arg.op.op_ic;
12558a7d6542SBram Moolenaar 
12568a7d6542SBram Moolenaar 		    switch (iptr->isn_arg.op.op_type)
12578a7d6542SBram Moolenaar 		    {
12588a7d6542SBram Moolenaar 			case EXPR_EQUAL: cmp =
12598a7d6542SBram Moolenaar 				      list_equal(arg1, arg2, ic, FALSE); break;
12608a7d6542SBram Moolenaar 			case EXPR_NEQUAL: cmp =
12618a7d6542SBram Moolenaar 				     !list_equal(arg1, arg2, ic, FALSE); break;
12628a7d6542SBram Moolenaar 			case EXPR_IS: cmp = arg1 == arg2; break;
12638a7d6542SBram Moolenaar 			case EXPR_ISNOT: cmp = arg1 != arg2; break;
12648a7d6542SBram Moolenaar 			default: cmp = 0; break;
12658a7d6542SBram Moolenaar 		    }
12668a7d6542SBram Moolenaar 		    --ectx.ec_stack.ga_len;
12678a7d6542SBram Moolenaar 		    clear_tv(tv1);
12688a7d6542SBram Moolenaar 		    clear_tv(tv2);
12698a7d6542SBram Moolenaar 		    tv1->v_type = VAR_BOOL;
12708a7d6542SBram Moolenaar 		    tv1->vval.v_number = cmp ? VVAL_TRUE : VVAL_FALSE;
12718a7d6542SBram Moolenaar 		}
12728a7d6542SBram Moolenaar 		break;
12738a7d6542SBram Moolenaar 
12748a7d6542SBram Moolenaar 	    case ISN_COMPAREBLOB:
12758a7d6542SBram Moolenaar 		{
12768a7d6542SBram Moolenaar 		    typval_T	*tv1 = STACK_TV_BOT(-2);
12778a7d6542SBram Moolenaar 		    typval_T	*tv2 = STACK_TV_BOT(-1);
12788a7d6542SBram Moolenaar 		    blob_T	*arg1 = tv1->vval.v_blob;
12798a7d6542SBram Moolenaar 		    blob_T	*arg2 = tv2->vval.v_blob;
12808a7d6542SBram Moolenaar 		    int		cmp = FALSE;
12818a7d6542SBram Moolenaar 
12828a7d6542SBram Moolenaar 		    switch (iptr->isn_arg.op.op_type)
12838a7d6542SBram Moolenaar 		    {
12848a7d6542SBram Moolenaar 			case EXPR_EQUAL: cmp = blob_equal(arg1, arg2); break;
12858a7d6542SBram Moolenaar 			case EXPR_NEQUAL: cmp = !blob_equal(arg1, arg2); break;
12868a7d6542SBram Moolenaar 			case EXPR_IS: cmp = arg1 == arg2; break;
12878a7d6542SBram Moolenaar 			case EXPR_ISNOT: cmp = arg1 != arg2; break;
12888a7d6542SBram Moolenaar 			default: cmp = 0; break;
12898a7d6542SBram Moolenaar 		    }
12908a7d6542SBram Moolenaar 		    --ectx.ec_stack.ga_len;
12918a7d6542SBram Moolenaar 		    clear_tv(tv1);
12928a7d6542SBram Moolenaar 		    clear_tv(tv2);
12938a7d6542SBram Moolenaar 		    tv1->v_type = VAR_BOOL;
12948a7d6542SBram Moolenaar 		    tv1->vval.v_number = cmp ? VVAL_TRUE : VVAL_FALSE;
12958a7d6542SBram Moolenaar 		}
12968a7d6542SBram Moolenaar 		break;
12978a7d6542SBram Moolenaar 
12988a7d6542SBram Moolenaar 		// TODO: handle separately
12998a7d6542SBram Moolenaar 	    case ISN_COMPARESTRING:
13008a7d6542SBram Moolenaar 	    case ISN_COMPAREDICT:
13018a7d6542SBram Moolenaar 	    case ISN_COMPAREFUNC:
13028a7d6542SBram Moolenaar 	    case ISN_COMPAREPARTIAL:
13038a7d6542SBram Moolenaar 	    case ISN_COMPAREANY:
13048a7d6542SBram Moolenaar 		{
13058a7d6542SBram Moolenaar 		    typval_T	*tv1 = STACK_TV_BOT(-2);
13068a7d6542SBram Moolenaar 		    typval_T	*tv2 = STACK_TV_BOT(-1);
13078a7d6542SBram Moolenaar 		    exptype_T	exptype = iptr->isn_arg.op.op_type;
13088a7d6542SBram Moolenaar 		    int		ic = iptr->isn_arg.op.op_ic;
13098a7d6542SBram Moolenaar 
13108a7d6542SBram Moolenaar 		    typval_compare(tv1, tv2, exptype, ic);
13118a7d6542SBram Moolenaar 		    clear_tv(tv2);
13128a7d6542SBram Moolenaar 		    tv1->v_type = VAR_BOOL;
13138a7d6542SBram Moolenaar 		    tv1->vval.v_number = tv1->vval.v_number
13148a7d6542SBram Moolenaar 						      ? VVAL_TRUE : VVAL_FALSE;
13158a7d6542SBram Moolenaar 		    --ectx.ec_stack.ga_len;
13168a7d6542SBram Moolenaar 		}
13178a7d6542SBram Moolenaar 		break;
13188a7d6542SBram Moolenaar 
13198a7d6542SBram Moolenaar 	    case ISN_ADDLIST:
13208a7d6542SBram Moolenaar 	    case ISN_ADDBLOB:
13218a7d6542SBram Moolenaar 		{
13228a7d6542SBram Moolenaar 		    typval_T *tv1 = STACK_TV_BOT(-2);
13238a7d6542SBram Moolenaar 		    typval_T *tv2 = STACK_TV_BOT(-1);
13248a7d6542SBram Moolenaar 
13258a7d6542SBram Moolenaar 		    if (iptr->isn_type == ISN_ADDLIST)
13268a7d6542SBram Moolenaar 			eval_addlist(tv1, tv2);
13278a7d6542SBram Moolenaar 		    else
13288a7d6542SBram Moolenaar 			eval_addblob(tv1, tv2);
13298a7d6542SBram Moolenaar 		    clear_tv(tv2);
13308a7d6542SBram Moolenaar 		    --ectx.ec_stack.ga_len;
13318a7d6542SBram Moolenaar 		}
13328a7d6542SBram Moolenaar 		break;
13338a7d6542SBram Moolenaar 
13348a7d6542SBram Moolenaar 	    // Computation with two arguments of unknown type
13358a7d6542SBram Moolenaar 	    case ISN_OPANY:
13368a7d6542SBram Moolenaar 		{
13378a7d6542SBram Moolenaar 		    typval_T	*tv1 = STACK_TV_BOT(-2);
13388a7d6542SBram Moolenaar 		    typval_T	*tv2 = STACK_TV_BOT(-1);
13398a7d6542SBram Moolenaar 		    varnumber_T	n1, n2;
13408a7d6542SBram Moolenaar #ifdef FEAT_FLOAT
13418a7d6542SBram Moolenaar 		    float_T	f1 = 0, f2 = 0;
13428a7d6542SBram Moolenaar #endif
13438a7d6542SBram Moolenaar 		    int		error = FALSE;
13448a7d6542SBram Moolenaar 
13458a7d6542SBram Moolenaar 		    if (iptr->isn_arg.op.op_type == EXPR_ADD)
13468a7d6542SBram Moolenaar 		    {
13478a7d6542SBram Moolenaar 			if (tv1->v_type == VAR_LIST && tv2->v_type == VAR_LIST)
13488a7d6542SBram Moolenaar 			{
13498a7d6542SBram Moolenaar 			    eval_addlist(tv1, tv2);
13508a7d6542SBram Moolenaar 			    clear_tv(tv2);
13518a7d6542SBram Moolenaar 			    --ectx.ec_stack.ga_len;
13528a7d6542SBram Moolenaar 			    break;
13538a7d6542SBram Moolenaar 			}
13548a7d6542SBram Moolenaar 			else if (tv1->v_type == VAR_BLOB
13558a7d6542SBram Moolenaar 						    && tv2->v_type == VAR_BLOB)
13568a7d6542SBram Moolenaar 			{
13578a7d6542SBram Moolenaar 			    eval_addblob(tv1, tv2);
13588a7d6542SBram Moolenaar 			    clear_tv(tv2);
13598a7d6542SBram Moolenaar 			    --ectx.ec_stack.ga_len;
13608a7d6542SBram Moolenaar 			    break;
13618a7d6542SBram Moolenaar 			}
13628a7d6542SBram Moolenaar 		    }
13638a7d6542SBram Moolenaar #ifdef FEAT_FLOAT
13648a7d6542SBram Moolenaar 		    if (tv1->v_type == VAR_FLOAT)
13658a7d6542SBram Moolenaar 		    {
13668a7d6542SBram Moolenaar 			f1 = tv1->vval.v_float;
13678a7d6542SBram Moolenaar 			n1 = 0;
13688a7d6542SBram Moolenaar 		    }
13698a7d6542SBram Moolenaar 		    else
13708a7d6542SBram Moolenaar #endif
13718a7d6542SBram Moolenaar 		    {
13728a7d6542SBram Moolenaar 			n1 = tv_get_number_chk(tv1, &error);
13738a7d6542SBram Moolenaar 			if (error)
13748a7d6542SBram Moolenaar 			    goto failed;
13758a7d6542SBram Moolenaar #ifdef FEAT_FLOAT
13768a7d6542SBram Moolenaar 			if (tv2->v_type == VAR_FLOAT)
13778a7d6542SBram Moolenaar 			    f1 = n1;
13788a7d6542SBram Moolenaar #endif
13798a7d6542SBram Moolenaar 		    }
13808a7d6542SBram Moolenaar #ifdef FEAT_FLOAT
13818a7d6542SBram Moolenaar 		    if (tv2->v_type == VAR_FLOAT)
13828a7d6542SBram Moolenaar 		    {
13838a7d6542SBram Moolenaar 			f2 = tv2->vval.v_float;
13848a7d6542SBram Moolenaar 			n2 = 0;
13858a7d6542SBram Moolenaar 		    }
13868a7d6542SBram Moolenaar 		    else
13878a7d6542SBram Moolenaar #endif
13888a7d6542SBram Moolenaar 		    {
13898a7d6542SBram Moolenaar 			n2 = tv_get_number_chk(tv2, &error);
13908a7d6542SBram Moolenaar 			if (error)
13918a7d6542SBram Moolenaar 			    goto failed;
13928a7d6542SBram Moolenaar #ifdef FEAT_FLOAT
13938a7d6542SBram Moolenaar 			if (tv1->v_type == VAR_FLOAT)
13948a7d6542SBram Moolenaar 			    f2 = n2;
13958a7d6542SBram Moolenaar #endif
13968a7d6542SBram Moolenaar 		    }
13978a7d6542SBram Moolenaar #ifdef FEAT_FLOAT
13988a7d6542SBram Moolenaar 		    // if there is a float on either side the result is a float
13998a7d6542SBram Moolenaar 		    if (tv1->v_type == VAR_FLOAT || tv2->v_type == VAR_FLOAT)
14008a7d6542SBram Moolenaar 		    {
14018a7d6542SBram Moolenaar 			switch (iptr->isn_arg.op.op_type)
14028a7d6542SBram Moolenaar 			{
14038a7d6542SBram Moolenaar 			    case EXPR_MULT: f1 = f1 * f2; break;
14048a7d6542SBram Moolenaar 			    case EXPR_DIV:  f1 = f1 / f2; break;
14058a7d6542SBram Moolenaar 			    case EXPR_SUB:  f1 = f1 - f2; break;
14068a7d6542SBram Moolenaar 			    case EXPR_ADD:  f1 = f1 + f2; break;
14078a7d6542SBram Moolenaar 			    default: emsg(_(e_modulus)); goto failed;
14088a7d6542SBram Moolenaar 			}
14098a7d6542SBram Moolenaar 			clear_tv(tv1);
14108a7d6542SBram Moolenaar 			clear_tv(tv2);
14118a7d6542SBram Moolenaar 			tv1->v_type = VAR_FLOAT;
14128a7d6542SBram Moolenaar 			tv1->vval.v_float = f1;
14138a7d6542SBram Moolenaar 			--ectx.ec_stack.ga_len;
14148a7d6542SBram Moolenaar 		    }
14158a7d6542SBram Moolenaar 		    else
14168a7d6542SBram Moolenaar #endif
14178a7d6542SBram Moolenaar 		    {
14188a7d6542SBram Moolenaar 			switch (iptr->isn_arg.op.op_type)
14198a7d6542SBram Moolenaar 			{
14208a7d6542SBram Moolenaar 			    case EXPR_MULT: n1 = n1 * n2; break;
14218a7d6542SBram Moolenaar 			    case EXPR_DIV:  n1 = num_divide(n1, n2); break;
14228a7d6542SBram Moolenaar 			    case EXPR_SUB:  n1 = n1 - n2; break;
14238a7d6542SBram Moolenaar 			    case EXPR_ADD:  n1 = n1 + n2; break;
14248a7d6542SBram Moolenaar 			    default:	    n1 = num_modulus(n1, n2); break;
14258a7d6542SBram Moolenaar 			}
14268a7d6542SBram Moolenaar 			clear_tv(tv1);
14278a7d6542SBram Moolenaar 			clear_tv(tv2);
14288a7d6542SBram Moolenaar 			tv1->v_type = VAR_NUMBER;
14298a7d6542SBram Moolenaar 			tv1->vval.v_number = n1;
14308a7d6542SBram Moolenaar 			--ectx.ec_stack.ga_len;
14318a7d6542SBram Moolenaar 		    }
14328a7d6542SBram Moolenaar 		}
14338a7d6542SBram Moolenaar 		break;
14348a7d6542SBram Moolenaar 
14358a7d6542SBram Moolenaar 	    case ISN_CONCAT:
14368a7d6542SBram Moolenaar 		{
14378a7d6542SBram Moolenaar 		    char_u *str1 = STACK_TV_BOT(-2)->vval.v_string;
14388a7d6542SBram Moolenaar 		    char_u *str2 = STACK_TV_BOT(-1)->vval.v_string;
14398a7d6542SBram Moolenaar 		    char_u *res;
14408a7d6542SBram Moolenaar 
14418a7d6542SBram Moolenaar 		    res = concat_str(str1, str2);
14428a7d6542SBram Moolenaar 		    clear_tv(STACK_TV_BOT(-2));
14438a7d6542SBram Moolenaar 		    clear_tv(STACK_TV_BOT(-1));
14448a7d6542SBram Moolenaar 		    --ectx.ec_stack.ga_len;
14458a7d6542SBram Moolenaar 		    STACK_TV_BOT(-1)->vval.v_string = res;
14468a7d6542SBram Moolenaar 		}
14478a7d6542SBram Moolenaar 		break;
14488a7d6542SBram Moolenaar 
14498a7d6542SBram Moolenaar 	    case ISN_INDEX:
14508a7d6542SBram Moolenaar 		{
14518a7d6542SBram Moolenaar 		    list_T	*list;
14528a7d6542SBram Moolenaar 		    varnumber_T	n;
14538a7d6542SBram Moolenaar 		    listitem_T	*li;
14548a7d6542SBram Moolenaar 
14558a7d6542SBram Moolenaar 		    // list index: list is at stack-2, index at stack-1
14568a7d6542SBram Moolenaar 		    tv = STACK_TV_BOT(-2);
14578a7d6542SBram Moolenaar 		    if (tv->v_type != VAR_LIST)
14588a7d6542SBram Moolenaar 		    {
14598a7d6542SBram Moolenaar 			emsg(_(e_listreq));
14608a7d6542SBram Moolenaar 			goto failed;
14618a7d6542SBram Moolenaar 		    }
14628a7d6542SBram Moolenaar 		    list = tv->vval.v_list;
14638a7d6542SBram Moolenaar 
14648a7d6542SBram Moolenaar 		    tv = STACK_TV_BOT(-1);
14658a7d6542SBram Moolenaar 		    if (tv->v_type != VAR_NUMBER)
14668a7d6542SBram Moolenaar 		    {
14678a7d6542SBram Moolenaar 			emsg(_(e_number_exp));
14688a7d6542SBram Moolenaar 			goto failed;
14698a7d6542SBram Moolenaar 		    }
14708a7d6542SBram Moolenaar 		    n = tv->vval.v_number;
14718a7d6542SBram Moolenaar 		    clear_tv(tv);
14728a7d6542SBram Moolenaar 		    if ((li = list_find(list, n)) == NULL)
14738a7d6542SBram Moolenaar 		    {
14748a7d6542SBram Moolenaar 			semsg(_(e_listidx), n);
14758a7d6542SBram Moolenaar 			goto failed;
14768a7d6542SBram Moolenaar 		    }
14778a7d6542SBram Moolenaar 		    --ectx.ec_stack.ga_len;
14788a7d6542SBram Moolenaar 		    clear_tv(STACK_TV_BOT(-1));
14798a7d6542SBram Moolenaar 		    copy_tv(&li->li_tv, STACK_TV_BOT(-1));
14808a7d6542SBram Moolenaar 		}
14818a7d6542SBram Moolenaar 		break;
14828a7d6542SBram Moolenaar 
14838a7d6542SBram Moolenaar 	    // dict member with string key
14848a7d6542SBram Moolenaar 	    case ISN_MEMBER:
14858a7d6542SBram Moolenaar 		{
14868a7d6542SBram Moolenaar 		    dict_T	*dict;
14878a7d6542SBram Moolenaar 		    dictitem_T	*di;
14888a7d6542SBram Moolenaar 
14898a7d6542SBram Moolenaar 		    tv = STACK_TV_BOT(-1);
14908a7d6542SBram Moolenaar 		    if (tv->v_type != VAR_DICT || tv->vval.v_dict == NULL)
14918a7d6542SBram Moolenaar 		    {
14928a7d6542SBram Moolenaar 			emsg(_(e_dictreq));
14938a7d6542SBram Moolenaar 			goto failed;
14948a7d6542SBram Moolenaar 		    }
14958a7d6542SBram Moolenaar 		    dict = tv->vval.v_dict;
14968a7d6542SBram Moolenaar 
14978a7d6542SBram Moolenaar 		    if ((di = dict_find(dict, iptr->isn_arg.string, -1))
14988a7d6542SBram Moolenaar 								       == NULL)
14998a7d6542SBram Moolenaar 		    {
15008a7d6542SBram Moolenaar 			semsg(_(e_dictkey), iptr->isn_arg.string);
15018a7d6542SBram Moolenaar 			goto failed;
15028a7d6542SBram Moolenaar 		    }
15038a7d6542SBram Moolenaar 		    clear_tv(tv);
15048a7d6542SBram Moolenaar 		    copy_tv(&di->di_tv, tv);
15058a7d6542SBram Moolenaar 		}
15068a7d6542SBram Moolenaar 		break;
15078a7d6542SBram Moolenaar 
15088a7d6542SBram Moolenaar 	    case ISN_NEGATENR:
15098a7d6542SBram Moolenaar 		tv = STACK_TV_BOT(-1);
15108a7d6542SBram Moolenaar 		tv->vval.v_number = -tv->vval.v_number;
15118a7d6542SBram Moolenaar 		break;
15128a7d6542SBram Moolenaar 
15138a7d6542SBram Moolenaar 	    case ISN_CHECKNR:
15148a7d6542SBram Moolenaar 		{
15158a7d6542SBram Moolenaar 		    int		error = FALSE;
15168a7d6542SBram Moolenaar 
15178a7d6542SBram Moolenaar 		    tv = STACK_TV_BOT(-1);
15188a7d6542SBram Moolenaar 		    if (check_not_string(tv) == FAIL)
15198a7d6542SBram Moolenaar 		    {
15208a7d6542SBram Moolenaar 			--ectx.ec_stack.ga_len;
15218a7d6542SBram Moolenaar 			goto failed;
15228a7d6542SBram Moolenaar 		    }
15238a7d6542SBram Moolenaar 		    (void)tv_get_number_chk(tv, &error);
15248a7d6542SBram Moolenaar 		    if (error)
15258a7d6542SBram Moolenaar 			goto failed;
15268a7d6542SBram Moolenaar 		}
15278a7d6542SBram Moolenaar 		break;
15288a7d6542SBram Moolenaar 
15298a7d6542SBram Moolenaar 	    case ISN_CHECKTYPE:
15308a7d6542SBram Moolenaar 		{
15318a7d6542SBram Moolenaar 		    checktype_T *ct = &iptr->isn_arg.type;
15328a7d6542SBram Moolenaar 
15338a7d6542SBram Moolenaar 		    tv = STACK_TV_BOT(ct->ct_off);
15348a7d6542SBram Moolenaar 		    if (tv->v_type != ct->ct_type)
15358a7d6542SBram Moolenaar 		    {
15368a7d6542SBram Moolenaar 			semsg(_("E1029: Expected %s but got %s"),
15378a7d6542SBram Moolenaar 				    vartype_name(ct->ct_type),
15388a7d6542SBram Moolenaar 				    vartype_name(tv->v_type));
15398a7d6542SBram Moolenaar 			goto failed;
15408a7d6542SBram Moolenaar 		    }
15418a7d6542SBram Moolenaar 		}
15428a7d6542SBram Moolenaar 		break;
15438a7d6542SBram Moolenaar 
15448a7d6542SBram Moolenaar 	    case ISN_2BOOL:
15458a7d6542SBram Moolenaar 		{
15468a7d6542SBram Moolenaar 		    int n;
15478a7d6542SBram Moolenaar 
15488a7d6542SBram Moolenaar 		    tv = STACK_TV_BOT(-1);
15498a7d6542SBram Moolenaar 		    n = tv2bool(tv);
15508a7d6542SBram Moolenaar 		    if (iptr->isn_arg.number)  // invert
15518a7d6542SBram Moolenaar 			n = !n;
15528a7d6542SBram Moolenaar 		    clear_tv(tv);
15538a7d6542SBram Moolenaar 		    tv->v_type = VAR_BOOL;
15548a7d6542SBram Moolenaar 		    tv->vval.v_number = n ? VVAL_TRUE : VVAL_FALSE;
15558a7d6542SBram Moolenaar 		}
15568a7d6542SBram Moolenaar 		break;
15578a7d6542SBram Moolenaar 
15588a7d6542SBram Moolenaar 	    case ISN_2STRING:
15598a7d6542SBram Moolenaar 		{
15608a7d6542SBram Moolenaar 		    char_u *str;
15618a7d6542SBram Moolenaar 
15628a7d6542SBram Moolenaar 		    tv = STACK_TV_BOT(iptr->isn_arg.number);
15638a7d6542SBram Moolenaar 		    if (tv->v_type != VAR_STRING)
15648a7d6542SBram Moolenaar 		    {
15658a7d6542SBram Moolenaar 			str = typval_tostring(tv);
15668a7d6542SBram Moolenaar 			clear_tv(tv);
15678a7d6542SBram Moolenaar 			tv->v_type = VAR_STRING;
15688a7d6542SBram Moolenaar 			tv->vval.v_string = str;
15698a7d6542SBram Moolenaar 		    }
15708a7d6542SBram Moolenaar 		}
15718a7d6542SBram Moolenaar 		break;
15728a7d6542SBram Moolenaar 
15738a7d6542SBram Moolenaar 	    case ISN_DROP:
15748a7d6542SBram Moolenaar 		--ectx.ec_stack.ga_len;
15758a7d6542SBram Moolenaar 		clear_tv(STACK_TV_BOT(0));
15768a7d6542SBram Moolenaar 		break;
15778a7d6542SBram Moolenaar 	}
15788a7d6542SBram Moolenaar     }
15798a7d6542SBram Moolenaar 
15808a7d6542SBram Moolenaar done:
15818a7d6542SBram Moolenaar     // function finished, get result from the stack.
15828a7d6542SBram Moolenaar     tv = STACK_TV_BOT(-1);
15838a7d6542SBram Moolenaar     *rettv = *tv;
15848a7d6542SBram Moolenaar     tv->v_type = VAR_UNKNOWN;
15858a7d6542SBram Moolenaar     ret = OK;
15868a7d6542SBram Moolenaar 
15878a7d6542SBram Moolenaar failed:
15888a7d6542SBram Moolenaar     for (idx = 0; idx < ectx.ec_stack.ga_len; ++idx)
15898a7d6542SBram Moolenaar 	clear_tv(STACK_TV(idx));
15908a7d6542SBram Moolenaar     vim_free(ectx.ec_stack.ga_data);
15918a7d6542SBram Moolenaar     return ret;
15928a7d6542SBram Moolenaar }
15938a7d6542SBram Moolenaar 
15948a7d6542SBram Moolenaar /*
15958a7d6542SBram Moolenaar  * ":dissassemble".
1596777770fbSBram Moolenaar  * We don't really need this at runtime, but we do have tests that require it,
1597777770fbSBram Moolenaar  * so always include this.
15988a7d6542SBram Moolenaar  */
15998a7d6542SBram Moolenaar     void
16008a7d6542SBram Moolenaar ex_disassemble(exarg_T *eap)
16018a7d6542SBram Moolenaar {
160221456cdcSBram Moolenaar     char_u	*arg = eap->arg;
16030f18b6d1SBram Moolenaar     char_u	*fname;
16040f18b6d1SBram Moolenaar     ufunc_T	*ufunc;
16058a7d6542SBram Moolenaar     dfunc_T	*dfunc;
16068a7d6542SBram Moolenaar     isn_T	*instr;
16078a7d6542SBram Moolenaar     int		current;
16088a7d6542SBram Moolenaar     int		line_idx = 0;
16098a7d6542SBram Moolenaar     int		prev_current = 0;
16108a7d6542SBram Moolenaar 
161121456cdcSBram Moolenaar     fname = trans_function_name(&arg, FALSE,
16120f18b6d1SBram Moolenaar 	     TFN_INT | TFN_QUIET | TFN_NO_AUTOLOAD | TFN_NO_DEREF, NULL, NULL);
161321456cdcSBram Moolenaar     if (fname == NULL)
161421456cdcSBram Moolenaar     {
161521456cdcSBram Moolenaar 	semsg(_(e_invarg2), eap->arg);
161621456cdcSBram Moolenaar 	return;
161721456cdcSBram Moolenaar     }
161821456cdcSBram Moolenaar 
16190f18b6d1SBram Moolenaar     ufunc = find_func(fname, NULL);
16200f18b6d1SBram Moolenaar     vim_free(fname);
16218a7d6542SBram Moolenaar     if (ufunc == NULL)
16228a7d6542SBram Moolenaar     {
1623df2ecddfSBram Moolenaar 	semsg(_("E1061: Cannot find function %s"), eap->arg);
16248a7d6542SBram Moolenaar 	return;
16258a7d6542SBram Moolenaar     }
16268a7d6542SBram Moolenaar     if (ufunc->uf_dfunc_idx < 0)
16278a7d6542SBram Moolenaar     {
1628df2ecddfSBram Moolenaar 	semsg(_("E1062: Function %s is not compiled"), eap->arg);
16298a7d6542SBram Moolenaar 	return;
16308a7d6542SBram Moolenaar     }
16318a7d6542SBram Moolenaar     if (ufunc->uf_name_exp != NULL)
16328a7d6542SBram Moolenaar 	msg((char *)ufunc->uf_name_exp);
16338a7d6542SBram Moolenaar     else
16348a7d6542SBram Moolenaar 	msg((char *)ufunc->uf_name);
16358a7d6542SBram Moolenaar 
16368a7d6542SBram Moolenaar     dfunc = ((dfunc_T *)def_functions.ga_data) + ufunc->uf_dfunc_idx;
16378a7d6542SBram Moolenaar     instr = dfunc->df_instr;
16388a7d6542SBram Moolenaar     for (current = 0; current < dfunc->df_instr_count; ++current)
16398a7d6542SBram Moolenaar     {
16408a7d6542SBram Moolenaar 	isn_T	    *iptr = &instr[current];
1641f2460a3aSBram Moolenaar 	char	    *line;
16428a7d6542SBram Moolenaar 
16438a7d6542SBram Moolenaar 	while (line_idx < iptr->isn_lnum && line_idx < ufunc->uf_lines.ga_len)
16448a7d6542SBram Moolenaar 	{
16458a7d6542SBram Moolenaar 	    if (current > prev_current)
16468a7d6542SBram Moolenaar 	    {
16478a7d6542SBram Moolenaar 		msg_puts("\n\n");
16488a7d6542SBram Moolenaar 		prev_current = current;
16498a7d6542SBram Moolenaar 	    }
1650f2460a3aSBram Moolenaar 	    line = ((char **)ufunc->uf_lines.ga_data)[line_idx++];
1651f2460a3aSBram Moolenaar 	    if (line != NULL)
1652f2460a3aSBram Moolenaar 		msg(line);
16538a7d6542SBram Moolenaar 	}
16548a7d6542SBram Moolenaar 
16558a7d6542SBram Moolenaar 	switch (iptr->isn_type)
16568a7d6542SBram Moolenaar 	{
16578a7d6542SBram Moolenaar 	    case ISN_EXEC:
16588a7d6542SBram Moolenaar 		smsg("%4d EXEC %s", current, iptr->isn_arg.string);
16598a7d6542SBram Moolenaar 		break;
16608a7d6542SBram Moolenaar 	    case ISN_ECHO:
16618a7d6542SBram Moolenaar 		{
16628a7d6542SBram Moolenaar 		    echo_T *echo = &iptr->isn_arg.echo;
16638a7d6542SBram Moolenaar 
16648a7d6542SBram Moolenaar 		    smsg("%4d %s %d", current,
16658a7d6542SBram Moolenaar 			    echo->echo_with_white ? "ECHO" : "ECHON",
16668a7d6542SBram Moolenaar 			    echo->echo_count);
16678a7d6542SBram Moolenaar 		}
16688a7d6542SBram Moolenaar 		break;
16698a7d6542SBram Moolenaar 	    case ISN_LOAD:
16708a7d6542SBram Moolenaar 		if (iptr->isn_arg.number < 0)
16718a7d6542SBram Moolenaar 		    smsg("%4d LOAD arg[%lld]", current,
16728a7d6542SBram Moolenaar 				      iptr->isn_arg.number + STACK_FRAME_SIZE);
16738a7d6542SBram Moolenaar 		else
16748a7d6542SBram Moolenaar 		    smsg("%4d LOAD $%lld", current, iptr->isn_arg.number);
16758a7d6542SBram Moolenaar 		break;
16768a7d6542SBram Moolenaar 	    case ISN_LOADV:
16778a7d6542SBram Moolenaar 		smsg("%4d LOADV v:%s", current,
16788a7d6542SBram Moolenaar 				       get_vim_var_name(iptr->isn_arg.number));
16798a7d6542SBram Moolenaar 		break;
16808a7d6542SBram Moolenaar 	    case ISN_LOADSCRIPT:
16818a7d6542SBram Moolenaar 		{
16828a7d6542SBram Moolenaar 		    scriptitem_T *si =
168321b9e977SBram Moolenaar 				  SCRIPT_ITEM(iptr->isn_arg.script.script_sid);
16848a7d6542SBram Moolenaar 		    svar_T *sv = ((svar_T *)si->sn_var_vals.ga_data)
16858a7d6542SBram Moolenaar 					     + iptr->isn_arg.script.script_idx;
16868a7d6542SBram Moolenaar 
16878a7d6542SBram Moolenaar 		    smsg("%4d LOADSCRIPT %s from %s", current,
16888a7d6542SBram Moolenaar 						     sv->sv_name, si->sn_name);
16898a7d6542SBram Moolenaar 		}
16908a7d6542SBram Moolenaar 		break;
16918a7d6542SBram Moolenaar 	    case ISN_LOADS:
16928a7d6542SBram Moolenaar 		{
1693b283a8a6SBram Moolenaar 		    scriptitem_T *si = SCRIPT_ITEM(
1694b283a8a6SBram Moolenaar 					       iptr->isn_arg.loadstore.ls_sid);
16958a7d6542SBram Moolenaar 
16968a7d6542SBram Moolenaar 		    smsg("%4d LOADS s:%s from %s", current,
16978a7d6542SBram Moolenaar 					    iptr->isn_arg.string, si->sn_name);
16988a7d6542SBram Moolenaar 		}
16998a7d6542SBram Moolenaar 		break;
17008a7d6542SBram Moolenaar 	    case ISN_LOADG:
17018a7d6542SBram Moolenaar 		smsg("%4d LOADG g:%s", current, iptr->isn_arg.string);
17028a7d6542SBram Moolenaar 		break;
17038a7d6542SBram Moolenaar 	    case ISN_LOADOPT:
17048a7d6542SBram Moolenaar 		smsg("%4d LOADOPT %s", current, iptr->isn_arg.string);
17058a7d6542SBram Moolenaar 		break;
17068a7d6542SBram Moolenaar 	    case ISN_LOADENV:
17078a7d6542SBram Moolenaar 		smsg("%4d LOADENV %s", current, iptr->isn_arg.string);
17088a7d6542SBram Moolenaar 		break;
17098a7d6542SBram Moolenaar 	    case ISN_LOADREG:
17108a7d6542SBram Moolenaar 		smsg("%4d LOADREG @%c", current, iptr->isn_arg.number);
17118a7d6542SBram Moolenaar 		break;
17128a7d6542SBram Moolenaar 
17138a7d6542SBram Moolenaar 	    case ISN_STORE:
1714170fcfcfSBram Moolenaar 		if (iptr->isn_arg.number < 0)
1715170fcfcfSBram Moolenaar 		    smsg("%4d STORE arg[%lld]", current,
1716170fcfcfSBram Moolenaar 				      iptr->isn_arg.number + STACK_FRAME_SIZE);
1717170fcfcfSBram Moolenaar 		else
17188a7d6542SBram Moolenaar 		    smsg("%4d STORE $%lld", current, iptr->isn_arg.number);
17198a7d6542SBram Moolenaar 		break;
1720b283a8a6SBram Moolenaar 	    case ISN_STOREV:
1721b283a8a6SBram Moolenaar 		smsg("%4d STOREV v:%s", current,
1722b283a8a6SBram Moolenaar 				       get_vim_var_name(iptr->isn_arg.number));
1723b283a8a6SBram Moolenaar 		break;
17248a7d6542SBram Moolenaar 	    case ISN_STOREG:
1725b283a8a6SBram Moolenaar 		smsg("%4d STOREG %s", current, iptr->isn_arg.string);
1726b283a8a6SBram Moolenaar 		break;
1727b283a8a6SBram Moolenaar 	    case ISN_STORES:
1728b283a8a6SBram Moolenaar 		{
1729b283a8a6SBram Moolenaar 		    scriptitem_T *si = SCRIPT_ITEM(
1730b283a8a6SBram Moolenaar 					       iptr->isn_arg.loadstore.ls_sid);
1731b283a8a6SBram Moolenaar 
17320bbf722aSBram Moolenaar 		    smsg("%4d STORES %s in %s", current,
1733b283a8a6SBram Moolenaar 					    iptr->isn_arg.string, si->sn_name);
1734b283a8a6SBram Moolenaar 		}
17358a7d6542SBram Moolenaar 		break;
17368a7d6542SBram Moolenaar 	    case ISN_STORESCRIPT:
17378a7d6542SBram Moolenaar 		{
17388a7d6542SBram Moolenaar 		    scriptitem_T *si =
173921b9e977SBram Moolenaar 				  SCRIPT_ITEM(iptr->isn_arg.script.script_sid);
17408a7d6542SBram Moolenaar 		    svar_T *sv = ((svar_T *)si->sn_var_vals.ga_data)
17418a7d6542SBram Moolenaar 					     + iptr->isn_arg.script.script_idx;
17428a7d6542SBram Moolenaar 
17438a7d6542SBram Moolenaar 		    smsg("%4d STORESCRIPT %s in %s", current,
17448a7d6542SBram Moolenaar 						     sv->sv_name, si->sn_name);
17458a7d6542SBram Moolenaar 		}
17468a7d6542SBram Moolenaar 		break;
17478a7d6542SBram Moolenaar 	    case ISN_STOREOPT:
17488a7d6542SBram Moolenaar 		smsg("%4d STOREOPT &%s", current,
17498a7d6542SBram Moolenaar 					       iptr->isn_arg.storeopt.so_name);
17508a7d6542SBram Moolenaar 		break;
1751b283a8a6SBram Moolenaar 	    case ISN_STOREENV:
1752b283a8a6SBram Moolenaar 		smsg("%4d STOREENV $%s", current, iptr->isn_arg.string);
1753b283a8a6SBram Moolenaar 		break;
1754b283a8a6SBram Moolenaar 	    case ISN_STOREREG:
1755b283a8a6SBram Moolenaar 		smsg("%4d STOREREG @%c", current, iptr->isn_arg.number);
1756b283a8a6SBram Moolenaar 		break;
17578a7d6542SBram Moolenaar 	    case ISN_STORENR:
17588a7d6542SBram Moolenaar 		smsg("%4d STORE %lld in $%d", current,
17598a7d6542SBram Moolenaar 				iptr->isn_arg.storenr.str_val,
17608a7d6542SBram Moolenaar 				iptr->isn_arg.storenr.str_idx);
17618a7d6542SBram Moolenaar 		break;
17628a7d6542SBram Moolenaar 
17638a7d6542SBram Moolenaar 	    // constants
17648a7d6542SBram Moolenaar 	    case ISN_PUSHNR:
17658a7d6542SBram Moolenaar 		smsg("%4d PUSHNR %lld", current, iptr->isn_arg.number);
17668a7d6542SBram Moolenaar 		break;
17678a7d6542SBram Moolenaar 	    case ISN_PUSHBOOL:
17688a7d6542SBram Moolenaar 	    case ISN_PUSHSPEC:
17698a7d6542SBram Moolenaar 		smsg("%4d PUSH %s", current,
17708a7d6542SBram Moolenaar 				   get_var_special_name(iptr->isn_arg.number));
17718a7d6542SBram Moolenaar 		break;
17728a7d6542SBram Moolenaar 	    case ISN_PUSHF:
1773a5d5953dSBram Moolenaar #ifdef FEAT_FLOAT
17748a7d6542SBram Moolenaar 		smsg("%4d PUSHF %g", current, iptr->isn_arg.fnumber);
1775a5d5953dSBram Moolenaar #endif
17768a7d6542SBram Moolenaar 		break;
17778a7d6542SBram Moolenaar 	    case ISN_PUSHS:
17788a7d6542SBram Moolenaar 		smsg("%4d PUSHS \"%s\"", current, iptr->isn_arg.string);
17798a7d6542SBram Moolenaar 		break;
17808a7d6542SBram Moolenaar 	    case ISN_PUSHBLOB:
17818a7d6542SBram Moolenaar 		{
17828a7d6542SBram Moolenaar 		    char_u	*r;
17838a7d6542SBram Moolenaar 		    char_u	numbuf[NUMBUFLEN];
17848a7d6542SBram Moolenaar 		    char_u	*tofree;
17858a7d6542SBram Moolenaar 
17868a7d6542SBram Moolenaar 		    r = blob2string(iptr->isn_arg.blob, &tofree, numbuf);
1787ff80cb68SBram Moolenaar 		    smsg("%4d PUSHBLOB %s", current, r);
17888a7d6542SBram Moolenaar 		    vim_free(tofree);
17898a7d6542SBram Moolenaar 		}
17908a7d6542SBram Moolenaar 		break;
17918a7d6542SBram Moolenaar 	    case ISN_PUSHEXC:
17928a7d6542SBram Moolenaar 		smsg("%4d PUSH v:exception", current);
17938a7d6542SBram Moolenaar 		break;
17948a7d6542SBram Moolenaar 	    case ISN_NEWLIST:
17958a7d6542SBram Moolenaar 		smsg("%4d NEWLIST size %lld", current, iptr->isn_arg.number);
17968a7d6542SBram Moolenaar 		break;
17978a7d6542SBram Moolenaar 	    case ISN_NEWDICT:
17988a7d6542SBram Moolenaar 		smsg("%4d NEWDICT size %lld", current, iptr->isn_arg.number);
17998a7d6542SBram Moolenaar 		break;
18008a7d6542SBram Moolenaar 
18018a7d6542SBram Moolenaar 	    // function call
18028a7d6542SBram Moolenaar 	    case ISN_BCALL:
18038a7d6542SBram Moolenaar 		{
18048a7d6542SBram Moolenaar 		    cbfunc_T	*cbfunc = &iptr->isn_arg.bfunc;
18058a7d6542SBram Moolenaar 
18068a7d6542SBram Moolenaar 		    smsg("%4d BCALL %s(argc %d)", current,
18078a7d6542SBram Moolenaar 			    internal_func_name(cbfunc->cbf_idx),
18088a7d6542SBram Moolenaar 			    cbfunc->cbf_argcount);
18098a7d6542SBram Moolenaar 		}
18108a7d6542SBram Moolenaar 		break;
18118a7d6542SBram Moolenaar 	    case ISN_DCALL:
18128a7d6542SBram Moolenaar 		{
18138a7d6542SBram Moolenaar 		    cdfunc_T	*cdfunc = &iptr->isn_arg.dfunc;
18148a7d6542SBram Moolenaar 		    dfunc_T	*df = ((dfunc_T *)def_functions.ga_data)
18158a7d6542SBram Moolenaar 							     + cdfunc->cdf_idx;
18168a7d6542SBram Moolenaar 
18178a7d6542SBram Moolenaar 		    smsg("%4d DCALL %s(argc %d)", current,
18188a7d6542SBram Moolenaar 			    df->df_ufunc->uf_name_exp != NULL
18198a7d6542SBram Moolenaar 				? df->df_ufunc->uf_name_exp
18208a7d6542SBram Moolenaar 				: df->df_ufunc->uf_name, cdfunc->cdf_argcount);
18218a7d6542SBram Moolenaar 		}
18228a7d6542SBram Moolenaar 		break;
18238a7d6542SBram Moolenaar 	    case ISN_UCALL:
18248a7d6542SBram Moolenaar 		{
18258a7d6542SBram Moolenaar 		    cufunc_T	*cufunc = &iptr->isn_arg.ufunc;
18268a7d6542SBram Moolenaar 
18278a7d6542SBram Moolenaar 		    smsg("%4d UCALL %s(argc %d)", current,
18288a7d6542SBram Moolenaar 				       cufunc->cuf_name, cufunc->cuf_argcount);
18298a7d6542SBram Moolenaar 		}
18308a7d6542SBram Moolenaar 		break;
18318a7d6542SBram Moolenaar 	    case ISN_PCALL:
18328a7d6542SBram Moolenaar 		{
18338a7d6542SBram Moolenaar 		    cpfunc_T	*cpfunc = &iptr->isn_arg.pfunc;
18348a7d6542SBram Moolenaar 
18358a7d6542SBram Moolenaar 		    smsg("%4d PCALL%s (argc %d)", current,
18368a7d6542SBram Moolenaar 			   cpfunc->cpf_top ? " top" : "", cpfunc->cpf_argcount);
18378a7d6542SBram Moolenaar 		}
18388a7d6542SBram Moolenaar 		break;
18398a7d6542SBram Moolenaar 	    case ISN_RETURN:
18408a7d6542SBram Moolenaar 		smsg("%4d RETURN", current);
18418a7d6542SBram Moolenaar 		break;
18428a7d6542SBram Moolenaar 	    case ISN_FUNCREF:
18438a7d6542SBram Moolenaar 		{
18448a7d6542SBram Moolenaar 		    dfunc_T	*df = ((dfunc_T *)def_functions.ga_data)
18458a7d6542SBram Moolenaar 							+ iptr->isn_arg.number;
18468a7d6542SBram Moolenaar 
18478a7d6542SBram Moolenaar 		    smsg("%4d FUNCREF %s", current, df->df_ufunc->uf_name);
18488a7d6542SBram Moolenaar 		}
18498a7d6542SBram Moolenaar 		break;
18508a7d6542SBram Moolenaar 
18518a7d6542SBram Moolenaar 	    case ISN_JUMP:
18528a7d6542SBram Moolenaar 		{
18538a7d6542SBram Moolenaar 		    char *when = "?";
18548a7d6542SBram Moolenaar 
18558a7d6542SBram Moolenaar 		    switch (iptr->isn_arg.jump.jump_when)
18568a7d6542SBram Moolenaar 		    {
18578a7d6542SBram Moolenaar 			case JUMP_ALWAYS:
18588a7d6542SBram Moolenaar 			    when = "JUMP";
18598a7d6542SBram Moolenaar 			    break;
18608a7d6542SBram Moolenaar 			case JUMP_AND_KEEP_IF_TRUE:
18618a7d6542SBram Moolenaar 			    when = "JUMP_AND_KEEP_IF_TRUE";
18628a7d6542SBram Moolenaar 			    break;
18638a7d6542SBram Moolenaar 			case JUMP_IF_FALSE:
18648a7d6542SBram Moolenaar 			    when = "JUMP_IF_FALSE";
18658a7d6542SBram Moolenaar 			    break;
18668a7d6542SBram Moolenaar 			case JUMP_AND_KEEP_IF_FALSE:
18678a7d6542SBram Moolenaar 			    when = "JUMP_AND_KEEP_IF_FALSE";
18688a7d6542SBram Moolenaar 			    break;
18698a7d6542SBram Moolenaar 		    }
18708a7d6542SBram Moolenaar 		    smsg("%4d %s -> %lld", current, when,
18718a7d6542SBram Moolenaar 						iptr->isn_arg.jump.jump_where);
18728a7d6542SBram Moolenaar 		}
18738a7d6542SBram Moolenaar 		break;
18748a7d6542SBram Moolenaar 
18758a7d6542SBram Moolenaar 	    case ISN_FOR:
18768a7d6542SBram Moolenaar 		{
18778a7d6542SBram Moolenaar 		    forloop_T *forloop = &iptr->isn_arg.forloop;
18788a7d6542SBram Moolenaar 
18798a7d6542SBram Moolenaar 		    smsg("%4d FOR $%d -> %d", current,
18808a7d6542SBram Moolenaar 					   forloop->for_idx, forloop->for_end);
18818a7d6542SBram Moolenaar 		}
18828a7d6542SBram Moolenaar 		break;
18838a7d6542SBram Moolenaar 
18848a7d6542SBram Moolenaar 	    case ISN_TRY:
18858a7d6542SBram Moolenaar 		{
18868a7d6542SBram Moolenaar 		    try_T *try = &iptr->isn_arg.try;
18878a7d6542SBram Moolenaar 
18888a7d6542SBram Moolenaar 		    smsg("%4d TRY catch -> %d, finally -> %d", current,
18898a7d6542SBram Moolenaar 					     try->try_catch, try->try_finally);
18908a7d6542SBram Moolenaar 		}
18918a7d6542SBram Moolenaar 		break;
18928a7d6542SBram Moolenaar 	    case ISN_CATCH:
18938a7d6542SBram Moolenaar 		// TODO
18948a7d6542SBram Moolenaar 		smsg("%4d CATCH", current);
18958a7d6542SBram Moolenaar 		break;
18968a7d6542SBram Moolenaar 	    case ISN_ENDTRY:
18978a7d6542SBram Moolenaar 		smsg("%4d ENDTRY", current);
18988a7d6542SBram Moolenaar 		break;
18998a7d6542SBram Moolenaar 	    case ISN_THROW:
19008a7d6542SBram Moolenaar 		smsg("%4d THROW", current);
19018a7d6542SBram Moolenaar 		break;
19028a7d6542SBram Moolenaar 
19038a7d6542SBram Moolenaar 	    // expression operations on number
19048a7d6542SBram Moolenaar 	    case ISN_OPNR:
19058a7d6542SBram Moolenaar 	    case ISN_OPFLOAT:
19068a7d6542SBram Moolenaar 	    case ISN_OPANY:
19078a7d6542SBram Moolenaar 		{
19088a7d6542SBram Moolenaar 		    char *what;
19098a7d6542SBram Moolenaar 		    char *ins;
19108a7d6542SBram Moolenaar 
19118a7d6542SBram Moolenaar 		    switch (iptr->isn_arg.op.op_type)
19128a7d6542SBram Moolenaar 		    {
19138a7d6542SBram Moolenaar 			case EXPR_MULT: what = "*"; break;
19148a7d6542SBram Moolenaar 			case EXPR_DIV: what = "/"; break;
19158a7d6542SBram Moolenaar 			case EXPR_REM: what = "%"; break;
19168a7d6542SBram Moolenaar 			case EXPR_SUB: what = "-"; break;
19178a7d6542SBram Moolenaar 			case EXPR_ADD: what = "+"; break;
19188a7d6542SBram Moolenaar 			default:       what = "???"; break;
19198a7d6542SBram Moolenaar 		    }
19208a7d6542SBram Moolenaar 		    switch (iptr->isn_type)
19218a7d6542SBram Moolenaar 		    {
19228a7d6542SBram Moolenaar 			case ISN_OPNR: ins = "OPNR"; break;
19238a7d6542SBram Moolenaar 			case ISN_OPFLOAT: ins = "OPFLOAT"; break;
19248a7d6542SBram Moolenaar 			case ISN_OPANY: ins = "OPANY"; break;
19258a7d6542SBram Moolenaar 			default: ins = "???"; break;
19268a7d6542SBram Moolenaar 		    }
19278a7d6542SBram Moolenaar 		    smsg("%4d %s %s", current, ins, what);
19288a7d6542SBram Moolenaar 		}
19298a7d6542SBram Moolenaar 		break;
19308a7d6542SBram Moolenaar 
19318a7d6542SBram Moolenaar 	    case ISN_COMPAREBOOL:
19328a7d6542SBram Moolenaar 	    case ISN_COMPARESPECIAL:
19338a7d6542SBram Moolenaar 	    case ISN_COMPARENR:
19348a7d6542SBram Moolenaar 	    case ISN_COMPAREFLOAT:
19358a7d6542SBram Moolenaar 	    case ISN_COMPARESTRING:
19368a7d6542SBram Moolenaar 	    case ISN_COMPAREBLOB:
19378a7d6542SBram Moolenaar 	    case ISN_COMPARELIST:
19388a7d6542SBram Moolenaar 	    case ISN_COMPAREDICT:
19398a7d6542SBram Moolenaar 	    case ISN_COMPAREFUNC:
19408a7d6542SBram Moolenaar 	    case ISN_COMPAREPARTIAL:
19418a7d6542SBram Moolenaar 	    case ISN_COMPAREANY:
19428a7d6542SBram Moolenaar 		   {
19438a7d6542SBram Moolenaar 		       char *p;
19448a7d6542SBram Moolenaar 		       char buf[10];
19458a7d6542SBram Moolenaar 		       char *type;
19468a7d6542SBram Moolenaar 
19478a7d6542SBram Moolenaar 		       switch (iptr->isn_arg.op.op_type)
19488a7d6542SBram Moolenaar 		       {
19498a7d6542SBram Moolenaar 			   case EXPR_EQUAL:	 p = "=="; break;
19508a7d6542SBram Moolenaar 			   case EXPR_NEQUAL:    p = "!="; break;
19518a7d6542SBram Moolenaar 			   case EXPR_GREATER:   p = ">"; break;
19528a7d6542SBram Moolenaar 			   case EXPR_GEQUAL:    p = ">="; break;
19538a7d6542SBram Moolenaar 			   case EXPR_SMALLER:   p = "<"; break;
19548a7d6542SBram Moolenaar 			   case EXPR_SEQUAL:    p = "<="; break;
19558a7d6542SBram Moolenaar 			   case EXPR_MATCH:	 p = "=~"; break;
19568a7d6542SBram Moolenaar 			   case EXPR_IS:	 p = "is"; break;
19578a7d6542SBram Moolenaar 			   case EXPR_ISNOT:	 p = "isnot"; break;
19588a7d6542SBram Moolenaar 			   case EXPR_NOMATCH:	 p = "!~"; break;
19598a7d6542SBram Moolenaar 			   default:  p = "???"; break;
19608a7d6542SBram Moolenaar 		       }
19618a7d6542SBram Moolenaar 		       STRCPY(buf, p);
19628a7d6542SBram Moolenaar 		       if (iptr->isn_arg.op.op_ic == TRUE)
19638a7d6542SBram Moolenaar 			   strcat(buf, "?");
19648a7d6542SBram Moolenaar 		       switch(iptr->isn_type)
19658a7d6542SBram Moolenaar 		       {
19668a7d6542SBram Moolenaar 			   case ISN_COMPAREBOOL: type = "COMPAREBOOL"; break;
19678a7d6542SBram Moolenaar 			   case ISN_COMPARESPECIAL:
19688a7d6542SBram Moolenaar 						 type = "COMPARESPECIAL"; break;
19698a7d6542SBram Moolenaar 			   case ISN_COMPARENR: type = "COMPARENR"; break;
19708a7d6542SBram Moolenaar 			   case ISN_COMPAREFLOAT: type = "COMPAREFLOAT"; break;
19718a7d6542SBram Moolenaar 			   case ISN_COMPARESTRING:
19728a7d6542SBram Moolenaar 						  type = "COMPARESTRING"; break;
19738a7d6542SBram Moolenaar 			   case ISN_COMPAREBLOB: type = "COMPAREBLOB"; break;
19748a7d6542SBram Moolenaar 			   case ISN_COMPARELIST: type = "COMPARELIST"; break;
19758a7d6542SBram Moolenaar 			   case ISN_COMPAREDICT: type = "COMPAREDICT"; break;
19768a7d6542SBram Moolenaar 			   case ISN_COMPAREFUNC: type = "COMPAREFUNC"; break;
19778a7d6542SBram Moolenaar 			   case ISN_COMPAREPARTIAL:
19788a7d6542SBram Moolenaar 						 type = "COMPAREPARTIAL"; break;
19798a7d6542SBram Moolenaar 			   case ISN_COMPAREANY: type = "COMPAREANY"; break;
19808a7d6542SBram Moolenaar 			   default: type = "???"; break;
19818a7d6542SBram Moolenaar 		       }
19828a7d6542SBram Moolenaar 
19838a7d6542SBram Moolenaar 		       smsg("%4d %s %s", current, type, buf);
19848a7d6542SBram Moolenaar 		   }
19858a7d6542SBram Moolenaar 		   break;
19868a7d6542SBram Moolenaar 
19878a7d6542SBram Moolenaar 	    case ISN_ADDLIST: smsg("%4d ADDLIST", current); break;
19888a7d6542SBram Moolenaar 	    case ISN_ADDBLOB: smsg("%4d ADDBLOB", current); break;
19898a7d6542SBram Moolenaar 
19908a7d6542SBram Moolenaar 	    // expression operations
19918a7d6542SBram Moolenaar 	    case ISN_CONCAT: smsg("%4d CONCAT", current); break;
19928a7d6542SBram Moolenaar 	    case ISN_INDEX: smsg("%4d INDEX", current); break;
19938a7d6542SBram Moolenaar 	    case ISN_MEMBER: smsg("%4d MEMBER %s", current,
19948a7d6542SBram Moolenaar 						  iptr->isn_arg.string); break;
19958a7d6542SBram Moolenaar 	    case ISN_NEGATENR: smsg("%4d NEGATENR", current); break;
19968a7d6542SBram Moolenaar 
19978a7d6542SBram Moolenaar 	    case ISN_CHECKNR: smsg("%4d CHECKNR", current); break;
19988a7d6542SBram Moolenaar 	    case ISN_CHECKTYPE: smsg("%4d CHECKTYPE %s stack[%d]", current,
19998a7d6542SBram Moolenaar 				      vartype_name(iptr->isn_arg.type.ct_type),
20008a7d6542SBram Moolenaar 				      iptr->isn_arg.type.ct_off);
20018a7d6542SBram Moolenaar 				break;
20028a7d6542SBram Moolenaar 	    case ISN_2BOOL: if (iptr->isn_arg.number)
20038a7d6542SBram Moolenaar 				smsg("%4d INVERT (!val)", current);
20048a7d6542SBram Moolenaar 			    else
20058a7d6542SBram Moolenaar 				smsg("%4d 2BOOL (!!val)", current);
20068a7d6542SBram Moolenaar 			    break;
20078a7d6542SBram Moolenaar 	    case ISN_2STRING: smsg("%4d 2STRING stack[%d]", current,
20088a7d6542SBram Moolenaar 							 iptr->isn_arg.number);
20098a7d6542SBram Moolenaar 				break;
20108a7d6542SBram Moolenaar 
20118a7d6542SBram Moolenaar 	    case ISN_DROP: smsg("%4d DROP", current); break;
20128a7d6542SBram Moolenaar 	}
20138a7d6542SBram Moolenaar     }
20148a7d6542SBram Moolenaar }
20158a7d6542SBram Moolenaar 
20168a7d6542SBram Moolenaar /*
20178a7d6542SBram Moolenaar  * Return TRUE when "tv" is not falsey: non-zero, non-empty string, non-empty
20188a7d6542SBram Moolenaar  * list, etc.  Mostly like what JavaScript does, except that empty list and
20198a7d6542SBram Moolenaar  * empty dictionary are FALSE.
20208a7d6542SBram Moolenaar  */
20218a7d6542SBram Moolenaar     int
20228a7d6542SBram Moolenaar tv2bool(typval_T *tv)
20238a7d6542SBram Moolenaar {
20248a7d6542SBram Moolenaar     switch (tv->v_type)
20258a7d6542SBram Moolenaar     {
20268a7d6542SBram Moolenaar 	case VAR_NUMBER:
20278a7d6542SBram Moolenaar 	    return tv->vval.v_number != 0;
20288a7d6542SBram Moolenaar 	case VAR_FLOAT:
20298a7d6542SBram Moolenaar #ifdef FEAT_FLOAT
20308a7d6542SBram Moolenaar 	    return tv->vval.v_float != 0.0;
20318a7d6542SBram Moolenaar #else
20328a7d6542SBram Moolenaar 	    break;
20338a7d6542SBram Moolenaar #endif
20348a7d6542SBram Moolenaar 	case VAR_PARTIAL:
20358a7d6542SBram Moolenaar 	    return tv->vval.v_partial != NULL;
20368a7d6542SBram Moolenaar 	case VAR_FUNC:
20378a7d6542SBram Moolenaar 	case VAR_STRING:
20388a7d6542SBram Moolenaar 	    return tv->vval.v_string != NULL && *tv->vval.v_string != NUL;
20398a7d6542SBram Moolenaar 	case VAR_LIST:
20408a7d6542SBram Moolenaar 	    return tv->vval.v_list != NULL && tv->vval.v_list->lv_len > 0;
20418a7d6542SBram Moolenaar 	case VAR_DICT:
20428a7d6542SBram Moolenaar 	    return tv->vval.v_dict != NULL
20438a7d6542SBram Moolenaar 				    && tv->vval.v_dict->dv_hashtab.ht_used > 0;
20448a7d6542SBram Moolenaar 	case VAR_BOOL:
20458a7d6542SBram Moolenaar 	case VAR_SPECIAL:
20468a7d6542SBram Moolenaar 	    return tv->vval.v_number == VVAL_TRUE ? TRUE : FALSE;
20478a7d6542SBram Moolenaar 	case VAR_JOB:
20488a7d6542SBram Moolenaar #ifdef FEAT_JOB_CHANNEL
20498a7d6542SBram Moolenaar 	    return tv->vval.v_job != NULL;
20508a7d6542SBram Moolenaar #else
20518a7d6542SBram Moolenaar 	    break;
20528a7d6542SBram Moolenaar #endif
20538a7d6542SBram Moolenaar 	case VAR_CHANNEL:
20548a7d6542SBram Moolenaar #ifdef FEAT_JOB_CHANNEL
20558a7d6542SBram Moolenaar 	    return tv->vval.v_channel != NULL;
20568a7d6542SBram Moolenaar #else
20578a7d6542SBram Moolenaar 	    break;
20588a7d6542SBram Moolenaar #endif
20598a7d6542SBram Moolenaar 	case VAR_BLOB:
20608a7d6542SBram Moolenaar 	    return tv->vval.v_blob != NULL && tv->vval.v_blob->bv_ga.ga_len > 0;
20618a7d6542SBram Moolenaar 	case VAR_UNKNOWN:
20628a7d6542SBram Moolenaar 	case VAR_VOID:
20638a7d6542SBram Moolenaar 	    break;
20648a7d6542SBram Moolenaar     }
20658a7d6542SBram Moolenaar     return FALSE;
20668a7d6542SBram Moolenaar }
20678a7d6542SBram Moolenaar 
20688a7d6542SBram Moolenaar /*
20698a7d6542SBram Moolenaar  * If "tv" is a string give an error and return FAIL.
20708a7d6542SBram Moolenaar  */
20718a7d6542SBram Moolenaar     int
20728a7d6542SBram Moolenaar check_not_string(typval_T *tv)
20738a7d6542SBram Moolenaar {
20748a7d6542SBram Moolenaar     if (tv->v_type == VAR_STRING)
20758a7d6542SBram Moolenaar     {
20768a7d6542SBram Moolenaar 	emsg(_("E1030: Using a String as a Number"));
20778a7d6542SBram Moolenaar 	clear_tv(tv);
20788a7d6542SBram Moolenaar 	return FAIL;
20798a7d6542SBram Moolenaar     }
20808a7d6542SBram Moolenaar     return OK;
20818a7d6542SBram Moolenaar }
20828a7d6542SBram Moolenaar 
20838a7d6542SBram Moolenaar 
20848a7d6542SBram Moolenaar #endif // FEAT_EVAL
2085