summaryrefslogtreecommitdiff
path: root/src/scriptfile.c
diff options
context:
space:
mode:
authorYegappan Lakshmanan <yegappan@yahoo.com>2022-03-21 19:45:17 +0000
committerBram Moolenaar <Bram@vim.org>2022-03-21 19:45:17 +0000
commit85b43c6cb7d56919e245622f4e42db6d8bee4194 (patch)
tree96162d04de08fc6d8a9ae1aa02347f12d6206642 /src/scriptfile.c
parent397a87ac1c9857eb0d5ec3add69e3b9ab6b0c77c (diff)
downloadvim-git-85b43c6cb7d56919e245622f4e42db6d8bee4194.tar.gz
patch 8.2.4603: sourcing buffer lines is too complicatedv8.2.4603
Problem: Sourcing buffer lines is too complicated. Solution: Simplify the code. Make it possible to source Vim9 script lines. (Yegappan Lakshmanan, closes #9974)
Diffstat (limited to 'src/scriptfile.c')
-rw-r--r--src/scriptfile.c414
1 files changed, 134 insertions, 280 deletions
diff --git a/src/scriptfile.c b/src/scriptfile.c
index 977884342..3faed4a70 100644
--- a/src/scriptfile.c
+++ b/src/scriptfile.c
@@ -23,6 +23,8 @@ static garray_T ga_loaded = {0, 0, sizeof(char_u *), 4, NULL};
static int last_current_SID_seq = 0;
#endif
+static int do_source_ext(char_u *fname, int check_other, int is_vimrc, int *ret_sid, exarg_T *eap);
+
/*
* Initialize the execution stack.
*/
@@ -1079,251 +1081,6 @@ ExpandPackAddDir(
return OK;
}
-/*
- * Cookie used to source Ex commands from a buffer.
- */
-typedef struct
-{
- garray_T lines_to_source;
- int lnum;
- linenr_T sourcing_lnum;
-} bufline_cookie_T;
-
-/*
- * Concatenate a Vim script line if it starts with a line continuation into a
- * growarray (excluding the continuation chars and leading whitespace).
- * Growsize of the growarray may be changed to speed up concatenations!
- *
- * Returns TRUE if this line did begin with a continuation (the next line
- * should also be considered, if it exists); FALSE otherwise.
- */
- static int
-concat_continued_line(
- garray_T *ga,
- int init_growsize,
- char_u *nextline,
- int options)
-{
- int comment_char = in_vim9script() ? '#' : '"';
- char_u *p = skipwhite(nextline);
- int contline;
- int do_vim9_all = in_vim9script()
- && options == GETLINE_CONCAT_ALL;
- int do_bar_cont = do_vim9_all
- || options == GETLINE_CONCAT_CONTBAR;
-
- if (*p == NUL)
- return FALSE;
-
- // Concatenate the next line when it starts with a backslash.
- /* Also check for a comment in between continuation lines: "\ */
- // Also check for a Vim9 comment, empty line, line starting with '|',
- // but not "||".
- if ((p[0] == comment_char && p[1] == '\\' && p[2] == ' ')
- || (do_vim9_all && (*p == NUL
- || vim9_comment_start(p))))
- return TRUE;
-
- contline = (*p == '\\' || (do_bar_cont && p[0] == '|' && p[1] != '|'));
- if (!contline)
- return FALSE;
-
- // Adjust the growsize to the current length to speed up concatenating many
- // lines.
- if (ga->ga_len > init_growsize)
- ga->ga_growsize = ga->ga_len > 8000 ? 8000 : ga->ga_len;
- if (*p == '\\')
- ga_concat(ga, (char_u *)p + 1);
- else if (*p == '|')
- {
- ga_concat(ga, (char_u *)" ");
- ga_concat(ga, p);
- }
-
- return TRUE;
-}
-
-/*
- * Get one full line from a sourced string (in-memory, no file).
- * Called by do_cmdline() when it's called from source_using_linegetter().
- *
- * Returns a pointer to allocated line, or NULL for end-of-file.
- */
- static char_u *
-source_getbufline(
- int c UNUSED,
- void *cookie,
- int indent UNUSED,
- getline_opt_T opts)
-{
- bufline_cookie_T *p = cookie;
- char_u *line;
- garray_T ga;
-
- SOURCING_LNUM = p->sourcing_lnum + 1;
-
- if (p->lnum >= p->lines_to_source.ga_len)
- return NULL;
- line = ((char_u **)p->lines_to_source.ga_data)[p->lnum];
-
- ga_init2(&ga, sizeof(char_u), 400);
- ga_concat(&ga, (char_u *)line);
- p->lnum++;
-
- if ((opts != GETLINE_NONE) && vim_strchr(p_cpo, CPO_CONCAT) == NULL)
- {
- while (p->lnum < p->lines_to_source.ga_len)
- {
- line = ((char_u **)p->lines_to_source.ga_data)[p->lnum];
- if (!concat_continued_line(&ga, 400, line, opts))
- break;
- p->sourcing_lnum++;
- p->lnum++;
- }
- }
- ga_append(&ga, NUL);
- p->sourcing_lnum++;
-
- return ga.ga_data;
-}
-
-/*
- * Source Ex commands from the lines in 'cookie'.
- */
- static int
-do_sourcebuffer(
- void *cookie,
- char_u *scriptname)
-{
- char_u *save_sourcing_name = SOURCING_NAME;
- linenr_T save_sourcing_lnum = SOURCING_LNUM;
- char_u sourcing_name_buf[256];
- sctx_T save_current_sctx;
-#ifdef FEAT_EVAL
- int sid;
- funccal_entry_T funccalp_entry;
- int save_estack_compiling = estack_compiling;
- scriptitem_T *si = NULL;
-#endif
- int save_sticky_cmdmod_flags = sticky_cmdmod_flags;
- int retval = FAIL;
- ESTACK_CHECK_DECLARATION
-
- if (save_sourcing_name == NULL)
- SOURCING_NAME = (char_u *)scriptname;
- else
- {
- vim_snprintf((char *)sourcing_name_buf, sizeof(sourcing_name_buf),
- "%s called at %s:%ld", scriptname, save_sourcing_name,
- save_sourcing_lnum);
- SOURCING_NAME = sourcing_name_buf;
- }
- SOURCING_LNUM = 0;
-
- // Keep the sourcing name/lnum, for recursive calls.
- estack_push(ETYPE_SCRIPT, scriptname, 0);
- ESTACK_CHECK_SETUP
-
- // "legacy" does not apply to commands in the script
- sticky_cmdmod_flags = 0;
-
- save_current_sctx = current_sctx;
- current_sctx.sc_version = 1; // default script version
-#ifdef FEAT_EVAL
- estack_compiling = FALSE;
- // Always use a new sequence number.
- current_sctx.sc_seq = ++last_current_SID_seq;
- current_sctx.sc_lnum = save_sourcing_lnum;
- save_funccal(&funccalp_entry);
-
- sid = find_script_by_name(scriptname);
- if (sid < 0)
- {
- int error = OK;
-
- // First time sourcing this buffer, create a new script item.
-
- sid = get_new_scriptitem(&error);
- if (error == FAIL)
- goto theend;
- current_sctx.sc_sid = sid;
- si = SCRIPT_ITEM(current_sctx.sc_sid);
- si->sn_name = vim_strsave(scriptname);
- si->sn_state = SN_STATE_NEW;
- }
- else
- {
- // the buffer was sourced previously, reuse the script ID.
- current_sctx.sc_sid = sid;
- si = SCRIPT_ITEM(current_sctx.sc_sid);
- si->sn_state = SN_STATE_RELOAD;
- }
-#endif
-
- retval = do_cmdline(NULL, source_getbufline, cookie,
- DOCMD_VERBOSE | DOCMD_NOWAIT | DOCMD_REPEAT);
-
- if (got_int)
- emsg(_(e_interrupted));
-
-#ifdef FEAT_EVAL
-theend:
-#endif
- ESTACK_CHECK_NOW
- estack_pop();
- current_sctx = save_current_sctx;
- SOURCING_LNUM = save_sourcing_lnum;
- SOURCING_NAME = save_sourcing_name;
- sticky_cmdmod_flags = save_sticky_cmdmod_flags;
-#ifdef FEAT_EVAL
- restore_funccal();
- estack_compiling = save_estack_compiling;
-#endif
-
- return retval;
-}
-
-/*
- * :source Ex commands from the current buffer
- */
- static void
-cmd_source_buffer(exarg_T *eap)
-{
- char_u *line = NULL;
- linenr_T curr_lnum;
- bufline_cookie_T cp;
- char_u sname[32];
-
- if (curbuf == NULL)
- return;
-
- // Use ":source buffer=<num>" as the script name
- vim_snprintf((char *)sname, sizeof(sname), ":source buffer=%d",
- curbuf->b_fnum);
-
- ga_init2(&cp.lines_to_source, sizeof(char_u *), 100);
-
- // Copy the lines from the buffer into a grow array
- for (curr_lnum = eap->line1; curr_lnum <= eap->line2; curr_lnum++)
- {
- line = vim_strsave(ml_get(curr_lnum));
- if (line == NULL)
- goto errret;
- if (ga_add_string(&cp.lines_to_source, line) == FAIL)
- goto errret;
- line = NULL;
- }
- cp.sourcing_lnum = 0;
- cp.lnum = 0;
-
- // Execute the Ex commands
- do_sourcebuffer((void *)&cp, (char_u *)sname);
-
-errret:
- vim_free(line);
- ga_clear_strings(&cp.lines_to_source);
-}
-
static void
cmd_source(char_u *fname, exarg_T *eap)
{
@@ -1341,7 +1098,7 @@ cmd_source(char_u *fname, exarg_T *eap)
emsg(_(e_argument_required));
else
// source ex commands from the current buffer
- cmd_source_buffer(eap);
+ do_source_ext(NULL, FALSE, FALSE, NULL, eap);
}
else if (eap != NULL && eap->forceit)
// ":source!": read Normal mode commands
@@ -1480,21 +1237,73 @@ fopen_noinh_readbin(char *filename)
#endif
/*
- * do_source: Read the file "fname" and execute its lines as EX commands.
+ * Initialization for sourcing lines from the current buffer. Reads all the
+ * lines from the buffer and stores it in the cookie grow array.
+ * Returns a pointer to the name ":source buffer=<n>" on success and NULL on
+ * failure.
+ */
+ static char_u *
+do_source_buffer_init(source_cookie_T *sp, exarg_T *eap)
+{
+ linenr_T curr_lnum;
+ char_u *line = NULL;
+ char_u *fname;
+
+ CLEAR_FIELD(*sp);
+
+ if (curbuf == NULL)
+ return NULL;
+
+ // Use ":source buffer=<num>" as the script name
+ vim_snprintf((char *)IObuff, IOSIZE, ":source buffer=%d", curbuf->b_fnum);
+ fname = vim_strsave(IObuff);
+ if (fname == NULL)
+ return NULL;
+
+ ga_init2(&sp->buflines, sizeof(char_u *), 100);
+
+ // Copy the lines from the buffer into a grow array
+ for (curr_lnum = eap->line1; curr_lnum <= eap->line2; curr_lnum++)
+ {
+ line = vim_strsave(ml_get(curr_lnum));
+ if (line == NULL)
+ goto errret;
+ if (ga_add_string(&sp->buflines, line) == FAIL)
+ goto errret;
+ line = NULL;
+ }
+ sp->buf_lnum = 0;
+ sp->source_from_buf = TRUE;
+
+ return fname;
+
+errret:
+ vim_free(fname);
+ vim_free(line);
+ ga_clear_strings(&sp->buflines);
+ return NULL;
+}
+
+/*
+ * Read the file "fname" and execute its lines as EX commands.
* When "ret_sid" is not NULL and we loaded the script before, don't load it
* again.
*
+ * The 'eap' argument is used when sourcing lines from a buffer instead of a
+ * file.
+ *
* This function may be called recursively!
*
* Return FAIL if file could not be opened, OK otherwise.
* If a scriptitem_T was found or created "*ret_sid" is set to the SID.
*/
- int
-do_source(
+ static int
+do_source_ext(
char_u *fname,
int check_other, // check for .vimrc and _vimrc
int is_vimrc, // DOSO_ value
- int *ret_sid UNUSED)
+ int *ret_sid UNUSED,
+ exarg_T *eap)
{
source_cookie_T cookie;
char_u *p;
@@ -1520,17 +1329,28 @@ do_source(
int trigger_source_post = FALSE;
ESTACK_CHECK_DECLARATION
- p = expand_env_save(fname);
- if (p == NULL)
- return retval;
- fname_exp = fix_fname(p);
- vim_free(p);
- if (fname_exp == NULL)
- return retval;
- if (mch_isdir(fname_exp))
- {
- smsg(_("Cannot source a directory: \"%s\""), fname);
- goto theend;
+ CLEAR_FIELD(cookie);
+ if (fname == NULL)
+ {
+ // sourcing lines from a buffer
+ fname_exp = do_source_buffer_init(&cookie, eap);
+ if (fname_exp == NULL)
+ return FAIL;
+ }
+ else
+ {
+ p = expand_env_save(fname);
+ if (p == NULL)
+ return retval;
+ fname_exp = fix_fname(p);
+ vim_free(p);
+ if (fname_exp == NULL)
+ return retval;
+ if (mch_isdir(fname_exp))
+ {
+ smsg(_("Cannot source a directory: \"%s\""), fname);
+ goto theend;
+ }
}
#ifdef FEAT_EVAL
estack_compiling = FALSE;
@@ -1567,11 +1387,14 @@ do_source(
// Apply SourcePre autocommands, they may get the file.
apply_autocmds(EVENT_SOURCEPRE, fname_exp, fname_exp, FALSE, curbuf);
+ if (!cookie.source_from_buf)
+ {
#ifdef USE_FOPEN_NOINH
- cookie.fp = fopen_noinh_readbin((char *)fname_exp);
+ cookie.fp = fopen_noinh_readbin((char *)fname_exp);
#else
- cookie.fp = mch_fopen((char *)fname_exp, READBIN);
+ cookie.fp = mch_fopen((char *)fname_exp, READBIN);
#endif
+ }
if (cookie.fp == NULL && check_other)
{
// Try again, replacing file name ".vimrc" by "_vimrc" or vice versa,
@@ -1594,7 +1417,7 @@ do_source(
}
}
- if (cookie.fp == NULL)
+ if (cookie.fp == NULL && !cookie.source_from_buf)
{
if (p_verbose > 0)
{
@@ -1632,12 +1455,14 @@ do_source(
cookie.fileformat = EOL_DOS;
else
cookie.fileformat = EOL_UNKNOWN;
- cookie.error = FALSE;
#endif
- cookie.nextline = NULL;
- cookie.sourcing_lnum = 0;
- cookie.finished = FALSE;
+ if (fname == NULL)
+ // When sourcing a range of lines from a buffer, use the buffer line
+ // number.
+ cookie.sourcing_lnum = eap->line1 - 1;
+ else
+ cookie.sourcing_lnum = 0;
#ifdef FEAT_EVAL
// Check if this script has a breakpoint.
@@ -1661,7 +1486,12 @@ do_source(
sticky_cmdmod_flags = 0;
save_current_sctx = current_sctx;
- current_sctx.sc_version = 1; // default script version
+ if (cmdmod.cmod_flags & CMOD_VIM9CMD)
+ // When the ":vim9cmd" command modifier is used, source the script as a
+ // Vim9 script.
+ current_sctx.sc_version = SCRIPT_VERSION_VIM9;
+ else
+ current_sctx.sc_version = 1; // default script version
#ifdef FEAT_EVAL
# ifdef FEAT_PROFILE
@@ -1874,7 +1704,10 @@ almosttheend:
#endif
current_sctx = save_current_sctx;
- fclose(cookie.fp);
+ if (cookie.fp != NULL)
+ fclose(cookie.fp);
+ if (cookie.source_from_buf)
+ ga_clear_strings(&cookie.buflines);
vim_free(cookie.nextline);
vim_free(firstline);
convert_setup(&cookie.conv, NULL, NULL);
@@ -1891,6 +1724,17 @@ theend:
return retval;
}
+ int
+do_source(
+ char_u *fname,
+ int check_other, // check for .vimrc and _vimrc
+ int is_vimrc, // DOSO_ value
+ int *ret_sid UNUSED)
+{
+ return do_source_ext(fname, check_other, is_vimrc, ret_sid, NULL);
+}
+
+
#if defined(FEAT_EVAL) || defined(PROTO)
/*
@@ -2038,11 +1882,21 @@ get_one_sourceline(source_cookie_T *sp)
// make room to read at least 120 (more) characters
if (ga_grow(&ga, 120) == FAIL)
break;
- buf = (char_u *)ga.ga_data;
-
- if (fgets((char *)buf + ga.ga_len, ga.ga_maxlen - ga.ga_len,
- sp->fp) == NULL)
- break;
+ if (sp->source_from_buf)
+ {
+ if (sp->buf_lnum >= sp->buflines.ga_len)
+ break; // all the lines are processed
+ ga_concat(&ga, ((char_u **)sp->buflines.ga_data)[sp->buf_lnum]);
+ sp->buf_lnum++;
+ buf = (char_u *)ga.ga_data;
+ }
+ else
+ {
+ buf = (char_u *)ga.ga_data;
+ if (fgets((char *)buf + ga.ga_len, ga.ga_maxlen - ga.ga_len,
+ sp->fp) == NULL)
+ break;
+ }
len = ga.ga_len + (int)STRLEN(buf + ga.ga_len);
#ifdef USE_CRNL
// Ignore a trailing CTRL-Z, when in Dos mode. Only recognize the
@@ -2145,7 +1999,7 @@ getsourceline(
#ifdef FEAT_EVAL
// If breakpoints have been added/deleted need to check for it.
- if (sp->dbg_tick < debug_tick)
+ if ((sp->dbg_tick < debug_tick) && !sp->source_from_buf)
{
sp->breakpoint = dbg_find_breakpoint(TRUE, sp->fname, SOURCING_LNUM);
sp->dbg_tick = debug_tick;
@@ -2161,7 +2015,7 @@ getsourceline(
// Get current line. If there is a read-ahead line, use it, otherwise get
// one now. "fp" is NULL if actually using a string.
- if (sp->finished || sp->fp == NULL)
+ if (sp->finished || (!sp->source_from_buf && sp->fp == NULL))
line = NULL;
else if (sp->nextline == NULL)
line = get_one_sourceline(sp);
@@ -2265,7 +2119,8 @@ getsourceline(
#ifdef FEAT_EVAL
// Did we encounter a breakpoint?
- if (sp->breakpoint != 0 && sp->breakpoint <= SOURCING_LNUM)
+ if (!sp->source_from_buf && sp->breakpoint != 0
+ && sp->breakpoint <= SOURCING_LNUM)
{
dbg_breakpoint(sp->fname, SOURCING_LNUM);
// Find next breakpoint.
@@ -2284,8 +2139,7 @@ getsourceline(
int
sourcing_a_script(exarg_T *eap)
{
- return (getline_equal(eap->getline, eap->cookie, getsourceline)
- || getline_equal(eap->getline, eap->cookie, source_getbufline));
+ return (getline_equal(eap->getline, eap->cookie, getsourceline));
}
/*