xref: /f-stack/freebsd/ddb/db_input.c (revision 22ce4aff)
1a9643ea8Slogwang /*-
2*22ce4affSfengbojiang  * SPDX-License-Identifier: MIT-CMU
3*22ce4affSfengbojiang  *
4a9643ea8Slogwang  * Mach Operating System
5a9643ea8Slogwang  * Copyright (c) 1991,1990 Carnegie Mellon University
6a9643ea8Slogwang  * All Rights Reserved.
7a9643ea8Slogwang  *
8a9643ea8Slogwang  * Permission to use, copy, modify and distribute this software and its
9a9643ea8Slogwang  * documentation is hereby granted, provided that both the copyright
10a9643ea8Slogwang  * notice and this permission notice appear in all copies of the
11a9643ea8Slogwang  * software, derivative works or modified versions, and any portions
12a9643ea8Slogwang  * thereof, and that both notices appear in supporting documentation.
13a9643ea8Slogwang  *
14a9643ea8Slogwang  * CARNEGIE MELLON ALLOWS FREE USE OF THIS SOFTWARE IN ITS
15a9643ea8Slogwang  * CONDITION.  CARNEGIE MELLON DISCLAIMS ANY LIABILITY OF ANY KIND FOR
16a9643ea8Slogwang  * ANY DAMAGES WHATSOEVER RESULTING FROM THE USE OF THIS SOFTWARE.
17a9643ea8Slogwang  *
18a9643ea8Slogwang  * Carnegie Mellon requests users of this software to return to
19a9643ea8Slogwang  *
20a9643ea8Slogwang  *  Software Distribution Coordinator  or  [email protected]
21a9643ea8Slogwang  *  School of Computer Science
22a9643ea8Slogwang  *  Carnegie Mellon University
23a9643ea8Slogwang  *  Pittsburgh PA 15213-3890
24a9643ea8Slogwang  *
25a9643ea8Slogwang  * any improvements or extensions that they make and grant Carnegie the
26a9643ea8Slogwang  * rights to redistribute these changes.
27a9643ea8Slogwang  */
28a9643ea8Slogwang /*
29a9643ea8Slogwang  *	Author: David B. Golub, Carnegie Mellon University
30a9643ea8Slogwang  *	Date:	7/90
31a9643ea8Slogwang  */
32a9643ea8Slogwang 
33a9643ea8Slogwang #include <sys/cdefs.h>
34a9643ea8Slogwang __FBSDID("$FreeBSD$");
35a9643ea8Slogwang 
36a9643ea8Slogwang #include <sys/param.h>
37a9643ea8Slogwang #include <sys/systm.h>
38a9643ea8Slogwang #include <sys/cons.h>
39a9643ea8Slogwang 
40a9643ea8Slogwang #include <ddb/ddb.h>
41a9643ea8Slogwang #include <ddb/db_output.h>
42a9643ea8Slogwang 
43a9643ea8Slogwang /*
44a9643ea8Slogwang  * Character input and editing.
45a9643ea8Slogwang  */
46a9643ea8Slogwang 
47a9643ea8Slogwang /*
48a9643ea8Slogwang  * We don't track output position while editing input,
49a9643ea8Slogwang  * since input always ends with a new-line.  We just
50a9643ea8Slogwang  * reset the line position at the end.
51a9643ea8Slogwang  */
52a9643ea8Slogwang static char *	db_lbuf_start;	/* start of input line buffer */
53a9643ea8Slogwang static char *	db_lbuf_end;	/* end of input line buffer */
54a9643ea8Slogwang static char *	db_lc;		/* current character */
55a9643ea8Slogwang static char *	db_le;		/* one past last character */
56a9643ea8Slogwang 
57a9643ea8Slogwang /*
58a9643ea8Slogwang  * Simple input line history support.
59a9643ea8Slogwang  */
60a9643ea8Slogwang static char	db_lhistory[2048];
61a9643ea8Slogwang static int	db_lhistlsize, db_lhistidx, db_lhistcur;
62a9643ea8Slogwang static int	db_lhist_nlines;
63a9643ea8Slogwang 
64a9643ea8Slogwang #define	CTRL(c)		((c) & 0x1f)
65a9643ea8Slogwang #define	BLANK		' '
66a9643ea8Slogwang #define	BACKUP		'\b'
67a9643ea8Slogwang 
68a9643ea8Slogwang static int	cnmaygetc(void);
69a9643ea8Slogwang static void	db_delete(int n, int bwd);
70a9643ea8Slogwang static int	db_inputchar(int c);
71a9643ea8Slogwang static void	db_putnchars(int c, int count);
72a9643ea8Slogwang static void	db_putstring(char *s, int count);
73a9643ea8Slogwang 
74a9643ea8Slogwang static void
db_putstring(s,count)75a9643ea8Slogwang db_putstring(s, count)
76a9643ea8Slogwang 	char	*s;
77a9643ea8Slogwang 	int	count;
78a9643ea8Slogwang {
79a9643ea8Slogwang 	while (--count >= 0)
80a9643ea8Slogwang 	    cnputc(*s++);
81a9643ea8Slogwang }
82a9643ea8Slogwang 
83a9643ea8Slogwang static void
db_putnchars(c,count)84a9643ea8Slogwang db_putnchars(c, count)
85a9643ea8Slogwang 	int	c;
86a9643ea8Slogwang 	int	count;
87a9643ea8Slogwang {
88a9643ea8Slogwang 	while (--count >= 0)
89a9643ea8Slogwang 	    cnputc(c);
90a9643ea8Slogwang }
91a9643ea8Slogwang 
92a9643ea8Slogwang /*
93a9643ea8Slogwang  * Delete N characters, forward or backward
94a9643ea8Slogwang  */
95a9643ea8Slogwang #define	DEL_FWD		0
96a9643ea8Slogwang #define	DEL_BWD		1
97a9643ea8Slogwang static void
db_delete(n,bwd)98a9643ea8Slogwang db_delete(n, bwd)
99a9643ea8Slogwang 	int	n;
100a9643ea8Slogwang 	int	bwd;
101a9643ea8Slogwang {
102a9643ea8Slogwang 	char *p;
103a9643ea8Slogwang 
104a9643ea8Slogwang 	if (bwd) {
105a9643ea8Slogwang 	    db_lc -= n;
106a9643ea8Slogwang 	    db_putnchars(BACKUP, n);
107a9643ea8Slogwang 	}
108a9643ea8Slogwang 	for (p = db_lc; p < db_le-n; p++) {
109a9643ea8Slogwang 	    *p = *(p+n);
110a9643ea8Slogwang 	    cnputc(*p);
111a9643ea8Slogwang 	}
112a9643ea8Slogwang 	db_putnchars(BLANK, n);
113a9643ea8Slogwang 	db_putnchars(BACKUP, db_le - db_lc);
114a9643ea8Slogwang 	db_le -= n;
115a9643ea8Slogwang }
116a9643ea8Slogwang 
117a9643ea8Slogwang /* returns true at end-of-line */
118a9643ea8Slogwang static int
db_inputchar(c)119a9643ea8Slogwang db_inputchar(c)
120a9643ea8Slogwang 	int	c;
121a9643ea8Slogwang {
122a9643ea8Slogwang 	static int escstate;
123a9643ea8Slogwang 
124a9643ea8Slogwang 	if (escstate == 1) {
125a9643ea8Slogwang 		/* ESC seen, look for [ or O */
126a9643ea8Slogwang 		if (c == '[' || c == 'O')
127a9643ea8Slogwang 			escstate++;
128a9643ea8Slogwang 		else
129a9643ea8Slogwang 			escstate = 0; /* re-init state machine */
130a9643ea8Slogwang 		return (0);
131a9643ea8Slogwang 	} else if (escstate == 2) {
132a9643ea8Slogwang 		escstate = 0;
133a9643ea8Slogwang 		/*
134a9643ea8Slogwang 		 * If a valid cursor key has been found, translate
135a9643ea8Slogwang 		 * into an emacs-style control key, and fall through.
136a9643ea8Slogwang 		 * Otherwise, drop off.
137a9643ea8Slogwang 		 */
138a9643ea8Slogwang 		switch (c) {
139a9643ea8Slogwang 		case 'A':	/* up */
140a9643ea8Slogwang 			c = CTRL('p');
141a9643ea8Slogwang 			break;
142a9643ea8Slogwang 		case 'B':	/* down */
143a9643ea8Slogwang 			c = CTRL('n');
144a9643ea8Slogwang 			break;
145a9643ea8Slogwang 		case 'C':	/* right */
146a9643ea8Slogwang 			c = CTRL('f');
147a9643ea8Slogwang 			break;
148a9643ea8Slogwang 		case 'D':	/* left */
149a9643ea8Slogwang 			c = CTRL('b');
150a9643ea8Slogwang 			break;
151a9643ea8Slogwang 		default:
152a9643ea8Slogwang 			return (0);
153a9643ea8Slogwang 		}
154a9643ea8Slogwang 	}
155a9643ea8Slogwang 
156a9643ea8Slogwang 	switch (c) {
157a9643ea8Slogwang 	    case CTRL('['):
158a9643ea8Slogwang 		escstate = 1;
159a9643ea8Slogwang 		break;
160a9643ea8Slogwang 	    case CTRL('b'):
161a9643ea8Slogwang 		/* back up one character */
162a9643ea8Slogwang 		if (db_lc > db_lbuf_start) {
163a9643ea8Slogwang 		    cnputc(BACKUP);
164a9643ea8Slogwang 		    db_lc--;
165a9643ea8Slogwang 		}
166a9643ea8Slogwang 		break;
167a9643ea8Slogwang 	    case CTRL('f'):
168a9643ea8Slogwang 		/* forward one character */
169a9643ea8Slogwang 		if (db_lc < db_le) {
170a9643ea8Slogwang 		    cnputc(*db_lc);
171a9643ea8Slogwang 		    db_lc++;
172a9643ea8Slogwang 		}
173a9643ea8Slogwang 		break;
174a9643ea8Slogwang 	    case CTRL('a'):
175a9643ea8Slogwang 		/* beginning of line */
176a9643ea8Slogwang 		while (db_lc > db_lbuf_start) {
177a9643ea8Slogwang 		    cnputc(BACKUP);
178a9643ea8Slogwang 		    db_lc--;
179a9643ea8Slogwang 		}
180a9643ea8Slogwang 		break;
181a9643ea8Slogwang 	    case CTRL('e'):
182a9643ea8Slogwang 		/* end of line */
183a9643ea8Slogwang 		while (db_lc < db_le) {
184a9643ea8Slogwang 		    cnputc(*db_lc);
185a9643ea8Slogwang 		    db_lc++;
186a9643ea8Slogwang 		}
187a9643ea8Slogwang 		break;
188a9643ea8Slogwang 	    case CTRL('h'):
189a9643ea8Slogwang 	    case 0177:
190a9643ea8Slogwang 		/* erase previous character */
191a9643ea8Slogwang 		if (db_lc > db_lbuf_start)
192a9643ea8Slogwang 		    db_delete(1, DEL_BWD);
193a9643ea8Slogwang 		break;
194a9643ea8Slogwang 	    case CTRL('d'):
195a9643ea8Slogwang 		/* erase next character */
196a9643ea8Slogwang 		if (db_lc < db_le)
197a9643ea8Slogwang 		    db_delete(1, DEL_FWD);
198a9643ea8Slogwang 		break;
199a9643ea8Slogwang 	    case CTRL('u'):
200*22ce4affSfengbojiang 	    case CTRL('c'):
201a9643ea8Slogwang 		/* kill entire line: */
202a9643ea8Slogwang 		/* at first, delete to beginning of line */
203a9643ea8Slogwang 		if (db_lc > db_lbuf_start)
204a9643ea8Slogwang 		    db_delete(db_lc - db_lbuf_start, DEL_BWD);
205a9643ea8Slogwang 		/* FALLTHROUGH */
206a9643ea8Slogwang 	    case CTRL('k'):
207a9643ea8Slogwang 		/* delete to end of line */
208a9643ea8Slogwang 		if (db_lc < db_le)
209a9643ea8Slogwang 		    db_delete(db_le - db_lc, DEL_FWD);
210a9643ea8Slogwang 		break;
211a9643ea8Slogwang 	    case CTRL('t'):
212a9643ea8Slogwang 		/* twiddle last 2 characters */
213a9643ea8Slogwang 		if (db_lc >= db_lbuf_start + 2) {
214a9643ea8Slogwang 		    c = db_lc[-2];
215a9643ea8Slogwang 		    db_lc[-2] = db_lc[-1];
216a9643ea8Slogwang 		    db_lc[-1] = c;
217a9643ea8Slogwang 		    cnputc(BACKUP);
218a9643ea8Slogwang 		    cnputc(BACKUP);
219a9643ea8Slogwang 		    cnputc(db_lc[-2]);
220a9643ea8Slogwang 		    cnputc(db_lc[-1]);
221a9643ea8Slogwang 		}
222a9643ea8Slogwang 		break;
223*22ce4affSfengbojiang 	    case CTRL('w'):
224*22ce4affSfengbojiang 		/* erase previous word */
225*22ce4affSfengbojiang 		for (; db_lc > db_lbuf_start;) {
226*22ce4affSfengbojiang 		    if (*(db_lc - 1) != ' ')
227*22ce4affSfengbojiang 			break;
228*22ce4affSfengbojiang 		    db_delete(1, DEL_BWD);
229*22ce4affSfengbojiang 		}
230*22ce4affSfengbojiang 		for (; db_lc > db_lbuf_start;) {
231*22ce4affSfengbojiang 		    if (*(db_lc - 1) == ' ')
232*22ce4affSfengbojiang 			break;
233*22ce4affSfengbojiang 		    db_delete(1, DEL_BWD);
234*22ce4affSfengbojiang 		}
235*22ce4affSfengbojiang 		break;
236a9643ea8Slogwang 	    case CTRL('r'):
237a9643ea8Slogwang 		db_putstring("^R\n", 3);
238a9643ea8Slogwang 	    redraw:
239a9643ea8Slogwang 		if (db_le > db_lbuf_start) {
240a9643ea8Slogwang 		    db_putstring(db_lbuf_start, db_le - db_lbuf_start);
241a9643ea8Slogwang 		    db_putnchars(BACKUP, db_le - db_lc);
242a9643ea8Slogwang 		}
243a9643ea8Slogwang 		break;
244a9643ea8Slogwang 	    case CTRL('p'):
245a9643ea8Slogwang 		/* Make previous history line the active one. */
246a9643ea8Slogwang 		if (db_lhistcur >= 0) {
247a9643ea8Slogwang 		    bcopy(db_lhistory + db_lhistcur * db_lhistlsize,
248a9643ea8Slogwang 			  db_lbuf_start, db_lhistlsize);
249a9643ea8Slogwang 		    db_lhistcur--;
250a9643ea8Slogwang 		    goto hist_redraw;
251a9643ea8Slogwang 		}
252a9643ea8Slogwang 		break;
253a9643ea8Slogwang 	    case CTRL('n'):
254a9643ea8Slogwang 		/* Make next history line the active one. */
255a9643ea8Slogwang 		if (db_lhistcur < db_lhistidx - 1) {
256a9643ea8Slogwang 		    db_lhistcur += 2;
257a9643ea8Slogwang 		    bcopy(db_lhistory + db_lhistcur * db_lhistlsize,
258a9643ea8Slogwang 			  db_lbuf_start, db_lhistlsize);
259a9643ea8Slogwang 		} else {
260a9643ea8Slogwang 		    /*
261a9643ea8Slogwang 		     * ^N through tail of history, reset the
262a9643ea8Slogwang 		     * buffer to zero length.
263a9643ea8Slogwang 		     */
264a9643ea8Slogwang 		    *db_lbuf_start = '\0';
265a9643ea8Slogwang 		    db_lhistcur = db_lhistidx;
266a9643ea8Slogwang 		}
267a9643ea8Slogwang 
268a9643ea8Slogwang 	    hist_redraw:
269a9643ea8Slogwang 		db_putnchars(BACKUP, db_lc - db_lbuf_start);
270a9643ea8Slogwang 		db_putnchars(BLANK, db_le - db_lbuf_start);
271a9643ea8Slogwang 		db_putnchars(BACKUP, db_le - db_lbuf_start);
272a9643ea8Slogwang 		db_le = strchr(db_lbuf_start, '\0');
273a9643ea8Slogwang 		if (db_le[-1] == '\r' || db_le[-1] == '\n')
274a9643ea8Slogwang 		    *--db_le = '\0';
275a9643ea8Slogwang 		db_lc = db_le;
276a9643ea8Slogwang 		goto redraw;
277a9643ea8Slogwang 
278a9643ea8Slogwang 	    case -1:
279a9643ea8Slogwang 		/*
280a9643ea8Slogwang 		 * eek! the console returned eof.
281a9643ea8Slogwang 		 * probably that means we HAVE no console.. we should try bail
282a9643ea8Slogwang 		 * XXX
283a9643ea8Slogwang 		 */
284a9643ea8Slogwang 		c = '\r';
285a9643ea8Slogwang 	    case '\n':
286a9643ea8Slogwang 		/* FALLTHROUGH */
287a9643ea8Slogwang 	    case '\r':
288a9643ea8Slogwang 		*db_le++ = c;
289a9643ea8Slogwang 		return (1);
290a9643ea8Slogwang 	    default:
291a9643ea8Slogwang 		if (db_le == db_lbuf_end) {
292a9643ea8Slogwang 		    cnputc('\007');
293a9643ea8Slogwang 		}
294a9643ea8Slogwang 		else if (c >= ' ' && c <= '~') {
295a9643ea8Slogwang 		    char *p;
296a9643ea8Slogwang 
297a9643ea8Slogwang 		    for (p = db_le; p > db_lc; p--)
298a9643ea8Slogwang 			*p = *(p-1);
299a9643ea8Slogwang 		    *db_lc++ = c;
300a9643ea8Slogwang 		    db_le++;
301a9643ea8Slogwang 		    cnputc(c);
302a9643ea8Slogwang 		    db_putstring(db_lc, db_le - db_lc);
303a9643ea8Slogwang 		    db_putnchars(BACKUP, db_le - db_lc);
304a9643ea8Slogwang 		}
305a9643ea8Slogwang 		break;
306a9643ea8Slogwang 	}
307a9643ea8Slogwang 	return (0);
308a9643ea8Slogwang }
309a9643ea8Slogwang 
310a9643ea8Slogwang static int
cnmaygetc()311a9643ea8Slogwang cnmaygetc()
312a9643ea8Slogwang {
313a9643ea8Slogwang 	return (-1);
314a9643ea8Slogwang }
315a9643ea8Slogwang 
316a9643ea8Slogwang int
db_readline(lstart,lsize)317a9643ea8Slogwang db_readline(lstart, lsize)
318a9643ea8Slogwang 	char *	lstart;
319a9643ea8Slogwang 	int	lsize;
320a9643ea8Slogwang {
321a9643ea8Slogwang 
322a9643ea8Slogwang 	if (lsize < 2)
323a9643ea8Slogwang 		return (0);
324a9643ea8Slogwang 	if (lsize != db_lhistlsize) {
325a9643ea8Slogwang 		/*
326a9643ea8Slogwang 		 * (Re)initialize input line history.  Throw away any
327a9643ea8Slogwang 		 * existing history.
328a9643ea8Slogwang 		 */
329a9643ea8Slogwang 		db_lhist_nlines = sizeof(db_lhistory) / lsize;
330a9643ea8Slogwang 		db_lhistlsize = lsize;
331a9643ea8Slogwang 		db_lhistidx = -1;
332a9643ea8Slogwang 	}
333a9643ea8Slogwang 	db_lhistcur = db_lhistidx;
334a9643ea8Slogwang 
335a9643ea8Slogwang 	db_force_whitespace();	/* synch output position */
336a9643ea8Slogwang 
337a9643ea8Slogwang 	db_lbuf_start = lstart;
338a9643ea8Slogwang 	db_lbuf_end   = lstart + lsize - 2;	/* Will append NL and NUL. */
339a9643ea8Slogwang 	db_lc = lstart;
340a9643ea8Slogwang 	db_le = lstart;
341a9643ea8Slogwang 
342a9643ea8Slogwang 	while (!db_inputchar(cngetc()))
343a9643ea8Slogwang 	    continue;
344a9643ea8Slogwang 
345a9643ea8Slogwang 	db_capture_write(lstart, db_le - db_lbuf_start);
346a9643ea8Slogwang 	db_printf("\n");	/* synch output position */
347a9643ea8Slogwang 	*db_le = 0;
348a9643ea8Slogwang 
349a9643ea8Slogwang 	if (db_le - db_lbuf_start > 1) {
350a9643ea8Slogwang 	    /* Maintain input line history for non-empty lines. */
351a9643ea8Slogwang 	    if (++db_lhistidx == db_lhist_nlines) {
352a9643ea8Slogwang 		/* Rotate history. */
353a9643ea8Slogwang 		bcopy(db_lhistory + db_lhistlsize, db_lhistory,
354a9643ea8Slogwang 		      db_lhistlsize * (db_lhist_nlines - 1));
355a9643ea8Slogwang 		db_lhistidx--;
356a9643ea8Slogwang 	    }
357a9643ea8Slogwang 	    bcopy(lstart, db_lhistory + db_lhistidx * db_lhistlsize,
358a9643ea8Slogwang 		  db_lhistlsize);
359a9643ea8Slogwang 	}
360a9643ea8Slogwang 
361a9643ea8Slogwang 	return (db_le - db_lbuf_start);
362a9643ea8Slogwang }
363a9643ea8Slogwang 
364a9643ea8Slogwang void
db_check_interrupt(void)365a9643ea8Slogwang db_check_interrupt(void)
366a9643ea8Slogwang {
367a9643ea8Slogwang 	int	c;
368a9643ea8Slogwang 
369a9643ea8Slogwang 	c = cnmaygetc();
370a9643ea8Slogwang 	switch (c) {
371a9643ea8Slogwang 	    case -1:		/* no character */
372a9643ea8Slogwang 		return;
373a9643ea8Slogwang 
374a9643ea8Slogwang 	    case CTRL('c'):
375a9643ea8Slogwang 		db_error((char *)0);
376a9643ea8Slogwang 		/*NOTREACHED*/
377a9643ea8Slogwang 
378a9643ea8Slogwang 	    case CTRL('s'):
379a9643ea8Slogwang 		do {
380a9643ea8Slogwang 		    c = cnmaygetc();
381a9643ea8Slogwang 		    if (c == CTRL('c'))
382a9643ea8Slogwang 			db_error((char *)0);
383a9643ea8Slogwang 		} while (c != CTRL('q'));
384a9643ea8Slogwang 		break;
385a9643ea8Slogwang 
386a9643ea8Slogwang 	    default:
387a9643ea8Slogwang 		/* drop on floor */
388a9643ea8Slogwang 		break;
389a9643ea8Slogwang 	}
390a9643ea8Slogwang }
391