[PATCH v3 2/5] android: Add line editing to haltest

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]

 



Android does not have readline.
This patch allows to edit command line.
---
 android/Android.mk        |    1 +
 android/client/haltest.c  |   39 +++-
 android/client/terminal.c |  431 +++++++++++++++++++++++++++++++++++++++++++++
 android/client/terminal.h |   59 +++++++
 4 files changed, 527 insertions(+), 3 deletions(-)
 create mode 100644 android/client/terminal.c
 create mode 100644 android/client/terminal.h

diff --git a/android/Android.mk b/android/Android.mk
index c2a0797..2eac3ec 100644
--- a/android/Android.mk
+++ b/android/Android.mk
@@ -59,6 +59,7 @@ include $(CLEAR_VARS)
 LOCAL_SRC_FILES := \
 	client/haltest.c \
 	client/pollhandler.c \
+	client/terminal.c \
 
 LOCAL_SHARED_LIBRARIES := libhardware
 
diff --git a/android/client/haltest.c b/android/client/haltest.c
index 11cdd97..de98079 100644
--- a/android/client/haltest.c
+++ b/android/client/haltest.c
@@ -19,8 +19,40 @@
 #include <poll.h>
 #include <unistd.h>
 
+#include "terminal.h"
 #include "pollhandler.h"
 
+/*
+ * This function changes input parameter line_buffer so it has
+ * null termination after each token (due to strtok)
+ * Output argv is filled with pointers to arguments
+ * returns number of tokens parsed - argc
+ */
+static int command_line_to_argv(char *line_buffer,
+				char *argv[], int argv_size)
+{
+	static const char *token_breaks = "\r\n\t ";
+	char *token;
+	int argc = 0;
+
+	token = strtok(line_buffer, token_breaks);
+	while (token != NULL && argc < (int) argv_size) {
+		argv[argc++] = token;
+		token = strtok(NULL, token_breaks);
+	}
+
+	return argc;
+}
+
+static void process_line(char *line_buffer)
+{
+	char *argv[10];
+	int argc;
+
+	argc = command_line_to_argv(line_buffer, argv, 10);
+
+	/* TODO: process command line */
+}
 
 /* called when there is something on stdin */
 static void stdin_handler(struct pollfd *pollfd)
