703 lines
11 KiB
C
703 lines
11 KiB
C
/*++
|
||
|
||
Copyright (c) 1990 Microsoft Corporation
|
||
|
||
Module Name:
|
||
|
||
display.c
|
||
|
||
Author:
|
||
|
||
Thomas Parslow [TomP] Feb-13-1991
|
||
Reworked substantially in Tokyo 7-July-95 (tedm)
|
||
|
||
Abstract:
|
||
|
||
This file contains an interface to the screen that is independent
|
||
of the screen type actually being written to. It is layered on top
|
||
of modules pecific to vga text mode and vga graphics mode.
|
||
|
||
--*/
|
||
|
||
|
||
#include "bootx86.h"
|
||
#include "displayp.h"
|
||
|
||
|
||
#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))
|
||
|
||
//
|
||
// Current screen position.
|
||
//
|
||
USHORT TextColumn = 0;
|
||
USHORT TextRow = 0;
|
||
|
||
//
|
||
// Current text attribute
|
||
//
|
||
UCHAR TextCurrentAttribute = 0x07; // start with white on black.
|
||
|
||
//
|
||
// Internal routines
|
||
//
|
||
VOID
|
||
puti(
|
||
LONG
|
||
);
|
||
|
||
VOID
|
||
putx(
|
||
ULONG
|
||
);
|
||
|
||
VOID
|
||
putu(
|
||
ULONG
|
||
);
|
||
|
||
VOID
|
||
pTextCharOut(
|
||
IN UCHAR c
|
||
);
|
||
|
||
VOID
|
||
putwS(
|
||
PUNICODE_STRING String
|
||
);
|
||
|
||
|
||
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
|
||
|
||
Does not do:
|
||
|
||
- field width specification
|
||
- floating point.
|
||
|
||
Arguments:
|
||
|
||
cp - pointer to the format string, text string.
|
||
|
||
Returns:
|
||
|
||
Nothing
|
||
|
||
--*/
|
||
|
||
{
|
||
USHORT b,c,w,len;
|
||
PUCHAR ap;
|
||
ULONG l;
|
||
ULONG Count;
|
||
CHAR ch;
|
||
ULONG DeviceId;
|
||
|
||
|
||
if (BlConsoleOutDeviceId == 0) {
|
||
DeviceId = 1;
|
||
} else {
|
||
DeviceId = BlConsoleOutDeviceId;
|
||
}
|
||
|
||
//
|
||
// Cast a pointer to the first word on the stack
|
||
//
|
||
ap = (PUCHAR)&cp + sizeof(PCHAR);
|
||
|
||
//
|
||
// Process the arguments using the descriptor string
|
||
//
|
||
while((b = *cp++) != 0) {
|
||
if(b == '%') {
|
||
|
||
c = *cp++;
|
||
|
||
switch (c) {
|
||
|
||
case 'd':
|
||
puti((long)*((int *)ap));
|
||
ap += sizeof(int);
|
||
break;
|
||
|
||
case 's':
|
||
ArcWrite(DeviceId, *((PCHAR *)ap), strlen(*((PCHAR *)ap)), &Count);
|
||
ap += sizeof(char *);
|
||
break;
|
||
|
||
case 'c':
|
||
//
|
||
// Does not handle dbcs chars
|
||
//
|
||
ArcWrite(DeviceId, ((char *)ap), 1, &Count);
|
||
ap += sizeof(int);
|
||
break;
|
||
|
||
case 'x':
|
||
w = *((USHORT *)ap);
|
||
len = (USHORT)ZLEN_SHORT(w);
|
||
ch = '0';
|
||
while (len--) {
|
||
ArcWrite(DeviceId, &ch, 1, &Count);
|
||
}
|
||
putx((ULONG)*((USHORT *)ap));
|
||
ap += sizeof(int);
|
||
break;
|
||
|
||
case 'u':
|
||
putu((ULONG)*((USHORT *)ap));
|
||
ap += sizeof(int);
|
||
break;
|
||
|
||
case 'w':
|
||
c = *cp++;
|
||
switch (c) {
|
||
case 'S':
|
||
case 'Z':
|
||
putwS(*((PUNICODE_STRING *)ap));
|
||
ap += sizeof(PUNICODE_STRING);
|
||
break;
|
||
}
|
||
break;
|
||
|
||
case 'l':
|
||
c = *cp++;
|
||
|
||
switch(c) {
|
||
|
||
case '0':
|
||
break;
|
||
|
||
case 'u':
|
||
putu(*((ULONG *)ap));
|
||
ap += sizeof(long);
|
||
break;
|
||
|
||
case 'x':
|
||
l = *((ULONG *)ap);
|
||
len = (USHORT)ZLEN_LONG(l);
|
||
ch = '0';
|
||
while (len--) {
|
||
ArcWrite(DeviceId, &ch, 1, &Count);
|
||
}
|
||
putx(*((ULONG *)ap));
|
||
ap += sizeof(long);
|
||
break;
|
||
|
||
case 'd':
|
||
puti(*((ULONG *)ap));
|
||
ap += sizeof(long);
|
||
break;
|
||
}
|
||
break;
|
||
|
||
default :
|
||
ch = (char)b;
|
||
ArcWrite(DeviceId, &ch, 1, &Count);
|
||
ch = (char)c;
|
||
ArcWrite(DeviceId, &ch, 1, &Count);
|
||
}
|
||
} else {
|
||
|
||
if (DbcsLangId && GrIsDBCSLeadByte(*(cp - 1))) {
|
||
//
|
||
// A double-byte char.
|
||
//
|
||
ArcWrite(DeviceId, cp - 1, 2, &Count);
|
||
cp++;
|
||
} else {
|
||
ArcWrite(DeviceId, cp - 1, 1, &Count);
|
||
}
|
||
|
||
}
|
||
|
||
}
|
||
|
||
}
|
||
|
||
|
||
VOID
|
||
putwS(
|
||
PUNICODE_STRING String
|
||
)
|
||
|
||
/*++
|
||
|
||
Routine Description:
|
||
|
||
Writes unicode string to the display at the current cursor position.
|
||
|
||
Arguments:
|
||
|
||
String - pointer to unicode string to display
|
||
|
||
Returns:
|
||
|
||
Nothing
|
||
|
||
--*/
|
||
|
||
{
|
||
ULONG i;
|
||
ULONG Count;
|
||
UCHAR ch;
|
||
|
||
for(i=0; i < String->Length/sizeof(WCHAR); i++) {
|
||
ch = (UCHAR)String->Buffer[i];
|
||
ArcWrite(BlConsoleOutDeviceId, &ch, 1, &Count);
|
||
}
|
||
}
|
||
|
||
|
||
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;
|
||
ULONG Count;
|
||
UCHAR ch;
|
||
|
||
if(x/16) {
|
||
putx(x/16);
|
||
}
|
||
|
||
if((j=x%16) > 9) {
|
||
ch = (UCHAR)(j+'A'-10);
|
||
ArcWrite(BlConsoleOutDeviceId, &ch, 1, &Count);
|
||
} else {
|
||
ch = (UCHAR)(j+'0');
|
||
ArcWrite(BlConsoleOutDeviceId, &ch, 1, &Count);
|
||
}
|
||
}
|
||
|
||
|
||
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
|
||
|
||
--*/
|
||
|
||
{
|
||
ULONG Count;
|
||
UCHAR ch;
|
||
|
||
if(i<0) {
|
||
i = -i;
|
||
ch = '-';
|
||
ArcWrite(BlConsoleOutDeviceId, &ch, 1, &Count);
|
||
}
|
||
|
||
if(i/10) {
|
||
puti(i/10);
|
||
}
|
||
|
||
ch = (UCHAR)((i%10)+'0');
|
||
ArcWrite(BlConsoleOutDeviceId, &ch, 1, &Count);
|
||
}
|
||
|
||
|
||
VOID
|
||
putu(
|
||
ULONG u
|
||
)
|
||
|
||
/*++
|
||
|
||
Routine Description:
|
||
|
||
Write an unsigned long to display
|
||
|
||
Arguments:
|
||
|
||
u - unsigned
|
||
|
||
Returns:
|
||
|
||
Nothing
|
||
|
||
--*/
|
||
|
||
{
|
||
ULONG Count;
|
||
UCHAR ch;
|
||
|
||
if(u/10) {
|
||
putu(u/10);
|
||
}
|
||
|
||
ch = (UCHAR)((u%10)+'0');
|
||
ArcWrite(BlConsoleOutDeviceId, &ch, 1, &Count);
|
||
}
|
||
|
||
|
||
VOID
|
||
pTextCharOut(
|
||
IN UCHAR c
|
||
)
|
||
{
|
||
if(DbcsLangId) {
|
||
//
|
||
// Single-byte only
|
||
//
|
||
TextGrCharOut(&c);
|
||
} else {
|
||
TextTmCharOut(&c);
|
||
}
|
||
}
|
||
|
||
|
||
PUCHAR
|
||
TextCharOut(
|
||
IN PUCHAR pc
|
||
)
|
||
{
|
||
if(DbcsLangId) {
|
||
return(TextGrCharOut(pc));
|
||
} else {
|
||
return(TextTmCharOut(pc));
|
||
}
|
||
}
|
||
|
||
|
||
VOID
|
||
TextStringOut(
|
||
IN PUCHAR String
|
||
)
|
||
{
|
||
if(DbcsLangId) {
|
||
TextGrStringOut(String);
|
||
} else {
|
||
TextTmStringOut(String);
|
||
}
|
||
}
|
||
|
||
|
||
VOID
|
||
TextClearToEndOfLine(
|
||
VOID
|
||
)
|
||
|
||
/*++
|
||
|
||
Routine Description:
|
||
|
||
Clears from the current cursor position to the end of the line
|
||
by writing blanks with the current video attribute.
|
||
|
||
Arguments:
|
||
|
||
None
|
||
|
||
Returns:
|
||
|
||
Nothing
|
||
|
||
|
||
--*/
|
||
|
||
{
|
||
if(DbcsLangId) {
|
||
TextGrClearToEndOfLine();
|
||
} else {
|
||
TextTmClearToEndOfLine();
|
||
}
|
||
}
|
||
|
||
|
||
VOID
|
||
TextClearFromStartOfLine(
|
||
VOID
|
||
)
|
||
|
||
/*++
|
||
|
||
Routine Description:
|
||
|
||
Clears from the start of the line to the current cursor position
|
||
by writing blanks with the current video attribute.
|
||
The cursor position is not changed.
|
||
|
||
Arguments:
|
||
|
||
None
|
||
|
||
Returns:
|
||
|
||
Nothing
|
||
|
||
--*/
|
||
|
||
{
|
||
if(DbcsLangId) {
|
||
TextGrClearFromStartOfLine();
|
||
} else {
|
||
TextTmClearFromStartOfLine();
|
||
}
|
||
}
|
||
|
||
|
||
VOID
|
||
TextClearToEndOfDisplay(
|
||
VOID
|
||
)
|
||
|
||
/*++
|
||
|
||
Routine Description:
|
||
|
||
Clears from the current cursor position to the end of the video
|
||
display by writing blanks with the current video attribute.
|
||
The cursor position is not changed.
|
||
|
||
Arguments:
|
||
|
||
None
|
||
|
||
Returns:
|
||
|
||
Nothing
|
||
|
||
--*/
|
||
|
||
{
|
||
if(DbcsLangId) {
|
||
TextGrClearToEndOfDisplay();
|
||
} else {
|
||
TextTmClearToEndOfDisplay();
|
||
}
|
||
}
|
||
|
||
|
||
VOID
|
||
TextClearDisplay(
|
||
VOID
|
||
)
|
||
|
||
/*++
|
||
|
||
Routine Description:
|
||
|
||
Clears the video display and positions the cursor
|
||
at the upper left corner of the screen (0,0).
|
||
|
||
Arguments:
|
||
|
||
None
|
||
|
||
Returns:
|
||
|
||
Nothing
|
||
|
||
--*/
|
||
|
||
{
|
||
if(DbcsLangId) {
|
||
TextGrClearDisplay();
|
||
} else {
|
||
TextTmClearDisplay();
|
||
}
|
||
TextSetCursorPosition(0,0);
|
||
}
|
||
|
||
|
||
VOID
|
||
TextSetCursorPosition(
|
||
IN ULONG X,
|
||
IN ULONG Y
|
||
)
|
||
|
||
/*++
|
||
|
||
Routine Description:
|
||
|
||
Moves the location of the software cursor to the specified X,Y position
|
||
on screen.
|
||
|
||
Arguments:
|
||
|
||
X - Supplies the X-position of the cursor
|
||
|
||
Y - Supplies the Y-position of the cursor
|
||
|
||
Return Value:
|
||
|
||
None.
|
||
|
||
--*/
|
||
|
||
{
|
||
TextColumn = (USHORT)X;
|
||
TextRow = (USHORT)Y;
|
||
|
||
if(DbcsLangId) {
|
||
TextGrPositionCursor((USHORT)Y,(USHORT)X);
|
||
} else {
|
||
TextTmPositionCursor((USHORT)Y,(USHORT)X);
|
||
}
|
||
}
|
||
|
||
|
||
VOID
|
||
TextGetCursorPosition(
|
||
OUT PULONG X,
|
||
OUT PULONG Y
|
||
)
|
||
|
||
/*++
|
||
|
||
Routine Description:
|
||
|
||
Gets the position of the soft cursor.
|
||
|
||
Arguments:
|
||
|
||
X - Receives column coordinate of where character would be written.
|
||
|
||
Y - Receives row coordinate of where next character would be written.
|
||
|
||
Returns:
|
||
|
||
Nothing.
|
||
|
||
--*/
|
||
|
||
{
|
||
*X = (ULONG)TextColumn;
|
||
*Y = (ULONG)TextRow;
|
||
}
|
||
|
||
|
||
VOID
|
||
TextSetCurrentAttribute(
|
||
IN UCHAR Attribute
|
||
)
|
||
|
||
/*++
|
||
|
||
Routine Description:
|
||
|
||
Sets the character attribute to be used for subsequent text display.
|
||
|
||
Arguments:
|
||
|
||
Returns:
|
||
|
||
Nothing.
|
||
|
||
--*/
|
||
|
||
{
|
||
TextCurrentAttribute = Attribute;
|
||
|
||
if(DbcsLangId) {
|
||
TextGrSetCurrentAttribute(Attribute);
|
||
} else {
|
||
TextTmSetCurrentAttribute(Attribute);
|
||
}
|
||
}
|
||
|
||
|
||
UCHAR
|
||
TextGetCurrentAttribute(
|
||
VOID
|
||
)
|
||
{
|
||
return(TextCurrentAttribute);
|
||
}
|
||
|
||
VOID
|
||
TextFillAttribute(
|
||
IN UCHAR Attribute,
|
||
IN ULONG Length
|
||
)
|
||
|
||
/*++
|
||
|
||
Routine Description:
|
||
|
||
Changes the screen attribute starting at the current cursor position.
|
||
The cursor is not moved.
|
||
|
||
Arguments:
|
||
|
||
Attribute - Supplies the new attribute
|
||
|
||
Length - Supplies the length of the area to change (in bytes)
|
||
|
||
Return Value:
|
||
|
||
None.
|
||
|
||
--*/
|
||
|
||
{
|
||
if(DbcsLangId) {
|
||
TextGrFillAttribute(Attribute,Length);
|
||
} else {
|
||
TextTmFillAttribute(Attribute,Length);
|
||
}
|
||
}
|
||
|
||
|
||
UCHAR
|
||
TextGetGraphicsCharacter(
|
||
IN GraphicsChar WhichOne
|
||
)
|
||
{
|
||
return((WhichOne < GraphicsCharMax)
|
||
? (DbcsLangId ? TextGrGetGraphicsChar(WhichOne) : TextTmGetGraphicsChar(WhichOne))
|
||
: ' ');
|
||
}
|
||
|
||
|
||
|