1dd3cb568SWarner Losh /*-
2796df753SPedro F. Giffuni * SPDX-License-Identifier: MIT-CMU
3796df753SPedro F. Giffuni *
45b81b6b3SRodney W. Grimes * Mach Operating System
55b81b6b3SRodney W. Grimes * Copyright (c) 1991,1990 Carnegie Mellon University
65b81b6b3SRodney W. Grimes * All Rights Reserved.
75b81b6b3SRodney W. Grimes *
85b81b6b3SRodney W. Grimes * Permission to use, copy, modify and distribute this software and its
95b81b6b3SRodney W. Grimes * documentation is hereby granted, provided that both the copyright
105b81b6b3SRodney W. Grimes * notice and this permission notice appear in all copies of the
115b81b6b3SRodney W. Grimes * software, derivative works or modified versions, and any portions
125b81b6b3SRodney W. Grimes * thereof, and that both notices appear in supporting documentation.
135b81b6b3SRodney W. Grimes *
145b81b6b3SRodney W. Grimes * CARNEGIE MELLON ALLOWS FREE USE OF THIS SOFTWARE IN ITS
155b81b6b3SRodney W. Grimes * CONDITION. CARNEGIE MELLON DISCLAIMS ANY LIABILITY OF ANY KIND FOR
165b81b6b3SRodney W. Grimes * ANY DAMAGES WHATSOEVER RESULTING FROM THE USE OF THIS SOFTWARE.
175b81b6b3SRodney W. Grimes *
185b81b6b3SRodney W. Grimes * Carnegie Mellon requests users of this software to return to
195b81b6b3SRodney W. Grimes *
205b81b6b3SRodney W. Grimes * Software Distribution Coordinator or Software.Distribution@CS.CMU.EDU
215b81b6b3SRodney W. Grimes * School of Computer Science
225b81b6b3SRodney W. Grimes * Carnegie Mellon University
235b81b6b3SRodney W. Grimes * Pittsburgh PA 15213-3890
245b81b6b3SRodney W. Grimes *
255b81b6b3SRodney W. Grimes * any improvements or extensions that they make and grant Carnegie the
265b81b6b3SRodney W. Grimes * rights to redistribute these changes.
275b81b6b3SRodney W. Grimes */
285b81b6b3SRodney W. Grimes /*
295b81b6b3SRodney W. Grimes * Author: David B. Golub, Carnegie Mellon University
305b81b6b3SRodney W. Grimes * Date: 7/90
315b81b6b3SRodney W. Grimes */
325b81b6b3SRodney W. Grimes
33f540b106SGarrett Wollman #include <sys/param.h>
34f540b106SGarrett Wollman #include <sys/systm.h>
35ce9edcf5SPoul-Henning Kamp #include <sys/cons.h>
363e5e9939SRyan Libby #include <sys/sysctl.h>
375ccbc3ccSBruce Evans
38f540b106SGarrett Wollman #include <ddb/ddb.h>
39f540b106SGarrett Wollman #include <ddb/db_output.h>
405b81b6b3SRodney W. Grimes
415b81b6b3SRodney W. Grimes /*
425b81b6b3SRodney W. Grimes * Character input and editing.
435b81b6b3SRodney W. Grimes */
445b81b6b3SRodney W. Grimes
455b81b6b3SRodney W. Grimes /*
465b81b6b3SRodney W. Grimes * We don't track output position while editing input,
475b81b6b3SRodney W. Grimes * since input always ends with a new-line. We just
485b81b6b3SRodney W. Grimes * reset the line position at the end.
495b81b6b3SRodney W. Grimes */
5025eb640dSPoul-Henning Kamp static char * db_lbuf_start; /* start of input line buffer */
5125eb640dSPoul-Henning Kamp static char * db_lbuf_end; /* end of input line buffer */
5225eb640dSPoul-Henning Kamp static char * db_lc; /* current character */
5325eb640dSPoul-Henning Kamp static char * db_le; /* one past last character */
545b81b6b3SRodney W. Grimes
55b4630773SJoerg Wunsch /*
563e5e9939SRyan Libby * Raw input buffer, processed only for certain control characters.
573e5e9939SRyan Libby */
583e5e9939SRyan Libby #define DB_RAW_SIZE 512
593e5e9939SRyan Libby static char db_raw[DB_RAW_SIZE];
603e5e9939SRyan Libby static u_int db_raw_pos;
613e5e9939SRyan Libby static u_int db_raw_cnt;
623e5e9939SRyan Libby static int db_raw_warned;
633e5e9939SRyan Libby static int ddb_prioritize_control_input = 1;
643e5e9939SRyan Libby SYSCTL_INT(_debug_ddb, OID_AUTO, prioritize_control_input, CTLFLAG_RWTUN,
653e5e9939SRyan Libby &ddb_prioritize_control_input, 0,
663e5e9939SRyan Libby "Drop input when the buffer fills in order to keep servicing ^C/^S/^Q");
673e5e9939SRyan Libby
683e5e9939SRyan Libby /*
69b4630773SJoerg Wunsch * Simple input line history support.
70b4630773SJoerg Wunsch */
7141630a01SBruce Evans static char db_lhistory[2048];
72b4630773SJoerg Wunsch static int db_lhistlsize, db_lhistidx, db_lhistcur;
73d984ae1eSMike Smith static int db_lhist_nlines;
74b4630773SJoerg Wunsch
755b81b6b3SRodney W. Grimes #define CTRL(c) ((c) & 0x1f)
765b81b6b3SRodney W. Grimes #define BLANK ' '
775b81b6b3SRodney W. Grimes #define BACKUP '\b'
785b81b6b3SRodney W. Grimes
7914e10f99SAlfred Perlstein static void db_delete(int n, int bwd);
8014e10f99SAlfred Perlstein static int db_inputchar(int c);
8114e10f99SAlfred Perlstein static void db_putnchars(int c, int count);
8214e10f99SAlfred Perlstein static void db_putstring(char *s, int count);
833e5e9939SRyan Libby static int db_raw_pop(void);
843e5e9939SRyan Libby static void db_raw_push(int);
853e5e9939SRyan Libby static int db_raw_space(void);
86058284fcSBruce Evans
8737c84183SPoul-Henning Kamp static void
db_putstring(char * s,int count)88*fb1b78bfSMateusz Guzik db_putstring(char *s, int count)
895b81b6b3SRodney W. Grimes {
905b81b6b3SRodney W. Grimes while (--count >= 0)
915b81b6b3SRodney W. Grimes cnputc(*s++);
925b81b6b3SRodney W. Grimes }
935b81b6b3SRodney W. Grimes
9437c84183SPoul-Henning Kamp static void
db_putnchars(int c,int count)95*fb1b78bfSMateusz Guzik db_putnchars(int c, int count)
965b81b6b3SRodney W. Grimes {
975b81b6b3SRodney W. Grimes while (--count >= 0)
985b81b6b3SRodney W. Grimes cnputc(c);
995b81b6b3SRodney W. Grimes }
1005b81b6b3SRodney W. Grimes
1015b81b6b3SRodney W. Grimes /*
1025b81b6b3SRodney W. Grimes * Delete N characters, forward or backward
1035b81b6b3SRodney W. Grimes */
1045b81b6b3SRodney W. Grimes #define DEL_FWD 0
1055b81b6b3SRodney W. Grimes #define DEL_BWD 1
10637c84183SPoul-Henning Kamp static void
db_delete(int n,int bwd)107*fb1b78bfSMateusz Guzik db_delete(int n, int bwd)
1085b81b6b3SRodney W. Grimes {
1090a95ab74SPedro F. Giffuni char *p;
1105b81b6b3SRodney W. Grimes
1115b81b6b3SRodney W. Grimes if (bwd) {
1125b81b6b3SRodney W. Grimes db_lc -= n;
1135b81b6b3SRodney W. Grimes db_putnchars(BACKUP, n);
1145b81b6b3SRodney W. Grimes }
1155b81b6b3SRodney W. Grimes for (p = db_lc; p < db_le-n; p++) {
1165b81b6b3SRodney W. Grimes *p = *(p+n);
1175b81b6b3SRodney W. Grimes cnputc(*p);
1185b81b6b3SRodney W. Grimes }
1195b81b6b3SRodney W. Grimes db_putnchars(BLANK, n);
1205b81b6b3SRodney W. Grimes db_putnchars(BACKUP, db_le - db_lc);
1215b81b6b3SRodney W. Grimes db_le -= n;
1225b81b6b3SRodney W. Grimes }
1235b81b6b3SRodney W. Grimes
1242b490bc7SPedro F. Giffuni /* returns true at end-of-line */
12537c84183SPoul-Henning Kamp static int
db_inputchar(int c)126*fb1b78bfSMateusz Guzik db_inputchar(int c)
1275b81b6b3SRodney W. Grimes {
128eae6ab5eSJoerg Wunsch static int escstate;
129eae6ab5eSJoerg Wunsch
130eae6ab5eSJoerg Wunsch if (escstate == 1) {
131eae6ab5eSJoerg Wunsch /* ESC seen, look for [ or O */
132eae6ab5eSJoerg Wunsch if (c == '[' || c == 'O')
133eae6ab5eSJoerg Wunsch escstate++;
134eae6ab5eSJoerg Wunsch else
135eae6ab5eSJoerg Wunsch escstate = 0; /* re-init state machine */
136eae6ab5eSJoerg Wunsch return (0);
137eae6ab5eSJoerg Wunsch } else if (escstate == 2) {
138eae6ab5eSJoerg Wunsch escstate = 0;
139eae6ab5eSJoerg Wunsch /*
140eae6ab5eSJoerg Wunsch * If a valid cursor key has been found, translate
141eae6ab5eSJoerg Wunsch * into an emacs-style control key, and fall through.
142eae6ab5eSJoerg Wunsch * Otherwise, drop off.
143eae6ab5eSJoerg Wunsch */
1445b81b6b3SRodney W. Grimes switch (c) {
145eae6ab5eSJoerg Wunsch case 'A': /* up */
146eae6ab5eSJoerg Wunsch c = CTRL('p');
147eae6ab5eSJoerg Wunsch break;
148eae6ab5eSJoerg Wunsch case 'B': /* down */
149eae6ab5eSJoerg Wunsch c = CTRL('n');
150eae6ab5eSJoerg Wunsch break;
151eae6ab5eSJoerg Wunsch case 'C': /* right */
152eae6ab5eSJoerg Wunsch c = CTRL('f');
153eae6ab5eSJoerg Wunsch break;
154eae6ab5eSJoerg Wunsch case 'D': /* left */
155eae6ab5eSJoerg Wunsch c = CTRL('b');
156eae6ab5eSJoerg Wunsch break;
157eae6ab5eSJoerg Wunsch default:
158eae6ab5eSJoerg Wunsch return (0);
159eae6ab5eSJoerg Wunsch }
160eae6ab5eSJoerg Wunsch }
161eae6ab5eSJoerg Wunsch
162eae6ab5eSJoerg Wunsch switch (c) {
163eae6ab5eSJoerg Wunsch case CTRL('['):
164eae6ab5eSJoerg Wunsch escstate = 1;
165eae6ab5eSJoerg Wunsch break;
1665b81b6b3SRodney W. Grimes case CTRL('b'):
1675b81b6b3SRodney W. Grimes /* back up one character */
1685b81b6b3SRodney W. Grimes if (db_lc > db_lbuf_start) {
1695b81b6b3SRodney W. Grimes cnputc(BACKUP);
1705b81b6b3SRodney W. Grimes db_lc--;
1715b81b6b3SRodney W. Grimes }
1725b81b6b3SRodney W. Grimes break;
1735b81b6b3SRodney W. Grimes case CTRL('f'):
1745b81b6b3SRodney W. Grimes /* forward one character */
1755b81b6b3SRodney W. Grimes if (db_lc < db_le) {
1765b81b6b3SRodney W. Grimes cnputc(*db_lc);
1775b81b6b3SRodney W. Grimes db_lc++;
1785b81b6b3SRodney W. Grimes }
1795b81b6b3SRodney W. Grimes break;
1805b81b6b3SRodney W. Grimes case CTRL('a'):
1815b81b6b3SRodney W. Grimes /* beginning of line */
1825b81b6b3SRodney W. Grimes while (db_lc > db_lbuf_start) {
1835b81b6b3SRodney W. Grimes cnputc(BACKUP);
1845b81b6b3SRodney W. Grimes db_lc--;
1855b81b6b3SRodney W. Grimes }
1865b81b6b3SRodney W. Grimes break;
1875b81b6b3SRodney W. Grimes case CTRL('e'):
1885b81b6b3SRodney W. Grimes /* end of line */
1895b81b6b3SRodney W. Grimes while (db_lc < db_le) {
1905b81b6b3SRodney W. Grimes cnputc(*db_lc);
1915b81b6b3SRodney W. Grimes db_lc++;
1925b81b6b3SRodney W. Grimes }
1935b81b6b3SRodney W. Grimes break;
1945b81b6b3SRodney W. Grimes case CTRL('h'):
1955b81b6b3SRodney W. Grimes case 0177:
1965b81b6b3SRodney W. Grimes /* erase previous character */
1975b81b6b3SRodney W. Grimes if (db_lc > db_lbuf_start)
1985b81b6b3SRodney W. Grimes db_delete(1, DEL_BWD);
1995b81b6b3SRodney W. Grimes break;
2005b81b6b3SRodney W. Grimes case CTRL('d'):
2015b81b6b3SRodney W. Grimes /* erase next character */
2025b81b6b3SRodney W. Grimes if (db_lc < db_le)
2035b81b6b3SRodney W. Grimes db_delete(1, DEL_FWD);
2045b81b6b3SRodney W. Grimes break;
20509ffa9adSYaroslav Tykhiy case CTRL('u'):
20681e12dd1SEdward Tomasz Napierala case CTRL('c'):
20709ffa9adSYaroslav Tykhiy /* kill entire line: */
20809ffa9adSYaroslav Tykhiy /* at first, delete to beginning of line */
20909ffa9adSYaroslav Tykhiy if (db_lc > db_lbuf_start)
21009ffa9adSYaroslav Tykhiy db_delete(db_lc - db_lbuf_start, DEL_BWD);
21109ffa9adSYaroslav Tykhiy /* FALLTHROUGH */
2125b81b6b3SRodney W. Grimes case CTRL('k'):
2135b81b6b3SRodney W. Grimes /* delete to end of line */
2145b81b6b3SRodney W. Grimes if (db_lc < db_le)
2155b81b6b3SRodney W. Grimes db_delete(db_le - db_lc, DEL_FWD);
2165b81b6b3SRodney W. Grimes break;
2175b81b6b3SRodney W. Grimes case CTRL('t'):
2185b81b6b3SRodney W. Grimes /* twiddle last 2 characters */
2195b81b6b3SRodney W. Grimes if (db_lc >= db_lbuf_start + 2) {
2205b81b6b3SRodney W. Grimes c = db_lc[-2];
2215b81b6b3SRodney W. Grimes db_lc[-2] = db_lc[-1];
2225b81b6b3SRodney W. Grimes db_lc[-1] = c;
2235b81b6b3SRodney W. Grimes cnputc(BACKUP);
2245b81b6b3SRodney W. Grimes cnputc(BACKUP);
2255b81b6b3SRodney W. Grimes cnputc(db_lc[-2]);
2265b81b6b3SRodney W. Grimes cnputc(db_lc[-1]);
2275b81b6b3SRodney W. Grimes }
2285b81b6b3SRodney W. Grimes break;
229eeb6101eSEdward Tomasz Napierala case CTRL('w'):
230eeb6101eSEdward Tomasz Napierala /* erase previous word */
231eeb6101eSEdward Tomasz Napierala for (; db_lc > db_lbuf_start;) {
232eeb6101eSEdward Tomasz Napierala if (*(db_lc - 1) != ' ')
233eeb6101eSEdward Tomasz Napierala break;
234eeb6101eSEdward Tomasz Napierala db_delete(1, DEL_BWD);
235eeb6101eSEdward Tomasz Napierala }
236eeb6101eSEdward Tomasz Napierala for (; db_lc > db_lbuf_start;) {
237eeb6101eSEdward Tomasz Napierala if (*(db_lc - 1) == ' ')
238eeb6101eSEdward Tomasz Napierala break;
239eeb6101eSEdward Tomasz Napierala db_delete(1, DEL_BWD);
240eeb6101eSEdward Tomasz Napierala }
241eeb6101eSEdward Tomasz Napierala break;
2425b81b6b3SRodney W. Grimes case CTRL('r'):
2435b81b6b3SRodney W. Grimes db_putstring("^R\n", 3);
244b4630773SJoerg Wunsch redraw:
2455b81b6b3SRodney W. Grimes if (db_le > db_lbuf_start) {
2465b81b6b3SRodney W. Grimes db_putstring(db_lbuf_start, db_le - db_lbuf_start);
2475b81b6b3SRodney W. Grimes db_putnchars(BACKUP, db_le - db_lc);
2485b81b6b3SRodney W. Grimes }
2495b81b6b3SRodney W. Grimes break;
250b4630773SJoerg Wunsch case CTRL('p'):
251b4630773SJoerg Wunsch /* Make previous history line the active one. */
252b4630773SJoerg Wunsch if (db_lhistcur >= 0) {
253b4630773SJoerg Wunsch bcopy(db_lhistory + db_lhistcur * db_lhistlsize,
254b4630773SJoerg Wunsch db_lbuf_start, db_lhistlsize);
255b4630773SJoerg Wunsch db_lhistcur--;
256b4630773SJoerg Wunsch goto hist_redraw;
257b4630773SJoerg Wunsch }
258b4630773SJoerg Wunsch break;
259b4630773SJoerg Wunsch case CTRL('n'):
260b4630773SJoerg Wunsch /* Make next history line the active one. */
261b4630773SJoerg Wunsch if (db_lhistcur < db_lhistidx - 1) {
262b4630773SJoerg Wunsch db_lhistcur += 2;
263b4630773SJoerg Wunsch bcopy(db_lhistory + db_lhistcur * db_lhistlsize,
264b4630773SJoerg Wunsch db_lbuf_start, db_lhistlsize);
265b4630773SJoerg Wunsch } else {
266b4630773SJoerg Wunsch /*
267b4630773SJoerg Wunsch * ^N through tail of history, reset the
268b4630773SJoerg Wunsch * buffer to zero length.
269b4630773SJoerg Wunsch */
270b4630773SJoerg Wunsch *db_lbuf_start = '\0';
271b4630773SJoerg Wunsch db_lhistcur = db_lhistidx;
272b4630773SJoerg Wunsch }
273b4630773SJoerg Wunsch
274b4630773SJoerg Wunsch hist_redraw:
275ea1c6a39SRobert Watson db_putnchars(BACKUP, db_lc - db_lbuf_start);
276b4630773SJoerg Wunsch db_putnchars(BLANK, db_le - db_lbuf_start);
277b4630773SJoerg Wunsch db_putnchars(BACKUP, db_le - db_lbuf_start);
278dc15eac0SEd Schouten db_le = strchr(db_lbuf_start, '\0');
279b4630773SJoerg Wunsch if (db_le[-1] == '\r' || db_le[-1] == '\n')
280b4630773SJoerg Wunsch *--db_le = '\0';
281b4630773SJoerg Wunsch db_lc = db_le;
282b4630773SJoerg Wunsch goto redraw;
283b4630773SJoerg Wunsch
28475680b05SJulian Elischer case -1:
28575680b05SJulian Elischer /*
28675680b05SJulian Elischer * eek! the console returned eof.
28775680b05SJulian Elischer * probably that means we HAVE no console.. we should try bail
28875680b05SJulian Elischer * XXX
28975680b05SJulian Elischer */
29075680b05SJulian Elischer c = '\r';
2915b81b6b3SRodney W. Grimes case '\n':
2929a4b535cSPoul-Henning Kamp /* FALLTHROUGH */
2935b81b6b3SRodney W. Grimes case '\r':
2945b81b6b3SRodney W. Grimes *db_le++ = c;
2955b81b6b3SRodney W. Grimes return (1);
2965b81b6b3SRodney W. Grimes default:
2975b81b6b3SRodney W. Grimes if (db_le == db_lbuf_end) {
2985b81b6b3SRodney W. Grimes cnputc('\007');
2995b81b6b3SRodney W. Grimes }
3005b81b6b3SRodney W. Grimes else if (c >= ' ' && c <= '~') {
3010a95ab74SPedro F. Giffuni char *p;
3025b81b6b3SRodney W. Grimes
3035b81b6b3SRodney W. Grimes for (p = db_le; p > db_lc; p--)
3045b81b6b3SRodney W. Grimes *p = *(p-1);
3055b81b6b3SRodney W. Grimes *db_lc++ = c;
3065b81b6b3SRodney W. Grimes db_le++;
3075b81b6b3SRodney W. Grimes cnputc(c);
3085b81b6b3SRodney W. Grimes db_putstring(db_lc, db_le - db_lc);
3095b81b6b3SRodney W. Grimes db_putnchars(BACKUP, db_le - db_lc);
3105b81b6b3SRodney W. Grimes }
3115b81b6b3SRodney W. Grimes break;
3125b81b6b3SRodney W. Grimes }
3135b81b6b3SRodney W. Grimes return (0);
3145b81b6b3SRodney W. Grimes }
3155b81b6b3SRodney W. Grimes
3163e5e9939SRyan Libby /* Get a character from the console, first checking the raw input buffer. */
3173e5e9939SRyan Libby int
db_getc(void)3183e5e9939SRyan Libby db_getc(void)
319c2718b42SMark Johnston {
3203e5e9939SRyan Libby int c;
3213e5e9939SRyan Libby
3223e5e9939SRyan Libby if (db_raw_cnt == 0) {
3233e5e9939SRyan Libby c = cngetc();
3243e5e9939SRyan Libby } else {
3253e5e9939SRyan Libby c = db_raw_pop();
3263e5e9939SRyan Libby if (c == '\r')
3273e5e9939SRyan Libby c = '\n';
3283e5e9939SRyan Libby }
3293e5e9939SRyan Libby return (c);
3303e5e9939SRyan Libby }
3313e5e9939SRyan Libby
3323e5e9939SRyan Libby /* Whether the raw input buffer has space to accept another character. */
3333e5e9939SRyan Libby static int
db_raw_space(void)3343e5e9939SRyan Libby db_raw_space(void)
3353e5e9939SRyan Libby {
3363e5e9939SRyan Libby
3373e5e9939SRyan Libby return (db_raw_cnt < DB_RAW_SIZE);
3383e5e9939SRyan Libby }
3393e5e9939SRyan Libby
3403e5e9939SRyan Libby /* Un-get a character from the console by buffering it. */
3413e5e9939SRyan Libby static void
db_raw_push(int c)3423e5e9939SRyan Libby db_raw_push(int c)
3433e5e9939SRyan Libby {
3443e5e9939SRyan Libby
3453e5e9939SRyan Libby if (!db_raw_space())
3463e5e9939SRyan Libby db_error(NULL);
3473e5e9939SRyan Libby db_raw[(db_raw_pos + db_raw_cnt++) % DB_RAW_SIZE] = c;
3483e5e9939SRyan Libby }
3493e5e9939SRyan Libby
3503e5e9939SRyan Libby /* Drain a character from the raw input buffer. */
3513e5e9939SRyan Libby static int
db_raw_pop(void)3523e5e9939SRyan Libby db_raw_pop(void)
3533e5e9939SRyan Libby {
3543e5e9939SRyan Libby
3553e5e9939SRyan Libby if (db_raw_cnt == 0)
356c2718b42SMark Johnston return (-1);
3573e5e9939SRyan Libby db_raw_cnt--;
3583e5e9939SRyan Libby db_raw_warned = 0;
3593e5e9939SRyan Libby return (db_raw[db_raw_pos++ % DB_RAW_SIZE]);
360c2718b42SMark Johnston }
361c2718b42SMark Johnston
362f23b4c91SGarrett Wollman int
db_readline(char * lstart,int lsize)363*fb1b78bfSMateusz Guzik db_readline(char *lstart, int lsize)
3645b81b6b3SRodney W. Grimes {
365233f8184SRobert Watson
366233f8184SRobert Watson if (lsize < 2)
367233f8184SRobert Watson return (0);
36841630a01SBruce Evans if (lsize != db_lhistlsize) {
36941630a01SBruce Evans /*
37041630a01SBruce Evans * (Re)initialize input line history. Throw away any
37141630a01SBruce Evans * existing history.
37241630a01SBruce Evans */
37341630a01SBruce Evans db_lhist_nlines = sizeof(db_lhistory) / lsize;
374b4630773SJoerg Wunsch db_lhistlsize = lsize;
375b4630773SJoerg Wunsch db_lhistidx = -1;
376b4630773SJoerg Wunsch }
377b4630773SJoerg Wunsch db_lhistcur = db_lhistidx;
378b4630773SJoerg Wunsch
3795b81b6b3SRodney W. Grimes db_force_whitespace(); /* synch output position */
3805b81b6b3SRodney W. Grimes
3815b81b6b3SRodney W. Grimes db_lbuf_start = lstart;
382233f8184SRobert Watson db_lbuf_end = lstart + lsize - 2; /* Will append NL and NUL. */
3835b81b6b3SRodney W. Grimes db_lc = lstart;
3845b81b6b3SRodney W. Grimes db_le = lstart;
3855b81b6b3SRodney W. Grimes
3863e5e9939SRyan Libby while (!db_inputchar(db_getc()))
3875b81b6b3SRodney W. Grimes continue;
3885b81b6b3SRodney W. Grimes
389086fec57SRobert Watson db_capture_write(lstart, db_le - db_lbuf_start);
3906ddbf1e2SGary Palmer db_printf("\n"); /* synch output position */
3915b81b6b3SRodney W. Grimes *db_le = 0;
392b4630773SJoerg Wunsch
39341630a01SBruce Evans if (db_le - db_lbuf_start > 1) {
394b4630773SJoerg Wunsch /* Maintain input line history for non-empty lines. */
395d984ae1eSMike Smith if (++db_lhistidx == db_lhist_nlines) {
396b4630773SJoerg Wunsch /* Rotate history. */
3978543efaeSDag-Erling Smørgrav bcopy(db_lhistory + db_lhistlsize, db_lhistory,
398d984ae1eSMike Smith db_lhistlsize * (db_lhist_nlines - 1));
399b4630773SJoerg Wunsch db_lhistidx--;
400b4630773SJoerg Wunsch }
40141630a01SBruce Evans bcopy(lstart, db_lhistory + db_lhistidx * db_lhistlsize,
402b4630773SJoerg Wunsch db_lhistlsize);
403b4630773SJoerg Wunsch }
404b4630773SJoerg Wunsch
4055b81b6b3SRodney W. Grimes return (db_le - db_lbuf_start);
4065b81b6b3SRodney W. Grimes }
4075b81b6b3SRodney W. Grimes
4083e5e9939SRyan Libby static void
db_do_interrupt(const char * reason)4093e5e9939SRyan Libby db_do_interrupt(const char *reason)
4103e5e9939SRyan Libby {
4113e5e9939SRyan Libby
4123e5e9939SRyan Libby /* Do a pager quit too because some commands have jmpbuf handling. */
4133e5e9939SRyan Libby db_disable_pager();
4143e5e9939SRyan Libby db_pager_quit = 1;
4153e5e9939SRyan Libby db_error(reason);
4163e5e9939SRyan Libby }
4173e5e9939SRyan Libby
4185b81b6b3SRodney W. Grimes void
db_check_interrupt(void)419a41dd031SPedro F. Giffuni db_check_interrupt(void)
4205b81b6b3SRodney W. Grimes {
4210a95ab74SPedro F. Giffuni int c;
4225b81b6b3SRodney W. Grimes
4233e5e9939SRyan Libby /*
4243e5e9939SRyan Libby * Check console input for control characters. Non-control input is
4253e5e9939SRyan Libby * buffered. When buffer space is exhausted, either stop responding to
4263e5e9939SRyan Libby * control input or drop further non-control input on the floor.
4273e5e9939SRyan Libby */
4283e5e9939SRyan Libby for (;;) {
4293e5e9939SRyan Libby if (!ddb_prioritize_control_input && !db_raw_space())
4303e5e9939SRyan Libby return;
4313e5e9939SRyan Libby c = cncheckc();
4325b81b6b3SRodney W. Grimes switch (c) {
4335b81b6b3SRodney W. Grimes case -1: /* no character */
4345b81b6b3SRodney W. Grimes return;
4355b81b6b3SRodney W. Grimes
4365b81b6b3SRodney W. Grimes case CTRL('c'):
4373e5e9939SRyan Libby db_do_interrupt("^C");
4385b81b6b3SRodney W. Grimes /*NOTREACHED*/
4395b81b6b3SRodney W. Grimes
4405b81b6b3SRodney W. Grimes case CTRL('s'):
4415b81b6b3SRodney W. Grimes do {
4423e5e9939SRyan Libby c = cncheckc();
4435b81b6b3SRodney W. Grimes if (c == CTRL('c'))
4443e5e9939SRyan Libby db_do_interrupt("^C");
4455b81b6b3SRodney W. Grimes } while (c != CTRL('q'));
4465b81b6b3SRodney W. Grimes break;
4475b81b6b3SRodney W. Grimes
4485b81b6b3SRodney W. Grimes default:
4493e5e9939SRyan Libby if (db_raw_space()) {
4503e5e9939SRyan Libby db_raw_push(c);
4513e5e9939SRyan Libby } else if (!db_raw_warned) {
4523e5e9939SRyan Libby db_raw_warned = 1;
4533e5e9939SRyan Libby db_printf("\n--Exceeded input buffer--\n");
4543e5e9939SRyan Libby }
4555b81b6b3SRodney W. Grimes break;
4565b81b6b3SRodney W. Grimes }
4575b81b6b3SRodney W. Grimes }
4583e5e9939SRyan Libby }
459