@@ -33,15 +65,16 @@ static void stdin_handler(struct pollfd *pollfd)
 		if (count > 0) {
 			int i;
 
-			for (i = 0; i < count; ++i) {
-				/* TODO: process input */
-			}
+			for (i = 0; i < count; ++i)
+				terminal_process_char(buf[i], process_line);
 		}
 	}
 }
 
 int main(int argc, char **argv)
 {
+	terminal_setup();
+
 	/* Register command line handler */
 	poll_register_fd(0, POLLIN, stdin_handler);
 
diff --git a/android/client/terminal.c b/android/client/terminal.c
new file mode 100644
index 0000000..95108a0
--- /dev/null
+++ b/android/client/terminal.c
@@ -0,0 +1,431 @@
+/*
+ * Copyright (C) 2013 Intel Corporation
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ *      http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ *
+ */
+
+#include <stdio.h>
+#include <string.h>
+#include <ctype.h>
+#include <stdbool.h>
+#include <termios.h>
+
+#include "terminal.h"
+
+/*
+ * Character sequences recognized by code in this file
+ * Leading ESC 0x1B is not included
+ */
+#define SEQ_INSERT "[2~"
+#define SEQ_DELETE "[3~"
+#define SEQ_HOME   "OH"
+#define SEQ_END    "OF"
+#define SEQ_PGUP   "[5~"
+#define SEQ_PGDOWN "[6~"
+#define SEQ_LEFT   "[D"
+#define SEQ_RIGHT  "[C"
+#define SEQ_UP     "[A"
+#define SEQ_DOWN   "[B"
+#define SEQ_STAB   "[Z"
+#define SEQ_M_n    "n"
+#define SEQ_M_p    "p"
+#define SEQ_CLEFT  "[1;5D"
+#define SEQ_CRIGHT "[1;5C"
+#define SEQ_CUP    "[1;5A"
+#define SEQ_CDOWN  "[1;5B"
+#define SEQ_SLEFT  "[1;2D"
+#define SEQ_SRIGHT "[1;2C"
+#define SEQ_SUP    "[1;2A"
+#define SEQ_SDOWN  "[1;2B"
+#define SEQ_MLEFT  "[1;3D"
+#define SEQ_MRIGHT "[1;3C"
+#define SEQ_MUP    "[1;3A"
+#define SEQ_MDOWN  "[1;3B"
+
+#define KEY_SEQUENCE(k) { KEY_##k, SEQ_##k }
+struct ansii_sequence {
+	int code;
+	const char *sequence;
+};
+
+/* Table connects single int key codes with character sequences */
+static const struct ansii_sequence ansii_sequnces[] = {
+	KEY_SEQUENCE(INSERT),
+	KEY_SEQUENCE(DELETE),
+	KEY_SEQUENCE(HOME),
+	KEY_SEQUENCE(END),
+	KEY_SEQUENCE(PGUP),
+	KEY_SEQUENCE(PGDOWN),
+	KEY_SEQUENCE(LEFT),
+	KEY_SEQUENCE(RIGHT),
+	KEY_SEQUENCE(UP),
+	KEY_SEQUENCE(DOWN),
+	KEY_SEQUENCE(CLEFT),
+	KEY_SEQUENCE(CRIGHT),
+	KEY_SEQUENCE(CUP),
+	KEY_SEQUENCE(CDOWN),
+	KEY_SEQUENCE(SLEFT),
+	KEY_SEQUENCE(SRIGHT),
+	KEY_SEQUENCE(SUP),
+	KEY_SEQUENCE(SDOWN),
+	KEY_SEQUENCE(MLEFT),
+	KEY_SEQUENCE(MRIGHT),
+	KEY_SEQUENCE(MUP),
+	KEY_SEQUENCE(MDOWN),
+	KEY_SEQUENCE(STAB),
+	KEY_SEQUENCE(M_p),
+	KEY_SEQUENCE(M_n),
+	{ 0, NULL }
+};
+
+#define isseqence(c) ((c) == 0x1B)
+
+/*
+ * Number of characters that consist of ANSII sequence
+ * Should not be less then longest string in ansii_sequnces
+ */
+#define MAX_ASCII_SEQUENCE 10
+
+static char current_sequence[MAX_ASCII_SEQUENCE];
+static int current_sequence_len = -1;
+
+/* single line typed by user goes here */
+static char line_buf[LINE_BUF_MAX];
+/* index of cursor in input line */
+static int line_buf_ix = 0;
+/* current length of input line */
+static int line_len = 0;
+
+/*
+ * Moves cursor to right or left
+ *
+ * n - positive - moves cursor right
+ * n - negative - moves cursor left
+ */
+static void terminal_move_cursor(int n)
+{
+	if (n < 0) {
+		for (; n < 0; n++)
+			putchar('\b');
+	} else if (n > 0) {
+		printf("%*s", n, line_buf + line_buf_ix);
+	}
+}
+
+/* Draw command line */
+void terminal_draw_command_line(void)
+{
+	/*
+	 * this needs to be checked here since line_buf is not cleard
+	 * before parsing event though line_len and line_buf_ix are
+	 */
+	if (line_len > 0)
+		printf(">%s", line_buf);
+	else
+		putchar('>');
+
+	/* move cursor to it's place */
+	terminal_move_cursor(line_len - line_buf_ix);
+}
+
+/* inserts string into command line at cursor position */
+void terminal_insert_into_command_line(const char *p)
+{
+	int len = strlen(p);
+
+	if (line_len == line_buf_ix) {
+		strcat(line_buf, p);
+		printf("%s", p);
+		line_len = line_len + len;
+		line_buf_ix = line_len;
+	} else {
+		memmove(line_buf + line_buf_ix + len,
+			line_buf + line_buf_ix, line_len - line_buf_ix + 1);
+		memmove(line_buf + line_buf_ix, p, len);
+		printf("%s", line_buf + line_buf_ix);
+		line_buf_ix += len;
+		line_len += len;
+		terminal_move_cursor(line_buf_ix - line_len);
+	}
+}
+
+/* Prints string and redraws command line */
+int terminal_print(const char *format, ...)
+{
+	va_list args;
+	int ret;
+
+	va_start(args, format);
+
+	ret = terminal_vprint(format, args);
+
+	va_end(args);
+	return ret;
+}
+
+/* Prints string and redraws command line */
+int terminal_vprint(const char *format, va_list args)
+{
+	int ret;
+
+	printf("\r%*s\r", (int) line_len + 1, " ");
+
+	ret = vprintf(format, args);
+
+	terminal_draw_command_line();
+
+	return ret;
+}
+
+/*
+ * Converts terminal character sequences to single value representing
+ * keyboard keys
+ */
+static int terminal_convert_sequence(int c)
+{
+	int i;
+
+	/* Not in sequence yet? */
+	if (current_sequence_len == -1) {
+		/* Is ansii sequence detected by 0x1B ? */
+		if (isseqence(c)) {
+			current_sequence_len++;
+			return 0;
+	       }
+	       return c;
+	}
+	/* Inside sequence */
+	current_sequence[current_sequence_len++] = c;
+	current_sequence[current_sequence_len] = '\0';
+	for (i = 0; ansii_sequnces[i].code; ++i) {
+		/* Matches so far? */
+		if (0 != strncmp(current_sequence, ansii_sequnces[i].sequence,
+							current_sequence_len))
+			continue;
+
+		/* Matches as a whole? */
+		if (ansii_sequnces[i].sequence[current_sequence_len] == 0) {
+			current_sequence_len = -1;
+			return ansii_sequnces[i].code;
+		}
+		/* partial match (not whole sequence yet) */
+		return 0;
+	}
+	terminal_print("ansii char 0x%X %c\n", c);
+	/*
+	 * Sequence does not match
+	 * mark that no in sequence any more, return char
+	 */
+	current_sequence_len = -1;
+	return c;
+}
+
+void terminal_process_char(int c, void (*process_line)(char *line))
+{
+	int refresh_from = -1;
+	int old_pos;
+
+	c = terminal_convert_sequence(c);
+
+	switch (c) {
+	case 0:
+		break;
+	case KEY_LEFT:
+		/* if not at the beginning move to previous character */
+		if (line_buf_ix <= 0)
+			break;
+		line_buf_ix--;
+		terminal_move_cursor(-1);
+		break;
+	case KEY_RIGHT:
+		/*
+		 * If not at the end, just print current character
+		 * and modify position
+		 */
+		if (line_buf_ix < line_len)
+			putchar(line_buf[line_buf_ix++]);
+		break;
+	case KEY_HOME:
+		/* move to beginning of line and update position */
+		putchar('\r');
+		putchar('>');
+		line_buf_ix = 0;
+		break;
+	case KEY_END:
+		/* if not at the end of line */
+		if (line_buf_ix < line_len) {
+			/* print everything from cursor */
+			printf("%s", line_buf + line_buf_ix);
+			/* just modify current position */
+			line_buf_ix = line_len;
+		}
+		break;
+	case KEY_DELETE:
+		/* delete character under cursor if not at the very end */
+		if (line_buf_ix >= line_len)
+			break;
+		/*
+		 * Prepare buffer with one character missing
+		 * trailing 0 is moved
+		 */
+		line_len--;
+		memmove(line_buf + line_buf_ix,
+			line_buf + line_buf_ix + 1,
+			line_len - line_buf_ix + 1);
+		/* print rest of line from current cursor position */
+		printf("%s \b", line_buf + line_buf_ix);
+		/* move back cursor */
+		terminal_move_cursor(line_buf_ix - line_len);
+		break;
+	case KEY_CLEFT:
+		/*
+		 * Move by word left
+		 *
+		 * Are we at the beginning of line?
+		 */
+		if (line_buf_ix <= 0)
+			break;
+
+		old_pos = line_buf_ix;
+		line_buf_ix--;
+		/* skip spaces left */
+		while (line_buf_ix && isspace(line_buf[line_buf_ix]))
+			line_buf_ix--;
+		/* skip all non spaces to the left */
+		while (line_buf_ix > 0 &&
+		       !isspace(line_buf[line_buf_ix - 1]))
+			line_buf_ix--;
+		/* move cursor to new position */
+		terminal_move_cursor(line_buf_ix - old_pos);
+		break;
+	case KEY_CRIGHT:
+		/*
+		 * Move by word right
+		 *
+		 * are we at the end of line?
+		 */
+		if (line_buf_ix >= line_len)
+			break;
+
+		old_pos = line_buf_ix;
+		/* skip all spaces */
+		while (line_buf_ix < line_len &&
+			isspace(line_buf[line_buf_ix]))
+			line_buf_ix++;
+		/* skip all non spaces */
+		while (line_buf_ix < line_len &&
+			!isspace(line_buf[line_buf_ix]))
+			line_buf_ix++;
+		/*
+		 * Move cursor to right by printing text
+		 * between old cursor and new
+		 */
+		if (line_buf_ix > old_pos)
+			printf("%.*s", (int) (line_buf_ix - old_pos),
+							line_buf + old_pos);
+		break;
+	case KEY_UP:
+	case KEY_DOWN:
+		break;
+	case '\n':
+	case '\r':
+		line_len = 0;
+		line_buf_ix = 0;
+		/* print new line */
+		putchar(c);
+		process_line(line_buf);
+		/* clear current line */
+		line_buf[0] = '\0';
+		putchar('>');
+		break;
+	case '\t':
+		/* tab processing */
+		/* TODO Add completion here */
+		break;
+	case KEY_BACKSPACE:
+		if (line_buf_ix <= 0)
+			break;
+
+		if (line_buf_ix == line_len) {
+			printf("\b \b");
+			line_len = --line_buf_ix;
+			line_buf[line_len] = 0;
+		} else {
+			putchar('\b');
+			refresh_from = --line_buf_ix;
+			line_len--;
+			memmove(line_buf + line_buf_ix,
+				line_buf + line_buf_ix + 1,
+				line_len - line_buf_ix + 1);
+		}
+		break;
+	case KEY_INSERT:
+	case KEY_PGUP:
+	case KEY_PGDOWN:
+	case KEY_CUP:
+	case KEY_CDOWN:
+	case KEY_SLEFT:
+	case KEY_SRIGHT:
+	case KEY_MLEFT:
+	case KEY_MRIGHT:
+	case KEY_MUP:
+	case KEY_MDOWN:
+	case KEY_STAB:
+	case KEY_M_n:
+	case KEY_M_p:
+		break;
+	default:
+		if (!isprint(c)) {
+			/*
+			 * TODO: remove this print once all meaningful sequences
+			 * are identified
+			 */
+			printf("char-0x%02x\n", c);
+			break;
+		}
+		if (line_buf_ix < LINE_BUF_MAX - 1) {
+			if (line_len == line_buf_ix) {
+				putchar(c);
+				line_buf[line_buf_ix++] = (char) c;
+				line_len++;
+				line_buf[line_len] = '\0';
+			} else {
+				memmove(line_buf + line_buf_ix + 1,
+					line_buf + line_buf_ix,
+					line_len - line_buf_ix + 1);
+				line_buf[line_buf_ix] = c;
+				refresh_from = line_buf_ix++;
+				line_len++;
+			}
+		}
+		break;
+	}
+
+	if (refresh_from >= 0) {
+		printf("%s \b", line_buf + refresh_from);
+		terminal_move_cursor(line_buf_ix - line_len);
+	}
+}
+
+void terminal_setup(void)
+{
+	struct termios tios;
+
+	/* Turn off echo since all editing is done by hand */
+	tcgetattr(0, &tios);
+	tios.c_lflag &= ~(ICANON | ECHO);
+	tcsetattr(0, TCSANOW, &tios);
+	putchar('>');
+}
+
diff --git a/android/client/terminal.h b/android/client/terminal.h
new file mode 100644
index 0000000..e53750f
--- /dev/null
+++ b/android/client/terminal.h
@@ -0,0 +1,59 @@
+/*
+ * Copyright (C) 2013 Intel Corporation
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ *      http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ *
+ */
+
+#include <stdarg.h>
+
+/* size of supported line */
+#define LINE_BUF_MAX 1024
+
+enum key_codes {
+	KEY_BACKSPACE = 0x7F,
+	KEY_INSERT = 1000, /* arbitrary value */
+	KEY_DELETE,
+	KEY_HOME,
+	KEY_END,
+	KEY_PGUP,
+	KEY_PGDOWN,
+	KEY_LEFT,
+	KEY_RIGHT,
+	KEY_UP,
+	KEY_DOWN,
+	KEY_CLEFT,
+	KEY_CRIGHT,
+	KEY_CUP,
+	KEY_CDOWN,
+	KEY_SLEFT,
+	KEY_SRIGHT,
+	KEY_SUP,
+	KEY_SDOWN,
+	KEY_MLEFT,
+	KEY_MRIGHT,
+	KEY_MUP,
+	KEY_MDOWN,
+	KEY_STAB,
+	KEY_M_p,
+	KEY_M_n
+};
+
+void terminal_setup(void);
+int terminal_print(const char *format, ...);
+int terminal_vprint(const char *format, va_list args);
+void terminal_process_char(int c, void (*process_line)(char *line));
+void terminal_insert_into_command_line(const char *p);
+void terminal_draw_command_line(void);
+
+void process_tab(const char *line, int len);
-- 
1.7.9.5

--
To unsubscribe from this list: send the line "unsubscribe linux-bluetooth" in
the body of a message to majordomo@xxxxxxxxxxxxxxx
More majordomo info at  http://vger.kernel.org/majordomo-info.html




[Index of Archives]     [Bluez Devel]     [Linux Wireless Networking]     [Linux Wireless Personal Area Networking]     [Linux ATH6KL]     [Linux USB Devel]     [Linux Media Drivers]     [Linux Audio Users]     [Linux Kernel]     [Linux SCSI]     [Big List of Linux Books]

  Powered by Linux