642 lines
10 KiB
C
642 lines
10 KiB
C
/*++
|
|
|
|
Copyright (c) 1990 Microsoft Corporation
|
|
|
|
Module Name:
|
|
|
|
display.c
|
|
|
|
Author:
|
|
|
|
Thomas Parslow (tomp) Mar-01-90
|
|
|
|
Abstract:
|
|
|
|
Video support routines.
|
|
|
|
The SU module only need to be able to write to the video display
|
|
in order to report errors, traps, etc.
|
|
|
|
The routines in this file all write to a video buffer assumed to be
|
|
at realmode address b800:0000, and 4k bytes in length. The segment
|
|
portion of the far pointers used to access the video buffer are stamped
|
|
with a protmode selector value when we switch to protect mode. This is
|
|
done in the routine "ProtMode" in "misc386.asm".
|
|
|
|
|
|
--*/
|
|
|
|
#include "su.h"
|
|
|
|
|
|
#define ZLEN_BYTE(x) (x < 0x10)
|
|
#define ZLEN_SHORT(x) ((x < 0x10) + (x < 0x100) + (x < 0x1000))
|
|
#define ZLEN_LONG(x) ((x < 0x10) + (x < 0x100) + (x < 0x1000) + (x < 0x10000) + (x < 0x100000)+(x < 0x1000000)+(x < 0x10000000))
|
|
|
|
#ifdef DEBUG1
|
|
#define ROWS 43
|
|
#else
|
|
#define ROWS 25
|
|
#endif
|
|
#define COLUMNS 80
|
|
#define SCREEN_WIDTH COLUMNS
|
|
#define SCREEN_SIZE ROWS * COLUMNS
|
|
#define NORMAL_ATTRIB 0x07
|
|
#define REVERSE_ATTRIB 0x70
|
|
#define SCREEN_START 0xb8000000
|
|
|
|
#define VIDEO_BIOS 0x10
|
|
#define LINES_400_CONFIGURATION 0x1202
|
|
#define SELECT_SCAN_LINE 0x301
|
|
#define SET_80X25_16_COLOR_MODE 0x3
|
|
#define LOAD_8X8_CHARACTER_SET 0x1112
|
|
|
|
//
|
|
// Internal routines
|
|
//
|
|
|
|
static
|
|
VOID
|
|
tab(
|
|
VOID
|
|
);
|
|
|
|
static
|
|
VOID
|
|
newline(
|
|
VOID
|
|
);
|
|
|
|
static
|
|
VOID
|
|
putzeros(
|
|
USHORT,
|
|
USHORT
|
|
);
|
|
|
|
|
|
USHORT
|
|
Redirect = 0;
|
|
|
|
VOID
|
|
InitializeVideoSubSystem(
|
|
VOID
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Initializes the video mode to 80x50 alphanumeric mode with 400 lines
|
|
vertical resolution.
|
|
|
|
Arguments:
|
|
|
|
None
|
|
|
|
Returns:
|
|
|
|
Nothing
|
|
|
|
|
|
--*/
|
|
|
|
{
|
|
BIOSREGS ps;
|
|
UCHAR _far *BiosArea;
|
|
|
|
//
|
|
// Set 40:10 to indicate color is the default display
|
|
// *(40:10) &= ~0x30;
|
|
// *(40:10) |= 0x20;
|
|
//
|
|
// Fixes obscure situation where both monochrome and VGA adapters
|
|
// are installed and the monochrome is the default display.
|
|
//
|
|
BiosArea = (UCHAR _far *)(0x410L);
|
|
|
|
*BiosArea &= ~0x30;
|
|
*BiosArea |= 0x20;
|
|
|
|
//
|
|
// Establish 80x25 alphanumeric mode with 400-lines vertical resolution
|
|
//
|
|
ps.fn = VIDEO_BIOS;
|
|
ps.ax = LINES_400_CONFIGURATION;
|
|
ps.bx = SELECT_SCAN_LINE;
|
|
biosint(&ps);
|
|
|
|
ps.fn = VIDEO_BIOS;
|
|
ps.ax = SET_80X25_16_COLOR_MODE;
|
|
biosint(&ps);
|
|
|
|
DBG1(
|
|
ps.ax = LOAD_8X8_CHARACTER_SET;
|
|
ps.bx = 0;
|
|
biosint(&ps);
|
|
)
|
|
|
|
//
|
|
// HACK-O-RAMA - Make some random video BIOS calls here to make sure the
|
|
// BIOS is initialized and warmed up and ready to go. Otherwise,
|
|
// some Number 9 S3 cards don't quite work right later in the game.
|
|
// John Vert (jvert) 9-Jun-1993
|
|
//
|
|
|
|
//
|
|
// set cursor position to 0,0
|
|
//
|
|
ps.fn = VIDEO_BIOS;
|
|
ps.ax = 0x2000;
|
|
ps.bx = 0;
|
|
ps.dx = 0;
|
|
biosint(&ps);
|
|
|
|
//
|
|
// write character (' ' in this case)
|
|
//
|
|
ps.fn = VIDEO_BIOS;
|
|
ps.ax = 0x0a00 | (USHORT)' ';
|
|
ps.bx = 0;
|
|
ps.cx = 1;
|
|
biosint(&ps);
|
|
|
|
clrscrn();
|
|
return ;
|
|
}
|
|
|
|
|
|
//
|
|
// Used by all BlPrint subordinate routines for padding computations.
|
|
//
|
|
|
|
CHAR sc=0;
|
|
ULONG fw=0;
|
|
|
|
|
|
|
|
VOID
|
|
BlPrint(
|
|
PCHAR cp,
|
|
...
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Standard printf function with a subset of formating features supported.
|
|
|
|
Currently handles
|
|
|
|
%d, %ld - signed short, signed long
|
|
%u, %lu - unsigned short, unsigned long
|
|
%c, %s - character, string
|
|
%x, %lx - unsigned print in hex, unsigned long print in hex
|
|
%b - byte in hex
|
|
|
|
Does not do:
|
|
|
|
- field width specification
|
|
- floating point.
|
|
|
|
Arguments:
|
|
|
|
cp - pointer to the format string, text string.
|
|
|
|
|
|
Returns:
|
|
|
|
Nothing
|
|
|
|
|
|
--*/
|
|
|
|
{
|
|
UCHAR uc;
|
|
USHORT b,c,w,len;
|
|
PUCHAR ap;
|
|
ULONG l;
|
|
|
|
//
|
|
// Cast a pointer to the first word on the stack
|
|
//
|
|
|
|
ap = (PUCHAR)&cp + sizeof(PCHAR);
|
|
sc = ' '; // default padding char is space
|
|
|
|
//
|
|
// Process the arguments using the descriptor string
|
|
//
|
|
|
|
|
|
while (b = *cp++)
|
|
{
|
|
if (b == '%')
|
|
{
|
|
c = *cp++;
|
|
|
|
switch (c)
|
|
{
|
|
case 'd':
|
|
puti((long)*((int *)ap));
|
|
ap += sizeof(int);
|
|
break;
|
|
|
|
case 's':
|
|
puts(*((PCHAR *)ap));
|
|
ap += sizeof (char *);
|
|
break;
|
|
|
|
case 'c':
|
|
putc(*((char *)ap));
|
|
ap += sizeof(int);
|
|
break;
|
|
|
|
case 'x':
|
|
w = *((USHORT *)ap);
|
|
len = ZLEN_SHORT(w);
|
|
while(len--) putc('0');
|
|
putx((ULONG)*((USHORT *)ap));
|
|
ap += sizeof(int);
|
|
break;
|
|
|
|
case 'u':
|
|
putu((ULONG)*((USHORT *)ap));
|
|
ap += sizeof(int);
|
|
break;
|
|
|
|
case 'b':
|
|
uc = *((UCHAR *)ap);
|
|
len = ZLEN_BYTE(uc);
|
|
while(len--) putc('0');
|
|
putx((ULONG)uc);
|
|
ap += sizeof(int);
|
|
break;
|
|
|
|
case 'l':
|
|
c = *cp++;
|
|
|
|
switch(c) {
|
|
|
|
case 'u':
|
|
putu(*((ULONG *)ap));
|
|
ap += sizeof(long);
|
|
break;
|
|
|
|
case 'x':
|
|
l = *((ULONG *)ap);
|
|
len = ZLEN_LONG(l);
|
|
while(len--) putc('0');
|
|
putx(*((ULONG *)ap));
|
|
ap += sizeof(long);
|
|
break;
|
|
|
|
case 'd':
|
|
puti(*((ULONG *)ap));
|
|
ap += sizeof(long);
|
|
break;
|
|
|
|
}
|
|
break;
|
|
|
|
default :
|
|
putc((char)b);
|
|
putc((char)c);
|
|
}
|
|
}
|
|
else
|
|
putc((char)b);
|
|
}
|
|
|
|
}
|
|
|
|
FPUCHAR vp = (FPUCHAR)SCREEN_START;
|
|
FPUCHAR ScreenStart = (FPUCHAR)SCREEN_START;
|
|
|
|
static int lcnt = 0;
|
|
static int row = 0;
|
|
|
|
|
|
VOID puts(
|
|
PCHAR cp
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Writes a string on the display at the current cursor position
|
|
|
|
Arguments:
|
|
|
|
cp - pointer to ASCIIZ string to display.
|
|
|
|
|
|
Returns:
|
|
|
|
Nothing
|
|
|
|
|
|
|
|
--*/
|
|
|
|
|
|
{
|
|
char c;
|
|
|
|
while(c = *cp++)
|
|
putc(c);
|
|
}
|
|
|
|
|
|
//
|
|
// Write a hex short to display
|
|
//
|
|
|
|
|
|
VOID putx(
|
|
ULONG x
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Writes hex long to the display at the current cursor position.
|
|
|
|
Arguments:
|
|
|
|
x - ulong to write.
|
|
|
|
Returns:
|
|
|
|
Nothing
|
|
|
|
|
|
--*/
|
|
|
|
{
|
|
ULONG j;
|
|
|
|
if (x/16)
|
|
putx(x/16);
|
|
|
|
if((j=x%16) > 9) {
|
|
putc((char)(j+'A'- 10));
|
|
} else {
|
|
putc((char)(j+'0'));
|
|
}
|
|
}
|
|
|
|
|
|
VOID puti(
|
|
LONG i
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Writes a long integer on the display at the current cursor position.
|
|
|
|
Arguments:
|
|
|
|
i - the integer to write to the display.
|
|
|
|
Returns:
|
|
|
|
Nothing
|
|
|
|
|
|
--*/
|
|
|
|
|
|
{
|
|
if (i<0)
|
|
{
|
|
i = -i;
|
|
putc((char)'-');
|
|
}
|
|
|
|
if (i/10)
|
|
puti(i/10);
|
|
|
|
putc((char)((i%10)+'0'));
|
|
}
|
|
|
|
|
|
|
|
VOID putu(
|
|
ULONG u
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Write an unsigned long to display
|
|
|
|
Arguments:
|
|
|
|
u - unsigned
|
|
|
|
|
|
--*/
|
|
|
|
{
|
|
if (u/10)
|
|
putu(u/10);
|
|
|
|
putc((char)((u%10)+'0'));
|
|
|
|
}
|
|
|
|
|
|
VOID putc(
|
|
CHAR c
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Writes a character on the display at the current position.
|
|
|
|
Arguments:
|
|
|
|
c - character to write
|
|
|
|
|
|
Returns:
|
|
|
|
Nothing
|
|
|
|
|
|
--*/
|
|
|
|
{
|
|
switch (c)
|
|
{
|
|
case '\n':
|
|
newline();
|
|
break;
|
|
|
|
case '\t':
|
|
tab();
|
|
break;
|
|
|
|
default :
|
|
if (FP_OFF(vp) >= (SCREEN_SIZE * 2)) {
|
|
vp = (FPUCHAR)((ScreenStart + (2*SCREEN_WIDTH*(ROWS-1))));
|
|
scroll();
|
|
}
|
|
*vp = c;
|
|
vp += 2;
|
|
++lcnt;
|
|
}
|
|
}
|
|
|
|
|
|
VOID newline(
|
|
VOID
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Moves the cursor to the beginning of the next line. If the bottom
|
|
of the display has been reached, the screen is scrolled one line up.
|
|
|
|
Arguments:
|
|
|
|
None
|
|
|
|
|
|
Returns:
|
|
|
|
Nothing
|
|
|
|
|
|
--*/
|
|
|
|
{
|
|
vp += (SCREEN_WIDTH - lcnt)<<1;
|
|
|
|
if (++row > ROWS-1) {
|
|
|
|
vp = (FPUCHAR)((ScreenStart + (2*SCREEN_WIDTH*(ROWS-1))));
|
|
scroll();
|
|
|
|
}
|
|
|
|
lcnt = 0;
|
|
|
|
}
|
|
|
|
|
|
VOID scroll(
|
|
VOID
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Scrolls the display UP one line.
|
|
|
|
Arguments:
|
|
|
|
None
|
|
|
|
Returns:
|
|
|
|
Nothing
|
|
|
|
Notes:
|
|
|
|
Currently we scroll the display by reading and writing directly from
|
|
and to the video display buffer. We optionally switch to real mode
|
|
and to int 10s
|
|
|
|
--*/
|
|
|
|
{
|
|
USHORT i,j;
|
|
USHORT far *p1 = (USHORT far *)ScreenStart;
|
|
USHORT far *p2 = (USHORT far *)(ScreenStart + 2*SCREEN_WIDTH) ;
|
|
|
|
for (i=0; i < ROWS - 1; i++)
|
|
for (j=0; j < SCREEN_WIDTH; j++)
|
|
*p1++ = *p2++;
|
|
|
|
for (i=0; i < SCREEN_WIDTH; i++)
|
|
*p1++ = REVERSE_ATTRIB*256 + ' ';
|
|
|
|
}
|
|
|
|
|
|
static
|
|
VOID tab(
|
|
VOID
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
|
|
Computes the next tab stop and moves the cursor to that location.
|
|
|
|
|
|
Arguments:
|
|
|
|
|
|
None
|
|
|
|
|
|
Returns:
|
|
|
|
Nothing
|
|
|
|
--*/
|
|
|
|
{
|
|
int inc;
|
|
|
|
inc = 8 - (lcnt % 8);
|
|
vp += inc<<1;
|
|
lcnt += inc;
|
|
}
|
|
|
|
|
|
VOID clrscrn(
|
|
VOID
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Clears the video display by writing blanks with the current
|
|
video attribute over the entire display.
|
|
|
|
|
|
Arguments:
|
|
|
|
None
|
|
|
|
Returns:
|
|
|
|
Nothing
|
|
|
|
|
|
--*/
|
|
|
|
{
|
|
int i,a;
|
|
unsigned far *vwp = (unsigned far *)ScreenStart;
|
|
a = NORMAL_ATTRIB*256 + ' ';
|
|
|
|
for (i = SCREEN_SIZE ; i ; i--)
|
|
*vwp++ = a;
|
|
|
|
row = 0;
|
|
lcnt = 0;
|
|
vp = (FPUCHAR)ScreenStart;
|
|
|
|
}
|
|
|
|
// END OF FILE
|