1998-02-27 07:54:42 +03:00
|
|
|
/* Client interface for General purpose Linux console save/restore server
|
|
|
|
Copyright (C) 1994 Janne Kukonlehto <jtklehto@stekt.oulu.fi>
|
|
|
|
|
|
|
|
This program is free software; you can redistribute it and/or modify
|
|
|
|
it under the terms of the GNU General Public License as published by
|
|
|
|
the Free Software Foundation; either version 2 of the License, or
|
|
|
|
(at your option) any later version.
|
|
|
|
|
|
|
|
This program is distributed in the hope that it will be useful,
|
|
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
GNU General Public License for more details.
|
|
|
|
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
|
|
along with this program; if not, write to the Free Software
|
2000-08-23 02:50:00 +04:00
|
|
|
Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. */
|
1998-02-27 07:54:42 +03:00
|
|
|
|
|
|
|
#include <config.h>
|
|
|
|
|
|
|
|
#ifdef HAVE_UNISTD_H
|
|
|
|
# include <unistd.h>
|
|
|
|
#endif
|
|
|
|
#include <sys/types.h>
|
|
|
|
#include <signal.h>
|
2003-03-05 12:29:31 +03:00
|
|
|
#include <stdio.h>
|
2001-09-07 21:52:19 +04:00
|
|
|
|
2003-02-26 22:14:52 +03:00
|
|
|
#ifdef SCO_FLAVOR
|
|
|
|
#include <sys/types.h>
|
|
|
|
#include <sys/vid.h>
|
|
|
|
#include <sys/console.h>
|
|
|
|
#include <sys/vtkd.h>
|
|
|
|
#include <memory.h>
|
|
|
|
#include <signal.h>
|
|
|
|
#endif /* SCO_FLAVOR */
|
|
|
|
|
|
|
|
#ifdef __FreeBSD__
|
|
|
|
#include <sys/consio.h>
|
|
|
|
#include <sys/ioctl.h>
|
|
|
|
#endif /* __FreeBSD__ */
|
|
|
|
|
2001-06-23 03:29:18 +04:00
|
|
|
#include "global.h"
|
2001-09-07 21:52:19 +04:00
|
|
|
#include "tty.h"
|
|
|
|
#include "cons.saver.h"
|
2001-06-23 03:29:18 +04:00
|
|
|
|
|
|
|
signed char console_flag = 0;
|
|
|
|
|
2003-02-27 07:57:58 +03:00
|
|
|
#ifdef __linux__
|
1998-02-27 07:54:42 +03:00
|
|
|
|
2003-02-26 22:14:52 +03:00
|
|
|
/* The cons saver can't have a pid of 1, used to prevent bunches of
|
|
|
|
* #ifdef linux */
|
|
|
|
|
|
|
|
int cons_saver_pid = 1;
|
2003-02-27 07:57:58 +03:00
|
|
|
static int pipefd1[2] = { -1, -1 };
|
|
|
|
static int pipefd2[2] = { -1, -1 };
|
1998-02-27 07:54:42 +03:00
|
|
|
|
2003-02-27 07:57:58 +03:00
|
|
|
static void
|
|
|
|
show_console_contents_linux (int starty, unsigned char begin_line,
|
|
|
|
unsigned char end_line)
|
1998-02-27 07:54:42 +03:00
|
|
|
{
|
|
|
|
unsigned char message = 0;
|
|
|
|
unsigned short bytes = 0;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
/* Is tty console? */
|
|
|
|
if (!console_flag)
|
|
|
|
return;
|
|
|
|
/* Paranoid: Is the cons.saver still running? */
|
2003-02-26 22:14:52 +03:00
|
|
|
if (cons_saver_pid < 1 || kill (cons_saver_pid, SIGCONT)) {
|
1998-02-27 07:54:42 +03:00
|
|
|
cons_saver_pid = 0;
|
|
|
|
console_flag = 0;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Send command to the console handler */
|
|
|
|
message = CONSOLE_CONTENTS;
|
|
|
|
write (pipefd1[1], &message, 1);
|
|
|
|
/* Check for outdated cons.saver */
|
|
|
|
read (pipefd2[0], &message, 1);
|
|
|
|
if (message != CONSOLE_CONTENTS)
|
|
|
|
return;
|
|
|
|
|
|
|
|
/* Send the range of lines that we want */
|
|
|
|
write (pipefd1[1], &begin_line, 1);
|
|
|
|
write (pipefd1[1], &end_line, 1);
|
|
|
|
/* Read the corresponding number of bytes */
|
|
|
|
read (pipefd2[0], &bytes, 2);
|
|
|
|
|
|
|
|
/* Read the bytes and output them */
|
2003-02-26 22:14:52 +03:00
|
|
|
for (i = 0; i < bytes; i++) {
|
1998-02-27 07:54:42 +03:00
|
|
|
if ((i % COLS) == 0)
|
2003-02-26 22:14:52 +03:00
|
|
|
move (starty + (i / COLS), 0);
|
1998-02-27 07:54:42 +03:00
|
|
|
read (pipefd2[0], &message, 1);
|
|
|
|
addch (message);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Read the value of the console_flag */
|
|
|
|
read (pipefd2[0], &message, 1);
|
|
|
|
}
|
|
|
|
|
2003-02-27 07:57:58 +03:00
|
|
|
static void
|
|
|
|
handle_console_linux (unsigned char action)
|
1998-02-27 07:54:42 +03:00
|
|
|
{
|
|
|
|
char *tty_name;
|
1998-08-27 00:23:10 +04:00
|
|
|
char *mc_conssaver;
|
1998-02-27 07:54:42 +03:00
|
|
|
int status;
|
|
|
|
|
2003-02-26 22:14:52 +03:00
|
|
|
switch (action) {
|
1998-02-27 07:54:42 +03:00
|
|
|
case CONSOLE_INIT:
|
|
|
|
/* Close old pipe ends in case it is the 2nd time we run cons.saver */
|
|
|
|
close (pipefd1[1]);
|
|
|
|
close (pipefd2[0]);
|
|
|
|
/* Create two pipes for communication */
|
|
|
|
pipe (pipefd1);
|
|
|
|
pipe (pipefd2);
|
|
|
|
/* Get the console saver running */
|
|
|
|
cons_saver_pid = fork ();
|
2003-02-26 22:14:52 +03:00
|
|
|
if (cons_saver_pid < 0) {
|
2001-08-14 04:55:34 +04:00
|
|
|
/* Cannot fork */
|
1998-02-27 07:54:42 +03:00
|
|
|
/* Delete pipes */
|
|
|
|
close (pipefd1[1]);
|
|
|
|
close (pipefd1[0]);
|
|
|
|
close (pipefd2[1]);
|
|
|
|
close (pipefd2[0]);
|
|
|
|
console_flag = 0;
|
2003-02-26 22:14:52 +03:00
|
|
|
} else if (cons_saver_pid > 0) {
|
1998-02-27 07:54:42 +03:00
|
|
|
/* Parent */
|
|
|
|
/* Close the extra pipe ends */
|
|
|
|
close (pipefd1[0]);
|
|
|
|
close (pipefd2[1]);
|
|
|
|
/* Was the child successful? */
|
|
|
|
read (pipefd2[0], &console_flag, 1);
|
2003-02-26 22:14:52 +03:00
|
|
|
if (!console_flag) {
|
1998-02-27 07:54:42 +03:00
|
|
|
close (pipefd1[1]);
|
|
|
|
close (pipefd2[0]);
|
|
|
|
waitpid (cons_saver_pid, &status, 0);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
/* Child */
|
|
|
|
/* Close the extra pipe ends */
|
|
|
|
close (pipefd1[1]);
|
|
|
|
close (pipefd2[0]);
|
|
|
|
tty_name = ttyname (0);
|
|
|
|
/* Bind the pipe 0 to the standard input */
|
|
|
|
close (0);
|
|
|
|
dup (pipefd1[0]);
|
|
|
|
close (pipefd1[0]);
|
|
|
|
/* Bind the pipe 1 to the standard output */
|
|
|
|
close (1);
|
|
|
|
dup (pipefd2[1]);
|
|
|
|
close (pipefd2[1]);
|
|
|
|
/* Bind standard error to /dev/null */
|
|
|
|
close (2);
|
|
|
|
open ("/dev/null", O_WRONLY);
|
2001-01-27 10:08:05 +03:00
|
|
|
if (tty_name) {
|
|
|
|
/* Exec the console save/restore handler */
|
2002-09-22 04:13:32 +04:00
|
|
|
mc_conssaver = concat_dir_and_file (LIBDIR, "cons.saver");
|
2001-01-27 10:08:05 +03:00
|
|
|
execl (mc_conssaver, "cons.saver", tty_name, NULL);
|
|
|
|
}
|
|
|
|
/* Console is not a tty or execl() failed */
|
1998-02-27 07:54:42 +03:00
|
|
|
console_flag = 0;
|
|
|
|
write (1, &console_flag, 1);
|
|
|
|
close (1);
|
|
|
|
close (0);
|
2002-08-20 04:04:49 +04:00
|
|
|
_exit (3);
|
2003-02-26 22:14:52 +03:00
|
|
|
} /* if (cons_saver_pid ...) */
|
1998-02-27 07:54:42 +03:00
|
|
|
break;
|
|
|
|
|
|
|
|
case CONSOLE_DONE:
|
|
|
|
case CONSOLE_SAVE:
|
|
|
|
case CONSOLE_RESTORE:
|
|
|
|
/* Is tty console? */
|
|
|
|
if (!console_flag)
|
|
|
|
return;
|
|
|
|
/* Paranoid: Is the cons.saver still running? */
|
2003-02-26 22:14:52 +03:00
|
|
|
if (cons_saver_pid < 1 || kill (cons_saver_pid, SIGCONT)) {
|
1998-02-27 07:54:42 +03:00
|
|
|
cons_saver_pid = 0;
|
|
|
|
console_flag = 0;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
/* Send command to the console handler */
|
|
|
|
write (pipefd1[1], &action, 1);
|
2003-02-26 22:14:52 +03:00
|
|
|
if (action != CONSOLE_DONE) {
|
1998-02-27 07:54:42 +03:00
|
|
|
/* Wait the console handler to do its job */
|
|
|
|
read (pipefd2[0], &console_flag, 1);
|
|
|
|
}
|
2003-02-26 22:14:52 +03:00
|
|
|
if (action == CONSOLE_DONE || !console_flag) {
|
1998-02-27 07:54:42 +03:00
|
|
|
/* We are done -> Let's clean up */
|
2003-02-26 22:14:52 +03:00
|
|
|
close (pipefd1[1]);
|
|
|
|
close (pipefd2[0]);
|
1998-02-27 07:54:42 +03:00
|
|
|
waitpid (cons_saver_pid, &status, 0);
|
|
|
|
console_flag = 0;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2003-02-26 22:14:52 +03:00
|
|
|
#elif defined(SCO_FLAVOR)
|
1998-02-27 07:54:42 +03:00
|
|
|
|
|
|
|
/*
|
|
|
|
** SCO console save/restore handling routines
|
|
|
|
** Copyright (C) 1997 Alex Tkachenko <alex@bcs.zaporizhzhe.ua>
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "color.h"
|
|
|
|
|
|
|
|
static int FD_OUT = 2;
|
|
|
|
|
2003-02-26 22:14:52 +03:00
|
|
|
static unsigned short *vidbuf = NULL;
|
|
|
|
static unsigned short *screen = NULL;
|
1998-02-27 07:54:42 +03:00
|
|
|
static int height = 0, width = 0, saved_attr = 0;
|
|
|
|
static int mode = 0;
|
|
|
|
|
2003-02-26 22:14:52 +03:00
|
|
|
#define SIG_ACQUIRE 21 /* originally: handset, line status change (?) */
|
1998-02-27 07:54:42 +03:00
|
|
|
|
|
|
|
static int
|
2003-02-26 22:14:52 +03:00
|
|
|
vt_active ()
|
1998-02-27 07:54:42 +03:00
|
|
|
{
|
2003-02-26 22:14:52 +03:00
|
|
|
struct vid_info vi;
|
|
|
|
int adapter = ioctl (FD_OUT, CONS_CURRENT, 0);
|
1998-02-27 07:54:42 +03:00
|
|
|
|
2003-02-26 22:14:52 +03:00
|
|
|
vi.size = sizeof (struct vid_info);
|
|
|
|
ioctl (FD_OUT, CONS_GETINFO, &(vi));
|
|
|
|
return (vi.m_num == ioctl (FD_OUT, CONSADP, adapter));
|
1998-02-27 07:54:42 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
2003-02-26 22:14:52 +03:00
|
|
|
console_acquire_vt ()
|
1998-02-27 07:54:42 +03:00
|
|
|
{
|
2003-02-26 22:14:52 +03:00
|
|
|
struct vt_mode smode;
|
1998-02-27 07:54:42 +03:00
|
|
|
|
2003-02-26 22:14:52 +03:00
|
|
|
signal (SIG_ACQUIRE, SIG_DFL);
|
|
|
|
smode.mode = VT_AUTO;
|
|
|
|
smode.waitv = smode.relsig = smode.acqsig = smode.frsig = 0;
|
|
|
|
ioctl (FD_OUT, VT_SETMODE, &smode);
|
|
|
|
ioctl (FD_OUT, VT_RELDISP, VT_ACKACQ);
|
1998-02-27 07:54:42 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
2003-02-26 22:14:52 +03:00
|
|
|
console_shutdown ()
|
1998-02-27 07:54:42 +03:00
|
|
|
{
|
2003-02-26 22:14:52 +03:00
|
|
|
if (!console_flag) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (screen != NULL) {
|
|
|
|
g_free (screen);
|
|
|
|
}
|
|
|
|
console_flag = 0;
|
1998-02-27 07:54:42 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
2003-02-26 22:14:52 +03:00
|
|
|
console_save ()
|
1998-02-27 07:54:42 +03:00
|
|
|
{
|
2003-02-26 22:14:52 +03:00
|
|
|
struct m6845_info mi;
|
1998-02-27 07:54:42 +03:00
|
|
|
|
2003-02-26 22:14:52 +03:00
|
|
|
if (!console_flag) {
|
|
|
|
return;
|
|
|
|
}
|
1998-02-27 07:54:42 +03:00
|
|
|
|
2003-02-26 22:14:52 +03:00
|
|
|
if (!vt_active ()) {
|
|
|
|
struct vt_mode smode;
|
1998-02-27 07:54:42 +03:00
|
|
|
|
2003-02-26 22:14:52 +03:00
|
|
|
/*
|
|
|
|
** User switched out of our vt. Let's wait until we get SIG_ACQUIRE,
|
|
|
|
** otherwise we could save wrong screen image
|
|
|
|
*/
|
|
|
|
signal (SIG_ACQUIRE, console_acquire_vt);
|
|
|
|
smode.mode = VT_PROCESS;
|
|
|
|
smode.waitv = 0;
|
|
|
|
smode.waitv = smode.relsig = smode.acqsig = smode.frsig =
|
|
|
|
SIG_ACQUIRE;
|
|
|
|
ioctl (FD_OUT, VT_SETMODE, &smode);
|
|
|
|
|
|
|
|
pause ();
|
|
|
|
}
|
1998-02-27 07:54:42 +03:00
|
|
|
|
2003-02-26 22:14:52 +03:00
|
|
|
saved_attr = ioctl (FD_OUT, GIO_ATTR, 0);
|
1998-02-27 07:54:42 +03:00
|
|
|
|
2003-02-26 22:14:52 +03:00
|
|
|
vidbuf = (unsigned short *) ioctl (FD_OUT, MAPCONS, 0);
|
1998-02-27 07:54:42 +03:00
|
|
|
|
2003-02-26 22:14:52 +03:00
|
|
|
mi.size = sizeof (struct m6845_info);
|
|
|
|
ioctl (FD_OUT, CONS_6845INFO, &mi);
|
|
|
|
|
|
|
|
{
|
|
|
|
unsigned short *start = vidbuf + mi.screen_top;
|
|
|
|
memcpy (screen, start, width * height * 2);
|
|
|
|
}
|
1998-02-27 07:54:42 +03:00
|
|
|
|
2003-02-26 22:14:52 +03:00
|
|
|
write (FD_OUT, "\0337", 2); /* save cursor position */
|
1998-02-27 07:54:42 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
2003-02-26 22:14:52 +03:00
|
|
|
console_restore ()
|
1998-02-27 07:54:42 +03:00
|
|
|
{
|
2003-02-26 22:14:52 +03:00
|
|
|
struct m6845_info mi;
|
|
|
|
unsigned short *start;
|
1998-02-27 07:54:42 +03:00
|
|
|
|
2003-02-26 22:14:52 +03:00
|
|
|
if (!console_flag) {
|
|
|
|
return;
|
|
|
|
}
|
1998-02-27 07:54:42 +03:00
|
|
|
|
|
|
|
write (FD_OUT, "\033[2J", 4);
|
|
|
|
|
2003-02-26 22:14:52 +03:00
|
|
|
mi.size = sizeof (struct m6845_info);
|
|
|
|
ioctl (FD_OUT, CONS_6845INFO, &mi);
|
1998-02-27 07:54:42 +03:00
|
|
|
|
2003-02-26 22:14:52 +03:00
|
|
|
start = vidbuf + mi.screen_top;
|
|
|
|
memcpy (start, screen, width * height * 2);
|
|
|
|
write (FD_OUT, "\0338", 2); /* restore cursor position */
|
1998-02-27 07:54:42 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
2003-02-26 22:14:52 +03:00
|
|
|
console_init ()
|
1998-02-27 07:54:42 +03:00
|
|
|
{
|
2003-02-26 22:14:52 +03:00
|
|
|
struct vid_info vi;
|
|
|
|
int adapter = ioctl (FD_OUT, CONS_CURRENT, 0);
|
1998-02-27 07:54:42 +03:00
|
|
|
|
2003-02-26 22:14:52 +03:00
|
|
|
console_flag = 0;
|
|
|
|
|
|
|
|
if (adapter != -1) {
|
|
|
|
vi.size = sizeof (struct vid_info);
|
|
|
|
ioctl (FD_OUT, CONS_GETINFO, &(vi));
|
|
|
|
|
|
|
|
if (vt_active ()) {
|
|
|
|
console_flag = 1;
|
1998-02-27 07:54:42 +03:00
|
|
|
|
2003-02-26 22:14:52 +03:00
|
|
|
height = vi.mv_rsz;
|
|
|
|
width = vi.mv_csz;
|
|
|
|
|
|
|
|
screen = (unsigned short *) g_malloc (height * width * 2);
|
|
|
|
if (screen == NULL) {
|
|
|
|
console_shutdown ();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
console_save ();
|
|
|
|
mode = ioctl (FD_OUT, CONS_GET, 0);
|
|
|
|
ioctl (FD_OUT, MODESWITCH | mode, 0);
|
|
|
|
console_restore ();
|
1998-02-27 07:54:42 +03:00
|
|
|
}
|
2003-02-26 22:14:52 +03:00
|
|
|
}
|
1998-02-27 07:54:42 +03:00
|
|
|
}
|
|
|
|
|
2003-02-27 07:57:58 +03:00
|
|
|
static void
|
|
|
|
handle_console_sco (unsigned char action)
|
1998-02-27 07:54:42 +03:00
|
|
|
{
|
2003-02-26 22:14:52 +03:00
|
|
|
switch (action) {
|
|
|
|
case CONSOLE_INIT:
|
|
|
|
console_init ();
|
|
|
|
break;
|
|
|
|
|
|
|
|
case CONSOLE_DONE:
|
|
|
|
console_shutdown ();
|
|
|
|
break;
|
|
|
|
|
|
|
|
case CONSOLE_SAVE:
|
|
|
|
console_save ();
|
|
|
|
break;
|
|
|
|
|
|
|
|
case CONSOLE_RESTORE:
|
|
|
|
console_restore ();
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
/* Nothing */ ;
|
1998-02-27 07:54:42 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2003-02-27 07:57:58 +03:00
|
|
|
static void
|
|
|
|
show_console_contents_sco (int starty, unsigned char begin_line,
|
|
|
|
unsigned char end_line)
|
1998-02-27 07:54:42 +03:00
|
|
|
{
|
2003-02-26 22:14:52 +03:00
|
|
|
register int i, len = (end_line - begin_line) * width;
|
1998-02-27 07:54:42 +03:00
|
|
|
|
2003-02-26 22:14:52 +03:00
|
|
|
attrset (DEFAULT_COLOR);
|
|
|
|
for (i = 0; i < len; i++) {
|
|
|
|
if ((i % width) == 0)
|
|
|
|
move (starty + (i / width), 0);
|
|
|
|
addch ((unsigned char) screen[width * starty + i]);
|
|
|
|
}
|
1998-02-27 07:54:42 +03:00
|
|
|
}
|
|
|
|
|
2003-02-26 22:14:52 +03:00
|
|
|
#elif defined(__FreeBSD__)
|
1998-02-27 07:54:42 +03:00
|
|
|
|
2003-02-26 22:14:52 +03:00
|
|
|
/*
|
|
|
|
* FreeBSD support copyright (C) 2003 Alexander Serkov <serkov@ukrpost.net>.
|
|
|
|
* Support for screenmaps by Max Khon <fjoe@FreeBSD.org>
|
|
|
|
*/
|
1998-02-27 07:54:42 +03:00
|
|
|
|
2003-02-26 22:14:52 +03:00
|
|
|
#define FD_OUT 1
|
1998-02-27 07:54:42 +03:00
|
|
|
|
2003-02-26 22:14:52 +03:00
|
|
|
static struct scrshot screen_shot;
|
|
|
|
static struct vid_info screen_info;
|
1998-02-27 07:54:42 +03:00
|
|
|
|
2003-02-26 22:14:52 +03:00
|
|
|
static void
|
|
|
|
console_init (void)
|
|
|
|
{
|
|
|
|
if (console_flag)
|
|
|
|
return;
|
|
|
|
|
|
|
|
screen_info.size = sizeof (screen_info);
|
|
|
|
if (ioctl (FD_OUT, CONS_GETINFO, &screen_info) == -1)
|
|
|
|
return;
|
|
|
|
|
2003-02-27 07:57:58 +03:00
|
|
|
memset (&screen_shot, 0, sizeof (screen_shot));
|
2003-02-26 22:14:52 +03:00
|
|
|
screen_shot.xsize = screen_info.mv_csz;
|
|
|
|
screen_shot.ysize = screen_info.mv_rsz;
|
|
|
|
if ((screen_shot.buf =
|
2003-02-27 07:57:58 +03:00
|
|
|
g_malloc (screen_info.mv_csz * screen_info.mv_rsz)) == NULL)
|
2003-02-26 22:14:52 +03:00
|
|
|
return;
|
|
|
|
|
|
|
|
console_flag = 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
set_attr (unsigned attr)
|
|
|
|
{
|
2003-03-05 12:29:31 +03:00
|
|
|
/*
|
|
|
|
* Convert color indices returned by SCRSHOT (red=4, green=2, blue=1)
|
|
|
|
* to indices for ANSI sequences (red=1, green=2, blue=4).
|
|
|
|
*/
|
|
|
|
static const int color_map[8] = { 0, 4, 2, 6, 1, 5, 3, 7 };
|
2003-02-26 22:14:52 +03:00
|
|
|
char cmd[17];
|
|
|
|
int bc, tc;
|
|
|
|
|
|
|
|
tc = attr & 0xF;
|
|
|
|
bc = (attr >> 4) & 0xF;
|
|
|
|
|
|
|
|
strcpy (cmd, "\x1B[");
|
|
|
|
strcat (cmd, (bc & 8) ? "5;" : "25;");
|
|
|
|
strcat (cmd, (tc & 8) ? "1;" : "22;");
|
|
|
|
strcat (cmd, "3%d;4%dm");
|
|
|
|
printf (cmd, color_map[tc & 7], color_map[bc & 7]);
|
|
|
|
}
|
|
|
|
|
|
|
|
#define cursor_to(x, y) do { \
|
|
|
|
printf("\x1B[%d;%df", (y) + 1, (x) + 1); \
|
|
|
|
fflush(stdout); \
|
|
|
|
} while (0)
|
|
|
|
|
|
|
|
static void
|
|
|
|
console_restore (void)
|
|
|
|
{
|
|
|
|
int i, last;
|
|
|
|
|
|
|
|
if (!console_flag)
|
|
|
|
return;
|
|
|
|
|
|
|
|
cursor_to (0, 0);
|
|
|
|
|
|
|
|
/* restoring all content up to cursor position */
|
|
|
|
last = screen_info.mv_row * screen_info.mv_csz + screen_info.mv_col;
|
|
|
|
for (i = 0; i < last; ++i) {
|
|
|
|
set_attr ((screen_shot.buf[i] >> 8) & 0xFF);
|
|
|
|
putc (screen_shot.buf[i] & 0xFF, stdout);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* restoring cursor color */
|
|
|
|
set_attr ((screen_shot.buf[last] >> 8) & 0xFF);
|
|
|
|
|
|
|
|
fflush (stdout);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
console_shutdown (void)
|
|
|
|
{
|
|
|
|
if (!console_flag)
|
|
|
|
return;
|
|
|
|
|
2003-02-27 07:57:58 +03:00
|
|
|
g_free (screen_shot.buf);
|
2003-02-26 22:14:52 +03:00
|
|
|
|
|
|
|
console_flag = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
console_save (void)
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
scrmap_t map;
|
|
|
|
scrmap_t revmap;
|
|
|
|
|
|
|
|
if (!console_flag)
|
|
|
|
return;
|
|
|
|
|
|
|
|
/* screen_info.size is already set in console_init() */
|
|
|
|
if (ioctl (FD_OUT, CONS_GETINFO, &screen_info) == -1) {
|
|
|
|
console_shutdown ();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* handle console resize */
|
|
|
|
if (screen_info.mv_csz != screen_shot.xsize
|
|
|
|
|| screen_info.mv_rsz != screen_shot.ysize) {
|
|
|
|
console_shutdown ();
|
|
|
|
console_init ();
|
|
|
|
}
|
|
|
|
|
|
|
|
if (ioctl (FD_OUT, CONS_SCRSHOT, &screen_shot) == -1) {
|
|
|
|
console_shutdown ();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (ioctl (FD_OUT, GIO_SCRNMAP, &map) == -1) {
|
|
|
|
console_shutdown ();
|
|
|
|
exit (1);
|
|
|
|
}
|
|
|
|
|
|
|
|
for (i = 0; i < 256; i++) {
|
|
|
|
char *p = memchr (map.scrmap, i, 256);
|
|
|
|
revmap.scrmap[i] = p ? p - map.scrmap : i;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (i = 0; i < screen_shot.xsize * screen_shot.ysize; i++) {
|
|
|
|
screen_shot.buf[i] =
|
|
|
|
(screen_shot.buf[i] & 0xff00) | (unsigned char) revmap.
|
|
|
|
scrmap[screen_shot.buf[i] & 0xff];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2003-02-27 07:57:58 +03:00
|
|
|
static void
|
|
|
|
show_console_contents_freebsd (int starty, unsigned char begin_line,
|
|
|
|
unsigned char end_line)
|
2003-02-26 22:14:52 +03:00
|
|
|
{
|
2003-03-05 12:29:31 +03:00
|
|
|
int col, line;
|
|
|
|
char c;
|
2003-02-26 22:14:52 +03:00
|
|
|
|
|
|
|
if (!console_flag)
|
|
|
|
return;
|
|
|
|
|
2003-03-05 12:29:31 +03:00
|
|
|
for (line = begin_line; line <= end_line; line++) {
|
|
|
|
move (starty + line - begin_line, 0);
|
|
|
|
for (col = 0; col < min (COLS, screen_info.mv_csz); col++) {
|
|
|
|
c = screen_shot.buf[line * screen_info.mv_csz + col] & 0xFF;
|
|
|
|
addch (c);
|
|
|
|
}
|
2003-02-26 22:14:52 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2003-02-27 07:57:58 +03:00
|
|
|
static void
|
|
|
|
handle_console_freebsd (unsigned char action)
|
2003-02-26 22:14:52 +03:00
|
|
|
{
|
|
|
|
switch (action) {
|
|
|
|
case CONSOLE_INIT:
|
|
|
|
console_init ();
|
|
|
|
break;
|
|
|
|
|
|
|
|
case CONSOLE_DONE:
|
|
|
|
console_shutdown ();
|
|
|
|
break;
|
|
|
|
|
|
|
|
case CONSOLE_SAVE:
|
|
|
|
console_save ();
|
|
|
|
break;
|
|
|
|
|
|
|
|
case CONSOLE_RESTORE:
|
|
|
|
console_restore ();
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2003-03-05 12:29:31 +03:00
|
|
|
#endif /* __FreeBSD__ */
|
2003-02-26 22:14:52 +03:00
|
|
|
|
|
|
|
void
|
|
|
|
show_console_contents (int starty, unsigned char begin_line,
|
|
|
|
unsigned char end_line)
|
1998-02-27 07:54:42 +03:00
|
|
|
{
|
|
|
|
standend ();
|
|
|
|
|
|
|
|
if (look_for_rxvt_extensions ()) {
|
|
|
|
show_rxvt_contents (starty, begin_line, end_line);
|
|
|
|
return;
|
|
|
|
}
|
2003-02-27 07:57:58 +03:00
|
|
|
#ifdef __linux__
|
|
|
|
show_console_contents_linux (starty, begin_line, end_line);
|
|
|
|
#elif defined (__FreeBSD__)
|
|
|
|
show_console_contents_freebsd (starty, begin_line, end_line);
|
|
|
|
#elif defined (SCO_FLAVOR)
|
|
|
|
show_console_contents_sco (starty, begin_line, end_line);
|
|
|
|
#else
|
1998-02-27 07:54:42 +03:00
|
|
|
console_flag = 0;
|
2003-02-27 07:57:58 +03:00
|
|
|
#endif
|
1998-02-27 07:54:42 +03:00
|
|
|
}
|
|
|
|
|
2003-02-26 22:14:52 +03:00
|
|
|
void
|
|
|
|
handle_console (unsigned char action)
|
1998-02-27 07:54:42 +03:00
|
|
|
{
|
2003-02-27 07:57:58 +03:00
|
|
|
if (look_for_rxvt_extensions ())
|
|
|
|
return;
|
1998-02-27 07:54:42 +03:00
|
|
|
|
2003-02-27 07:57:58 +03:00
|
|
|
#ifdef __linux__
|
|
|
|
handle_console_linux (action);
|
|
|
|
#elif defined (__FreeBSD__)
|
|
|
|
handle_console_freebsd (action);
|
|
|
|
#elif defined (SCO_FLAVOR)
|
|
|
|
handle_console_sco (action);
|
|
|
|
#endif
|
|
|
|
}
|