wok-current rev 5711

busybox: add conspy
author Pascal Bellard <pascal.bellard@slitaz.org>
date Wed Jun 23 21:48:13 2010 +0200 (2010-06-23)
parents 932522d70bdb
children 386214f606f5
files busybox/receipt busybox/stuff/busybox-1.16.1-modinfo.u busybox/stuff/busybox-1.16.1-modprobe.u busybox/stuff/busybox-1.16.1-stat.u busybox/stuff/busybox-1.16.1-vcsa2txt.u
line diff
     1.1 --- a/busybox/receipt	Wed Jun 23 16:11:06 2010 +0200
     1.2 +++ b/busybox/receipt	Wed Jun 23 21:48:13 2010 +0200
     1.3 @@ -36,8 +36,13 @@
     1.4  EOT
     1.5      cp ../stuff/$PACKAGE-$VERSION.config .config
     1.6      make oldconfig
     1.7 -    # "CFLAGS=-O0" is a workaround for GCC 4.5.0
     1.8 -    make -j 4 "CFLAGS=-O0" && make "CFLAGS=-O0" install
     1.9 +    if [ "$(gcc --version | awk '{ print $3; exit }')" == "4.5.0" ]; then
    1.10 +	# "CFLAGS=-O0" is a workaround for GCC 4.5.0 (sed crach)
    1.11 +	# "CFLAGS=-fno-tree-pta" may be a workaround for GCC 4.5.0 (sed garbage)
    1.12 +	make -j 4 "CFLAGS=-O0" && make "CFLAGS=-O0" install
    1.13 +    else
    1.14 +	make -j 4 && make install
    1.15 +    fi
    1.16      echo "Chmod 4755 on busybox binary..."
    1.17      chmod 4755 _install/bin/busybox
    1.18  }
    1.19 @@ -88,10 +93,10 @@
    1.20  		[ -e $ROOT$i ] || continue
    1.21  		if [ -z "$answer" ]; then
    1.22  			echo -n "Keep installed GNU utilities ? "
    1.23 -			read answer
    1.24 +			read -t 30 answer	# by default: keep
    1.25  			case "$answer" in
    1.26 -			y*|Y*|o*|O*);;
    1.27 -			*) break;;
    1.28 +			n*|N*) break;;
    1.29 +			*) ;;
    1.30  			esac
    1.31  		fi
    1.32  		mv $ROOT$i $ROOT$i-busybox-install
     2.1 --- a/busybox/stuff/busybox-1.16.1-modinfo.u	Wed Jun 23 16:11:06 2010 +0200
     2.2 +++ b/busybox/stuff/busybox-1.16.1-modinfo.u	Wed Jun 23 21:48:13 2010 +0200
     2.3 @@ -1,6 +1,6 @@
     2.4  --- busybox-1.16.1/include/applets.h
     2.5  +++ busybox-1.16.1/include/applets.h
     2.6 -@@ -272,6 +272,7 @@
     2.7 +@@ -273,6 +273,7 @@
     2.8   IF_CRYPTPW(APPLET_ODDNAME(mkpasswd, cryptpw, _BB_DIR_USR_BIN, _BB_SUID_DROP, mkpasswd))
     2.9   IF_MKSWAP(APPLET(mkswap, _BB_DIR_SBIN, _BB_SUID_DROP))
    2.10   IF_MKTEMP(APPLET(mktemp, _BB_DIR_BIN, _BB_SUID_DROP))
    2.11 @@ -11,7 +11,7 @@
    2.12  
    2.13  --- busybox-1.16.1/include/usage.h
    2.14  +++ busybox-1.16.1/include/usage.h
    2.15 -@@ -2967,6 +2967,20 @@
    2.16 +@@ -2984,6 +2984,20 @@
    2.17          "   which are the default for alias 'tulip2' overridden by the options 'irq=2 io=0x210'\n\n" \
    2.18          "   from the command line\n"
    2.19   
     3.1 --- a/busybox/stuff/busybox-1.16.1-modprobe.u	Wed Jun 23 16:11:06 2010 +0200
     3.2 +++ b/busybox/stuff/busybox-1.16.1-modprobe.u	Wed Jun 23 21:48:13 2010 +0200
     3.3 @@ -1,6 +1,6 @@
     3.4  --- busybox-1.16.1/include/usage.h
     3.5  +++ busybox-1.16.1/include/usage.h
     3.6 -@@ -2881,7 +2881,7 @@
     3.7 +@@ -2898,7 +2898,7 @@
     3.8   #define modprobe_trivial_usage \
     3.9   	IF_MODPROBE_SMALL("[-qfwrsv] MODULE [symbol=value]...") \
    3.10   	IF_NOT_MODPROBE_SMALL("[-" \
    3.11 @@ -9,7 +9,7 @@
    3.12   		IF_FEATURE_MODPROBE_BLACKLIST("b")"] MODULE [symbol=value]...")
    3.13   #define modprobe_full_usage "\n\n" \
    3.14          "Options:" \
    3.15 -@@ -2898,6 +2898,7 @@
    3.16 +@@ -2915,6 +2915,7 @@
    3.17        "\n	-k	Make module autoclean-able" \
    3.18   	) \
    3.19        "\n	-n	Dry run" \
     4.1 --- a/busybox/stuff/busybox-1.16.1-stat.u	Wed Jun 23 16:11:06 2010 +0200
     4.2 +++ b/busybox/stuff/busybox-1.16.1-stat.u	Wed Jun 23 21:48:13 2010 +0200
     4.3 @@ -65,7 +65,7 @@
     4.4  
     4.5  --- busybox-1.16.1/include/usage.h
     4.6  +++ busybox-1.16.1/include/usage.h
     4.7 -@@ -4184,6 +4184,7 @@
     4.8 +@@ -4201,6 +4201,7 @@
     4.9        "\n	-f	Display filesystem status" \
    4.10        "\n	-L	Follow links" \
    4.11        "\n	-t	Display info in terse form" \
     5.1 --- a/busybox/stuff/busybox-1.16.1-vcsa2txt.u	Wed Jun 23 16:11:06 2010 +0200
     5.2 +++ b/busybox/stuff/busybox-1.16.1-vcsa2txt.u	Wed Jun 23 21:48:13 2010 +0200
     5.3 @@ -1,6 +1,14 @@
     5.4  --- busybox-1.16.1/include/applets.h
     5.5  +++ busybox-1.16.1/include/applets.h
     5.6 -@@ -420,6 +420,7 @@
     5.7 +@@ -105,6 +105,7 @@
     5.8 + IF_CLEAR(APPLET(clear, _BB_DIR_USR_BIN, _BB_SUID_DROP))
     5.9 + IF_CMP(APPLET(cmp, _BB_DIR_USR_BIN, _BB_SUID_DROP))
    5.10 + IF_COMM(APPLET(comm, _BB_DIR_USR_BIN, _BB_SUID_DROP))
    5.11 ++IF_VCSA2TXT(APPLET(conspy, _BB_DIR_USR_BIN, _BB_SUID_DROP))
    5.12 + IF_CP(APPLET_NOEXEC(cp, cp, _BB_DIR_BIN, _BB_SUID_DROP, cp))
    5.13 + IF_CPIO(APPLET(cpio, _BB_DIR_BIN, _BB_SUID_DROP))
    5.14 + IF_CROND(APPLET(crond, _BB_DIR_USR_SBIN, _BB_SUID_DROP))
    5.15 +@@ -420,6 +421,7 @@
    5.16   IF_UUDECODE(APPLET(uudecode, _BB_DIR_USR_BIN, _BB_SUID_DROP))
    5.17   IF_UUENCODE(APPLET(uuencode, _BB_DIR_USR_BIN, _BB_SUID_DROP))
    5.18   IF_VCONFIG(APPLET(vconfig, _BB_DIR_SBIN, _BB_SUID_DROP))
    5.19 @@ -11,14 +19,38 @@
    5.20  
    5.21  --- busybox-1.16.1/include/usage.h
    5.22  +++ busybox-1.16.1/include/usage.h
    5.23 -@@ -5011,6 +5011,13 @@
    5.24 +@@ -440,6 +440,23 @@
    5.25 +      "\n	-1		Close stdout" \
    5.26 +      "\n	-2		Close stderr" \
    5.27 + 
    5.28 ++#define conspy_trivial_usage \
    5.29 ++	"[-vcsnd] [-x rows] [-y lines] [virtual_console]"
    5.30 ++#define conspy_full_usage "\n\n" \
    5.31 ++	"A text-mode VNC like program for Linux virtual terminals.\n" \
    5.32 ++	"\nTo exit, quickly press escape 3 times.\n" \
    5.33 ++	"\nvirtual_console:\n" \
    5.34 ++	"  omitted  Track the current console.\n" \
    5.35 ++	"  1..63    Virtual console N.\n" \
    5.36 ++	"\nOptions:" \
    5.37 ++	"\n	-v	Don't send keystrokes to the console." \
    5.38 ++	"\n	-c	May create device in /dev." \
    5.39 ++	"\n	-s	Open a SHELL session." \
    5.40 ++	"\n	-n	No colors. Black & white only." \
    5.41 ++	"\n	-d	Dump console to stdout."\
    5.42 ++	"\n	-x r	Skip the r first rows."\
    5.43 ++	"\n	-y l	Skip the l first lines."\
    5.44 ++
    5.45 + #define setuidgid_trivial_usage \
    5.46 +        "USER PROG ARGS"
    5.47 + #define setuidgid_full_usage "\n\n" \
    5.48 +@@ -5011,6 +5028,13 @@
    5.49        "\n	set_ingress_map	[vlan-name] [skb_priority] [vlan_qos]" \
    5.50        "\n	set_name_type	[name-type]" \
    5.51   
    5.52  +#define vcsa2txt_trivial_usage \
    5.53  +       "stdin"
    5.54  +#define vcsa2txt_full_usage \
    5.55 -+       "Filter /dev/vcsa* to ansi escape sequences"
    5.56 ++       "	Filter /dev/vcsa* to ansi escape sequences"
    5.57  +#define vcsa2txt_example_usage \
    5.58  +       "# vcsa2txt < /dev/vcsa1\n"
    5.59  +
    5.60 @@ -33,7 +65,7 @@
    5.61   	  Write a message to all users that are logged in.
    5.62   
    5.63  +config VCSA2TXT
    5.64 -+	bool "vcsa2txt"
    5.65 ++	bool "vcsa2txt/conspy"
    5.66  +	default n
    5.67  +	help
    5.68  +	  Filter /dev/vcsa* output to ansi escape sequences.
    5.69 @@ -52,83 +84,493 @@
    5.70  
    5.71  --- busybox-1.16.1/util-linux/vcsa2txt.c
    5.72  +++ busybox-1.16.1/util-linux/vcsa2txt.c
    5.73 -@@ -0,0 +1,79 @@
    5.74 +@@ -0,0 +1,489 @@
    5.75  +/* vi: set sw=4 ts=4: */
    5.76  +/*
    5.77 -+ * /dev/vcsa* filter for busybox
    5.78 ++ * A text-mode VNC like program for Linux virtual terminals.
    5.79  + *
    5.80  + * pascal.bellard@ads-lu.com
    5.81  + *
    5.82 ++ * Based on Russell Stuart's conspy.c
    5.83 ++ *   http://ace-host.stuart.id.au/russell/files/conspy.c
    5.84 ++ *
    5.85  + * Licensed under GPLv2 or later, see file License in this tarball for details.
    5.86 ++ *
    5.87 ++ * example :	conspy num		shared access to console num
    5.88 ++ * or		conspy -d num		screenshot of console num
    5.89 ++ * or		conspy -cs num		poor man's GNU screen like
    5.90  + */
    5.91 -+ 
    5.92 ++
    5.93 ++//applet:IF_CONSPY(APPLET(conspy, _BB_DIR_BIN, _BB_SUID_DROP))
    5.94 ++
    5.95 ++//kbuild:lib-$(CONFIG_CONSPY) += conspy.o
    5.96 ++
    5.97 ++//config:config CONSPY
    5.98 ++//config:	bool "conspy"
    5.99 ++//config:	default n
   5.100 ++//config:	help
   5.101 ++//config:	  A text-mode VNC like program for Linux virtual terminals.
   5.102 ++//config:	  example : conspy num      shared access to console num
   5.103 ++//config:	  or        conspy -d num   screenshot of console num
   5.104 ++//config:	  or        conspy -cs num  poor man's GNU screen like
   5.105 ++
   5.106 ++//usage:#define conspy_trivial_usage
   5.107 ++//usage:     "[-vcsndf] [-x ROW] [-y LINE] [CONSOLE_NO]"
   5.108 ++//usage:#define conspy_full_usage "\n\n"
   5.109 ++//usage:     "A text-mode VNC like program for Linux virtual consoles."
   5.110 ++//usage:     "\nTo exit, quickly press ESC 3 times."
   5.111 ++//usage:     "\n"
   5.112 ++//usage:     "\nOptions:"
   5.113 ++//usage:     "\n	-v	Don't send keystrokes to the console"
   5.114 ++//usage:     "\n	-c	Create missing devices in /dev"
   5.115 ++//usage:     "\n	-s	Open a SHELL session"
   5.116 ++//usage:     "\n	-n	Black & white"
   5.117 ++//usage:     "\n	-d	Dump console to stdout"
   5.118 ++//usage:     "\n	-f	Follow cursor"
   5.119 ++//usage:     "\n	-x ROW	Starting row"
   5.120 ++//usage:     "\n	-y LINE	Starting line"
   5.121 ++
   5.122  +#include "libbb.h"
   5.123 ++#include <sys/kd.h>
   5.124 ++
   5.125 ++struct screen_info {
   5.126 ++	unsigned char lines, rows, cursor_x, cursor_y;
   5.127 ++};
   5.128 ++
   5.129 ++#define CHAR(x) ((x)[0])
   5.130 ++#define ATTR(x) ((x)[1])
   5.131 ++#define NEXT(x) ((x)+=2)
   5.132 ++#define DATA(x) (* (short *) (x))
   5.133 ++
   5.134 ++struct globals {
   5.135 ++	char* data;
   5.136 ++	int size;
   5.137 ++	int x, y;
   5.138 ++	int kbd_fd;
   5.139 ++	unsigned width;
   5.140 ++	unsigned height;
   5.141 ++	char last_attr;
   5.142 ++	int ioerror_count;
   5.143 ++	int key_count;
   5.144 ++	int escape_count;
   5.145 ++	int nokeys;
   5.146 ++	int current;
   5.147 ++	int vcsa_fd;
   5.148 ++	struct screen_info info;
   5.149 ++	struct termios term_orig;
   5.150 ++};
   5.151 ++
   5.152 ++#define G (*ptr_to_globals)
   5.153 ++#define INIT_G() do { \
   5.154 ++	SET_PTR_TO_GLOBALS(xzalloc(sizeof(G))); \
   5.155 ++} while (0)
   5.156 ++
   5.157 ++enum {
   5.158 ++	FLAG_v,  // view only
   5.159 ++	FLAG_c,  // create device if need
   5.160 ++	FLAG_s,  // session
   5.161 ++	FLAG_n,  // no colors
   5.162 ++	FLAG_d,  // dump screen
   5.163 ++	FLAG_f,  // follow cursor
   5.164 ++};
   5.165 ++#define FLAG(x) (1 << FLAG_##x)
   5.166 ++#define BW (option_mask32 & FLAG(n))
   5.167 ++
   5.168 ++static void screen_read_close(void)
   5.169 ++{
   5.170 ++	unsigned i, j;
   5.171 ++	char *data = G.data + G.current;
   5.172 ++
   5.173 ++	xread(G.vcsa_fd, data, G.size);
   5.174 ++	G.last_attr = 0;
   5.175 ++	for (i = 0; i < G.info.lines; i++) {
   5.176 ++		for (j = 0; j < G.info.rows; j++, NEXT(data)) {
   5.177 ++			unsigned x = j - G.x; // if will catch j < G.x too
   5.178 ++			unsigned y = i - G.y; // if will catch i < G.y too
   5.179 ++
   5.180 ++			if (CHAR(data) < ' ')
   5.181 ++				CHAR(data) = ' ';
   5.182 ++			if (y >= G.height || x >= G.width)
   5.183 ++				DATA(data) = 0;
   5.184 ++		}
   5.185 ++	}
   5.186 ++	close(G.vcsa_fd);
   5.187 ++}
   5.188 ++
   5.189 ++static void screen_char(char *data)
   5.190 ++{
   5.191 ++	if (!BW && G.last_attr != ATTR(data)) {
   5.192 ++		//                            BLGCRMOW
   5.193 ++		static const char color[8] = "04261537";
   5.194 ++
   5.195 ++		printf("\033[%c;4%c;3%cm",
   5.196 ++			(ATTR(data) & 8) ? '1'  // bold
   5.197 ++					 : '0', // defaults
   5.198 ++			color[(ATTR(data) >> 4) & 7], color[ATTR(data) & 7]);
   5.199 ++		G.last_attr = ATTR(data);
   5.200 ++	}
   5.201 ++	bb_putchar(CHAR(data));
   5.202 ++}
   5.203 ++
   5.204 ++#define clrscr()  printf("\033[1;1H" "\033[0J")
   5.205 ++#define curoff()  printf("\033[?25l")
   5.206 ++
   5.207 ++static void curon(void)
   5.208 ++{
   5.209 ++	printf("\033[?25h");
   5.210 ++}
   5.211 ++
   5.212 ++static void gotoxy(int row, int line)
   5.213 ++{
   5.214 ++	printf("\033[%u;%uH", line + 1, row + 1);
   5.215 ++}
   5.216 ++
   5.217 ++static void screen_dump(void)
   5.218 ++{
   5.219 ++	int linefeed_cnt;
   5.220 ++	int line, row;
   5.221 ++	int linecnt = G.info.lines - G.y;
   5.222 ++	char *data = G.data + G.current + (2 * G.y * G.info.rows);
   5.223 ++
   5.224 ++	linefeed_cnt = 0;
   5.225 ++	for (line = 0; line < linecnt && line < G.height; line++) {
   5.226 ++		int space_cnt = 0;
   5.227 ++		for (row = 0; row < G.info.rows; row++, NEXT(data)) {
   5.228 ++			unsigned tty_row = row - G.x; // if will catch row < G.x too
   5.229 ++
   5.230 ++			if (tty_row >= G.width)
   5.231 ++				continue;
   5.232 ++			space_cnt++;
   5.233 ++			if (BW && (CHAR(data) | ' ') == ' ')
   5.234 ++				continue;
   5.235 ++			while (linefeed_cnt != 0) {
   5.236 ++				bb_putchar('\r');
   5.237 ++				bb_putchar('\n');
   5.238 ++				linefeed_cnt--;
   5.239 ++			}
   5.240 ++			while (--space_cnt)
   5.241 ++				bb_putchar(' ');
   5.242 ++			screen_char(data);
   5.243 ++		}
   5.244 ++		linefeed_cnt++;
   5.245 ++	}
   5.246 ++}
   5.247 ++
   5.248 ++static void curmove(void)
   5.249 ++{
   5.250 ++	unsigned cx = G.info.cursor_x - G.x;
   5.251 ++	unsigned cy = G.info.cursor_y - G.y;
   5.252 ++
   5.253 ++	if (cx >= G.width || cy >= G.height) {
   5.254 ++		curoff();
   5.255 ++	} else {
   5.256 ++		curon();
   5.257 ++		gotoxy(cx, cy);
   5.258 ++	}
   5.259 ++	fflush_all();
   5.260 ++}
   5.261 ++
   5.262 ++static void cleanup(int code)
   5.263 ++{
   5.264 ++	curon();
   5.265 ++	fflush_all();
   5.266 ++	tcsetattr(G.kbd_fd, TCSANOW, &G.term_orig);
   5.267 ++	if (ENABLE_FEATURE_CLEAN_UP) {
   5.268 ++		free(ptr_to_globals);
   5.269 ++		close(G.kbd_fd);
   5.270 ++	}
   5.271 ++	// Reset attributes
   5.272 ++	if (!BW)
   5.273 ++		printf("\033[0m");
   5.274 ++	bb_putchar('\n');
   5.275 ++	if (code > 1)
   5.276 ++		kill_myself_with_sig(code); // does not return
   5.277 ++	exit(code);
   5.278 ++}
   5.279 ++
   5.280 ++static void get_initial_data(const char* vcsa_name)
   5.281 ++{
   5.282 ++	G.vcsa_fd = xopen(vcsa_name, O_RDONLY);
   5.283 ++	xread(G.vcsa_fd, &G.info, 4);
   5.284 ++	G.size = G.info.rows * G.info.lines * 2;
   5.285 ++	G.width = G.height = UINT_MAX;
   5.286 ++	G.data = xzalloc(2 * G.size);
   5.287 ++	screen_read_close();
   5.288 ++}
   5.289 ++
   5.290 ++static void create_cdev_if_doesnt_exist(const char* name, dev_t dev)
   5.291 ++{
   5.292 ++	int fd = open(name, O_RDONLY);
   5.293 ++	if (fd != -1)
   5.294 ++		close(fd);
   5.295 ++	else if (errno == ENOENT)
   5.296 ++		mknod(name, S_IFCHR | 0660, dev);
   5.297 ++}
   5.298 ++
   5.299 ++static NOINLINE void start_shell_in_child(const char* tty_name)
   5.300 ++{
   5.301 ++	int pid = vfork();
   5.302 ++	if (pid < 0) {
   5.303 ++		bb_perror_msg_and_die("vfork");
   5.304 ++	}
   5.305 ++	if (pid == 0) {
   5.306 ++		struct termios termchild;
   5.307 ++		char *shell = getenv("SHELL");
   5.308 ++
   5.309 ++		if (!shell)
   5.310 ++			shell = (char *) DEFAULT_SHELL;
   5.311 ++		signal(SIGHUP, SIG_IGN);
   5.312 ++		// set tty as a controlling tty
   5.313 ++		setsid();
   5.314 ++		// make tty to be input, output, error
   5.315 ++		close(0);
   5.316 ++		xopen(tty_name, O_RDWR); // uses fd 0
   5.317 ++		xdup2(0, 1);
   5.318 ++		xdup2(0, 2);
   5.319 ++		ioctl(0, TIOCSCTTY, 1);
   5.320 ++		tcsetpgrp(0, getpid());
   5.321 ++		tcgetattr(0, &termchild);
   5.322 ++		termchild.c_lflag |= ECHO;
   5.323 ++		termchild.c_oflag |= ONLCR | XTABS;
   5.324 ++		termchild.c_iflag |= ICRNL;
   5.325 ++		termchild.c_iflag &= ~IXOFF;
   5.326 ++		tcsetattr_stdin_TCSANOW(&termchild);
   5.327 ++		execl(shell, shell, "-i", (char *) NULL);
   5.328 ++		bb_simple_perror_msg_and_die(shell);
   5.329 ++	}
   5.330 ++}
   5.331  +
   5.332  +int vcsa2txt_main(int argc) MAIN_EXTERNALLY_VISIBLE;
   5.333  +int vcsa2txt_main(int argc)
   5.334  +{
   5.335 -+    struct {
   5.336 -+	unsigned char l, c, x, y; // man 4 console_codes
   5.337 -+    } scrn;
   5.338 -+    unsigned char last = 0, ch[2];                     // BLGCRMOW
   5.339 -+    static  unsigned char end[5] = "\e[0m\n", color[8] = "04261537";
   5.340 -+    int sp, lf, x;
   5.341 -+    
   5.342 -+    if (safe_read(0, &scrn, 4) < 0) return 1;
   5.343 -+    for (lf = 0; scrn.l; lf++, scrn.l--) {
   5.344 -+	for (sp = x = 0; ++x <= scrn.c;) {
   5.345 -+	    if (safe_read(0, &ch[0], 2) < 0) return 1;
   5.346 -+	    if (argc > 1) ch[1] = 0;
   5.347 -+	    sp++;
   5.348 -+	    if (last == ch[1] && ch[0] == ' ') continue;
   5.349 -+	    for (lf++; --lf;) bb_putchar('\n');
   5.350 -+	    while (--sp) bb_putchar(' ');
   5.351 -+#define ENABLE_VCSA_PACKED 1
   5.352 -+#if ENABLE_VCSA_PACKED
   5.353 -+	    if (last ^= ch[1]) {
   5.354 -+		char esc[16],*s;
   5.355 -+		struct offsets {
   5.356 -+		    char mask, type, shr;
   5.357 -+		} *p;
   5.358 -+		static struct offsets offset[3] = {
   5.359 -+		    {8,0,1}, {0x70,'4',4}, {7,'3',0}
   5.360 -+		};
   5.361 -+		static char init = 0x7F;
   5.362 ++	INIT_G();
   5.363 ++	option_mask32 = FLAG(n);
   5.364 ++	if (argc < 2) option_mask32 = 0;
   5.365 ++	xread(0, &G.info, 4);
   5.366 ++	G.size = G.info.rows * G.info.lines * 2;
   5.367 ++	G.width = G.height = UINT_MAX;
   5.368 ++	G.data = xzalloc(G.size);
   5.369 ++	screen_read_close();
   5.370 ++	screen_dump();
   5.371 ++	bb_putchar('\n');
   5.372 ++	if (ENABLE_FEATURE_CLEAN_UP) {
   5.373 ++		free(ptr_to_globals);
   5.374 ++		close(G.kbd_fd);
   5.375 ++	}
   5.376 ++	return 0;
   5.377 ++}
   5.378 ++
   5.379 ++int conspy_main(int argc UNUSED_PARAM, char **argv) MAIN_EXTERNALLY_VISIBLE;
   5.380 ++int conspy_main(int argc UNUSED_PARAM, char **argv)
   5.381 ++{
   5.382 ++	char vcsa_name[sizeof("/dev/vcsa") + 2];
   5.383 ++	char tty_name[sizeof("/dev/tty") + 2];
   5.384 ++#define keybuf bb_common_bufsiz1
   5.385 ++	struct termios termbuf;
   5.386 ++	unsigned opts;
   5.387 ++	unsigned ttynum;
   5.388 ++	int poll_timeout_ms;
   5.389 ++#if ENABLE_LONG_OPTS
   5.390 ++	static const char getopt_longopts[] ALIGN1 =
   5.391 ++		"viewonly\0"     No_argument "v"
   5.392 ++		"createdevice\0" No_argument "c"
   5.393 ++		"session\0"      No_argument "s"
   5.394 ++		"nocolors\0"     No_argument "n"
   5.395 ++		"dump\0"         No_argument "d"
   5.396 ++		"follow\0"       No_argument "f"
   5.397 ++		;
   5.398 ++
   5.399 ++	applet_long_options = getopt_longopts;
   5.400 ++#endif
   5.401 ++	INIT_G();
   5.402 ++
   5.403 ++	opt_complementary = "x+:y+"; // numeric params
   5.404 ++	opts = getopt32(argv, "vcsndfx:y:", &G.x, &G.y);
   5.405 ++	argv += optind;
   5.406 ++	ttynum = 0;
   5.407 ++	if (argv[0]) {
   5.408 ++		ttynum = xatou_range(argv[0], 0, 63);
   5.409 ++	}
   5.410 ++	sprintf(vcsa_name, "/dev/vcsa%u", ttynum);
   5.411 ++	sprintf(tty_name, "%s%u", "/dev/tty", ttynum);
   5.412 ++	if (opts & FLAG(c)) {
   5.413 ++		if ((opts & (FLAG(s)|FLAG(v))) != FLAG(v))
   5.414 ++			create_cdev_if_doesnt_exist(tty_name, makedev(4, ttynum));
   5.415 ++		create_cdev_if_doesnt_exist(vcsa_name, makedev(7, 128 + ttynum));
   5.416 ++	}
   5.417 ++	if ((opts & FLAG(s)) && ttynum) {
   5.418 ++		start_shell_in_child(tty_name);
   5.419 ++	}
   5.420 ++
   5.421 ++	get_initial_data(vcsa_name);
   5.422 ++	G.kbd_fd = xopen(CURRENT_TTY, O_RDONLY);
   5.423 ++	if (opts & FLAG(d)) {
   5.424 ++		screen_dump();
   5.425 ++		bb_putchar('\n');
   5.426 ++		if (ENABLE_FEATURE_CLEAN_UP) {
   5.427 ++			free(ptr_to_globals);
   5.428 ++			close(G.kbd_fd);
   5.429 ++		}
   5.430 ++		return 0;
   5.431 ++	}
   5.432 ++
   5.433 ++	bb_signals(BB_FATAL_SIGS, cleanup);
   5.434 ++	// All characters must be passed through to us unaltered
   5.435 ++	tcgetattr(G.kbd_fd, &G.term_orig);
   5.436 ++	termbuf = G.term_orig;
   5.437 ++	termbuf.c_iflag &= ~(BRKINT|INLCR|ICRNL|IXON|IXOFF|IUCLC|IXANY|IMAXBEL);
   5.438 ++	termbuf.c_oflag &= ~(OPOST);
   5.439 ++	termbuf.c_lflag &= ~(ISIG|ICANON|ECHO);
   5.440 ++	termbuf.c_cc[VMIN] = 1;
   5.441 ++	termbuf.c_cc[VTIME] = 0;
   5.442 ++	tcsetattr(G.kbd_fd, TCSANOW, &termbuf);
   5.443 ++	poll_timeout_ms = 250;
   5.444 ++	while (1) {
   5.445 ++		struct pollfd pfd;
   5.446 ++		int bytes_read;
   5.447 ++		int i, j;
   5.448 ++		char *data, *old;
   5.449 ++
   5.450 ++		old = G.data + G.current;
   5.451 ++		G.current = G.size - G.current;
   5.452 ++		data = G.data + G.current;
   5.453  +		
   5.454 -+		s = esc+2;
   5.455 -+		*(short *)esc = ntohs(256*'\e'+'[');
   5.456 -+		p = offset;
   5.457 -+		do {
   5.458 -+		    if ((init|last) & p->mask) {
   5.459 -+			int c = (ch[1] & p->mask) >> p->shr;
   5.460 -+		    
   5.461 -+			if ((*s = p->type) != 0) s++;
   5.462 -+			else if (c == 0) {
   5.463 -+			    c = 2;
   5.464 -+			    *s++ = '2'; /* normal */
   5.465 ++		// Close & re-open vcsa in case they have
   5.466 ++		// swapped virtual consoles
   5.467 ++		G.vcsa_fd = xopen(vcsa_name, O_RDONLY);
   5.468 ++		xread(G.vcsa_fd, &G.info, 4);
   5.469 ++		if (G.size != (G.info.rows * G.info.lines * 2)) {
   5.470 ++			cleanup(1);
   5.471 ++		}
   5.472 ++		i = G.width;
   5.473 ++		j = G.height;
   5.474 ++		get_terminal_width_height(G.kbd_fd, &G.width, &G.height);
   5.475 ++		if ((option_mask32 & FLAG(f))) {
   5.476 ++			int nx = G.info.cursor_x - G.width + 1;
   5.477 ++			int ny = G.info.cursor_y - G.height + 1;
   5.478 ++
   5.479 ++			if (G.info.cursor_x < G.x) {
   5.480 ++				G.x = G.info.cursor_x;
   5.481 ++				i = 0;	// force refresh
   5.482  +			}
   5.483 -+			*s++ = color[c];
   5.484 -+			*s++ = ';'; 
   5.485 -+		    }
   5.486 -+		} while (p++->shr);
   5.487 -+		s[-1] = 'm';
   5.488 -+		init = 0;
   5.489 -+		fwrite(esc,s-esc,1,stdout);
   5.490 -+	    }
   5.491 -+	    last = ch[1];
   5.492 -+#else
   5.493 -+	    if (last != ch[1]) {
   5.494 -+		static char esc[10] = "\e[0;47;37m";
   5.495 -+		
   5.496 -+		esc[2] = ((last = ch[1]) & 8) ? '1' /* bold */ : '0' /* defaults */;
   5.497 -+		esc[sizeof(esc)-5] = color[(ch[1] >> 4) & 7];
   5.498 -+		esc[sizeof(esc)-2] = color[ch[1] & 7];
   5.499 -+		fwrite(esc,sizeof(esc),1,stdout);
   5.500 -+	    }
   5.501 -+#endif
   5.502 -+	    bb_putchar(ch[0]);
   5.503 ++			if (nx > G.x) {
   5.504 ++				G.x = nx;
   5.505 ++				i = 0;	// force refresh
   5.506 ++			}
   5.507 ++			if (G.info.cursor_y < G.y) {
   5.508 ++				G.y = G.info.cursor_y;
   5.509 ++				i = 0;	// force refresh
   5.510 ++			}
   5.511 ++			if (ny > G.y) {
   5.512 ++				G.y = ny;
   5.513 ++				i = 0;	// force refresh
   5.514 ++			}
   5.515 ++		}
   5.516 ++
   5.517 ++		// Scan console data and redraw our tty where needed
   5.518 ++		screen_read_close();
   5.519 ++		if (i != G.width || j != G.height) {
   5.520 ++			clrscr();
   5.521 ++			screen_dump();
   5.522 ++		}
   5.523 ++		else for (i = 0; i < G.info.lines; i++) {
   5.524 ++			char *last = last;
   5.525 ++			char *first = NULL;
   5.526 ++			int iy = i - G.y;
   5.527 ++
   5.528 ++			if (iy >= (int) G.height)
   5.529 ++				break;
   5.530 ++			for (j = 0; j < G.info.rows; j++) {
   5.531 ++				last = data;
   5.532 ++				if (DATA(data) != DATA(old) && iy >= 0) {
   5.533 ++					unsigned jx = j - G.x;
   5.534 ++
   5.535 ++					last = NULL;
   5.536 ++					if (first == NULL && jx < G.width) {
   5.537 ++						first = data;
   5.538 ++						gotoxy(jx, iy);
   5.539 ++					}
   5.540 ++				}
   5.541 ++				NEXT(old);
   5.542 ++				NEXT(data);
   5.543 ++			}
   5.544 ++			if (first == NULL)
   5.545 ++				continue;
   5.546 ++			if (last == NULL)
   5.547 ++				last = data;
   5.548 ++
   5.549 ++			// Write the data to the screen
   5.550 ++			for (; first < last; NEXT(first))
   5.551 ++				screen_char(first);
   5.552 ++		}
   5.553 ++		curmove();
   5.554 ++
   5.555 ++		// Wait for local user keypresses
   5.556 ++		pfd.fd = G.kbd_fd;
   5.557 ++		pfd.events = POLLIN;
   5.558 ++		bytes_read = 0;
   5.559 ++		switch (poll(&pfd, 1, poll_timeout_ms)) {
   5.560 ++			char *k;
   5.561 ++		case -1:
   5.562 ++			if (errno != EINTR)
   5.563 ++				cleanup(1);
   5.564 ++			break;
   5.565 ++		case 0:
   5.566 ++			if (++G.nokeys >= 4)
   5.567 ++				G.nokeys = G.escape_count = 0;
   5.568 ++			break;
   5.569 ++		default:
   5.570 ++			// Read the keys pressed
   5.571 ++			k = keybuf + G.key_count;
   5.572 ++			bytes_read = read(G.kbd_fd, k, sizeof(keybuf) - G.key_count);
   5.573 ++			if (bytes_read < 0)
   5.574 ++				cleanup(1);
   5.575 ++
   5.576 ++			// Do exit processing
   5.577 ++			for (i = 0; i < bytes_read; i++) {
   5.578 ++				if (k[i] != '\033') G.escape_count = 0;
   5.579 ++				else if (++G.escape_count >= 3)
   5.580 ++					cleanup(0);
   5.581 ++			}
   5.582 ++		}
   5.583 ++		poll_timeout_ms = 250;
   5.584 ++
   5.585 ++		// Insert all keys pressed into the virtual console's input
   5.586 ++		// buffer.  Don't do this if the virtual console is in scan
   5.587 ++		// code mode - giving ASCII characters to a program expecting
   5.588 ++		// scan codes will confuse it.
   5.589 ++		if (!(option_mask32 & FLAG(v)) && G.escape_count == 0) {
   5.590 ++			int handle, result;
   5.591 ++			long kbd_mode;
   5.592 ++
   5.593 ++			G.key_count += bytes_read;
   5.594 ++			handle = xopen(tty_name, O_WRONLY);
   5.595 ++			result = ioctl(handle, KDGKBMODE, &kbd_mode);
   5.596 ++			if (result == -1)
   5.597 ++				/* nothing */;
   5.598 ++			else if (kbd_mode != K_XLATE && kbd_mode != K_UNICODE)
   5.599 ++				G.key_count = 0; // scan code mode
   5.600 ++			else {
   5.601 ++				for (i = 0; i < G.key_count && result != -1; i++)
   5.602 ++					result = ioctl(handle, TIOCSTI, keybuf + i);
   5.603 ++				G.key_count -= i;
   5.604 ++				if (G.key_count)
   5.605 ++					memmove(keybuf, keybuf + i, G.key_count);
   5.606 ++				// If there is an application on console which reacts
   5.607 ++				// to keypresses, we need to make our first sleep
   5.608 ++				// shorter to quickly redraw whatever it printed there.
   5.609 ++				poll_timeout_ms = 20;
   5.610 ++			}
   5.611 ++			// Close & re-open tty in case they have
   5.612 ++			// swapped virtual consoles
   5.613 ++			close(handle);
   5.614 ++
   5.615 ++			// We sometimes get spurious IO errors on the TTY
   5.616 ++			// as programs close and re-open it
   5.617 ++			if (result != -1)
   5.618 ++				G.ioerror_count = 0;
   5.619 ++			else if (errno != EIO || ++G.ioerror_count > 4)
   5.620 ++				cleanup(1);
   5.621 ++		}
   5.622  +	}
   5.623 -+    }
   5.624 -+    fwrite(end,sizeof(end),1,stdout);
   5.625 -+    return 0;
   5.626  +}