* cons.saver.c: Remove support for Linux kernels before 2.0.

Massive cleanup.
(send_contents): Remove incorrect optimization - restore even
empty lines because they may be touched by the viewer.
From Denis Vlasenko <vda_unique@iname.com>
This commit is contained in:
Pavel Roskin 2002-07-28 08:52:27 +00:00
parent fa689098a9
commit edebd94750
2 changed files with 32 additions and 142 deletions

View File

@ -1,3 +1,11 @@
2002-07-28 Pavel Roskin <proski@gnu.org>
* cons.saver.c: Remove support for Linux kernels before 2.0.
Massive cleanup.
(send_contents): Remove incorrect optimization - restore even
empty lines because they may be touched by the viewer.
From Denis Vlasenko <vda_unique@iname.com>
2002-07-26 Pavel Roskin <proski@gnu.org> 2002-07-26 Pavel Roskin <proski@gnu.org>
* utilunix.c (canonicalize_pathname): Don't extend empty path * utilunix.c (canonicalize_pathname): Don't extend empty path

View File

@ -52,18 +52,12 @@
/* Meaning of console_flag: /* Meaning of console_flag:
-1 == to be detected, -1 == to be detected,
0 == not a console 0 == not a console or Linux < 1.1.92
1 == is a console, Linux < 1.1.67 (black & white) 1 == obsolete, not used
2 == is a console, Linux >= 1.1.67 (color) 2 == obsolete, not used
3 == is a console, Linux >= 1.1.92 (color, use /dev/vcsa$num 3 == is a console, Linux >= 1.1.92 (color, use /dev/vcsa$num
*/ */
static signed char console_flag = -1; static signed char console_flag = -1;
/*
Meaning of console_fd:
-1 == not opened,
>=0 == opened
*/
static int console_fd = -1;
static char *tty_name; static char *tty_name;
static int console_minor = 0; static int console_minor = 0;
static char *buffer = NULL; static char *buffer = NULL;
@ -71,12 +65,8 @@ static int buffer_size = 0;
static int columns, rows; static int columns, rows;
static int vcs_fd; static int vcs_fd;
static void dwrite (int fd, char *buffer)
{
write (fd, buffer, strlen (buffer));
}
static void tty_getsize (void) static void tty_getsize (int console_fd)
{ {
struct winsize winsz; struct winsize winsz;
@ -87,22 +77,12 @@ static void tty_getsize (void)
rows = winsz.ws_row; rows = winsz.ws_row;
} else { } else {
/* Never happens (I think) */ /* Never happens (I think) */
dwrite (2, "TIOCGWINSZ failed\n");
columns = 80; columns = 80;
rows = 25; rows = 25;
console_flag = 0; console_flag = 0;
} }
} }
static inline void tty_cursormove(int y, int x)
{
char buffer [128];
/* Standard ANSI escape sequence for cursor positioning */
snprintf (buffer, sizeof (buffer), "\33[%d;%dH", y + 1, x + 1);
dwrite (console_fd, buffer);
}
static int check_file (char *filename, int check_console) static int check_file (char *filename, int check_console)
{ {
int fd; int fd;
@ -124,14 +104,6 @@ static int check_file (char *filename, int check_console)
break; break;
} }
#ifdef DEBUG
fprintf (stderr,
"Device %s: major %d, minor %d\r\n",
filename,
((int) stat_buf.st_rdev & 0xff00) >> 8,
((int) stat_buf.st_rdev & 0xff));
#endif
if (check_console){ if (check_console){
/* Major number must be 4 */ /* Major number must be 4 */
if ((stat_buf.st_rdev & 0xff00) != 0x0400){ if ((stat_buf.st_rdev & 0xff00) != 0x0400){
@ -165,6 +137,7 @@ static int detect_console (void)
{ {
char console_name [16]; char console_name [16];
static char vcs_name [16]; static char vcs_name [16];
int console_fd;
/* Must be console */ /* Must be console */
console_fd = check_file (tty_name, 1); console_fd = check_file (tty_name, 1);
@ -202,117 +175,36 @@ static int detect_console (void)
vcs_fd = check_file (vcs_name, 0); vcs_fd = check_file (vcs_name, 0);
} }
#ifdef DEBUG
fprintf (stderr, "vcs_fd = %d console_fd = %d\r\n", vcs_fd, console_fd);
#endif
if (vcs_fd != -1){ if (vcs_fd != -1){
console_flag = 3; console_flag = 3;
tty_getsize (console_fd);
close (console_fd);
return 0;
} }
return 0; close (console_fd);
return -1;
} }
static void save_console (void) static void save_console (void)
{ {
int i;
if (!console_flag)
return;
buffer [1] = console_minor;
if (console_flag >= 2){
/* Linux >= 1.1.67 */
/* Get screen contents and cursor position */
buffer [0] = 8;
if (console_flag == 2){
if ((i = ioctl (console_fd, TIOCLINUX, buffer)) == -1){
/* Oops, this is not Linux 1.1.67 */
console_flag = 1;
}
} else {
lseek (vcs_fd, 0, 0); lseek (vcs_fd, 0, 0);
read (vcs_fd, buffer, buffer_size); read (vcs_fd, buffer, buffer_size);
} }
}
if (console_flag == 1){
int index, x, y;
/* Linux < 1.1.67 */
/* Get screen contents */
buffer [0] = 0;
if (ioctl(console_fd, TIOCLINUX, buffer) == -1){
buffer[0] = buffer[1] = 0;
/* Linux bug: bad ioctl on console 8 */
if (ioctl(console_fd, TIOCLINUX, buffer) == -1){
/* Oops, this is not a console after all */
console_flag = 0;
return;
}
}
/* Select the beginning of the bottommost empty line
to be the cursor position */
index = 2 + rows * columns;
for (y = rows - 1; y >= 0; y--)
for (x = columns - 1; x >= 0; x--)
if (buffer[--index] != ' ')
goto non_space_found;
non_space_found:
buffer[0] = y + 1;
buffer[1] = 0;
/*tty_cursormove(y + 1, 0);*/
}
}
static void restore_console (void) static void restore_console (void)
{ {
if (!console_flag)
return;
if (console_flag == 2){
/* Linux >= 1.1.67 */
/* Restore screen contents and cursor position */
buffer [0] = 9;
buffer [1] = console_minor;
ioctl (console_fd, TIOCLINUX, buffer);
}
if (console_flag == 3){
lseek (vcs_fd, 0, 0); lseek (vcs_fd, 0, 0);
write (vcs_fd, buffer, buffer_size); write (vcs_fd, buffer, buffer_size);
} }
if (console_flag == 1){
/* Clear screen */
write(console_fd, "\033[H\033[2J", 7);
/* Output saved screen contents */
write(console_fd, buffer + 2, rows * columns);
/* Move the cursor to the previously selected position */
tty_cursormove(buffer[0], buffer[1]);
}
}
static void send_contents (void) static void send_contents (void)
{ {
unsigned char begin_line=0, end_line=0; unsigned char begin_line=0, end_line=0;
int index, x, y; int index;
int lastline;
unsigned char message; unsigned char message;
unsigned short bytes; unsigned short bytes;
int bytes_per_char; const int bytes_per_char = 2;
bytes_per_char = console_flag == 1 ? 1 : 2;
/* Calculate the number of used lines */
if (console_flag == 2 || console_flag == 1 || console_flag == 3){
index = (2 + rows * columns) * bytes_per_char;
for (y = rows - 1; y >= 0; y--)
for (x = columns - 1; x >= 0; x--){
index -= bytes_per_char;
if (buffer[index] != ' ')
goto non_space_found;
}
non_space_found:
lastline = y + 1;
} else
return;
/* Inform the invoker that we can handle this command */ /* Inform the invoker that we can handle this command */
message = CONSOLE_CONTENTS; message = CONSOLE_CONTENTS;
@ -321,10 +213,10 @@ static void send_contents (void)
/* Read the range of lines wanted */ /* Read the range of lines wanted */
read (cmd_input, &begin_line, 1); read (cmd_input, &begin_line, 1);
read (cmd_input, &end_line, 1); read (cmd_input, &end_line, 1);
if (begin_line > lastline) if (begin_line > rows)
begin_line = lastline; begin_line = rows;
if (end_line > lastline) if (end_line > rows)
end_line = lastline; end_line = rows;
/* Tell the invoker how many bytes it will be */ /* Tell the invoker how many bytes it will be */
bytes = (end_line - begin_line) * columns; bytes = (end_line - begin_line) * columns;
@ -345,11 +237,13 @@ int main (int argc, char **argv)
int stderr_fd; int stderr_fd;
/* /*
* Make sure Stderr points to a valid place * Make sure stderr points to a valid place
*/ */
close (2); close (2);
stderr_fd = open ("/dev/tty", O_RDWR); stderr_fd = open ("/dev/tty", O_RDWR);
if (stderr_fd == -1) /* This may well happen if program is running non-root */
/* This may well happen if program is running non-root */
if (stderr_fd == -1)
stderr_fd = open ("/dev/null", O_RDWR); stderr_fd = open ("/dev/null", O_RDWR);
if (stderr_fd == -1) if (stderr_fd == -1)
@ -362,7 +256,6 @@ int main (int argc, char **argv)
if (argc != 2){ if (argc != 2){
/* Wrong number of arguments */ /* Wrong number of arguments */
dwrite (2, "Usage: cons.saver <ttyname>\n");
console_flag = 0; console_flag = 0;
write (cmd_output, &console_flag, 1); write (cmd_output, &console_flag, 1);
return 3; return 3;
@ -377,24 +270,13 @@ int main (int argc, char **argv)
if (detect_console () == -1){ if (detect_console () == -1){
/* Not a console -> no need for privileges */ /* Not a console -> no need for privileges */
setuid (getuid ()); setuid (getuid ());
/* dwrite (2, error); */
console_flag = 0; console_flag = 0;
if (console_fd >= 0)
close (console_fd);
} else { } else {
/* Console was detected */
if (console_flag != 3)
console_flag = 2; /* Default to Linux >= 1.1.67 */
/* Allocate buffer for screen image */ /* Allocate buffer for screen image */
tty_getsize ();
buffer_size = 4 + 2 * columns * rows; buffer_size = 4 + 2 * columns * rows;
buffer = (char*) malloc (buffer_size); buffer = (char*) malloc (buffer_size);
} }
/* If using /dev/vcs*, we don't need anymore the console fd */
if (console_flag == 3)
close (console_fd);
/* Inform the invoker about the result of the tests */ /* Inform the invoker about the result of the tests */
write (cmd_output, &console_flag, 1); write (cmd_output, &console_flag, 1);