]> granicus.if.org Git - vim/commitdiff
patch 8.2.3147: Vim9: profiling does not work with a nested function v8.2.3147
authorBram Moolenaar <Bram@vim.org>
Sun, 11 Jul 2021 15:55:01 +0000 (17:55 +0200)
committerBram Moolenaar <Bram@vim.org>
Sun, 11 Jul 2021 15:55:01 +0000 (17:55 +0200)
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.

src/testdir/test_vim9_script.vim
src/version.c
src/vim9compile.c
src/vim9execute.c

index 0602b2ae5baf5e4e0dea3f6d6ef07a9d4ca543ce..10f41c7bf8e1ce480f0d0bc5b05e588b10e2f2b6 100644 (file)
@@ -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.
index 8e1d88e3e9634b68ca97478e07d45988da968713..993d1631855d8e653035b39813d3cb1f3f69a61e 100644 (file)
@@ -755,6 +755,8 @@ static char *(features[]) =
 
 static int included_patches[] =
 {   /* Add new patch number below this line */
+/**/
+    3147,
 /**/
     3146,
 /**/
index 9aa11f6210af10edb186f7987045e968b761783f..b04e750ef564e8172ec32d31ab8e00f3611477c8 100644 (file)
@@ -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,
index 87f3c424ea311d73bf29d898dbf0fea9b13d1d97..41249284dc0f7c5474b6a7999e6e4916d3f7a043 100644 (file)
@@ -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),