syscons.c revision 1235
1/*-
2 * Copyright (c) 1992-1994 S�ren Schmidt
3 * Copyright (c) 1990 The Regents of the University of California.
4 * All rights reserved.
5 *
6 * This code is derived from software contributed to Berkeley by
7 * William Jolitz and Don Ahn.
8 *
9 * Redistribution and use in source and binary forms, with or without
10 * modification, are permitted provided that the following conditions
11 * are met:
12 * 1. Redistributions of source code must retain the above copyright
13 *    notice, this list of conditions and the following disclaimer.
14 * 2. Redistributions in binary form must reproduce the above copyright
15 *    notice, this list of conditions and the following disclaimer in the
16 *    documentation and/or other materials provided with the distribution.
17 * 3. All advertising materials mentioning features or use of this software
18 *    must display the following acknowledgement:
19 *	This product includes software developed by the University of
20 *	California, Berkeley and its contributors.
21 * 4. Neither the name of the University nor the names of its contributors
22 *    may be used to endorse or promote products derived from this software
23 *    without specific prior written permission.
24 *
25 * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
26 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
27 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
28 * ARE DISCLAIMED.  IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
29 * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
30 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
31 * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
32 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
33 * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
34 * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
35 * SUCH DAMAGE.
36 *
37 *	from:@(#)syscons.c	1.3 940129
38 *	$Id: syscons.c,v 1.35 1994/02/07 02:14:27 davidg Exp $
39 *
40 */
41
42#if !defined(__FreeBSD__)
43#define FAT_CURSOR
44#endif
45
46#include "param.h"
47#include "conf.h"
48#include "ioctl.h"
49#include "proc.h"
50#include "user.h"
51#include "tty.h"
52#include "uio.h"
53#include "callout.h"
54#include "systm.h"
55#include "kernel.h"
56#include "syslog.h"
57#include "errno.h"
58#include "malloc.h"
59#include "i386/isa/isa.h"
60#include "i386/isa/isa_device.h"
61#include "i386/isa/timerreg.h"
62#include "i386/i386/cons.h"
63#include "machine/console.h"
64#include "machine/psl.h"
65#include "machine/frame.h"
66#include "machine/pc/display.h"
67#include "iso8859.font"
68#include "kbdtables.h"
69#include "sc.h"
70
71#if NSC > 0
72
73#if !defined(NCONS)
74#define NCONS 12
75#endif
76
77/* status flags */
78#define LOCK_KEY_MASK	0x0000F
79#define LED_MASK	0x00007
80#define UNKNOWN_MODE	0x00010
81#define KBD_RAW_MODE	0x00020
82#define SWITCH_WAIT_REL	0x00040
83#define SWITCH_WAIT_ACQ	0x00080
84
85/* video hardware memory addresses */
86#define VIDEOMEM	0x000A0000
87
88/* misc defines */
89#define MAX_ESC_PAR 	3
90#define TEXT80x25	1
91#define TEXT80x50	2
92#define	COL		80
93#define	ROW		25
94#define BELL_DURATION	5
95#define BELL_PITCH	800
96#define TIMER_FREQ	1193182			/* should be in isa.h */
97#define PCBURST		128
98
99/* defines related to hardware addresses */
100#define	MONO_BASE	0x3B4			/* crt controller base mono */
101#define	COLOR_BASE	0x3D4			/* crt controller base color */
102#define ATC		IO_VGA+0x00		/* attribute controller */
103#define TSIDX		IO_VGA+0x04		/* timing sequencer idx */
104#define TSREG		IO_VGA+0x05		/* timing sequencer data */
105#define PIXMASK		IO_VGA+0x06		/* pixel write mask */
106#define PALRADR		IO_VGA+0x07		/* palette read address */
107#define PALWADR		IO_VGA+0x08		/* palette write address */
108#define PALDATA		IO_VGA+0x09		/* palette data register */
109#define GDCIDX		IO_VGA+0x0E		/* graph data controller idx */
110#define GDCREG		IO_VGA+0x0F		/* graph data controller data */
111
112/* special characters */
113#define cntlc	0x03
114#define cntld	0x04
115#define bs	0x08
116#define lf	0x0a
117#define cr	0x0d
118#define del	0x7f
119
120typedef struct term_stat {
121	int 		esc;			/* processing escape sequence */
122	int 		num_param;		/* # of parameters to ESC */
123	int	 	last_param;		/* last parameter # */
124	int 		param[MAX_ESC_PAR];	/* contains ESC parameters */
125	int 		cur_attr;		/* current attributes */
126	int 		std_attr;		/* normal attributes */
127	int 		rev_attr;		/* reverse attributes */
128} term_stat;
129
130typedef struct scr_stat {
131	u_short 	*crt_base;		/* address of screen memory */
132	u_short 	*scr_buf;		/* buffer when off screen */
133	u_short 	*crtat;			/* cursor address */
134	int 		xpos;			/* current X position */
135	int 		ypos;			/* current Y position */
136	int 		xsize;			/* X size */
137	int 		ysize;			/* Y size */
138	term_stat 	term;			/* terminal emulation stuff */
139	char		cursor_start;		/* cursor start line # */
140	char		cursor_end;		/* cursor end line # */
141	u_char		border;			/* border color */
142	u_short		bell_duration;
143	u_short		bell_pitch;
144	u_short 	status;			/* status (bitfield) */
145	u_short 	mode;			/* mode */
146	pid_t 		pid;			/* pid of controlling proc */
147	struct proc 	*proc;			/* proc* of controlling proc */
148	struct vt_mode 	smode;			/* switch mode */
149} scr_stat;
150
151typedef struct default_attr {
152	int             std_attr;               /* normal attributes */
153	int 		rev_attr;		/* reverse attributes */
154} default_attr;
155
156static default_attr user_default = {
157	(FG_LIGHTGREY | BG_BLACK) << 8,
158	(FG_BLACK | BG_LIGHTGREY) << 8
159};
160
161static default_attr kernel_default = {
162	(FG_WHITE | BG_BLACK) << 8,
163	(FG_BLACK | BG_LIGHTGREY) << 8
164};
165
166static	scr_stat	console[NCONS];
167static	scr_stat	*cur_console = &console[0];
168static	scr_stat	*new_scp, *old_scp;
169static	term_stat	kernel_console;
170static	default_attr	*current_default;
171static	int		switch_in_progress = 0;
172static 	u_short	 	*crtat = 0;
173static	u_int		crtc_addr = MONO_BASE;
174static	char		crtc_vga = 0;
175static 	u_char		shfts = 0, ctls = 0, alts = 0, agrs = 0, metas = 0;
176static 	u_char		nlkcnt = 0, clkcnt = 0, slkcnt = 0, alkcnt = 0;
177static	char		palette[3*256];
178static 	const u_int 	n_fkey_tab = sizeof(fkey_tab) / sizeof(*fkey_tab);
179static	int 		cur_cursor_pos = -1;
180static	char 		in_putc = 0;
181static	char	 	polling = 0;
182static	int	 	delayed_next_scr;
183static	char		saved_console = -1;	/* saved console number	*/
184static	long		scrn_blank_time = 0;	/* screen saver timout value */
185static	int		scrn_blanked = 0;	/* screen saver active flag */
186static	int		scrn_saver = 0;		/* screen saver routine */
187static	long 		scrn_time_stamp;
188static  u_char		scr_map[256];
189extern	int hz;
190extern	struct timeval time;
191
192/* function prototypes */
193int pcprobe(struct isa_device *dev);
194int pcattach(struct isa_device *dev);
195int pcopen(dev_t dev, int flag, int mode, struct proc *p);
196int pcclose(dev_t dev, int flag, int mode, struct proc *p);
197int pcread(dev_t dev, struct uio *uio, int flag);
198int pcwrite(dev_t dev, struct uio *uio, int flag);
199int pcparam(struct tty *tp, struct termios *t);
200int pcioctl(dev_t dev, int cmd, caddr_t data, int flag, struct proc *p);
201void pcxint(dev_t dev);
202void pcstart(struct tty *tp);
203void pccnprobe(struct consdev *cp);
204void pccninit(struct consdev *cp);
205void pccnputc(dev_t dev, char c);
206int pccngetc(dev_t dev);
207void scintr(int unit);
208int pcmmap(dev_t dev, int offset, int nprot);
209u_int sgetc(int noblock);
210int getchar(void);
211static void scinit(void);
212static void scput(u_char c);
213static u_int scgetc(int noblock);
214static struct tty *get_tty_ptr(dev_t dev);
215static scr_stat *get_scr_stat(dev_t dev);
216static int get_scr_num();
217static void cursor_shape(int start, int end);
218static void get_cursor_shape(int *start, int *end);
219static void cursor_pos(int force);
220static void clear_screen(scr_stat *scp);
221static int switch_scr(u_int next_scr);
222static void exchange_scr(void);
223static void move_crsr(scr_stat *scp, int x, int y);
224static void move_up(u_short *s, u_short *d, u_int len);
225static void move_down(u_short *s, u_short *d, u_int len);
226static void scan_esc(scr_stat *scp, u_char c);
227static void ansi_put(scr_stat *scp, u_char c);
228static u_char *get_fstr(u_int c, u_int *len);
229static void update_leds(int which);
230static void kbd_wait(void);
231static void kbd_cmd(u_char command);
232static void kbd_cmd2(u_char command, u_char arg);
233static int kbd_reply(void);
234static void set_mode(scr_stat *scp);
235static void set_border(int color);
236static void load_font(int segment, int size, char* font);
237static void save_palette(void);
238static void load_palette(void);
239static void change_winsize(struct tty *tp, int x, int y);
240
241
242/* available screen savers */
243
244static void none_saver(int test);
245static void blank_saver(int test);
246static void fade_saver(int test);
247static void star_saver(int test);
248static void snake_saver(int test);
249
250static const struct {
251	char	*name;
252	void	(*routine)();
253} screen_savers[] = {
254	{ "none",	none_saver },	/* 0 */
255	{ "blank",	blank_saver },	/* 1 */
256	{ "fade",	fade_saver },	/* 2 */
257	{ "star",	star_saver },	/* 3 */
258	{ "snake",	snake_saver },	/* 4 */
259};
260#define SCRN_SAVER(arg)	(*screen_savers[scrn_saver].routine)(arg)
261#define NUM_SCRN_SAVERS	(sizeof(screen_savers) / sizeof(screen_savers[0]))
262
263/* OS specific stuff */
264
265#if defined(NetBSD)
266#define VIRTUAL_TTY(x)	pc_tty[x] ? (pc_tty[x]) : (pc_tty[x] = ttymalloc())
267#define	CONSOLE_TTY	pc_tty[NCONS] ? (pc_tty[NCONS]) : (pc_tty[NCONS] = ttymalloc())
268#define	frametype	struct trapframe
269#define eflags		tf_eflags
270extern	u_short		*Crtat;
271struct	tty 		*pc_tty[NCONS+1];
272int	ttrstrt();
273#endif
274
275#if defined(__FreeBSD__)
276#define VIRTUAL_TTY(x)	(pccons[x] = ttymalloc(pccons[x]))
277#define	CONSOLE_TTY	(pccons[NCONS] = ttymalloc(pccons[NCONS]))
278#define	frametype	struct trapframe
279#define eflags		tf_eflags
280#define	timeout_t	timeout_func_t
281#define	MONO_BUF	(KERNBASE+0xB0000)
282#define	CGA_BUF		(KERNBASE+0xB8000)
283struct	tty 		*pccons[NCONS+1];
284#endif
285
286#if defined(__386BSD__) && !defined(__FreeBSD__)
287#define VIRTUAL_TTY(x)	&pccons[x]
288#define	CONSOLE_TTY	&pccons[NCONS]
289#define	frametype	struct syscframe
290#define eflags		sf_eflags
291#define	timeout_t	caddr_t
292#define	MONO_BUF	(0xFE0B0000)
293#define	CGA_BUF		(0xFE0B8000)
294struct	tty 		pccons[NCONS+1];
295#endif
296
297#if defined(__386BSD__) || defined(__FreeBSD__)
298u_short			*Crtat = (u_short *)MONO_BUF;
299void 	consinit(void) 	{scinit();}
300#include "ddb.h"
301#if NDDB > 0
302#define DDB	1
303#endif
304#endif
305
306struct	isa_driver scdriver = {
307	pcprobe, pcattach, "sc",
308};
309
310
311int pcprobe(struct isa_device *dev)
312{
313	/* Enable interrupts and keyboard controller */
314	kbd_wait();
315	outb(KB_STAT, KB_WRITE);
316	kbd_cmd(0x4D);
317
318	/* Start keyboard stuff RESET */
319	for (;;) {
320		kbd_cmd(KB_RESET);
321		if (kbd_reply() == KB_ACK &&    /* command accepted */
322		    kbd_reply() == 0xaa)        /* self test passed */
323			break;
324		printf("Keyboard reset failed\n");
325	}
326	return (IO_KBDSIZE);
327}
328
329
330int pcattach(struct isa_device *dev)
331{
332	scr_stat *scp;
333	int start = -1, end = -1, i;
334
335	printf("sc%d: ", dev->id_unit);
336	if (crtc_vga)
337		if (crtc_addr == MONO_BASE)
338			printf("VGA mono");
339		else
340			printf("VGA color");
341	else
342		if (crtc_addr == MONO_BASE)
343			printf("MDA/hercules");
344		else
345			printf("CGA/EGA");
346
347	if (NCONS > 1)
348		printf(" <%d virtual consoles>\n", NCONS);
349	else
350		printf("\n");
351#if defined(FAT_CURSOR)
352                start = 0;
353                end = 18;
354	if (crtc_vga) {
355#else
356	if (crtc_vga) {
357		get_cursor_shape(&start, &end);
358#endif
359		save_palette();
360		load_font(0, 16, font_8x16);
361		load_font(1, 8, font_8x8);
362		load_font(2, 14, font_8x14);
363	}
364	current_default = &user_default;
365	for (i = 0; i < NCONS; i++) {
366		scp = &console[i];
367		scp->scr_buf = (u_short *)malloc(COL * ROW * 2, M_DEVBUF, M_NOWAIT);
368		scp->mode = TEXT80x25;
369		scp->term.esc = 0;
370		scp->term.std_attr = current_default->std_attr;
371		scp->term.rev_attr = current_default->rev_attr;
372		scp->term.cur_attr = scp->term.std_attr;
373		scp->border = BG_BLACK;
374		scp->cursor_start = start;
375		scp->cursor_end = end;
376		scp->xsize = COL;
377		scp->ysize = ROW;
378		scp->bell_pitch = BELL_PITCH;
379		scp->bell_duration = BELL_DURATION;
380		scp->status = 0;
381		scp->pid = 0;
382		scp->proc = NULL;
383		scp->smode.mode = VT_AUTO;
384		if (i > 0) {
385			scp->crt_base = scp->crtat = scp->scr_buf;
386			fillw(scp->term.cur_attr|scr_map[0x20], scp->scr_buf, COL*ROW);
387		}
388	}
389	/* get cursor going */
390#if defined(FAT_CURSOR)
391        cursor_shape(console[0].cursor_start,
392                     console[0].cursor_end);
393#endif
394	cursor_pos(1);
395	return 0;
396}
397
398
399static struct tty *get_tty_ptr(dev_t dev)
400{
401	int unit = minor(dev);
402
403	if (unit > NCONS)
404		return(NULL);
405	if (unit == NCONS)
406		return(CONSOLE_TTY);
407	return(VIRTUAL_TTY(unit));
408}
409
410
411static scr_stat *get_scr_stat(dev_t dev)
412{
413	int unit = minor(dev);
414
415	if (unit > NCONS)
416		return(NULL);
417	if (unit == NCONS)
418		return(&console[0]);
419	return(&console[unit]);
420}
421
422
423static int get_scr_num()
424{
425	int i = 0;
426
427	while ((i < NCONS) && (cur_console != &console[i])) i++;
428	return i < NCONS ? i : 0;
429}
430
431int pcopen(dev_t dev, int flag, int mode, struct proc *p)
432{
433	struct tty *tp = get_tty_ptr(dev);
434
435	if (!tp)
436		return(ENXIO);
437
438	tp->t_oproc = pcstart;
439	tp->t_param = pcparam;
440	tp->t_dev = dev;
441	if (!(tp->t_state & TS_ISOPEN)) {
442		tp->t_state |= TS_WOPEN;
443		ttychars(tp);
444		tp->t_iflag = TTYDEF_IFLAG;
445		tp->t_oflag = TTYDEF_OFLAG;
446		tp->t_cflag = TTYDEF_CFLAG;
447		tp->t_lflag = TTYDEF_LFLAG;
448		tp->t_ispeed = tp->t_ospeed = TTYDEF_SPEED;
449		pcparam(tp, &tp->t_termios);
450		ttsetwater(tp);
451	} else if (tp->t_state&TS_XCLUDE && p->p_ucred->cr_uid != 0)
452		return(EBUSY);
453	tp->t_state |= TS_CARR_ON;
454	tp->t_cflag |= CLOCAL;
455#if defined(__FreeBSD__)
456	return((*linesw[tp->t_line].l_open)(dev, tp, 0));
457#else
458	return((*linesw[tp->t_line].l_open)(dev, tp));
459#endif
460}
461
462
463int pcclose(dev_t dev, int flag, int mode, struct proc *p)
464{
465	struct tty *tp = get_tty_ptr(dev);
466	struct scr_stat *scp;
467
468	if (!tp)
469		return(ENXIO);
470	if (minor(dev) < NCONS) {
471		scp = get_scr_stat(tp->t_dev);
472		if (scp->status & SWITCH_WAIT_ACQ)
473			wakeup((caddr_t)&scp->smode);
474		scp->pid = 0;
475		scp->proc = NULL;
476		scp->smode.mode = VT_AUTO;
477	}
478	(*linesw[tp->t_line].l_close)(tp, flag);
479	ttyclose(tp);
480	return(0);
481}
482
483
484int pcread(dev_t dev, struct uio *uio, int flag)
485{
486	struct tty *tp = get_tty_ptr(dev);
487
488	if (!tp)
489		return(ENXIO);
490	return((*linesw[tp->t_line].l_read)(tp, uio, flag));
491}
492
493
494int pcwrite(dev_t dev, struct uio *uio, int flag)
495{
496	struct tty *tp = get_tty_ptr(dev);
497
498	if (!tp)
499		return(ENXIO);
500	return((*linesw[tp->t_line].l_write)(tp, uio, flag));
501}
502
503
504/*
505 * Got a console interrupt, keyboard action !
506 * Catch the character, and see who it goes to.
507 */
508void scintr(int unit)
509{
510	static struct tty *cur_tty;
511	int c, len;
512	u_char *cp;
513
514	/* make screensaver happy */
515	scrn_time_stamp = time.tv_sec;
516	if (scrn_blanked)
517		SCRN_SAVER(0);
518
519	c = scgetc(1);
520
521	cur_tty = VIRTUAL_TTY(get_scr_num());
522	if (!(cur_tty->t_state & TS_ISOPEN))
523		cur_tty = CONSOLE_TTY;
524
525	if (!(cur_tty->t_state & TS_ISOPEN) || polling)
526		return;
527
528	switch (c & 0xff00) {
529	case 0x0000: /* normal key */
530		(*linesw[cur_tty->t_line].l_rint)(c & 0xFF, cur_tty);
531		break;
532	case NOKEY:	/* nothing there */
533		break;
534	case FKEY:	/* function key, return string */
535		if (cp = get_fstr((u_int)c, (u_int *)&len)) {
536			while (len-- >  0)
537				(*linesw[cur_tty->t_line].l_rint)
538					(*cp++ & 0xFF, cur_tty);
539		}
540		break;
541	case MKEY:	/* meta is active, prepend ESC */
542		(*linesw[cur_tty->t_line].l_rint)(0x1b, cur_tty);
543		(*linesw[cur_tty->t_line].l_rint)(c & 0xFF, cur_tty);
544		break;
545	}
546}
547
548
549/*
550 * Set line parameters
551 */
552int pcparam(struct tty *tp, struct termios *t)
553{
554	int cflag = t->c_cflag;
555
556	/* and copy to tty */
557	tp->t_ispeed = t->c_ispeed;
558	tp->t_ospeed = t->c_ospeed;
559	tp->t_cflag = cflag;
560	return 0;
561}
562
563
564int pcioctl(dev_t dev, int cmd, caddr_t data, int flag, struct proc *p)
565{
566	int i, error;
567	struct tty *tp;
568	frametype *fp;
569	scr_stat *scp;
570
571	tp = get_tty_ptr(dev);
572	if (!tp)
573		return ENXIO;
574	scp = get_scr_stat(tp->t_dev);
575
576	switch (cmd) {	/* process console hardware related ioctl's */
577
578	case CONS_BLANKTIME:	/* set screen saver timeout (0 = no saver) */
579		scrn_blank_time = *(int*)data;
580		return 0;
581	case CONS_SSAVER:	/* set screen saver */
582		{
583		register ssaver_t *sav = (ssaver_t *)data;
584		if (sav->num < 0 || sav->num >= NUM_SCRN_SAVERS)
585			return EIO;
586		SCRN_SAVER(0);
587		scrn_saver = sav->num;
588		scrn_blank_time = sav->time;
589		return 0;
590		}
591	case CONS_GSAVER:	/* get screen saver info */
592		{
593		register ssaver_t *sav = (ssaver_t *)data;
594		if (sav->num < 0)
595			sav->num = scrn_saver;
596		else if (sav->num >= NUM_SCRN_SAVERS)
597			return EIO;
598		sav->time = scrn_blank_time;
599		strcpy(sav->name, screen_savers[sav->num].name);
600		return 0;
601		}
602	case CONS_80x25TEXT:	/* set 80x25 text mode */
603		if (!crtc_vga)
604			return ENXIO;
605		scp->mode = TEXT80x25;
606		scp->ysize = 25;
607		free(scp->scr_buf, M_DEVBUF);
608		scp->scr_buf = (u_short *)malloc(scp->xsize*scp->ysize*2,
609					     M_DEVBUF, M_NOWAIT);
610		if (scp != cur_console)
611			scp->crt_base = scp->scr_buf;
612		set_mode(scp);
613		clear_screen(scp);
614		change_winsize(tp, scp->xsize, scp->ysize);
615		return 0;
616
617	case CONS_80x50TEXT:	/* set 80x50 text mode */
618		if (!crtc_vga)
619			return ENXIO;
620		scp->mode = TEXT80x50;
621		scp->ysize = 50;
622		free(scp->scr_buf, M_DEVBUF);
623		scp->scr_buf = (u_short *)malloc(scp->xsize*scp->ysize*2,
624					     M_DEVBUF, M_NOWAIT);
625		if (scp != cur_console)
626			scp->crt_base = scp->scr_buf;
627		set_mode(scp);
628		clear_screen(scp);
629		change_winsize(tp, scp->xsize, scp->ysize);
630		return 0;
631
632	case CONS_GETVERS:	/* get version number */
633		*(int*)data = 0x103;	/* version 1.3 */
634		return 0;
635
636	case CONS_GETINFO:	/* get current (virtual) console info */
637		if (*data == sizeof(struct vid_info)) {
638			vid_info_t *ptr = (vid_info_t*)data;
639			ptr->m_num = get_scr_num();
640			ptr->mv_col = scp->xpos;
641			ptr->mv_row = scp->ypos;
642			ptr->mv_csz = scp->xsize;
643			ptr->mv_rsz = scp->ysize;
644			ptr->mv_norm.fore = (scp->term.std_attr & 0x0f00)>>8;
645			ptr->mv_norm.back = (scp->term.std_attr & 0xf000)>>12;
646			ptr->mv_rev.fore = (scp->term.rev_attr & 0x0f00)>>8;
647			ptr->mv_rev.back = (scp->term.rev_attr & 0xf000)>>12;
648			ptr->mv_grfc.fore = 0;		/* not supported */
649			ptr->mv_grfc.back = 0;		/* not supported */
650			ptr->mv_ovscan = scp->border;
651			ptr->mk_keylock = scp->status & LOCK_KEY_MASK;
652			return 0;
653		}
654		return EINVAL;
655
656	case VT_SETMODE:	/* set screen switcher mode */
657		bcopy(data, &scp->smode, sizeof(struct vt_mode));
658		if (scp->smode.mode == VT_PROCESS) {
659			scp->proc = p;
660			scp->pid = scp->proc->p_pid;
661		}
662		return 0;
663
664	case VT_GETMODE:	/* get screen switcher mode */
665		bcopy(&scp->smode, data, sizeof(struct vt_mode));
666		return 0;
667
668	case VT_RELDISP:	/* screen switcher ioctl */
669		switch(*data) {
670		case VT_FALSE:	/* user refuses to release screen, abort */
671			if (scp == old_scp && (scp->status & SWITCH_WAIT_REL)) {
672				old_scp->status &= ~SWITCH_WAIT_REL;
673				switch_in_progress = 0;
674				return 0;
675			}
676			return EINVAL;
677
678		case VT_TRUE:	/* user has released screen, go on */
679			if (scp == old_scp && (scp->status & SWITCH_WAIT_REL)) {
680				scp->status &= ~SWITCH_WAIT_REL;
681				exchange_scr();
682				if (new_scp->smode.mode == VT_PROCESS) {
683					new_scp->status |= SWITCH_WAIT_ACQ;
684					psignal(new_scp->proc,
685						new_scp->smode.acqsig);
686				}
687				else
688					switch_in_progress = 0;
689				return 0;
690			}
691			return EINVAL;
692
693		case VT_ACKACQ:	/* acquire acknowledged, switch completed */
694			if (scp == new_scp && (scp->status & SWITCH_WAIT_ACQ)) {
695				scp->status &= ~SWITCH_WAIT_ACQ;
696				switch_in_progress = 0;
697				return 0;
698			}
699			return EINVAL;
700
701		default:
702			return EINVAL;
703		}
704		/* NOT REACHED */
705
706	case VT_OPENQRY:	/* return free virtual console */
707 		for (i = 0; i < NCONS; i++) {
708			tp = VIRTUAL_TTY(i);
709 			if (!(tp->t_state & TS_ISOPEN)) {
710 				*data = i + 1;
711 				return 0;
712 			}
713		}
714 		return EINVAL;
715
716	case VT_ACTIVATE:	/* switch to screen *data */
717		return switch_scr((*data) - 1);
718
719	case VT_WAITACTIVE:	/* wait for switch to occur */
720		if (*data > NCONS)
721			return EINVAL;
722		if (minor(dev) == (*data) - 1)
723			return 0;
724		if (*data == 0) {
725			if (scp == cur_console)
726				return 0;
727			while ((error=tsleep((caddr_t)&scp->smode,
728			    	PZERO|PCATCH, "waitvt", 0)) == ERESTART) ;
729		}
730		else
731			while ((error=tsleep(
732      				(caddr_t)&console[*(data-1)].smode,
733			    	PZERO|PCATCH, "waitvt", 0)) == ERESTART) ;
734		return error;
735
736	case VT_GETACTIVE:
737		*data = get_scr_num()+1;
738		return 0;
739
740	case KDENABIO:		/* allow io operations */
741	 	fp = (frametype *)p->p_regs;
742	 	fp->eflags |= PSL_IOPL;
743		return 0;
744
745	case KDDISABIO:		/* disallow io operations (default) */
746	 	fp = (frametype *)p->p_regs;
747	 	fp->eflags &= ~PSL_IOPL;
748	 	return 0;
749
750        case KDSETMODE:		/* set current mode of this (virtual) console */
751		switch (*data) {
752		case KD_TEXT:	/* switch to TEXT (known) mode */
753				/* restore fonts & palette ! */
754			if (crtc_vga) {
755				load_font(0, 16, font_8x16);
756				load_font(1, 8, font_8x8);
757				load_font(2, 14, font_8x14);
758				load_palette();
759			}
760			/* FALL THROUGH */
761
762		case KD_TEXT1:	/* switch to TEXT (known) mode */
763				/* no restore fonts & palette */
764			scp->status &= ~UNKNOWN_MODE;
765			set_mode(scp);
766			clear_screen(scp);
767			return 0;
768
769		case KD_GRAPHICS:/* switch to GRAPHICS (unknown) mode */
770			scp->status |= UNKNOWN_MODE;
771			return 0;
772		default:
773			return EINVAL;
774		}
775		/* NOT REACHED */
776
777	case KDGETMODE:		/* get current mode of this (virtual) console */
778		*data = (scp->status & UNKNOWN_MODE) ? KD_GRAPHICS : KD_TEXT;
779		return 0;
780
781	case KDSBORDER:		/* set border color of this (virtual) console */
782		if (!crtc_vga)
783			return ENXIO;
784		scp->border = *data;
785		if (scp == cur_console)
786			set_border(scp->border);
787		return 0;
788
789	case KDSKBSTATE:	/* set keyboard state (locks) */
790		if (*data >= 0 && *data <= LOCK_KEY_MASK) {
791			scp->status &= ~LOCK_KEY_MASK;
792			scp->status |= *data;
793			if (scp == cur_console)
794				update_leds(scp->status & LED_MASK);
795			return 0;
796		}
797		return EINVAL;
798
799	case KDGKBSTATE:	/* get keyboard state (locks) */
800		*data = scp->status & LOCK_KEY_MASK;
801		return 0;
802
803	case KDSETRAD:		/* set keyboard repeat & delay rates */
804		if (*data & 0x80)
805			return EINVAL;
806		kbd_cmd2(KB_SETRAD, *data);
807		return 0;
808
809	case KDSKBMODE:		/* set keyboard mode */
810		switch (*data) {
811		case K_RAW:	/* switch to RAW scancode mode */
812			scp->status |= KBD_RAW_MODE;
813			return 0;
814
815		case K_XLATE:	/* switch to XLT ascii mode */
816			if (scp == cur_console && scp->status == KBD_RAW_MODE)
817				shfts = ctls = alts = agrs = metas = 0;
818			scp->status &= ~KBD_RAW_MODE;
819			return 0;
820		default:
821			return EINVAL;
822		}
823		/* NOT REACHED */
824
825	case KDGKBMODE:		/* get keyboard mode */
826		*data = (scp->status & KBD_RAW_MODE) ? K_RAW : K_XLATE;
827		return 0;
828
829	case KDMKTONE:		/* sound the bell */
830		if (scp == cur_console)
831			sysbeep(scp->bell_pitch, scp->bell_duration);
832		return 0;
833
834	case KIOCSOUND:		/* make tone (*data) hz */
835		if (scp == cur_console) {
836			if (*(int*)data) {
837			int pitch = TIMER_FREQ/(*(int*)data);
838				/* enable counter 2 */
839				outb(0x61, inb(0x61) | 3);
840				/* set command for counter 2, 2 byte write */
841				outb(TIMER_MODE,
842				     	TIMER_SEL2|TIMER_16BIT|TIMER_SQWAVE);
843				/* set pitch */
844				outb(TIMER_CNTR2, pitch);
845				outb(TIMER_CNTR2, (pitch>>8));
846			}
847			else {
848				/* disable counter 2 */
849				outb(0x61, inb(0x61) & 0xFC);
850			}
851		}
852		return 0;
853
854	case KDGKBTYPE:		/* get keyboard type */
855		*data = 0;	/* type not known (yet) */
856		return 0;
857
858	case KDSETLED:		/* set keyboard LED status */
859		if (*data >= 0 && *data <= LED_MASK) {
860			scp->status &= ~LED_MASK;
861			scp->status |= *data;
862			if (scp == cur_console)
863			update_leds(scp->status & LED_MASK);
864			return 0;
865		}
866		return EINVAL;
867
868	case KDGETLED:		/* get keyboard LED status */
869		*data = scp->status & LED_MASK;
870		return 0;
871
872	case GETFKEY:		/* get functionkey string */
873		if (*(u_short*)data < n_fkey_tab) {
874		 	fkeyarg_t *ptr = (fkeyarg_t*)data;
875			bcopy(&fkey_tab[ptr->keynum].str,
876			      ptr->keydef,
877			      fkey_tab[ptr->keynum].len);
878			ptr->flen = fkey_tab[ptr->keynum].len;
879			return 0;
880		}
881		else
882			return EINVAL;
883
884	case SETFKEY:		/* set functionkey string */
885		if (*(u_short*)data < n_fkey_tab) {
886		 	fkeyarg_t *ptr = (fkeyarg_t*)data;
887			bcopy(ptr->keydef,
888			      &fkey_tab[ptr->keynum].str,
889			      min(ptr->flen, MAXFK));
890			fkey_tab[ptr->keynum].len = min(ptr->flen, MAXFK);
891			return 0;
892		}
893		else
894			return EINVAL;
895
896	case GIO_SCRNMAP: 	/* get output translation table */
897		bcopy(&scr_map, data, sizeof(scr_map));
898		return 0;
899
900	case PIO_SCRNMAP:	/* set output translation table */
901		bcopy(data, &scr_map, sizeof(scr_map));
902		return 0;
903
904	case GIO_KEYMAP: 	/* get keyboard translation table */
905		bcopy(&key_map, data, sizeof(key_map));
906		return 0;
907
908	case PIO_KEYMAP:	/* set keyboard translation table */
909		bcopy(data, &key_map, sizeof(key_map));
910		return 0;
911
912	case PIO_FONT8x8:	/* set 8x8 dot font */
913		if (!crtc_vga)
914			return ENXIO;
915		bcopy(data, &font_8x8, sizeof(font_8x8));
916		load_font(1, 8, font_8x8);
917		return 0;
918
919	case GIO_FONT8x8:	/* get 8x8 dot font */
920		if (!crtc_vga)
921			return ENXIO;
922		bcopy(&font_8x8, data, sizeof(font_8x8));
923		return 0;
924
925	case PIO_FONT8x14:	/* set 8x14 dot font */
926		if (!crtc_vga)
927			return ENXIO;
928		bcopy(data, &font_8x14, sizeof(font_8x14));
929		load_font(2, 14, font_8x14);
930		return 0;
931
932	case GIO_FONT8x14:	/* get 8x14 dot font */
933		if (!crtc_vga)
934			return ENXIO;
935		bcopy(&font_8x14, data, sizeof(font_8x14));
936		return 0;
937
938	case PIO_FONT8x16:	/* set 8x16 dot font */
939		if (!crtc_vga)
940			return ENXIO;
941		bcopy(data, &font_8x16, sizeof(font_8x16));
942		load_font(0, 16, font_8x16);
943		return 0;
944
945	case GIO_FONT8x16:	/* get 8x16 dot font */
946		if (!crtc_vga)
947			return ENXIO;
948		bcopy(&font_8x16, data, sizeof(font_8x16));
949		return 0;
950
951	case CONSOLE_X_MODE_ON:	/* just to be compatible */
952		if (saved_console < 0) {
953			saved_console = get_scr_num();
954			switch_scr(minor(dev));
955	 		fp = (frametype *)p->p_regs;
956	 		fp->eflags |= PSL_IOPL;
957			scp->status |= UNKNOWN_MODE;
958			scp->status |= KBD_RAW_MODE;
959			return 0;
960		}
961		return EAGAIN;
962
963	case CONSOLE_X_MODE_OFF:/* just to be compatible */
964	 	fp = (frametype *)p->p_regs;
965	 	fp->eflags &= ~PSL_IOPL;
966		if (crtc_vga) {
967			load_font(0, 16, font_8x16);
968			load_font(1, 8, font_8x8);
969			load_font(2, 14, font_8x14);
970			load_palette();
971		}
972		scp->status &= ~UNKNOWN_MODE;
973		set_mode(scp);
974		clear_screen(scp);
975		scp->status &= ~KBD_RAW_MODE;
976		switch_scr(saved_console);
977		saved_console = -1;
978		return 0;
979
980	 case CONSOLE_X_BELL:	/* more compatibility */
981                /*
982                 * if set, data is a pointer to a length 2 array of
983                 * integers. data[0] is the pitch in Hz and data[1]
984                 * is the duration in msec.
985                 */
986                if (data)
987	    		sysbeep(TIMER_FREQ/((int*)data)[0],
988				((int*)data)[1]*hz/3000);
989                else
990			sysbeep(scp->bell_pitch, scp->bell_duration);
991                return 0;
992
993	default:
994		break;
995	}
996
997	error = (*linesw[tp->t_line].l_ioctl)(tp, cmd, data, flag);
998	if (error >= 0)
999		return(error);
1000	error = ttioctl(tp, cmd, data, flag);
1001	if (error >= 0)
1002		return(error);
1003	return(ENOTTY);
1004}
1005
1006
1007void pcxint(dev_t dev)
1008{
1009	struct tty *tp = get_tty_ptr(dev);
1010
1011	if (!tp)
1012		return;
1013	tp->t_state &= ~TS_BUSY;
1014	if (tp->t_line)
1015		(*linesw[tp->t_line].l_start)(tp);
1016	else
1017		pcstart(tp);
1018}
1019
1020
1021void pcstart(struct tty *tp)
1022{
1023#if defined(NetBSD)
1024	struct clist *rbp;
1025	int i, s, len;
1026	u_char buf[PCBURST];
1027	scr_stat *scp = get_scr_stat(tp->t_dev);
1028
1029	if (scp->status & SLKED)
1030		return;
1031	s = spltty();
1032	if (!(tp->t_state & (TS_TIMEOUT|TS_BUSY|TS_TTSTOP))) {
1033		tp->t_state |= TS_BUSY;
1034		splx(s);
1035		rbp = &tp->t_outq;
1036		len = q_to_b(rbp, buf, PCBURST);
1037		for (i=0; i<len; i++)
1038			if (buf[i]) ansi_put(scp, buf[i]);
1039		s = spltty();
1040		tp->t_state &= ~TS_BUSY;
1041		if (rbp->c_cc) {
1042			tp->t_state |= TS_TIMEOUT;
1043			timeout((timeout_t)ttrstrt, (caddr_t)tp, 1);
1044		}
1045		if (rbp->c_cc <= tp->t_lowat) {
1046			if (tp->t_state & TS_ASLEEP) {
1047				tp->t_state &= ~TS_ASLEEP;
1048				wakeup((caddr_t)rbp);
1049			}
1050			selwakeup(&tp->t_wsel);
1051		}
1052	}
1053	splx(s);
1054
1055#else /* __FreeBSD__ & __386BSD__ */
1056
1057	int c, s;
1058	scr_stat *scp = get_scr_stat(tp->t_dev);
1059
1060	if (scp->status & SLKED)
1061		return;
1062	s = spltty();
1063	if (!(tp->t_state & (TS_TIMEOUT|TS_BUSY|TS_TTSTOP)))
1064		for (;;) {
1065			if (RB_LEN(tp->t_out) <= tp->t_lowat) {
1066				if (tp->t_state & TS_ASLEEP) {
1067					tp->t_state &= ~TS_ASLEEP;
1068					wakeup((caddr_t)tp->t_out);
1069				}
1070				if (tp->t_wsel) {
1071					selwakeup(tp->t_wsel,
1072						  tp->t_state & TS_WCOLL);
1073					tp->t_wsel = 0;
1074					tp->t_state &= ~TS_WCOLL;
1075				}
1076			}
1077			if (RB_LEN(tp->t_out) == 0)
1078				break;
1079			if (scp->status & SLKED)
1080				break;
1081			c = getc(tp->t_out);
1082			tp->t_state |= TS_BUSY;
1083			splx(s);
1084			ansi_put(scp, c);
1085			s = spltty();
1086			tp->t_state &= ~TS_BUSY;
1087		}
1088	splx(s);
1089#endif
1090}
1091
1092
1093void pccnprobe(struct consdev *cp)
1094{
1095	int maj;
1096
1097	/* locate the major number */
1098	for (maj = 0; maj < nchrdev; maj++)
1099		if ((void*)cdevsw[maj].d_open == (void*)pcopen)
1100			break;
1101
1102	/* initialize required fields */
1103	cp->cn_dev = makedev(maj, NCONS);
1104	cp->cn_pri = CN_INTERNAL;
1105#warning Crude hack, do it better
1106/*#if defined(__FreeBSD__) || defined(__386BSD__)
1107	cp->cn_tp = CONSOLE_TTY;
1108#endif*/
1109}
1110
1111
1112void pccninit(struct consdev *cp)
1113{
1114	scinit();
1115}
1116
1117
1118void pccnputc(dev_t dev, char c)
1119{
1120	if (c == '\n')
1121		scput('\r');
1122	scput(c);
1123	if (cur_console == &console[0]) {
1124	int 	pos = cur_console->crtat - cur_console->crt_base;
1125		if (pos != cur_cursor_pos) {
1126			cur_cursor_pos = pos;
1127			outb(crtc_addr,14);
1128			outb(crtc_addr+1,pos >> 8);
1129			outb(crtc_addr,15);
1130			outb(crtc_addr+1,pos&0xff);
1131		}
1132	}
1133}
1134
1135
1136int pccngetc(dev_t dev)
1137{
1138	int s = spltty();		/* block scintr while we poll */
1139	int c = scgetc(0);
1140	splx(s);
1141	if (c == '\r') c = '\n';
1142	return(c);
1143}
1144
1145static void none_saver(int test)
1146{
1147}
1148
1149static void fade_saver(int test)
1150{
1151	static int count = 0;
1152	int i;
1153
1154	if (test) {
1155		scrn_blanked = 1;
1156		if (count < 64) {
1157  			outb(PIXMASK, 0xFF);		/* no pixelmask */
1158  			outb(PALWADR, 0x00);
1159    			outb(PALDATA, 0);
1160    			outb(PALDATA, 0);
1161    			outb(PALDATA, 0);
1162  			for (i = 3; i < 768; i++) {
1163  				if (palette[i] - count > 15)
1164    					outb(PALDATA, palette[i]-count);
1165				else
1166    					outb(PALDATA, 15);
1167			}
1168			inb(crtc_addr+6);		/* reset flip/flop */
1169			outb(ATC, 0x20);		/* enable palette */
1170			count++;
1171		}
1172	}
1173	else {
1174		count = scrn_blanked = 0;
1175		load_palette();
1176	}
1177}
1178
1179static void blank_saver(int test)
1180{
1181	u_char val;
1182	if (test) {
1183		scrn_blanked = 1;
1184  		outb(TSIDX, 0x01); val = inb(TSREG);
1185		outb(TSIDX, 0x01); outb(TSREG, val | 0x20);
1186	}
1187	else {
1188		scrn_blanked = 0;
1189  		outb(TSIDX, 0x01); val = inb(TSREG);
1190		outb(TSIDX, 0x01); outb(TSREG, val & 0xDF);
1191	}
1192}
1193
1194static u_long 	rand_next = 1;
1195
1196static int rand()
1197{
1198	return ((rand_next = rand_next * 1103515245 + 12345) & 0x7FFFFFFF);
1199}
1200
1201/*
1202 * Alternate saver that got its inspiration from a well known utility
1203 * package for an unfamous OS.
1204 */
1205
1206#define NUM_STARS	50
1207
1208static void star_saver(int test)
1209{
1210	scr_stat	*scp = cur_console;
1211	int		cell, i;
1212	char 		pattern[] = {"...........++++***   "};
1213	char		colors[] = {FG_DARKGREY, FG_LIGHTGREY,
1214				    FG_WHITE, FG_LIGHTCYAN};
1215	static u_short 	stars[NUM_STARS][2];
1216
1217	if (test) {
1218		if (!scrn_blanked) {
1219			bcopy(Crtat, scp->scr_buf,
1220			      scp->xsize * scp->ysize * 2);
1221			fillw((FG_LIGHTGREY|BG_BLACK)<<8 | scr_map[0x20], Crtat,
1222			      scp->xsize * scp->ysize);
1223			set_border(0);
1224			i = scp->ysize * scp->xsize + 5;
1225			outb(crtc_addr, 14);
1226			outb(crtc_addr+1, i >> 8);
1227			outb(crtc_addr, 15);
1228			outb(crtc_addr+1, i & 0xff);
1229			scrn_blanked = 1;
1230 			for(i=0; i<NUM_STARS; i++) {
1231  				stars[i][0] =
1232					rand() % (scp->xsize*scp->ysize);
1233  				stars[i][1] = 0;
1234 			}
1235		}
1236   		cell = rand() % NUM_STARS;
1237		*((u_short*)(Crtat + stars[cell][0])) =
1238			scr_map[pattern[stars[cell][1]]] |
1239			        colors[rand()%sizeof(colors)] << 8;
1240		if ((stars[cell][1]+=(rand()%4)) >= sizeof(pattern)-1) {
1241    			stars[cell][0] = rand() % (scp->xsize*scp->ysize);
1242   			stars[cell][1] = 0;
1243		}
1244	}
1245	else {
1246		if (scrn_blanked) {
1247			bcopy(scp->scr_buf, Crtat, scp->xsize*scp->ysize*2);
1248			cur_cursor_pos = -1;
1249			set_border(scp->border);
1250			scrn_blanked = 0;
1251		}
1252	}
1253}
1254
1255
1256static void snake_saver(int test)
1257{
1258	const char	saves[] = {"FreeBSD"};
1259	static u_char	*savs[sizeof(saves)-1];
1260	static int	dirx, diry;
1261	int		f;
1262	scr_stat	*scp = cur_console;
1263
1264	if (test) {
1265		if (!scrn_blanked) {
1266			bcopy(Crtat, scp->scr_buf,
1267			      scp->xsize * scp->ysize * 2);
1268			fillw((FG_LIGHTGREY|BG_BLACK)<<8 | scr_map[0x20],
1269			      Crtat, scp->xsize * scp->ysize);
1270			set_border(0);
1271			dirx = (scp->xpos ? 1 : -1);
1272			diry = (scp->ypos ?
1273				scp->xsize : -scp->xsize);
1274			for (f=0; f< sizeof(saves)-1; f++)
1275				savs[f] = (u_char *)Crtat + 2 *
1276					  (scp->xpos+scp->ypos*scp->xsize);
1277			*(savs[0]) = scr_map[*saves];
1278			f = scp->ysize * scp->xsize + 5;
1279			outb(crtc_addr, 14);
1280			outb(crtc_addr+1, f >> 8);
1281			outb(crtc_addr, 15);
1282			outb(crtc_addr+1, f & 0xff);
1283			scrn_blanked = 1;
1284		}
1285		if (scrn_blanked++ < 4)
1286			return;
1287		scrn_blanked = 1;
1288		*(savs[sizeof(saves)-2]) = scr_map[0x20];
1289		for (f=sizeof(saves)-2; f > 0; f--)
1290			savs[f] = savs[f-1];
1291		f = (savs[0] - (u_char *)Crtat) / 2;
1292		if ((f % scp->xsize) == 0 ||
1293		    (f % scp->xsize) == scp->xsize - 1 ||
1294		    (rand() % 50) == 0)
1295			dirx = -dirx;
1296		if ((f / scp->xsize) == 0 ||
1297		    (f / scp->xsize) == scp->ysize - 1 ||
1298		    (rand() % 20) == 0)
1299			diry = -diry;
1300		savs[0] += 2*dirx + 2*diry;
1301		for (f=sizeof(saves)-2; f>=0; f--)
1302			*(savs[f]) = scr_map[saves[f]];
1303	}
1304	else {
1305		if (scrn_blanked) {
1306			bcopy(scp->scr_buf, Crtat,
1307			      scp->xsize * scp->ysize * 2);
1308			cur_cursor_pos = -1;
1309			set_border(scp->border);
1310			scrn_blanked = 0;
1311		}
1312	}
1313}
1314
1315static void cursor_shape(int start, int end)
1316{
1317	outb(crtc_addr, 10);
1318	outb(crtc_addr+1, start & 0xFF);
1319	outb(crtc_addr, 11);
1320	outb(crtc_addr+1, end & 0xFF);
1321}
1322
1323
1324#if !defined(FAT_CURSOR)
1325static void get_cursor_shape(int *start, int *end)
1326{
1327	outb(crtc_addr, 10);
1328	*start = inb(crtc_addr+1) & 0x1F;
1329	outb(crtc_addr, 11);
1330	*end = inb(crtc_addr+1) & 0x1F;
1331}
1332#endif
1333
1334
1335static void cursor_pos(int force)
1336{
1337	int pos;
1338
1339	if (cur_console->status & UNKNOWN_MODE)
1340		return;
1341	if (scrn_blank_time && (time.tv_sec > scrn_time_stamp+scrn_blank_time))
1342		SCRN_SAVER(1);
1343	pos = cur_console->crtat - cur_console->crt_base;
1344	if (force || (!scrn_blanked && pos != cur_cursor_pos)) {
1345		cur_cursor_pos = pos;
1346		outb(crtc_addr, 14);
1347		outb(crtc_addr+1, pos>>8);
1348		outb(crtc_addr, 15);
1349		outb(crtc_addr+1, pos&0xff);
1350	}
1351	timeout((timeout_t)cursor_pos, 0, hz/20);
1352}
1353
1354
1355static void clear_screen(scr_stat *scp)
1356{
1357	move_crsr(scp, 0, 0);
1358	fillw(scp->term.cur_attr | scr_map[0x20], scp->crt_base,
1359	       scp->xsize * scp->ysize);
1360}
1361
1362
1363static int switch_scr(u_int next_scr)
1364{
1365	if (in_putc) {		/* delay switch if in putc */
1366		delayed_next_scr = next_scr+1;
1367		return 0;
1368	}
1369	if (switch_in_progress &&
1370	    (cur_console->proc != pfind(cur_console->pid)))
1371		switch_in_progress = 0;
1372
1373    	if (next_scr >= NCONS || switch_in_progress) {
1374		sysbeep(BELL_PITCH, BELL_DURATION);
1375		return EINVAL;
1376	}
1377
1378	/* is the wanted virtual console open ? */
1379	if (next_scr) {
1380		struct tty *tp = VIRTUAL_TTY(next_scr);
1381		if (!(tp->t_state & TS_ISOPEN)) {
1382			sysbeep(BELL_PITCH, BELL_DURATION);
1383			return EINVAL;
1384		}
1385	}
1386
1387	switch_in_progress = 1;
1388	old_scp = cur_console;
1389	new_scp = &console[next_scr];
1390	wakeup((caddr_t)&new_scp->smode);
1391	if (new_scp == old_scp) {
1392		switch_in_progress = 0;
1393		return 0;
1394	}
1395
1396	/* has controlling process died? */
1397	if (old_scp->proc && (old_scp->proc != pfind(old_scp->pid)))
1398		old_scp->smode.mode = VT_AUTO;
1399	if (new_scp->proc && (new_scp->proc != pfind(new_scp->pid)))
1400		new_scp->smode.mode = VT_AUTO;
1401
1402	/* check the modes and switch approbiatly */
1403	if (old_scp->smode.mode == VT_PROCESS) {
1404		old_scp->status |= SWITCH_WAIT_REL;
1405		psignal(old_scp->proc, old_scp->smode.relsig);
1406	}
1407	else {
1408		exchange_scr();
1409		if (new_scp->smode.mode == VT_PROCESS) {
1410			new_scp->status |= SWITCH_WAIT_ACQ;
1411			psignal(new_scp->proc, new_scp->smode.acqsig);
1412		}
1413		else
1414			switch_in_progress = 0;
1415	}
1416	return 0;
1417}
1418
1419
1420static void exchange_scr(void)
1421{
1422	struct tty *tp;
1423
1424	bcopy(Crtat, old_scp->scr_buf, old_scp->xsize * old_scp->ysize * 2);
1425	old_scp->crt_base = old_scp->scr_buf;
1426	move_crsr(old_scp, old_scp->xpos, old_scp->ypos);
1427	cur_console = new_scp;
1428	set_mode(new_scp);
1429	new_scp->crt_base = Crtat;
1430	move_crsr(new_scp, new_scp->xpos, new_scp->ypos);
1431	bcopy(new_scp->scr_buf, Crtat, new_scp->xsize * new_scp->ysize * 2);
1432	update_leds(new_scp->status & LED_MASK);
1433	if (old_scp->status & KBD_RAW_MODE || new_scp->status & KBD_RAW_MODE)
1434		shfts = ctls = alts = agrs = metas = 0;
1435	delayed_next_scr = 0;
1436}
1437
1438
1439static void move_crsr(scr_stat *scp, int x, int y)
1440{
1441	if (x < 0 || y < 0 || x >= scp->xsize || y >= scp->ysize)
1442		return;
1443	scp->xpos = x;
1444	scp->ypos = y;
1445	scp->crtat = scp->crt_base + scp->ypos * scp->xsize + scp->xpos;
1446}
1447
1448
1449static void move_up(u_short *s, u_short *d, u_int len)
1450{
1451	s += len;
1452	d += len;
1453	while (len-- > 0)
1454		*--d = *--s;
1455}
1456
1457
1458static void move_down(u_short *s, u_short *d, u_int len)
1459{
1460	while (len-- > 0)
1461		*d++ = *s++;
1462}
1463
1464
1465static void scan_esc(scr_stat *scp, u_char c)
1466{
1467	static u_char ansi_col[16] =
1468		{0, 4, 2, 6, 1, 5, 3, 7, 8, 12, 10, 14, 9, 13, 11, 15};
1469	int i, n;
1470	u_short *src, *dst, count;
1471
1472	if (scp->term.esc == 1) {
1473		switch (c) {
1474
1475		case '[': 	/* Start ESC [ sequence */
1476			scp->term.esc = 2;
1477			scp->term.last_param = -1;
1478			for (i = scp->term.num_param; i < MAX_ESC_PAR; i++)
1479				scp->term.param[i] = 1;
1480			scp->term.num_param = 0;
1481			return;
1482
1483		case 'M':	/* Move cursor up 1 line, scroll if at top */
1484			if (scp->ypos > 0)
1485				move_crsr(scp, scp->xpos, scp->ypos - 1);
1486			else {
1487				move_up(scp->crt_base,
1488					scp->crt_base + scp->xsize,
1489					(scp->ysize - 1) * scp->xsize);
1490				fillw(scp->term.cur_attr | scr_map[0x20],
1491				      scp->crt_base, scp->xsize);
1492			}
1493			break;
1494#if notyet
1495		case 'Q':
1496			scp->term.esc = 4;
1497			break;
1498#endif
1499		case 'c':	/* Clear screen & home */
1500			clear_screen(scp);
1501			break;
1502		}
1503	}
1504	else if (scp->term.esc == 2) {
1505		if (c >= '0' && c <= '9') {
1506			if (scp->term.num_param < MAX_ESC_PAR) {
1507				if (scp->term.last_param != scp->term.num_param) {
1508					scp->term.last_param = scp->term.num_param;
1509					scp->term.param[scp->term.num_param] = 0;
1510				}
1511				else
1512					scp->term.param[scp->term.num_param] *= 10;
1513				scp->term.param[scp->term.num_param] += c - '0';
1514				return;
1515			}
1516		}
1517		scp->term.num_param = scp->term.last_param + 1;
1518		switch (c) {
1519
1520		case ';':
1521			if (scp->term.num_param < MAX_ESC_PAR)
1522				return;
1523			break;
1524
1525		case '=':
1526			scp->term.esc = 3;
1527			scp->term.last_param = -1;
1528			for (i = scp->term.num_param; i < MAX_ESC_PAR; i++)
1529				scp->term.param[i] = 1;
1530			scp->term.num_param = 0;
1531			return;
1532
1533		case 'A': /* up n rows */
1534			n = scp->term.param[0]; if (n < 1) n = 1;
1535			move_crsr(scp, scp->xpos, scp->ypos - n);
1536			break;
1537
1538		case 'B': /* down n rows */
1539			n = scp->term.param[0]; if (n < 1) n = 1;
1540			move_crsr(scp, scp->xpos, scp->ypos + n);
1541			break;
1542
1543		case 'C': /* right n columns */
1544			n = scp->term.param[0]; if (n < 1) n = 1;
1545			move_crsr(scp, scp->xpos + n, scp->ypos);
1546			break;
1547
1548		case 'D': /* left n columns */
1549			n = scp->term.param[0]; if (n < 1) n = 1;
1550			move_crsr(scp, scp->xpos - n, scp->ypos);
1551			break;
1552
1553		case 'E': /* cursor to start of line n lines down */
1554			n = scp->term.param[0]; if (n < 1) n = 1;
1555			move_crsr(scp, 0, scp->ypos + n);
1556			break;
1557
1558		case 'F': /* cursor to start of line n lines up */
1559			n = scp->term.param[0]; if (n < 1) n = 1;
1560			move_crsr(scp, 0, scp->ypos - n);
1561			break;
1562
1563		case 'f': /* System V consoles .. */
1564		case 'H': /* Cursor move */
1565			if (scp->term.num_param == 0)
1566				move_crsr(scp, 0, 0);
1567			else if (scp->term.num_param == 2)
1568				move_crsr(scp, scp->term.param[1] - 1,
1569					  scp->term.param[0] - 1);
1570			break;
1571
1572		case 'J': /* Clear all or part of display */
1573			if (scp->term.num_param == 0)
1574				n = 0;
1575			else
1576				n = scp->term.param[0];
1577			switch (n) {
1578			case 0: /* clear form cursor to end of display */
1579				fillw(scp->term.cur_attr | scr_map[0x20],
1580				      scp->crtat, scp->crt_base +
1581				      scp->xsize * scp->ysize -
1582				      scp->crtat);
1583				break;
1584			case 1: /* clear from beginning of display to cursor */
1585				fillw(scp->term.cur_attr | scr_map[0x20],
1586				      scp->crt_base,
1587				      scp->crtat - scp->crt_base);
1588				break;
1589			case 2: /* clear entire display */
1590				clear_screen(scp);
1591				break;
1592			}
1593			break;
1594
1595		case 'K': /* Clear all or part of line */
1596			if (scp->term.num_param == 0)
1597				n = 0;
1598			else
1599				n = scp->term.param[0];
1600			switch (n) {
1601			case 0: /* clear form cursor to end of line */
1602				fillw(scp->term.cur_attr | scr_map[0x20],
1603				      scp->crtat, scp->xsize - scp->xpos);
1604				break;
1605			case 1: /* clear from beginning of line to cursor */
1606				fillw(scp->term.cur_attr|scr_map[0x20],
1607				      scp->crtat - (scp->xsize - scp->xpos),
1608				      (scp->xsize - scp->xpos) + 1);
1609				break;
1610			case 2: /* clear entire line */
1611				fillw(scp->term.cur_attr|scr_map[0x20],
1612				      scp->crtat - (scp->xsize - scp->xpos),
1613				      scp->xsize);
1614				break;
1615			}
1616			break;
1617
1618		case 'L':	/* Insert n lines */
1619			n = scp->term.param[0]; if (n < 1) n = 1;
1620			if (n > scp->ysize - scp->ypos)
1621				n = scp->ysize - scp->ypos;
1622			src = scp->crt_base + scp->ypos * scp->xsize;
1623			dst = src + n * scp->xsize;
1624			count = scp->ysize - (scp->ypos + n);
1625			move_up(src, dst, count * scp->xsize);
1626			fillw(scp->term.cur_attr | scr_map[0x20], src,
1627			      n * scp->xsize);
1628			break;
1629
1630		case 'M':	/* Delete n lines */
1631			n = scp->term.param[0]; if (n < 1) n = 1;
1632			if (n > scp->ysize - scp->ypos)
1633				n = scp->ysize - scp->ypos;
1634			dst = scp->crt_base + scp->ypos * scp->xsize;
1635			src = dst + n * scp->xsize;
1636			count = scp->ysize - (scp->ypos + n);
1637			move_down(src, dst, count * scp->xsize);
1638			src = dst + count * scp->xsize;
1639			fillw(scp->term.cur_attr | scr_map[0x20], src,
1640			      n * scp->xsize);
1641			break;
1642
1643		case 'P':	/* Delete n chars */
1644			n = scp->term.param[0]; if (n < 1) n = 1;
1645			if (n > scp->xsize - scp->xpos)
1646				n = scp->xsize - scp->xpos;
1647			dst = scp->crtat;
1648			src = dst + n;
1649			count = scp->xsize - (scp->xpos + n);
1650			move_down(src, dst, count);
1651			src = dst + count;
1652			fillw(scp->term.cur_attr | scr_map[0x20], src, n);
1653			break;
1654
1655		case '@':	/* Insert n chars */
1656			n = scp->term.param[0]; if (n < 1) n = 1;
1657			if (n > scp->xsize - scp->xpos)
1658				n = scp->xsize - scp->xpos;
1659			src = scp->crtat;
1660			dst = src + n;
1661			count = scp->xsize - (scp->xpos + n);
1662			move_up(src, dst, count);
1663			fillw(scp->term.cur_attr | scr_map[0x20], src, n);
1664			break;
1665
1666		case 'S':	/* scroll up n lines */
1667			n = scp->term.param[0]; if (n < 1)  n = 1;
1668			bcopy(scp->crt_base + (scp->xsize * n),
1669			      scp->crt_base,
1670			      scp->xsize * (scp->ysize - n) *
1671			      sizeof(u_short));
1672			fillw(scp->term.cur_attr | scr_map[0x20],
1673			      scp->crt_base + scp->xsize *
1674			      (scp->ysize - 1),
1675			      scp->xsize);
1676			break;
1677
1678		case 'T':	/* scroll down n lines */
1679			n = scp->term.param[0]; if (n < 1)  n = 1;
1680			bcopy(scp->crt_base,
1681			      scp->crt_base + (scp->xsize * n),
1682			      scp->xsize * (scp->ysize - n) *
1683			      sizeof(u_short));
1684			fillw(scp->term.cur_attr | scr_map[0x20], scp->crt_base,
1685			      scp->xsize);
1686			break;
1687
1688		case 'X':	/* delete n characters in line */
1689			n = scp->term.param[0]; if (n < 1)  n = 1;
1690			fillw(scp->term.cur_attr | scr_map[0x20],
1691                              scp->crt_base + scp->xpos +
1692			      ((scp->xsize*scp->ypos) * sizeof(u_short)), n);
1693			break;
1694
1695		case 'Z':	/* move n tabs backwards */
1696			n = scp->term.param[0]; if (n < 1)  n = 1;
1697			if ((i = scp->xpos & 0xf8) == scp->xpos)
1698				i -= 8*n;
1699			else
1700				i -= 8*(n-1);
1701			if (i < 0)
1702				i = 0;
1703			move_crsr(scp, i, scp->ypos);
1704			break;
1705
1706		case '`': 	/* move cursor to column n */
1707			n = scp->term.param[0]; if (n < 1)  n = 1;
1708			move_crsr(scp, n, scp->ypos);
1709			break;
1710
1711		case 'a': 	/* move cursor n columns to the right */
1712			n = scp->term.param[0]; if (n < 1)  n = 1;
1713			move_crsr(scp, scp->xpos + n, scp->ypos);
1714			break;
1715
1716		case 'd': 	/* move cursor to row n */
1717			n = scp->term.param[0]; if (n < 1)  n = 1;
1718			move_crsr(scp, scp->xpos, n);
1719			break;
1720
1721		case 'e': 	/* move cursor n rows down */
1722			n = scp->term.param[0]; if (n < 1)  n = 1;
1723			move_crsr(scp, scp->xpos, scp->ypos + n);
1724			break;
1725
1726		case 'm': 	/* change attribute */
1727			if (scp->term.num_param == 0)
1728				n = 0;
1729			else
1730				n = scp->term.param[0];
1731			switch (n) {
1732			case 0:	/* back to normal */
1733				scp->term.cur_attr = scp->term.std_attr;
1734				break;
1735			case 1:	/* highlight (bold) */
1736				scp->term.cur_attr &= 0xFF00;
1737				scp->term.cur_attr |= 0x0800;
1738				break;
1739			case 4: /* highlight (underline) */
1740				scp->term.cur_attr &= 0x0F00;
1741				scp->term.cur_attr |= 0x0800;
1742				break;
1743			case 5: /* blink */
1744				scp->term.cur_attr &= 0xFF00;
1745				scp->term.cur_attr |= 0x8000;
1746				break;
1747			case 7: /* reverse video */
1748				scp->term.cur_attr = scp->term.rev_attr;
1749				break;
1750			case 30: case 31: case 32: case 33: /* set fg color */
1751			case 34: case 35: case 36: case 37:
1752				scp->term.cur_attr = (scp->term.cur_attr & 0xF0FF)
1753					    | (ansi_col[(n - 30) & 7] << 8);
1754				break;
1755			case 40: case 41: case 42: case 43: /* set bg color */
1756			case 44: case 45: case 46: case 47:
1757				scp->term.cur_attr = (scp->term.cur_attr & 0x0FFF)
1758					    | (ansi_col[(n - 40) & 7] << 12);
1759				break;
1760			}
1761			break;
1762
1763		case 'x':
1764			if (scp->term.num_param == 0)
1765				n = 0;
1766			else
1767				n = scp->term.param[0];
1768			switch (n) {
1769			case 0: 	/* reset attributes */
1770				scp->term.cur_attr = scp->term.std_attr =
1771					current_default->std_attr;
1772				scp->term.rev_attr = current_default->rev_attr;
1773				break;
1774			case 1: 	/* set ansi background */
1775				scp->term.cur_attr = scp->term.std_attr =
1776					(scp->term.std_attr & 0x0F00) |
1777					(ansi_col[(scp->term.param[1])&0x0F]<<12);
1778				break;
1779			case 2: 	/* set ansi foreground */
1780				scp->term.cur_attr = scp->term.std_attr =
1781					(scp->term.std_attr & 0xF000) |
1782					(ansi_col[(scp->term.param[1])&0x0F]<<8);
1783				break;
1784			case 3: 	/* set ansi attribute directly */
1785				scp->term.cur_attr = scp->term.std_attr =
1786					(scp->term.param[1]&0xFF)<<8;
1787				break;
1788			case 5: 	/* set ansi reverse video background */
1789				scp->term.rev_attr =
1790					(scp->term.rev_attr & 0x0F00) |
1791					(ansi_col[(scp->term.param[1])&0x0F]<<12);
1792				break;
1793			case 6: 	/* set ansi reverse video foreground */
1794				scp->term.rev_attr =
1795					(scp->term.rev_attr & 0xF000) |
1796					(ansi_col[(scp->term.param[1])&0x0F]<<8);
1797				break;
1798			case 7: 	/* set ansi reverse video directly */
1799				scp->term.rev_attr = (scp->term.param[1]&0xFF)<<8;
1800				break;
1801			}
1802			break;
1803
1804		case 'z':	/* switch to (virtual) console n */
1805			if (scp->term.num_param == 1)
1806				switch_scr(scp->term.param[0]);
1807			break;
1808		}
1809	}
1810	else if (scp->term.esc == 3) {
1811		if (c >= '0' && c <= '9') {
1812			if (scp->term.num_param < MAX_ESC_PAR) {
1813				if (scp->term.last_param != scp->term.num_param) {
1814					scp->term.last_param = scp->term.num_param;
1815					scp->term.param[scp->term.num_param] = 0;
1816				}
1817				else
1818					scp->term.param[scp->term.num_param] *= 10;
1819				scp->term.param[scp->term.num_param] += c - '0';
1820				return;
1821			}
1822		}
1823		scp->term.num_param = scp->term.last_param + 1;
1824		switch (c) {
1825
1826		case ';':
1827			if (scp->term.num_param < MAX_ESC_PAR)
1828				return;
1829			break;
1830
1831		case 'A':	/* set display border color */
1832			if (scp->term.num_param == 1)
1833				scp->border=scp->term.param[0] & 0xff;
1834				if (scp == cur_console)
1835					set_border(scp->border);
1836			break;
1837
1838		case 'B':	/* set bell pitch and duration */
1839			if (scp->term.num_param == 2) {
1840				scp->bell_pitch = scp->term.param[0];
1841				scp->bell_duration = scp->term.param[1]*10;
1842			}
1843			break;
1844
1845		case 'C': 	/* set cursor shape (start & end line) */
1846			if (scp->term.num_param == 2) {
1847				scp->cursor_start = scp->term.param[0] & 0x1F;
1848				scp->cursor_end = scp->term.param[1] & 0x1F;
1849				if (scp == cur_console)
1850					cursor_shape(scp->cursor_start,
1851						     scp->cursor_end);
1852			}
1853			break;
1854
1855		case 'F':	/* set ansi foreground */
1856			if (scp->term.num_param == 1)
1857				scp->term.cur_attr = scp->term.std_attr =
1858					(scp->term.std_attr & 0xF000)
1859					| ((scp->term.param[0] & 0x0F) << 8);
1860			break;
1861
1862		case 'G': 	/* set ansi background */
1863			if (scp->term.num_param == 1)
1864				scp->term.cur_attr = scp->term.std_attr =
1865					(scp->term.std_attr & 0x0F00)
1866					| ((scp->term.param[0] & 0x0F) << 12);
1867			break;
1868
1869		case 'H':	/* set ansi reverse video foreground */
1870			if (scp->term.num_param == 1)
1871				scp->term.rev_attr =
1872					(scp->term.rev_attr & 0xF000)
1873					| ((scp->term.param[0] & 0x0F) << 8);
1874			break;
1875
1876		case 'I': 	/* set ansi reverse video background */
1877			if (scp->term.num_param == 1)
1878				scp->term.rev_attr =
1879					(scp->term.rev_attr & 0x0F00)
1880					| ((scp->term.param[0] & 0x0F) << 12);
1881			break;
1882		}
1883	}
1884	scp->term.esc = 0;
1885}
1886
1887
1888static void ansi_put(scr_stat *scp, u_char c)
1889{
1890	if (scp->status & UNKNOWN_MODE)
1891		return;
1892
1893	/* make screensaver happy */
1894	if (scp == cur_console) {
1895		scrn_time_stamp = time.tv_sec;
1896		if (scrn_blanked)
1897			SCRN_SAVER(0);
1898	}
1899	in_putc++;
1900	if (scp->term.esc)
1901		scan_esc(scp, c);
1902	else switch(c) {
1903	case 0x1B:	/* start escape sequence */
1904		scp->term.esc = 1;
1905		scp->term.num_param = 0;
1906		break;
1907	case 0x07:
1908		if (scp == cur_console)
1909		 	sysbeep(scp->bell_pitch, scp->bell_duration);
1910		break;
1911	case '\t':	/* non-destructive tab */
1912		scp->crtat += (8 - scp->xpos % 8);
1913		scp->xpos += (8 - scp->xpos % 8);
1914		break;
1915	case '\b':      /* non-destructive backspace */
1916		if (scp->crtat > scp->crt_base) {
1917			scp->crtat--;
1918			if (scp->xpos > 0)
1919				scp->xpos--;
1920			else {
1921				scp->xpos += scp->xsize - 1;
1922				scp->ypos--;
1923			}
1924		}
1925		break;
1926	case '\r':	/* return to pos 0 */
1927		move_crsr(scp, 0, scp->ypos);
1928		break;
1929	case '\n':	/* newline, same pos */
1930		scp->crtat += scp->xsize;
1931		scp->ypos++;
1932		break;
1933	case '\f':	/* form feed, clears screen */
1934		clear_screen(scp);
1935		break;
1936	default:
1937		/* Print only printables */
1938		*scp->crtat = (scp->term.cur_attr | scr_map[c]);
1939		scp->crtat++;
1940		if (++scp->xpos >= scp->xsize) {
1941			scp->xpos = 0;
1942			scp->ypos++;
1943		}
1944		break;
1945	}
1946	if (scp->crtat >= scp->crt_base + scp->ysize * scp->xsize) {
1947		bcopy(scp->crt_base + scp->xsize, scp->crt_base,
1948			scp->xsize * (scp->ysize - 1) * sizeof(u_short));
1949		fillw(scp->term.cur_attr | scr_map[0x20],
1950			scp->crt_base + scp->xsize * (scp->ysize - 1),
1951			scp->xsize);
1952		scp->crtat -= scp->xsize;
1953		scp->ypos--;
1954	}
1955	in_putc--;
1956	if (delayed_next_scr)
1957		switch_scr(delayed_next_scr - 1);
1958}
1959
1960static void scinit(void)
1961{
1962	u_short volatile *cp = Crtat + (CGA_BUF-MONO_BUF)/sizeof(u_short), was;
1963	unsigned cursorat;
1964	int i;
1965
1966	/*
1967	 * catch that once in a blue moon occurence when scinit is called
1968	 * TWICE, adding the CGA_BUF offset again -> poooff
1969	 */
1970	if (crtat != 0)
1971		return;
1972	/*
1973	 * Crtat initialized to point to MONO buffer, if not present change
1974	 * to CGA_BUF offset. ONLY ADD the difference since locore.s adds
1975	 * in the remapped offset at the "right" time
1976	 */
1977	was = *cp;
1978	*cp = (u_short) 0xA55A;
1979	if (*cp != 0xA55A) {
1980		crtc_addr = MONO_BASE;
1981	} else {
1982		*cp = was;
1983		crtc_addr = COLOR_BASE;
1984		Crtat = Crtat + (CGA_BUF-MONO_BUF)/sizeof(u_short);
1985	}
1986
1987	/* Extract cursor location */
1988	outb(crtc_addr,14);
1989	cursorat = inb(crtc_addr+1)<<8 ;
1990	outb(crtc_addr,15);
1991	cursorat |= inb(crtc_addr+1);
1992	crtat = Crtat + cursorat;
1993
1994	/* is this a VGA or higher ? */
1995	outb(crtc_addr, 7);
1996	if (inb(crtc_addr) == 7)
1997		crtc_vga = 1;
1998
1999	current_default = &user_default;
2000	console[0].crtat = crtat;
2001	console[0].crt_base = Crtat;
2002	console[0].term.esc = 0;
2003	console[0].term.std_attr = current_default->std_attr;
2004	console[0].term.rev_attr = current_default->rev_attr;
2005	console[0].term.cur_attr = current_default->std_attr;
2006	console[0].xpos = cursorat % COL;
2007	console[0].ypos = cursorat / COL;
2008	console[0].border = BG_BLACK;;
2009	console[0].xsize = COL;
2010	console[0].ysize = ROW;
2011	console[0].status = 0;
2012	console[0].pid = 0;
2013	console[0].proc = NULL;
2014	console[0].smode.mode = VT_AUTO;
2015	console[0].bell_pitch = BELL_PITCH;
2016	console[0].bell_duration = BELL_DURATION;
2017	kernel_console.esc = 0;
2018	kernel_console.std_attr = kernel_default.std_attr;
2019	kernel_console.rev_attr = kernel_default.rev_attr;
2020	kernel_console.cur_attr = kernel_default.std_attr;
2021	/* initialize mapscrn array to a one to one map */
2022	for (i=0; i<sizeof(scr_map); i++)
2023		scr_map[i] = i;
2024	clear_screen(&console[0]);
2025}
2026
2027
2028static void scput(u_char c)
2029{
2030	scr_stat *scp = &console[0];
2031	term_stat save;
2032
2033	if (crtat == 0)
2034		scinit();
2035	save = scp->term;
2036	scp->term = kernel_console;
2037	current_default = &kernel_default;
2038	ansi_put(scp, c);
2039	kernel_console = scp->term;
2040	current_default = &user_default;
2041	scp->term = save;
2042}
2043
2044
2045static u_char *get_fstr(u_int c, u_int *len)
2046{
2047	u_int i;
2048
2049	if (!(c & FKEY))
2050		return(NULL);
2051	i = (c & 0xFF) - F_FN;
2052	if (i > n_fkey_tab)
2053		return(NULL);
2054	*len = fkey_tab[i].len;
2055	return(fkey_tab[i].str);
2056}
2057
2058
2059static void update_leds(int which)
2060{
2061  	static u_char xlate_leds[8] = { 0, 4, 2, 6, 1, 5, 3, 7 };
2062
2063	kbd_cmd2(KB_SETLEDS, xlate_leds[which & LED_MASK]);
2064}
2065
2066
2067/*
2068 * scgetc(noblock) : get a character from the keyboard.
2069 * If noblock = 0 wait until a key is gotten.  Otherwise return NOKEY.
2070 */
2071u_int scgetc(int noblock)
2072{
2073	u_char val, code, release;
2074	u_int state, action;
2075	struct key_t *key;
2076	static u_char esc_flag = 0, compose = 0;
2077	static u_int chr = 0;
2078
2079next_code:
2080	kbd_wait();
2081	/* First see if there is something in the keyboard port */
2082	if (inb(KB_STAT) & KB_BUF_FULL)
2083		val = inb(KB_DATA);
2084	else if (noblock)
2085		return(NOKEY);
2086	else
2087		goto next_code;
2088
2089	if (cur_console->status & KBD_RAW_MODE)
2090		return val;
2091
2092	code = val & 0x7F;
2093	release = val & 0x80;
2094
2095	switch (esc_flag) {
2096	case 0x00:		/* normal scancode */
2097		switch(code) {
2098		case 0x38:	/* left alt  (compose key) */
2099			if (release && compose) {
2100				compose = 0;
2101				if (chr > 255) {
2102					sysbeep(BELL_PITCH, BELL_DURATION);
2103					chr = 0;
2104				}
2105			}
2106			else {
2107				if (!compose) {
2108					compose = 1;
2109					chr = 0;
2110				}
2111			}
2112			break;
2113		case 0x60:
2114		case 0x61:
2115			esc_flag = code;
2116			goto next_code;
2117		}
2118		break;
2119	case 0x60:		/* 0xE0 prefix */
2120		esc_flag = 0;
2121		switch (code) {
2122		case 0x1c:	/* right enter key */
2123			code = 0x59;
2124			break;
2125		case 0x1d:	/* right ctrl key */
2126			code = 0x5a;
2127			break;
2128		case 0x35:	/* keypad divide key */
2129			code = 0x5b;
2130			break;
2131		case 0x37:	/* print scrn key */
2132			code = 0x5c;
2133			break;
2134		case 0x38:	/* right alt key (alt gr) */
2135			code = 0x5d;
2136			break;
2137		case 0x47:	/* grey home key */
2138			code = 0x5e;
2139			break;
2140		case 0x48:	/* grey up arrow key */
2141			code = 0x5f;
2142			break;
2143		case 0x49:	/* grey page up key */
2144			code = 0x60;
2145			break;
2146		case 0x4b:	/* grey left arrow key */
2147			code = 0x61;
2148			break;
2149		case 0x4d:	/* grey right arrow key */
2150			code = 0x62;
2151			break;
2152		case 0x4f:	/* grey end key */
2153			code = 0x63;
2154			break;
2155		case 0x50:	/* grey down arrow key */
2156			code = 0x64;
2157			break;
2158		case 0x51:	/* grey page down key */
2159			code = 0x65;
2160			break;
2161		case 0x52:	/* grey insert key */
2162			code = 0x66;
2163			break;
2164		case 0x53:	/* grey delete key */
2165			code = 0x67;
2166			break;
2167		default:	/* ignore everything else */
2168			goto next_code;
2169		}
2170		break;
2171	case 0x61:		/* 0xE1 prefix */
2172		esc_flag = 0;
2173		if (code == 0x1D)
2174			esc_flag = 0x1D;
2175		goto next_code;
2176		/* NOT REACHED */
2177	case 0x1D:		/* pause / break */
2178		esc_flag = 0;
2179		if (code != 0x45)
2180			goto next_code;
2181		code = 0x68;
2182		break;
2183	}
2184
2185	if (compose) {
2186		switch (code) {
2187		case 0x47:
2188		case 0x48:				/* keypad 7,8,9 */
2189		case 0x49:
2190			if (!release)
2191				chr = (code - 0x40) + chr*10;
2192			goto next_code;
2193		case 0x4b:
2194		case 0x4c:				/* keypad 4,5,6 */
2195		case 0x4d:
2196			if (!release)
2197				chr = (code - 0x47) + chr*10;
2198			goto next_code;
2199		case 0x4f:
2200		case 0x50:				/* keypad 1,2,3 */
2201		case 0x51:
2202			if (!release)
2203				chr = (code - 0x4e) + chr*10;
2204			goto next_code;
2205		case 0x52:				/* keypad 0 */
2206			if (!release)
2207				chr *= 10;
2208			goto next_code;
2209		case 0x38:				/* left alt key */
2210			break;
2211		default:
2212			if (chr) {
2213				compose = chr = 0;
2214				sysbeep(BELL_PITCH, BELL_DURATION);
2215				goto next_code;
2216			}
2217			break;
2218		}
2219	}
2220
2221	state = (shfts ? 1 : 0 ) | (2 * (ctls ? 1 : 0)) | (4 * (alts ? 1 : 0));
2222	if ((!agrs && (cur_console->status & ALKED))
2223	    || (agrs && !(cur_console->status & ALKED)))
2224		code += ALTGR_OFFSET;
2225	key = &key_map.key[code];
2226	if ( ((key->flgs & FLAG_LOCK_C) && (cur_console->status & CLKED))
2227	     || ((key->flgs & FLAG_LOCK_N) && (cur_console->status & NLKED)) )
2228		state ^= 1;
2229
2230	/* Check for make/break */
2231	action = key->map[state];
2232	if (release) { 		/* key released */
2233		if (key->spcl & 0x80) {
2234			switch (action) {
2235			case LSH:
2236				shfts &= ~1;
2237				break;
2238			case RSH:
2239				shfts &= ~2;
2240				break;
2241			case LCTR:
2242				ctls &= ~1;
2243				break;
2244			case RCTR:
2245				ctls &= ~2;
2246				break;
2247			case LALT:
2248				alts &= ~1;
2249				break;
2250			case RALT:
2251				alts &= ~2;
2252				break;
2253			case NLK:
2254				nlkcnt = 0;
2255				break;
2256			case CLK:
2257				clkcnt = 0;
2258				break;
2259			case SLK:
2260				slkcnt = 0;
2261				break;
2262			case ASH:
2263				agrs = 0;
2264				break;
2265			case ALK:
2266				alkcnt = 0;
2267				break;
2268			case META:
2269				metas = 0;
2270				break;
2271			}
2272		}
2273		if (chr && !compose) {
2274			action = chr;
2275			chr = 0;
2276			return(action);
2277		}
2278	} else {
2279		/* key pressed */
2280		if (key->spcl & (0x80>>state)) {
2281			switch (action) {
2282			/* LOCKING KEYS */
2283			case NLK:
2284				if (!nlkcnt) {
2285					nlkcnt++;
2286					if (cur_console->status & NLKED)
2287						cur_console->status &= ~NLKED;
2288					else
2289						cur_console->status |= NLKED;
2290					update_leds(cur_console->status & LED_MASK);
2291				}
2292				break;
2293			case CLK:
2294				if (!clkcnt) {
2295					clkcnt++;
2296					if (cur_console->status & CLKED)
2297						cur_console->status &= ~CLKED;
2298					else
2299						cur_console->status |= CLKED;
2300					update_leds(cur_console->status & LED_MASK);
2301				}
2302				break;
2303			case SLK:
2304				if (!slkcnt) {
2305					slkcnt++;
2306					if (cur_console->status & SLKED) {
2307						cur_console->status &= ~SLKED;
2308						pcstart(VIRTUAL_TTY(get_scr_num()));
2309					}
2310					else
2311						cur_console->status |= SLKED;
2312					update_leds(cur_console->status & LED_MASK);
2313				}
2314				break;
2315 			case ALK:
2316				if (!alkcnt) {
2317					alkcnt++;
2318 					if (cur_console->status & ALKED)
2319 						cur_console->status &= ~ALKED;
2320 					else
2321 						cur_console->status |= ALKED;
2322				}
2323  				break;
2324
2325			/* NON-LOCKING KEYS */
2326			case NOP:
2327				break;
2328			case RBT:
2329#if defined(__FreeBSD__)
2330				shutdown_nice();
2331#else
2332				cpu_reset();
2333#endif
2334				break;
2335			case DBG:
2336#if DDB > 0			/* try to switch to console 0 */
2337				if (cur_console->smode.mode == VT_AUTO &&
2338		    		    console[0].smode.mode == VT_AUTO)
2339					switch_scr(0);
2340				Debugger("manual escape to debugger");
2341				return(NOKEY);
2342#else
2343				printf("No debugger in kernel\n");
2344#endif
2345				break;
2346			case LSH:
2347				shfts |= 1;
2348				break;
2349			case RSH:
2350				shfts |= 2;
2351				break;
2352			case LCTR:
2353				ctls |= 1;
2354				break;
2355			case RCTR:
2356				ctls |= 2;
2357				break;
2358			case LALT:
2359				alts |= 1;
2360				break;
2361			case RALT:
2362				alts |= 2;
2363				break;
2364			case ASH:
2365				agrs = 1;
2366				break;
2367			case META:
2368				metas = 1;
2369				break;
2370			case NEXT:
2371				switch_scr((get_scr_num()+1)%NCONS);
2372				break;
2373			default:
2374				if (action >= F_SCR && action <= L_SCR) {
2375					switch_scr(action - F_SCR);
2376					break;
2377				}
2378				if (action >= F_FN && action <= L_FN)
2379					action |= FKEY;
2380				return(action);
2381			}
2382		}
2383		else {
2384			if (metas)
2385				action |= MKEY;
2386 			return(action);
2387		}
2388	}
2389	goto next_code;
2390}
2391
2392
2393int getchar(void)
2394{
2395	u_char thechar;
2396	int s;
2397
2398	polling = 1;
2399	s = splhigh();
2400	scput('>');
2401	thechar = (u_char) scgetc(0);
2402	polling = 0;
2403	splx(s);
2404	switch (thechar) {
2405	default:
2406		if (thechar >= scr_map[0x20])
2407			scput(thechar);
2408		return(thechar);
2409	case cr:
2410	case lf:
2411		scput(cr); scput(lf);
2412		return(lf);
2413	case bs:
2414	case del:
2415		scput(bs); scput(scr_map[0x20]); scput(bs);
2416		return(thechar);
2417	case cntld:
2418		scput('^'); scput('D'); scput('\r'); scput('\n');
2419		return(0);
2420	}
2421}
2422
2423
2424u_int sgetc(int noblock)
2425{
2426	return (scgetc(noblock) & 0xff);
2427}
2428
2429int pcmmap(dev_t dev, int offset, int nprot)
2430{
2431	if (offset > 0x20000)
2432		return EINVAL;
2433	return i386_btop((VIDEOMEM + offset));
2434}
2435
2436
2437static void kbd_wait(void)
2438{
2439	int i;
2440
2441	for (i=0; i<1000; i++) {        /* up to 10 msec */
2442		if ((inb(KB_STAT) & KB_READY) == 0)
2443			break;
2444		DELAY (10);
2445	}
2446}
2447
2448
2449static void kbd_cmd(u_char command)
2450{
2451	kbd_wait();
2452	outb(KB_DATA, command);
2453}
2454
2455
2456static void kbd_cmd2(u_char command, u_char arg)
2457{
2458	int r, s = spltty();
2459	do {
2460		kbd_cmd(command);
2461		r = kbd_reply();
2462		if (r == KB_ACK) {
2463			kbd_cmd(arg & 0x7f);
2464			r = kbd_reply();
2465		}
2466	} while (r != KB_ACK);
2467	splx(s);
2468}
2469
2470
2471static int kbd_reply()
2472{
2473	int i;
2474
2475	kbd_wait();
2476	for (i=0; i<60000; i++) {       /* at least 300 msec, 600 msec enough */
2477		if (inb(KB_STAT) & KB_BUF_FULL)
2478			return ((u_char) inb(KB_DATA));
2479		DELAY (10);
2480	}
2481	return(-1);
2482}
2483
2484
2485static void set_mode(scr_stat *scp)
2486{
2487	u_char byte;
2488	int s;
2489
2490	if (scp != cur_console)
2491		return;
2492
2493	/* (re)activate cursor */
2494	untimeout((timeout_t)cursor_pos, 0);
2495	cursor_pos(1);
2496
2497	/* change cursor type if set */
2498	if (scp->cursor_start != -1 && scp->cursor_end != -1)
2499		cursor_shape(scp->cursor_start, scp->cursor_end);
2500
2501	/* mode change only on VGA's */
2502	if (!crtc_vga)
2503		return;
2504
2505	/* setup video hardware for the given mode */
2506	s = splhigh();
2507	switch(scp->mode) {
2508	case TEXT80x25:
2509		outb(crtc_addr, 9); byte = inb(crtc_addr+1);
2510		outb(crtc_addr, 9); outb(crtc_addr+1, byte | 0x0F);
2511    		outb(TSIDX, 0x03); outb(TSREG, 0x00);	/* select font 0 */
2512		break;
2513	case TEXT80x50:
2514		outb(crtc_addr, 9); byte = inb(crtc_addr+1);
2515		outb(crtc_addr, 9); outb(crtc_addr+1, (byte & 0xF0) | 0x07);
2516    		outb(TSIDX, 0x03); outb(TSREG, 0x05);	/* select font 1 */
2517		break;
2518	default:
2519		break;
2520	}
2521	splx(s);
2522
2523	/* set border color for this (virtual) console */
2524	set_border(scp->border);
2525	return;
2526}
2527
2528
2529static void set_border(int color)
2530{
2531	inb(crtc_addr+6); 				/* reset flip-flop */
2532	outb(ATC, 0x11); outb(ATC, color);
2533 	inb(crtc_addr+6); 				/* reset flip-flop */
2534 	outb(ATC, 0x20);			/* enable Palette */
2535}
2536
2537static void load_font(int segment, int size, char* font)
2538{
2539  	int ch, line, s;
2540	u_char val;
2541
2542 	outb(TSIDX, 0x01); val = inb(TSREG); 		/* blank screen */
2543	outb(TSIDX, 0x01); outb(TSREG, val | 0x20);
2544
2545	/* setup vga for loading fonts (graphics plane mode) */
2546	s = splhigh();
2547	inb(crtc_addr+6);				/* reset flip/flop */
2548	outb(ATC, 0x30); outb(ATC, 0x01);
2549	outb(TSIDX, 0x02); outb(TSREG, 0x04);
2550	outb(TSIDX, 0x04); outb(TSREG, 0x06);
2551	outb(GDCIDX, 0x04); outb(GDCREG, 0x02);
2552	outb(GDCIDX, 0x05); outb(GDCREG, 0x00);
2553	outb(GDCIDX, 0x06); outb(GDCREG, 0x05);		/* addr = a0000, 64kb */
2554	splx(s);
2555    	for (ch=0; ch < 256; ch++)
2556		for (line=0; line < size; line++)
2557			*((char *)atdevbase+(segment*0x4000)+(ch*32)+line) =
2558				font[(ch*size)+line];
2559	/* setup vga for text mode again */
2560	s = splhigh();
2561	inb(crtc_addr+6);				/* reset flip/flop */
2562	outb(ATC, 0x30); outb(ATC, 0x0C);
2563	outb(TSIDX, 0x02); outb(TSREG, 0x03);
2564	outb(TSIDX, 0x04); outb(TSREG, 0x02);
2565	outb(GDCIDX, 0x04); outb(GDCREG, 0x00);
2566	outb(GDCIDX, 0x05); outb(GDCREG, 0x10);
2567	if (crtc_addr == MONO_BASE) {
2568		outb(GDCIDX, 0x06); outb(GDCREG, 0x0A);	/* addr = b0000, 32kb */
2569	}
2570	else {
2571		outb(GDCIDX, 0x06); outb(GDCREG, 0x0E);	/* addr = b8000, 32kb */
2572	}
2573	splx(s);
2574 	outb(TSIDX, 0x01); val = inb(TSREG); 		/* unblank screen */
2575	outb(TSIDX, 0x01); outb(TSREG, val & 0xDF);
2576}
2577
2578
2579static void load_palette(void)
2580{
2581	int i;
2582
2583  	outb(PIXMASK, 0xFF);			/* no pixelmask */
2584  	outb(PALWADR, 0x00);
2585  	for (i=0x00; i<0x300; i++)
2586    		 outb(PALDATA, palette[i]);
2587	inb(crtc_addr+6);			/* reset flip/flop */
2588	outb(ATC, 0x20);			/* enable palette */
2589}
2590
2591static void save_palette(void)
2592{
2593	int i;
2594
2595  	outb(PALRADR, 0x00);
2596  	for (i=0x00; i<0x300; i++)
2597    		palette[i] = inb(PALDATA);
2598	inb(crtc_addr+6);			/* reset flip/flop */
2599}
2600
2601
2602static void change_winsize(struct tty *tp, int x, int y)
2603{
2604	if (tp->t_winsize.ws_col != x || tp->t_winsize.ws_row != y) {
2605		tp->t_winsize.ws_col = x;
2606		tp->t_winsize.ws_row = y;
2607		pgsignal(tp->t_pgrp, SIGWINCH, 1);
2608	}
2609}
2610
2611#endif /* NSC */
2612