xref: /freebsd/contrib/nvi/ex/ex_cmd.c (revision 884a2a699669ec61e2366e3e358342dbc94be24a)
1 /*-
2  * Copyright (c) 1992, 1993, 1994
3  *	The Regents of the University of California.  All rights reserved.
4  * Copyright (c) 1992, 1993, 1994, 1995, 1996
5  *	Keith Bostic.  All rights reserved.
6  *
7  * See the LICENSE file for redistribution information.
8  */
9 
10 #include "config.h"
11 
12 #ifndef lint
13 static const char sccsid[] = "@(#)ex_cmd.c	10.20 (Berkeley) 10/10/96";
14 #endif /* not lint */
15 
16 #include <sys/types.h>
17 #include <sys/queue.h>
18 
19 #include <bitstring.h>
20 #include <limits.h>
21 #include <stdio.h>
22 
23 #include "../common/common.h"
24 
25 /*
26  * This array maps ex command names to command functions.
27  *
28  * The order in which command names are listed below is important --
29  * ambiguous abbreviations are resolved to be the first possible match,
30  * e.g. "r" means "read", not "rewind", because "read" is listed before
31  * "rewind".
32  *
33  * The syntax of the ex commands is unbelievably irregular, and a special
34  * case from beginning to end.  Each command has an associated "syntax
35  * script" which describes the "arguments" that are possible.  The script
36  * syntax is as follows:
37  *
38  *	!		-- ! flag
39  *	1		-- flags: [+-]*[pl#][+-]*
40  *	2		-- flags: [-.+^]
41  *	3		-- flags: [-.+^=]
42  *	b		-- buffer
43  *	c[01+a]		-- count (0-N, 1-N, signed 1-N, address offset)
44  *	f[N#][or]	-- file (a number or N, optional or required)
45  *	l		-- line
46  *	S		-- string with file name expansion
47  *	s		-- string
48  *	W		-- word string
49  *	w[N#][or]	-- word (a number or N, optional or required)
50  */
51 EXCMDLIST const cmds[] = {
52 /* C_SCROLL */
53 	{"\004",	ex_pr,		E_ADDR2,
54 	    "",
55 	    "^D",
56 	    "scroll lines"},
57 /* C_BANG */
58 	{"!",		ex_bang,	E_ADDR2_NONE | E_SECURE,
59 	    "S",
60 	    "[line [,line]] ! command",
61 	    "filter lines through commands or run commands"},
62 /* C_HASH */
63 	{"#",		ex_number,	E_ADDR2|E_CLRFLAG,
64 	    "ca1",
65 	    "[line [,line]] # [count] [l]",
66 	    "display numbered lines"},
67 /* C_SUBAGAIN */
68 	{"&",		ex_subagain,	E_ADDR2,
69 	    "s",
70 	    "[line [,line]] & [cgr] [count] [#lp]",
71 	    "repeat the last substitution"},
72 /* C_STAR */
73 	{"*",		ex_at,		0,
74 	    "b",
75 	    "* [buffer]",
76 	    "execute a buffer"},
77 /* C_SHIFTL */
78 	{"<",		ex_shiftl,	E_ADDR2|E_AUTOPRINT,
79 	    "ca1",
80 	    "[line [,line]] <[<...] [count] [flags]",
81 	    "shift lines left"},
82 /* C_EQUAL */
83 	{"=",		ex_equal,	E_ADDR1|E_ADDR_ZERO|E_ADDR_ZERODEF,
84 	    "1",
85 	    "[line] = [flags]",
86 	    "display line number"},
87 /* C_SHIFTR */
88 	{">",		ex_shiftr,	E_ADDR2|E_AUTOPRINT,
89 	    "ca1",
90 	    "[line [,line]] >[>...] [count] [flags]",
91 	    "shift lines right"},
92 /* C_AT */
93 	{"@",		ex_at,		E_ADDR2,
94 	    "b",
95 	    "@ [buffer]",
96 	    "execute a buffer"},
97 /* C_APPEND */
98 	{"append",	ex_append,	E_ADDR1|E_ADDR_ZERO|E_ADDR_ZERODEF,
99 	    "!",
100 	    "[line] a[ppend][!]",
101 	    "append input to a line"},
102 /* C_ABBR */
103 	{"abbreviate", 	ex_abbr,	0,
104 	    "W",
105 	    "ab[brev] [word replace]",
106 	    "specify an input abbreviation"},
107 /* C_ARGS */
108 	{"args",	ex_args,	0,
109 	    "",
110 	    "ar[gs]",
111 	    "display file argument list"},
112 /* C_BG */
113 	{"bg",		ex_bg,		E_VIONLY,
114 	    "",
115 	    "bg",
116 	    "put a foreground screen into the background"},
117 /* C_CHANGE */
118 	{"change",	ex_change,	E_ADDR2|E_ADDR_ZERODEF,
119 	    "!ca",
120 	    "[line [,line]] c[hange][!] [count]",
121 	    "change lines to input"},
122 /* C_CD */
123 	{"cd",		ex_cd,		0,
124 	    "!f1o",
125 	    "cd[!] [directory]",
126 	    "change the current directory"},
127 /* C_CHDIR */
128 	{"chdir",	ex_cd,		0,
129 	    "!f1o",
130 	    "chd[ir][!] [directory]",
131 	    "change the current directory"},
132 /* C_COPY */
133 	{"copy",	ex_copy,	E_ADDR2|E_AUTOPRINT,
134 	    "l1",
135 	    "[line [,line]] co[py] line [flags]",
136 	    "copy lines elsewhere in the file"},
137 /* C_CSCOPE */
138 	{"cscope",      ex_cscope,      0,
139 	    "!s",
140 	    "cs[cope] command [args]",
141 	    "create a set of tags using a cscope command"},
142 /*
143  * !!!
144  * Adding new commands starting with 'd' may break the delete command code
145  * in ex_cmd() (the ex parser).  Read through the comments there, first.
146  */
147 /* C_DELETE */
148 	{"delete",	ex_delete,	E_ADDR2|E_AUTOPRINT,
149 	    "bca1",
150 	    "[line [,line]] d[elete][flags] [buffer] [count] [flags]",
151 	    "delete lines from the file"},
152 /* C_DISPLAY */
153 	{"display",	ex_display,	0,
154 	    "w1r",
155 	    "display b[uffers] | c[onnections] | s[creens] | t[ags]",
156 	    "display buffers, connections, screens or tags"},
157 /* C_EDIT */
158 	{"edit",	ex_edit,	E_NEWSCREEN,
159 	    "f1o",
160 	    "[Ee][dit][!] [+cmd] [file]",
161 	    "begin editing another file"},
162 /* C_EX */
163 	{"ex",		ex_edit,	E_NEWSCREEN,
164 	    "f1o",
165 	    "[Ee]x[!] [+cmd] [file]",
166 	    "begin editing another file"},
167 /* C_EXUSAGE */
168 	{"exusage",	ex_usage,	0,
169 	    "w1o",
170 	    "[exu]sage [command]",
171 	    "display ex command usage statement"},
172 /* C_FILE */
173 	{"file",	ex_file,	0,
174 	    "f1o",
175 	    "f[ile] [name]",
176 	    "display (and optionally set) file name"},
177 /* C_FG */
178 	{"fg",		ex_fg,		E_NEWSCREEN|E_VIONLY,
179 	    "f1o",
180 	    "[Ff]g [file]",
181 	    "bring a backgrounded screen into the foreground"},
182 /* C_GLOBAL */
183 	{"global",	ex_global,	E_ADDR2_ALL,
184 	    "!s",
185 	    "[line [,line]] g[lobal][!] [;/]RE[;/] [commands]",
186 	    "execute a global command on lines matching an RE"},
187 /* C_HELP */
188 	{"help",	ex_help,	0,
189 	    "",
190 	    "he[lp]",
191 	    "display help statement"},
192 /* C_INSERT */
193 	{"insert",	ex_insert,	E_ADDR1|E_ADDR_ZERO|E_ADDR_ZERODEF,
194 	    "!",
195 	    "[line] i[nsert][!]",
196 	    "insert input before a line"},
197 /* C_JOIN */
198 	{"join",	ex_join,	E_ADDR2|E_AUTOPRINT,
199 	    "!ca1",
200 	    "[line [,line]] j[oin][!] [count] [flags]",
201 	    "join lines into a single line"},
202 /* C_K */
203 	{"k",		ex_mark,	E_ADDR1,
204 	    "w1r",
205 	    "[line] k key",
206 	    "mark a line position"},
207 /* C_LIST */
208 	{"list",	ex_list,	E_ADDR2|E_CLRFLAG,
209 	    "ca1",
210 	    "[line [,line]] l[ist] [count] [#]",
211 	    "display lines in an unambiguous form"},
212 /* C_MOVE */
213 	{"move",	ex_move,	E_ADDR2|E_AUTOPRINT,
214 	    "l",
215 	    "[line [,line]] m[ove] line",
216 	    "move lines elsewhere in the file"},
217 /* C_MARK */
218 	{"mark",	ex_mark,	E_ADDR1,
219 	    "w1r",
220 	    "[line] ma[rk] key",
221 	    "mark a line position"},
222 /* C_MAP */
223 	{"map",		ex_map,		0,
224 	    "!W",
225 	    "map[!] [keys replace]",
226 	    "map input or commands to one or more keys"},
227 /* C_MKEXRC */
228 	{"mkexrc",	ex_mkexrc,	0,
229 	    "!f1r",
230 	    "mkexrc[!] file",
231 	    "write a .exrc file"},
232 /* C_NEXT */
233 	{"next",	ex_next,	E_NEWSCREEN,
234 	    "!fN",
235 	    "[Nn][ext][!] [+cmd] [file ...]",
236 	    "edit (and optionally specify) the next file"},
237 /* C_NUMBER */
238 	{"number",	ex_number,	E_ADDR2|E_CLRFLAG,
239 	    "ca1",
240 	    "[line [,line]] nu[mber] [count] [l]",
241 	    "change display to number lines"},
242 /* C_OPEN */
243 	{"open",	ex_open,	E_ADDR1,
244 	    "s",
245 	    "[line] o[pen] [/RE/] [flags]",
246 	    "enter \"open\" mode (not implemented)"},
247 /* C_PRINT */
248 	{"print",	ex_pr,		E_ADDR2|E_CLRFLAG,
249 	    "ca1",
250 	    "[line [,line]] p[rint] [count] [#l]",
251 	    "display lines"},
252 /* C_PERLCMD */
253 	{"perl",	ex_perl,	E_ADDR2_ALL|E_ADDR_ZERO|
254 					    E_ADDR_ZERODEF|E_SECURE,
255 	    "s",
256 	    "pe[rl] cmd",
257 	    "run the perl interpreter with the command"},
258 /* C_PERLDOCMD */
259 	{"perldo",	ex_perl,	E_ADDR2_ALL|E_ADDR_ZERO|
260 					    E_ADDR_ZERODEF|E_SECURE,
261 	    "s",
262 	    "perld[o] cmd",
263 	    "run the perl interpreter with the command, on each line"},
264 /* C_PRESERVE */
265 	{"preserve",	ex_preserve,	0,
266 	    "",
267 	    "pre[serve]",
268 	    "preserve an edit session for recovery"},
269 /* C_PREVIOUS */
270 	{"previous",	ex_prev,	E_NEWSCREEN,
271 	    "!",
272 	    "[Pp]rev[ious][!]",
273 	    "edit the previous file in the file argument list"},
274 /* C_PUT */
275 	{"put",		ex_put,
276 	    E_ADDR1|E_AUTOPRINT|E_ADDR_ZERO|E_ADDR_ZERODEF,
277 	    "b",
278 	    "[line] pu[t] [buffer]",
279 	    "append a cut buffer to the line"},
280 /* C_QUIT */
281 	{"quit",	ex_quit,	0,
282 	    "!",
283 	    "q[uit][!]",
284 	    "exit ex/vi"},
285 /* C_READ */
286 	{"read",	ex_read,	E_ADDR1|E_ADDR_ZERO|E_ADDR_ZERODEF,
287 	    "s",
288 	    "[line] r[ead] [!cmd | [file]]",
289 	    "append input from a command or file to the line"},
290 /* C_RECOVER */
291 	{"recover",	ex_recover,	0,
292 	    "!f1r",
293 	    "recover[!] file",
294 	    "recover a saved file"},
295 /* C_RESIZE */
296 	{"resize",	ex_resize,	E_VIONLY,
297 	    "c+",
298 	    "resize [+-]rows",
299 	    "grow or shrink the current screen"},
300 /* C_REWIND */
301 	{"rewind",	ex_rew,		0,
302 	    "!",
303 	    "rew[ind][!]",
304 	    "re-edit all the files in the file argument list"},
305 #ifdef GTAGS
306 /* C_RTAG */
307 	{"rtag",	ex_rtag_push,	E_NEWSCREEN,
308 	    "!w1o",
309 	    "[Rr]ta[g][!] [string]",
310 	    "edit the file containing the tag"},
311 #endif
312 /*
313  * !!!
314  * Adding new commands starting with 's' may break the substitute command code
315  * in ex_cmd() (the ex parser).  Read through the comments there, first.
316  */
317 /* C_SUBSTITUTE */
318 	{"s",		ex_s,		E_ADDR2,
319 	    "s",
320 	    "[line [,line]] s [[/;]RE[/;]repl[/;] [cgr] [count] [#lp]]",
321 	    "substitute on lines matching an RE"},
322 /* C_SCRIPT */
323 	{"script",	ex_script,	E_SECURE,
324 	    "!f1o",
325 	    "sc[ript][!] [file]",
326 	    "run a shell in a screen"},
327 /* C_SET */
328 	{"set",		ex_set,		0,
329 	    "wN",
330 	    "se[t] [option[=[value]]...] [nooption ...] [option? ...] [all]",
331 	    "set options (use \":set all\" to see all options)"},
332 /* C_SHELL */
333 	{"shell",	ex_shell,	E_SECURE,
334 	    "",
335 	    "sh[ell]",
336 	    "suspend editing and run a shell"},
337 /* C_SOURCE */
338 	{"source",	ex_source,	0,
339 	    "f1r",
340 	    "so[urce] file",
341 	    "read a file of ex commands"},
342 /* C_STOP */
343 	{"stop",	ex_stop,	E_SECURE,
344 	    "!",
345 	    "st[op][!]",
346 	    "suspend the edit session"},
347 /* C_SUSPEND */
348 	{"suspend",	ex_stop,	E_SECURE,
349 	    "!",
350 	    "su[spend][!]",
351 	    "suspend the edit session"},
352 /* C_T */
353 	{"t",		ex_copy,	E_ADDR2|E_AUTOPRINT,
354 	    "l1",
355 	    "[line [,line]] t line [flags]",
356 	    "copy lines elsewhere in the file"},
357 /* C_TAG */
358 	{"tag",		ex_tag_push,	E_NEWSCREEN,
359 	    "!w1o",
360 	    "[Tt]a[g][!] [string]",
361 	    "edit the file containing the tag"},
362 /* C_TAGNEXT */
363 	{"tagnext",	ex_tag_next,	0,
364 	    "!",
365 	    "tagn[ext][!]",
366 	    "move to the next tag"},
367 /* C_TAGPOP */
368 	{"tagpop",	ex_tag_pop,	0,
369 	    "!w1o",
370 	    "tagp[op][!] [number | file]",
371 	    "return to the previous group of tags"},
372 /* C_TAGPREV */
373 	{"tagprev",	ex_tag_prev,	0,
374 	    "!",
375 	    "tagpr[ev][!]",
376 	    "move to the previous tag"},
377 /* C_TAGTOP */
378 	{"tagtop",	ex_tag_top,	0,
379 	    "!",
380 	    "tagt[op][!]",
381 	    "discard all tags"},
382 /* C_TCLCMD */
383 	{"tcl",		ex_tcl,		E_ADDR2_ALL|E_ADDR_ZERO|
384 					    E_ADDR_ZERODEF|E_SECURE,
385 	    "s",
386 	    "tc[l] cmd",
387 	    "run the tcl interpreter with the command"},
388 /* C_UNDO */
389 	{"undo",	ex_undo,	E_AUTOPRINT,
390 	    "",
391 	    "u[ndo]",
392 	    "undo the most recent change"},
393 /* C_UNABBREVIATE */
394 	{"unabbreviate",ex_unabbr,	0,
395 	    "w1r",
396 	    "una[bbrev] word",
397 	    "delete an abbreviation"},
398 /* C_UNMAP */
399 	{"unmap",	ex_unmap,	0,
400 	    "!w1r",
401 	    "unm[ap][!] word",
402 	    "delete an input or command map"},
403 /* C_V */
404 	{"v",		ex_v,		E_ADDR2_ALL,
405 	    "s",
406 	    "[line [,line]] v [;/]RE[;/] [commands]",
407 	    "execute a global command on lines NOT matching an RE"},
408 /* C_VERSION */
409 	{"version",	ex_version,	0,
410 	    "",
411 	    "version",
412 	    "display the program version information"},
413 /* C_VISUAL_EX */
414 	{"visual",	ex_visual,	E_ADDR1|E_ADDR_ZERODEF,
415 	    "2c11",
416 	    "[line] vi[sual] [-|.|+|^] [window_size] [flags]",
417 	    "enter visual (vi) mode from ex mode"},
418 /* C_VISUAL_VI */
419 	{"visual",	ex_edit,	E_NEWSCREEN,
420 	    "f1o",
421 	    "[Vv]i[sual][!] [+cmd] [file]",
422 	    "edit another file (from vi mode only)"},
423 /* C_VIUSAGE */
424 	{"viusage",	ex_viusage,	0,
425 	    "w1o",
426 	    "[viu]sage [key]",
427 	    "display vi key usage statement"},
428 /* C_WRITE */
429 	{"write",	ex_write,	E_ADDR2_ALL|E_ADDR_ZERODEF,
430 	    "!s",
431 	    "[line [,line]] w[rite][!] [ !cmd | [>>] [file]]",
432 	    "write the file"},
433 /* C_WN */
434 	{"wn",		ex_wn,		E_ADDR2_ALL|E_ADDR_ZERODEF,
435 	    "!s",
436 	    "[line [,line]] wn[!] [>>] [file]",
437 	    "write the file and switch to the next file"},
438 /* C_WQ */
439 	{"wq",		ex_wq,		E_ADDR2_ALL|E_ADDR_ZERODEF,
440 	    "!s",
441 	    "[line [,line]] wq[!] [>>] [file]",
442 	    "write the file and exit"},
443 /* C_XIT */
444 	{"xit",		ex_xit,		E_ADDR2_ALL|E_ADDR_ZERODEF,
445 	    "!f1o",
446 	    "[line [,line]] x[it][!] [file]",
447 	    "exit"},
448 /* C_YANK */
449 	{"yank",	ex_yank,	E_ADDR2,
450 	    "bca",
451 	    "[line [,line]] ya[nk] [buffer] [count]",
452 	    "copy lines to a cut buffer"},
453 /* C_Z */
454 	{"z",		ex_z,		E_ADDR1,
455 	    "3c01",
456 	    "[line] z [-|.|+|^|=] [count] [flags]",
457 	    "display different screens of the file"},
458 /* C_SUBTILDE */
459 	{"~",		ex_subtilde,	E_ADDR2,
460 	    "s",
461 	    "[line [,line]] ~ [cgr] [count] [#lp]",
462 	    "replace previous RE with previous replacement string,"},
463 	{NULL},
464 };
465