xref: /minix3/sys/arch/i386/stand/lib/menuutils.c (revision d19d7d58aa5cd1165eefe1a0d807c8afe282db62)
1 /*	$NetBSD: menuutils.c,v 1.3 2008/12/14 18:46:33 christos Exp $	*/
2 
3 /*
4  * Copyright (c) 1996, 1997
5  * 	Matthias Drochner.  All rights reserved.
6  * Copyright (c) 1996, 1997
7  * 	Perry E. Metzger.  All rights reserved.
8  * Copyright (c) 1997
9  *	Jason R. Thorpe.  All rights reserved
10  *
11  * Redistribution and use in source and binary forms, with or without
12  * modification, are permitted provided that the following conditions
13  * are met:
14  * 1. Redistributions of source code must retain the above copyright
15  *    notice, this list of conditions and the following disclaimer.
16  * 2. Redistributions in binary form must reproduce the above copyright
17  *    notice, this list of conditions and the following disclaimer in the
18  *    documentation and/or other materials provided with the distribution.
19  * 3. All advertising materials mentioning features or use of this software
20  *    must display the following acknowledgements:
21  *	This product includes software developed for the NetBSD Project
22  *	by Matthias Drochner.
23  *	This product includes software developed for the NetBSD Project
24  *	by Perry E. Metzger.
25  * 4. The names of the authors may not be used to endorse or promote products
26  *    derived from this software without specific prior written permission.
27  *
28  * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
29  * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
30  * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
31  * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
32  * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
33  * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
34  * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
35  * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
36  * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
37  * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
38  */
39 
40 #include <lib/libkern/libkern.h>
41 #include <lib/libsa/stand.h>
42 
43 #include "libi386.h"
44 
45 void
46 docommand(char *arg)
47 {
48 	char *options;
49 	int i;
50 
51 	options = gettrailer(arg);
52 
53 	for (i = 0; commands[i].c_name != NULL; i++) {
54 		if (strcmp(arg, commands[i].c_name) == 0) {
55 			(*commands[i].c_fn)(options);
56 			return;
57 		}
58 	}
59 
60 	printf("unknown command\n");
61 	command_help(NULL);
62 }
63 
64 void
65 prompt(int allowreturn)
66 {
67 	char input[80];
68 
69 	for (;;) {
70 		char *c = input;
71 
72 		printf("> ");
73 		editline(input, sizeof(input), NULL);
74 
75 		/*
76 		 * Skip leading whitespace.
77 		 */
78 		while (*c == ' ')
79 			c++;
80 		if (allowreturn && !strcmp(c, "menu"))
81 			break;
82 		if (*c)
83 			docommand(c);
84 	}
85 }
86 
87 void
88 bootmenu(void)
89 {
90 	prompt(0);
91 }
92 
93 /* Derived from libsa gets(). */
94 void
95 editline(char *buf, size_t size, char *input)
96 {
97 	int c, i, pos, len = 0;
98 
99 	/* If an initial input has been given, copy and print this first. */
100 	if (input != NULL) {
101 		while (*input && len < size - 1)
102 			putchar(buf[len++] = *input++);
103 	}
104 	pos = len;
105 
106 	for (;;) {
107 		c = getchar_ex();
108 		switch (c & 0177) {
109 		case '\0':
110 			switch (c) {
111 			case 0x4b00: /* Left arrow: move cursor to left. */
112 				if (pos > 0) {
113 					putchar('\b');
114 					pos--;
115 				}
116 				break;
117 			case 0x4d00: /* Right arrow: move cursor to right. */
118 				if (pos < len) putchar(buf[pos++]);
119 				break;
120 			}
121 			break;
122 		case 'b' & 037: /* Ctrl+B: move cursor to left. */
123 			if (pos > 0) {
124 				putchar('\b');
125 				pos--;
126 			}
127 			break;
128 		case 'f' & 037: /* Ctrl+F: move cursor to right. */
129 			if (pos < len) putchar(buf[pos++]);
130 			break;
131 		case 'a' & 037: /* Ctrl+A: move cursor to start of line. */
132 			for ( ; pos > 0; pos--) putchar('\b');
133 			break;
134 		case 'e' & 037: /* Ctrl+E: move cursor to end of line. */
135 			for ( ; pos < len; pos++) putchar(buf[pos]);
136 			break;
137 		case '\n': /* Enter: return line. */
138 		case '\r':
139 			for ( ; pos < len; pos++) putchar(buf[pos]);
140 			buf[len] = '\0';
141 			putchar('\n');
142 			return;
143 #if HASH_ERASE
144 		case '#':
145 #endif
146 		case '\b': /* Backspace: erase character before cursor. */
147 		case '\177':
148 			if (pos > 0) {
149 				pos--;
150 				len--;
151 				putchar('\b');
152 				for (i = pos; i < len; i++)
153 					putchar(buf[i] = buf[i + 1]);
154 				putchar(' ');
155 				for (i = pos; i < len; i++) putchar('\b');
156 				putchar('\b');
157 			}
158 			break;
159 		case 'r' & 037: /* Ctrl+R: reprint line. */
160 			putchar('\n');
161 			for (i = 0; i < len; i++) putchar(buf[i]);
162 			for (i = len; i > pos; i--) putchar('\b');
163 			break;
164 #if AT_ERASE
165 		case '@':
166 #endif
167 		case 'u' & 037: /* Ctrl+U: clear entire line. */
168 		case 'w' & 037:
169 			for ( ; pos > 0; pos--) putchar('\b');
170 			for ( ; pos < len; pos++) putchar(' ');
171 			for ( ; pos > 0; pos--) putchar('\b');
172 			len = 0;
173 			break;
174 		case '\a': /* Ctrl+G: sound bell but do not store character. */
175 			putchar(c);
176 			break;
177 		case '\t': /* Tab: convert to single space. */
178 			c = ' ';
179 			/*FALLTHROUGH*/
180 		default: /* Insert character at cursor position. */
181 			if (len < size - 1) {
182 				for (i = len; i > pos; i--)
183 					buf[i] = buf[i - 1];
184 				buf[pos] = c;
185 				pos++;
186 				len++;
187 				putchar(c);
188 				for (i = pos; i < len; i++) putchar(buf[i]);
189 				for (i = pos; i < len; i++) putchar('\b');
190 			} else {
191 				putchar('\a');
192 			}
193 			break;
194 		}
195 	}
196 	/*NOTREACHED*/
197 }
198