summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorBram Moolenaar <Bram@vim.org>2021-07-11 17:55:01 +0200
committerBram Moolenaar <Bram@vim.org>2021-07-11 17:55:01 +0200
commit648594eaf703fe9a862cb12a35702a10aff6e5a9 (patch)
tree10c97df4f865c248292f255d496d794ecedc5ec5
parentc03fe66ade4c79a4eb5fc05d1d549c8f931a04b6 (diff)
patch 8.2.3147: Vim9: profiling does not work with a nested functionv8.2.3147
Problem: Vim9: profiling does not work with a nested function. Solution: Also compile a nested function without profiling. (closes #8543) Handle that compiling may cause the table of compiled functions to change.
-rw-r--r--src/testdir/test_vim9_script.vim16
-rw-r--r--src/version.c2
-rw-r--r--src/vim9compile.c10
-rw-r--r--src/vim9execute.c21
4 files changed, 34 insertions, 15 deletions
diff --git a/src/testdir/test_vim9_script.vim b/src/testdir/test_vim9_script.vim
index 0602b2ae5b..10f41c7bf8 100644
--- a/src/testdir/test_vim9_script.vim
+++ b/src/testdir/test_vim9_script.vim
@@ -4177,19 +4177,29 @@ def Test_xxx_echoerr_line_number()
CheckDefExecAndScriptFailure(lines, 'some error continued', 1)
enddef
-def ProfiledFunc()
+def ProfiledWithLambda()
var n = 3
echo [[1, 2], [3, 4]]->filter((_, l) => l[0] == n)
enddef
+def ProfiledNested()
+ var x = 0
+ def Nested(): any
+ return x
+ enddef
+ Nested()
+enddef
+
" Execute this near the end, profiling doesn't stop until Vim exists.
" This only tests that it works, not the profiling output.
def Test_xx_profile_with_lambda()
CheckFeature profile
profile start Xprofile.log
- profile func ProfiledFunc
- ProfiledFunc()
+ profile func ProfiledWithLambda
+ ProfiledWithLambda()
+ profile func ProfiledNested
+ ProfiledNested()
enddef
" Keep this last, it messes up highlighting.
diff --git a/src/version.c b/src/version.c
index 8e1d88e3e9..993d163185 100644
--- a/src/version.c
+++ b/src/version.c
@@ -756,6 +756,8 @@ static char *(features[]) =
static int included_patches[] =
{ /* Add new patch number below this line */
/**/
+ 3147,
+/**/
3146,
/**/
3145,
diff --git a/src/vim9compile.c b/src/vim9compile.c
index 9aa11f6210..b04e750ef5 100644
--- a/src/vim9compile.c
+++ b/src/vim9compile.c
@@ -3624,10 +3624,12 @@ compile_lambda(char_u **arg, cctx_T *cctx)
ufunc->uf_ret_type = &t_unknown;
compile_def_function(ufunc, FALSE, cctx->ctx_compile_type, cctx);
+#ifdef FEAT_PROFILE
// When the outer function is compiled for profiling, the lambda may be
// called without profiling. Compile it here in the right context.
if (cctx->ctx_compile_type == CT_PROFILE)
compile_def_function(ufunc, FALSE, CT_NONE, cctx);
+#endif
// evalarg.eval_tofree_cmdline may have a copy of the last line and "*arg"
// points into it. Point to the original line to avoid a dangling pointer.
@@ -5631,6 +5633,14 @@ compile_nested_function(exarg_T *eap, cctx_T *cctx)
goto theend;
}
+#ifdef FEAT_PROFILE
+ // When the outer function is compiled for profiling, the nested function
+ // may be called without profiling. Compile it here in the right context.
+ if (COMPILE_TYPE(ufunc) == CT_PROFILE
+ && func_needs_compiling(ufunc, CT_NONE))
+ compile_def_function(ufunc, FALSE, CT_NONE, cctx);
+#endif
+
if (is_global)
{
char_u *func_name = vim_strnsave(name_start + 2,
diff --git a/src/vim9execute.c b/src/vim9execute.c
index 87f3c424ea..41249284dc 100644
--- a/src/vim9execute.c
+++ b/src/vim9execute.c
@@ -197,6 +197,7 @@ call_dfunc(
int idx;
estack_T *entry;
funclocal_T *floc = NULL;
+ int res = OK;
if (dfunc->df_deleted)
{
@@ -219,14 +220,6 @@ call_dfunc(
(((dfunc_T *)def_functions.ga_data)
+ ectx->ec_dfunc_idx)->df_ufunc);
}
-
- // Profiling might be enabled/disabled along the way. This should not
- // fail, since the function was compiled before and toggling profiling
- // doesn't change any errors.
- if (func_needs_compiling(ufunc, COMPILE_TYPE(ufunc))
- && compile_def_function(ufunc, FALSE, COMPILE_TYPE(ufunc), NULL)
- == FAIL)
- return FAIL;
}
#endif
@@ -235,10 +228,14 @@ call_dfunc(
// When debugging and using "cont" switches to the not-debugged
// instructions, may need to still compile them.
- if ((func_needs_compiling(ufunc, COMPILE_TYPE(ufunc))
- && compile_def_function(ufunc, FALSE, COMPILE_TYPE(ufunc), NULL)
- == FAIL)
- || INSTRUCTIONS(dfunc) == NULL)
+ if (func_needs_compiling(ufunc, COMPILE_TYPE(ufunc)))
+ {
+ res = compile_def_function(ufunc, FALSE, COMPILE_TYPE(ufunc), NULL);
+
+ // compile_def_function() may cause def_functions.ga_data to change
+ dfunc = ((dfunc_T *)def_functions.ga_data) + cdf_idx;
+ }
+ if (res == FAIL || INSTRUCTIONS(dfunc) == NULL)
{
if (did_emsg_cumul + did_emsg == did_emsg_before)
semsg(_(e_function_is_not_compiled_str),