You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
3649 lines
119 KiB
3649 lines
119 KiB
/*++
|
|
|
|
Copyright (c) 1985 - 1999, Microsoft Corporation
|
|
|
|
Module Name:
|
|
|
|
private.c
|
|
|
|
Abstract:
|
|
|
|
This file implements private APIs for Hardware Desktop Support.
|
|
|
|
Author:
|
|
|
|
Therese Stowell (thereses) 12-13-1991
|
|
|
|
Revision History:
|
|
|
|
Notes:
|
|
|
|
--*/
|
|
|
|
#include "precomp.h"
|
|
#include "vdm.h"
|
|
#pragma hdrstop
|
|
|
|
#if defined(FE_SB)
|
|
BOOL fFullScreenGraphics ; // Do not trun graphics mode.
|
|
#if defined(i386)
|
|
extern ULONG gdwMachineId;
|
|
#endif // i386
|
|
#endif
|
|
|
|
//
|
|
// initial palette registers
|
|
//
|
|
|
|
#define PAL_BLACK 0
|
|
#define PAL_BLUE 1
|
|
#define PAL_GREEN 2
|
|
#define PAL_RED 4
|
|
#define PAL_YELLOW (PAL_RED | PAL_GREEN)
|
|
#define PAL_CYAN (PAL_GREEN | PAL_BLUE)
|
|
#define PAL_MAGENTA (PAL_BLUE | PAL_RED)
|
|
#define PAL_WHITE (PAL_RED | PAL_GREEN | PAL_BLUE)
|
|
|
|
#define PAL_I_BLACK (PAL_BLACK + (PAL_WHITE << 3))
|
|
#define PAL_I_RED (PAL_RED + (PAL_RED << 3))
|
|
#define PAL_I_GREEN (PAL_GREEN + (PAL_GREEN << 3))
|
|
#define PAL_I_YELLOW (PAL_YELLOW + (PAL_YELLOW << 3))
|
|
#define PAL_I_BLUE (PAL_BLUE + (PAL_BLUE << 3))
|
|
#define PAL_I_CYAN (PAL_CYAN + (PAL_CYAN << 3))
|
|
#define PAL_I_MAGENTA (PAL_MAGENTA + (PAL_MAGENTA << 3))
|
|
#define PAL_I_WHITE (PAL_WHITE + (PAL_WHITE << 3))
|
|
|
|
#define INITIAL_PALETTE_SIZE 18
|
|
|
|
USHORT InitialPalette[INITIAL_PALETTE_SIZE] = {
|
|
|
|
16, // 16 entries
|
|
0, // start with first palette register
|
|
0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 0x0A, 0x0B, 0x0C, 0x0D, 0x0E, 0x0F
|
|
};
|
|
|
|
#if defined(FE_SB)
|
|
PUSHORT RegInitialPalette = InitialPalette;
|
|
#endif
|
|
|
|
UCHAR ColorBuffer[] = {
|
|
|
|
16, // 16 entries
|
|
0,
|
|
0,
|
|
0, // start with first palette register
|
|
0x00, 0x00, 0x00, 0x00, // black
|
|
0x00, 0x00, 0x2A, 0x00, // blue
|
|
0x00, 0x2A, 0x00, 0x00, // green
|
|
0x00, 0x2A, 0x2A, 0x00, // cyan
|
|
0x2A, 0x00, 0x00, 0x00, // red
|
|
0x2A, 0x00, 0x2A, 0x00, // magenta
|
|
0x2A, 0x2A, 0x00, 0x00, // mustard/brown
|
|
0x36, 0x36, 0x36, 0x00, // light gray 39
|
|
0x28, 0x28, 0x28, 0x00, // dark gray 2A
|
|
0x00, 0x00, 0x3F, 0x00, // bright blue
|
|
0x00, 0x3F, 0x00, 0x00, // bright green
|
|
0x00, 0x3F, 0x3F, 0x00, // bright cyan
|
|
0x3F, 0x00, 0x00, 0x00, // bright red
|
|
0x3F, 0x00, 0x3F, 0x00, // bright magenta
|
|
0x3F, 0x3F, 0x00, 0x00, // bright yellow
|
|
0x3F, 0x3F, 0x3F, 0x00 // bright white
|
|
};
|
|
|
|
#if defined(FE_SB)
|
|
PUCHAR RegColorBuffer = ColorBuffer;
|
|
PUCHAR RegColorBufferNoTranslate = NULL;
|
|
#endif
|
|
|
|
#if defined(FE_SB)
|
|
MODE_FONT_PAIR ModeFontPairs[] = {
|
|
{FS_MODE_TEXT, 80, 21, 640, 350, 8, 16},
|
|
{FS_MODE_TEXT, 80, 25, 720, 400, 8, 16},
|
|
{FS_MODE_TEXT, 80, 28, 720, 400, 8, 14},
|
|
{FS_MODE_TEXT, 80, 43, 640, 350, 8, 8 },
|
|
{FS_MODE_TEXT, 80, 50, 720, 400, 8, 8 }
|
|
};
|
|
|
|
DWORD NUMBER_OF_MODE_FONT_PAIRS = sizeof(ModeFontPairs)/sizeof(MODE_FONT_PAIR);
|
|
PMODE_FONT_PAIR RegModeFontPairs = ModeFontPairs;
|
|
|
|
SINGLE_LIST_ENTRY gRegFullScreenCodePage; // This list contain FS_CODEPAGE data.
|
|
|
|
#else
|
|
typedef struct _MODE_FONT_PAIR {
|
|
ULONG Height;
|
|
COORD Resolution;
|
|
COORD FontSize;
|
|
} MODE_FONT_PAIR, PMODE_FONT_PAIR;
|
|
|
|
#define NUMBER_OF_MODE_FONT_PAIRS 5
|
|
|
|
MODE_FONT_PAIR ModeFontPairs[NUMBER_OF_MODE_FONT_PAIRS] = {
|
|
{21, 640, 350, 8, 16},
|
|
{25, 720, 400, 8, 16},
|
|
{28, 720, 400, 8, 14},
|
|
{43, 640, 350, 8, 8 },
|
|
{50, 720, 400, 8, 8 }
|
|
};
|
|
#endif
|
|
|
|
|
|
HANDLE hCPIFile; // handle to font file
|
|
|
|
typedef struct _FONTFILEHEADER {
|
|
BYTE ffhFileTag[8]; // SHOULD BE 0FFH,"FONT___"
|
|
BYTE ffhReserved[8];
|
|
WORD ffhPointers;
|
|
BYTE ffhPointerType;
|
|
BYTE ffhOffset1;
|
|
WORD ffhOffset2;
|
|
BYTE ffhOffset3;
|
|
} FONTFILEHEADER, *LPFONTFILEHEADER;
|
|
|
|
typedef struct _FONTINFOHEADER {
|
|
WORD fihCodePages;
|
|
} FONTINFOHEADER, *LPFONTINFOHEADER;
|
|
|
|
typedef struct _CPENTRYHEADER {
|
|
WORD cpeLength;
|
|
WORD cpeNext1;
|
|
WORD cpeNext2;
|
|
WORD cpeDevType;
|
|
BYTE cpeDevSubtype[8];
|
|
WORD cpeCodepageID;
|
|
BYTE cpeReserved[6];
|
|
DWORD cpeOffset;
|
|
} CPENTRYHEADER, *LPCPENTRYHEADER;
|
|
|
|
typedef struct _FONTDATAHEADER {
|
|
WORD fdhReserved;
|
|
WORD fdhFonts;
|
|
WORD fdhLength;
|
|
} FONTDATAHEADER, *LPFONTDATAHEADER;
|
|
|
|
typedef struct _SCREENFONTHEADER {
|
|
BYTE sfhHeight;
|
|
BYTE sfhWidth;
|
|
WORD sfhAspect;
|
|
WORD sfhCharacters;
|
|
} SCREENFONTHEADER, *LPSCREENFONTHEADER;
|
|
|
|
#define CONSOLE_WINDOWS_DIR_LENGTH 256
|
|
#define CONSOLE_EGACPI_LENGTH 9 // includes NULL
|
|
#define CONSOLE_EGACPI "\\ega.cpi"
|
|
#define CONSOLE_FONT_BUFFER_LENGTH 50
|
|
#define CONSOLE_DEFAULT_ROM_FONT 437
|
|
|
|
|
|
#ifdef i386
|
|
VOID
|
|
ReverseMousePointer(
|
|
IN PSCREEN_INFORMATION ScreenInfo,
|
|
IN PSMALL_RECT Region
|
|
);
|
|
|
|
VOID
|
|
ReadRectFromScreenBuffer(
|
|
IN PSCREEN_INFORMATION ScreenInfo,
|
|
IN COORD SourcePoint,
|
|
IN PCHAR_INFO Target,
|
|
IN COORD TargetSize,
|
|
IN PSMALL_RECT TargetRect
|
|
);
|
|
|
|
#endif
|
|
|
|
NTSTATUS
|
|
MapViewOfSection(
|
|
PHANDLE SectionHandle,
|
|
ULONG CommitSize,
|
|
PVOID *BaseAddress,
|
|
PSIZE_T ViewSize,
|
|
HANDLE ClientHandle,
|
|
PVOID *BaseClientAddress
|
|
);
|
|
|
|
NTSTATUS
|
|
ConnectToEmulator(
|
|
IN BOOL Connect,
|
|
IN PCONSOLE_INFORMATION Console
|
|
);
|
|
|
|
|
|
ULONG
|
|
SrvSetConsoleCursor(
|
|
IN OUT PCSR_API_MSG m,
|
|
IN OUT PCSR_REPLY_STATUS ReplyStatus
|
|
)
|
|
|
|
/*++
|
|
|
|
Description:
|
|
|
|
Sets the mouse pointer for the specified screen buffer.
|
|
|
|
Parameters:
|
|
|
|
hConsoleOutput - Supplies a console output handle.
|
|
|
|
hCursor - win32 cursor handle, should be NULL to set the default
|
|
cursor.
|
|
|
|
Return value:
|
|
|
|
TRUE - The operation was successful.
|
|
|
|
FALSE/NULL - The operation failed. Extended error status is available
|
|
using GetLastError.
|
|
|
|
--*/
|
|
|
|
{
|
|
PCONSOLE_SETCURSOR_MSG a = (PCONSOLE_SETCURSOR_MSG)&m->u.ApiMessageData;
|
|
NTSTATUS Status;
|
|
PCONSOLE_INFORMATION Console;
|
|
PHANDLE_DATA HandleData;
|
|
|
|
UNREFERENCED_PARAMETER(ReplyStatus);
|
|
|
|
Status = ApiPreamble(a->ConsoleHandle,
|
|
&Console
|
|
);
|
|
if (!NT_SUCCESS(Status)) {
|
|
return Status;
|
|
}
|
|
Status = DereferenceIoHandle(CONSOLE_PERPROCESSDATA(),
|
|
a->OutputHandle,
|
|
CONSOLE_GRAPHICS_OUTPUT_HANDLE,
|
|
GENERIC_WRITE,
|
|
&HandleData
|
|
);
|
|
if (NT_SUCCESS(Status)) {
|
|
if (a->CursorHandle == NULL) {
|
|
HandleData->Buffer.ScreenBuffer->CursorHandle = ghNormalCursor;
|
|
} else {
|
|
HandleData->Buffer.ScreenBuffer->CursorHandle = a->CursorHandle;
|
|
}
|
|
PostMessage(HandleData->Buffer.ScreenBuffer->Console->hWnd,
|
|
WM_SETCURSOR,
|
|
0,
|
|
-1
|
|
);
|
|
}
|
|
UnlockConsole(Console);
|
|
return Status;
|
|
}
|
|
|
|
#ifdef i386
|
|
VOID
|
|
FullScreenCursor(
|
|
IN PSCREEN_INFORMATION ScreenInfo,
|
|
IN BOOL On
|
|
)
|
|
{
|
|
if (On) {
|
|
if (ScreenInfo->CursorDisplayCount < 0) {
|
|
ScreenInfo->CursorDisplayCount = 0;
|
|
ReverseMousePointer(ScreenInfo, &ScreenInfo->Window);
|
|
}
|
|
} else {
|
|
if (ScreenInfo->CursorDisplayCount >= 0) {
|
|
ReverseMousePointer(ScreenInfo, &ScreenInfo->Window);
|
|
ScreenInfo->CursorDisplayCount = -1;
|
|
}
|
|
}
|
|
|
|
}
|
|
#endif
|
|
|
|
ULONG
|
|
SrvShowConsoleCursor(
|
|
IN OUT PCSR_API_MSG m,
|
|
IN OUT PCSR_REPLY_STATUS ReplyStatus
|
|
)
|
|
|
|
/*++
|
|
|
|
Description:
|
|
|
|
Sets the mouse pointer visibility counter. If the counter is less than
|
|
zero, the mouse pointer is not shown.
|
|
|
|
Parameters:
|
|
|
|
hOutput - Supplies a console output handle.
|
|
|
|
bShow - if TRUE, the display count is to be increased. if FALSE,
|
|
decreased.
|
|
|
|
Return value:
|
|
|
|
The return value specifies the new display count.
|
|
|
|
--*/
|
|
|
|
{
|
|
PCONSOLE_SHOWCURSOR_MSG a = (PCONSOLE_SHOWCURSOR_MSG)&m->u.ApiMessageData;
|
|
NTSTATUS Status;
|
|
PCONSOLE_INFORMATION Console;
|
|
PHANDLE_DATA HandleData;
|
|
|
|
UNREFERENCED_PARAMETER(ReplyStatus);
|
|
|
|
Status = ApiPreamble(a->ConsoleHandle,
|
|
&Console
|
|
);
|
|
if (!NT_SUCCESS(Status)) {
|
|
return Status;
|
|
}
|
|
Status = DereferenceIoHandle(CONSOLE_PERPROCESSDATA(),
|
|
a->OutputHandle,
|
|
CONSOLE_OUTPUT_HANDLE | CONSOLE_GRAPHICS_OUTPUT_HANDLE,
|
|
GENERIC_WRITE,
|
|
&HandleData
|
|
);
|
|
if (NT_SUCCESS(Status)) {
|
|
if (!(Console->FullScreenFlags & CONSOLE_FULLSCREEN_HARDWARE)) {
|
|
if (a->bShow) {
|
|
HandleData->Buffer.ScreenBuffer->CursorDisplayCount += 1;
|
|
} else {
|
|
HandleData->Buffer.ScreenBuffer->CursorDisplayCount -= 1;
|
|
}
|
|
if (HandleData->Buffer.ScreenBuffer == Console->CurrentScreenBuffer) {
|
|
PostMessage(HandleData->Buffer.ScreenBuffer->Console->hWnd,
|
|
WM_SETCURSOR,
|
|
0,
|
|
-1
|
|
);
|
|
}
|
|
} else {
|
|
#ifdef i386
|
|
if (HandleData->HandleType != CONSOLE_GRAPHICS_OUTPUT_HANDLE &&
|
|
Console->FullScreenFlags & CONSOLE_FULLSCREEN_HARDWARE &&
|
|
HandleData->Buffer.ScreenBuffer == Console->CurrentScreenBuffer) {
|
|
FullScreenCursor(HandleData->Buffer.ScreenBuffer, a->bShow);
|
|
}
|
|
#endif
|
|
}
|
|
a->DisplayCount = HandleData->Buffer.ScreenBuffer->CursorDisplayCount;
|
|
}
|
|
UnlockConsole(Console);
|
|
return Status;
|
|
}
|
|
|
|
|
|
ULONG
|
|
SrvConsoleMenuControl(
|
|
IN OUT PCSR_API_MSG m,
|
|
IN OUT PCSR_REPLY_STATUS ReplyStatus
|
|
)
|
|
|
|
/*++
|
|
|
|
Description:
|
|
|
|
Sets the command id range for the current screen buffer and returns the
|
|
menu handle.
|
|
|
|
Parameters:
|
|
|
|
hConsoleOutput - Supplies a console output handle.
|
|
|
|
dwCommandIdLow - Specifies the lowest command id to store in the input buffer.
|
|
|
|
dwCommandIdHigh - Specifies the highest command id to store in the input
|
|
buffer.
|
|
|
|
Return value:
|
|
|
|
TRUE - The operation was successful.
|
|
|
|
FALSE/NULL - The operation failed. Extended error status is available
|
|
using GetLastError.
|
|
|
|
--*/
|
|
|
|
{
|
|
PCONSOLE_MENUCONTROL_MSG a = (PCONSOLE_MENUCONTROL_MSG)&m->u.ApiMessageData;
|
|
NTSTATUS Status;
|
|
PCONSOLE_INFORMATION Console;
|
|
PHANDLE_DATA HandleData;
|
|
|
|
UNREFERENCED_PARAMETER(ReplyStatus);
|
|
|
|
Status = ApiPreamble(a->ConsoleHandle,
|
|
&Console
|
|
);
|
|
if (!NT_SUCCESS(Status)) {
|
|
return Status;
|
|
}
|
|
Status = DereferenceIoHandle(CONSOLE_PERPROCESSDATA(),
|
|
a->OutputHandle,
|
|
CONSOLE_OUTPUT_HANDLE | CONSOLE_GRAPHICS_OUTPUT_HANDLE,
|
|
GENERIC_WRITE,
|
|
&HandleData
|
|
);
|
|
if (NT_SUCCESS(Status)) {
|
|
a->hMenu = HandleData->Buffer.ScreenBuffer->Console->hMenu;
|
|
HandleData->Buffer.ScreenBuffer->CommandIdLow = a->CommandIdLow;
|
|
HandleData->Buffer.ScreenBuffer->CommandIdHigh = a->CommandIdHigh;
|
|
}
|
|
|
|
UnlockConsole(Console);
|
|
return Status;
|
|
}
|
|
|
|
ULONG
|
|
SrvSetConsolePalette(
|
|
IN OUT PCSR_API_MSG m,
|
|
IN OUT PCSR_REPLY_STATUS ReplyStatus
|
|
)
|
|
|
|
/*++
|
|
|
|
Description:
|
|
|
|
Sets the palette for the console screen buffer.
|
|
|
|
Parameters:
|
|
|
|
hOutput - Supplies a console output handle.
|
|
|
|
hPalette - Supplies a handle to the palette to set.
|
|
|
|
dwUsage - Specifies use of the system palette.
|
|
|
|
SYSPAL_NOSTATIC - System palette contains no static colors
|
|
except black and white.
|
|
|
|
SYSPAL_STATIC - System palette contains static colors
|
|
which will not change when an application
|
|
realizes its logical palette.
|
|
|
|
Return value:
|
|
|
|
TRUE - The operation was successful.
|
|
|
|
FALSE/NULL - The operation failed. Extended error status is available
|
|
using GetLastError.
|
|
|
|
--*/
|
|
|
|
{
|
|
PCONSOLE_SETPALETTE_MSG a = (PCONSOLE_SETPALETTE_MSG)&m->u.ApiMessageData;
|
|
NTSTATUS Status;
|
|
PCONSOLE_INFORMATION Console;
|
|
PHANDLE_DATA HandleData;
|
|
HPALETTE hOldPalette;
|
|
|
|
UNREFERENCED_PARAMETER(ReplyStatus);
|
|
|
|
Status = ApiPreamble(a->ConsoleHandle,
|
|
&Console
|
|
);
|
|
if (!NT_SUCCESS(Status)) {
|
|
return Status;
|
|
}
|
|
Status = DereferenceIoHandle(CONSOLE_PERPROCESSDATA(),
|
|
a->OutputHandle,
|
|
CONSOLE_GRAPHICS_OUTPUT_HANDLE,
|
|
GENERIC_WRITE,
|
|
&HandleData
|
|
);
|
|
if (NT_SUCCESS(Status)) {
|
|
USERTHREAD_USEDESKTOPINFO utudi;
|
|
BOOL bReset = FALSE;
|
|
|
|
/*
|
|
* Palette handle was converted in the client.
|
|
*/
|
|
if (GetCurrentThreadId() != HandleData->Buffer.ScreenBuffer->
|
|
Console->InputThreadInfo->ThreadId) {
|
|
bReset = TRUE;
|
|
utudi.hThread = HandleData->Buffer.ScreenBuffer->Console->InputThreadInfo->ThreadHandle;
|
|
utudi.drdRestore.pdeskRestore = NULL;
|
|
NtUserSetInformationThread(NtCurrentThread(),
|
|
UserThreadUseDesktop,
|
|
&utudi, sizeof(utudi));
|
|
}
|
|
|
|
NtUserConsoleControl(ConsolePublicPalette, &(a->hPalette), sizeof(HPALETTE));
|
|
|
|
hOldPalette = SelectPalette(
|
|
HandleData->Buffer.ScreenBuffer->Console->hDC,
|
|
a->hPalette,
|
|
FALSE);
|
|
|
|
if (hOldPalette == NULL) {
|
|
Status = STATUS_INVALID_PARAMETER;
|
|
} else {
|
|
if ((HandleData->Buffer.ScreenBuffer->hPalette != NULL) &&
|
|
(a->hPalette != HandleData->Buffer.ScreenBuffer->hPalette)) {
|
|
DeleteObject(HandleData->Buffer.ScreenBuffer->hPalette);
|
|
}
|
|
HandleData->Buffer.ScreenBuffer->hPalette = a->hPalette;
|
|
HandleData->Buffer.ScreenBuffer->dwUsage = a->dwUsage;
|
|
if (!(HandleData->Buffer.ScreenBuffer->Console->Flags & CONSOLE_IS_ICONIC) &&
|
|
HandleData->Buffer.ScreenBuffer->Console->FullScreenFlags == 0) {
|
|
|
|
SetSystemPaletteUse(HandleData->Buffer.ScreenBuffer->Console->hDC,
|
|
HandleData->Buffer.ScreenBuffer->dwUsage);
|
|
RealizePalette(HandleData->Buffer.ScreenBuffer->Console->hDC);
|
|
}
|
|
if (HandleData->Buffer.ScreenBuffer->Console->hSysPalette == NULL) {
|
|
HandleData->Buffer.ScreenBuffer->Console->hSysPalette = hOldPalette;
|
|
}
|
|
}
|
|
|
|
if (bReset) {
|
|
utudi.hThread = NULL;
|
|
NtUserSetInformationThread(NtCurrentThread(),
|
|
UserThreadUseDesktop, &utudi, sizeof(utudi));
|
|
}
|
|
}
|
|
UnlockConsole(Console);
|
|
return Status;
|
|
}
|
|
|
|
|
|
VOID
|
|
SetActivePalette(
|
|
IN PSCREEN_INFORMATION ScreenInfo
|
|
)
|
|
{
|
|
USERTHREAD_USEDESKTOPINFO utudi;
|
|
BOOL bReset = FALSE;
|
|
|
|
if (GetCurrentThreadId() != ScreenInfo->Console->InputThreadInfo->ThreadId) {
|
|
bReset = TRUE;
|
|
utudi.hThread = ScreenInfo->Console->InputThreadInfo->ThreadHandle;
|
|
utudi.drdRestore.pdeskRestore = NULL;
|
|
NtUserSetInformationThread(NtCurrentThread(),
|
|
UserThreadUseDesktop,
|
|
&utudi, sizeof(utudi));
|
|
}
|
|
|
|
SetSystemPaletteUse(ScreenInfo->Console->hDC,
|
|
ScreenInfo->dwUsage
|
|
);
|
|
RealizePalette(ScreenInfo->Console->hDC);
|
|
|
|
if (bReset == TRUE) {
|
|
utudi.hThread = NULL;
|
|
NtUserSetInformationThread(NtCurrentThread(),
|
|
UserThreadUseDesktop, &utudi, sizeof(utudi));
|
|
}
|
|
}
|
|
|
|
VOID
|
|
UnsetActivePalette(
|
|
IN PSCREEN_INFORMATION ScreenInfo
|
|
)
|
|
{
|
|
USERTHREAD_USEDESKTOPINFO utudi;
|
|
BOOL bReset = FALSE;
|
|
|
|
if (GetCurrentThreadId() != ScreenInfo->Console->InputThreadInfo->ThreadId) {
|
|
bReset = TRUE;
|
|
utudi.hThread = ScreenInfo->Console->InputThreadInfo->ThreadHandle;
|
|
utudi.drdRestore.pdeskRestore = NULL;
|
|
NtUserSetInformationThread(NtCurrentThread(),
|
|
UserThreadUseDesktop,
|
|
&utudi, sizeof(utudi));
|
|
}
|
|
|
|
SetSystemPaletteUse(ScreenInfo->Console->hDC,
|
|
SYSPAL_STATIC
|
|
);
|
|
RealizePalette(ScreenInfo->Console->hDC);
|
|
|
|
|
|
if (bReset == TRUE) {
|
|
utudi.hThread = NULL;
|
|
NtUserSetInformationThread(NtCurrentThread(),
|
|
UserThreadUseDesktop, &utudi, sizeof(utudi));
|
|
}
|
|
}
|
|
|
|
NTSTATUS
|
|
ConvertToFullScreen(
|
|
IN PCONSOLE_INFORMATION Console
|
|
)
|
|
{
|
|
#ifdef i386
|
|
PSCREEN_INFORMATION Cur;
|
|
COORD WindowedWindowSize, WindowSize;
|
|
|
|
// for each charmode screenbuffer
|
|
// match window size to a mode/font
|
|
// grow screen buffer if necessary
|
|
// save old window dimensions
|
|
// set new window dimensions
|
|
|
|
for (Cur=Console->ScreenBuffers;Cur!=NULL;Cur=Cur->Next) {
|
|
|
|
if (Cur->Flags & CONSOLE_GRAPHICS_BUFFER) {
|
|
continue;
|
|
}
|
|
|
|
// save old window dimensions
|
|
|
|
WindowedWindowSize.X = CONSOLE_WINDOW_SIZE_X(Cur);
|
|
WindowedWindowSize.Y = CONSOLE_WINDOW_SIZE_Y(Cur);
|
|
|
|
Cur->BufferInfo.TextInfo.WindowedWindowSize = WindowedWindowSize;
|
|
Cur->BufferInfo.TextInfo.WindowedScreenSize = Cur->ScreenBufferSize;
|
|
|
|
// match window size to a mode/font
|
|
|
|
Cur->BufferInfo.TextInfo.ModeIndex = MatchWindowSize(
|
|
Console->OutputCP,
|
|
Cur->ScreenBufferSize, &WindowSize);
|
|
|
|
// grow screen buffer if necessary
|
|
|
|
if (WindowSize.X > Cur->ScreenBufferSize.X ||
|
|
WindowSize.Y > Cur->ScreenBufferSize.Y) {
|
|
COORD NewScreenSize;
|
|
|
|
NewScreenSize.X = max(WindowSize.X, Cur->ScreenBufferSize.X);
|
|
NewScreenSize.Y = max(WindowSize.Y, Cur->ScreenBufferSize.Y);
|
|
|
|
if (ResizeScreenBuffer(Cur, NewScreenSize, FALSE) == STATUS_INVALID_HANDLE) {
|
|
return STATUS_INVALID_HANDLE;
|
|
}
|
|
}
|
|
|
|
//
|
|
// Set new window dimensions.
|
|
// We always resize horizontally from the right (change the
|
|
// right edge).
|
|
// We resize vertically from the bottom, keeping the cursor visible.
|
|
//
|
|
|
|
if (WindowedWindowSize.X != WindowSize.X) {
|
|
Cur->Window.Right -= WindowedWindowSize.X - WindowSize.X;
|
|
if (Cur->Window.Right >= Cur->ScreenBufferSize.X) {
|
|
Cur->Window.Left -= Cur->Window.Right - Cur->ScreenBufferSize.X + 1;
|
|
Cur->Window.Right -= Cur->Window.Right - Cur->ScreenBufferSize.X + 1;
|
|
}
|
|
}
|
|
if (WindowedWindowSize.Y > WindowSize.Y) {
|
|
Cur->Window.Bottom -= WindowedWindowSize.Y - WindowSize.Y;
|
|
if (Cur->Window.Bottom >= Cur->ScreenBufferSize.Y) {
|
|
Cur->Window.Top -= Cur->Window.Bottom - Cur->ScreenBufferSize.Y + 1;
|
|
Cur->Window.Bottom = Cur->ScreenBufferSize.Y - 1;
|
|
}
|
|
} else if (WindowedWindowSize.Y < WindowSize.Y) {
|
|
Cur->Window.Top -= WindowSize.Y - WindowedWindowSize.Y;
|
|
if (Cur->Window.Top < 0) {
|
|
Cur->Window.Bottom -= Cur->Window.Top;
|
|
Cur->Window.Top = 0;
|
|
}
|
|
}
|
|
if (Cur->BufferInfo.TextInfo.CursorPosition.Y > Cur->Window.Bottom) {
|
|
Cur->Window.Top += Cur->BufferInfo.TextInfo.CursorPosition.Y - Cur->Window.Bottom;
|
|
Cur->Window.Bottom = Cur->BufferInfo.TextInfo.CursorPosition.Y;
|
|
}
|
|
|
|
UserAssert(WindowSize.X == CONSOLE_WINDOW_SIZE_X(Cur));
|
|
UserAssert(WindowSize.Y == CONSOLE_WINDOW_SIZE_Y(Cur));
|
|
Cur->BufferInfo.TextInfo.MousePosition.X = Cur->Window.Left;
|
|
Cur->BufferInfo.TextInfo.MousePosition.Y = Cur->Window.Top;
|
|
|
|
if (Cur->Flags & CONSOLE_OEMFONT_DISPLAY) {
|
|
DBGCHARS(("ConvertToFullScreen converts UnicodeOem -> Unicode\n"));
|
|
FalseUnicodeToRealUnicode(
|
|
Cur->BufferInfo.TextInfo.TextRows,
|
|
Cur->ScreenBufferSize.X * Cur->ScreenBufferSize.Y,
|
|
Console->OutputCP);
|
|
} else {
|
|
DBGCHARS(("ConvertToFullScreen needs no conversion\n"));
|
|
}
|
|
DBGCHARS(("Cur->BufferInfo.TextInfo.Rows = %lx\n",
|
|
Cur->BufferInfo.TextInfo.Rows));
|
|
DBGCHARS(("Cur->BufferInfo.TextInfo.TextRows = %lx\n",
|
|
Cur->BufferInfo.TextInfo.TextRows));
|
|
}
|
|
|
|
Cur = Console->CurrentScreenBuffer;
|
|
|
|
if (Cur->Flags & CONSOLE_TEXTMODE_BUFFER) {
|
|
if (CONSOLE_IS_DBCS_OUTPUTCP(Console)) {
|
|
PCONVERSIONAREA_INFORMATION ConvAreaInfo;
|
|
ConvAreaInfo = Console->ConsoleIme.ConvAreaRoot;
|
|
while (ConvAreaInfo) {
|
|
NTSTATUS Status;
|
|
|
|
Status = StoreTextBufferFontInfo(ConvAreaInfo->ScreenBuffer,
|
|
SCR_FONTNUMBER(Cur),
|
|
SCR_FONTSIZE(Cur),
|
|
SCR_FAMILY(Cur),
|
|
SCR_FONTWEIGHT(Cur),
|
|
SCR_FACENAME(Cur),
|
|
SCR_FONTCODEPAGE(Cur));
|
|
if (!NT_SUCCESS(Status)) {
|
|
return((ULONG) Status);
|
|
}
|
|
|
|
ConvAreaInfo->ScreenBuffer->BufferInfo.TextInfo.ModeIndex = Cur->BufferInfo.TextInfo.ModeIndex;
|
|
ConvAreaInfo = ConvAreaInfo->ConvAreaNext;
|
|
}
|
|
}
|
|
Cur->BufferInfo.TextInfo.Flags &= ~TEXT_VALID_HINT;
|
|
}
|
|
|
|
SetWindowSize(Cur);
|
|
WriteToScreen(Cur, &Console->CurrentScreenBuffer->Window);
|
|
|
|
#else
|
|
UNREFERENCED_PARAMETER(Console);
|
|
#endif
|
|
return STATUS_SUCCESS;
|
|
}
|
|
|
|
NTSTATUS
|
|
ConvertToWindowed(
|
|
IN PCONSOLE_INFORMATION Console
|
|
)
|
|
{
|
|
#ifdef i386
|
|
PSCREEN_INFORMATION Cur;
|
|
SMALL_RECT WindowedWindow;
|
|
|
|
// for each charmode screenbuffer
|
|
// restore window dimensions
|
|
|
|
for (Cur=Console->ScreenBuffers;Cur!=NULL;Cur=Cur->Next) {
|
|
if ((Cur->Flags & CONSOLE_TEXTMODE_BUFFER) == 0) {
|
|
continue;
|
|
}
|
|
|
|
if (ResizeScreenBuffer(Cur,
|
|
Cur->BufferInfo.TextInfo.WindowedScreenSize,
|
|
FALSE) == STATUS_INVALID_HANDLE) {
|
|
/*
|
|
* Something really went wrong. All we can do is just to
|
|
* bail out.
|
|
*/
|
|
return STATUS_INVALID_HANDLE;
|
|
}
|
|
|
|
WindowedWindow.Right = Cur->Window.Right;
|
|
WindowedWindow.Bottom = Cur->Window.Bottom;
|
|
WindowedWindow.Left = Cur->Window.Right + 1 -
|
|
Cur->BufferInfo.TextInfo.WindowedWindowSize.X;
|
|
WindowedWindow.Top = Cur->Window.Bottom + 1 -
|
|
Cur->BufferInfo.TextInfo.WindowedWindowSize.Y;
|
|
if (WindowedWindow.Left > Cur->Window.Left) {
|
|
WindowedWindow.Right -= WindowedWindow.Left - Cur->Window.Left;
|
|
WindowedWindow.Left = Cur->Window.Left;
|
|
}
|
|
if (WindowedWindow.Right < Cur->BufferInfo.TextInfo.CursorPosition.X) {
|
|
WindowedWindow.Left += Cur->BufferInfo.TextInfo.CursorPosition.X - WindowedWindow.Right;
|
|
WindowedWindow.Right = Cur->BufferInfo.TextInfo.CursorPosition.X;
|
|
}
|
|
if (WindowedWindow.Top > Cur->Window.Top) {
|
|
WindowedWindow.Bottom -= WindowedWindow.Top - Cur->Window.Top;
|
|
WindowedWindow.Top = Cur->Window.Top;
|
|
}
|
|
if (WindowedWindow.Bottom < Cur->BufferInfo.TextInfo.CursorPosition.Y) {
|
|
WindowedWindow.Top += Cur->BufferInfo.TextInfo.CursorPosition.Y - WindowedWindow.Bottom;
|
|
WindowedWindow.Bottom = Cur->BufferInfo.TextInfo.CursorPosition.Y;
|
|
}
|
|
ResizeWindow(Cur, &WindowedWindow, FALSE);
|
|
|
|
if (CONSOLE_IS_DBCS_OUTPUTCP(Console)) {
|
|
SetFont(Cur);
|
|
}
|
|
|
|
if (Cur->Flags & CONSOLE_OEMFONT_DISPLAY) {
|
|
DBGCHARS(("ConvertToWindowed converts Unicode -> UnicodeOem\n"));
|
|
RealUnicodeToFalseUnicode(
|
|
Cur->BufferInfo.TextInfo.TextRows,
|
|
Cur->ScreenBufferSize.X * Cur->ScreenBufferSize.Y,
|
|
Console->OutputCP);
|
|
} else {
|
|
DBGCHARS(("ConvertToWindowed needs no conversion\n"));
|
|
}
|
|
DBGCHARS(("Cur->BufferInfo.TextInfo.Rows = %lx\n",
|
|
Cur->BufferInfo.TextInfo.Rows));
|
|
DBGCHARS(("Cur->BufferInfo.TextInfo.TextRows = %lx\n",
|
|
Cur->BufferInfo.TextInfo.TextRows));
|
|
}
|
|
|
|
Cur = Console->CurrentScreenBuffer;
|
|
|
|
if (Cur->Flags & CONSOLE_TEXTMODE_BUFFER) {
|
|
if (CONSOLE_IS_DBCS_OUTPUTCP(Console)) {
|
|
PCONVERSIONAREA_INFORMATION ConvAreaInfo;
|
|
ConvAreaInfo = Console->ConsoleIme.ConvAreaRoot;
|
|
while (ConvAreaInfo) {
|
|
NTSTATUS Status;
|
|
|
|
Status = StoreTextBufferFontInfo(ConvAreaInfo->ScreenBuffer,
|
|
SCR_FONTNUMBER(Cur),
|
|
SCR_FONTSIZE(Cur),
|
|
SCR_FAMILY(Cur),
|
|
SCR_FONTWEIGHT(Cur),
|
|
SCR_FACENAME(Cur),
|
|
SCR_FONTCODEPAGE(Cur));
|
|
if (!NT_SUCCESS(Status)) {
|
|
return((ULONG) Status);
|
|
}
|
|
|
|
ConvAreaInfo->ScreenBuffer->BufferInfo.TextInfo.ModeIndex = Cur->BufferInfo.TextInfo.ModeIndex;
|
|
ConvAreaInfo = ConvAreaInfo->ConvAreaNext;
|
|
}
|
|
}
|
|
Cur->BufferInfo.TextInfo.Flags &= ~TEXT_VALID_HINT;
|
|
}
|
|
|
|
SetWindowSize(Cur);
|
|
WriteToScreen(Cur, &Console->CurrentScreenBuffer->Window);
|
|
|
|
#else
|
|
UNREFERENCED_PARAMETER(Console);
|
|
#endif
|
|
return STATUS_SUCCESS;
|
|
}
|
|
|
|
ULONG
|
|
SrvSetConsoleDisplayMode(
|
|
IN OUT PCSR_API_MSG m,
|
|
IN OUT PCSR_REPLY_STATUS ReplyStatus
|
|
)
|
|
|
|
/*++
|
|
|
|
Description:
|
|
|
|
This routine sets the console display mode for an output buffer.
|
|
This API is only supported on x86 machines.
|
|
|
|
Parameters:
|
|
|
|
hConsoleOutput - Supplies a console output handle.
|
|
|
|
dwFlags - Specifies the display mode. Options are:
|
|
|
|
CONSOLE_FULLSCREEN_MODE - data is displayed fullscreen
|
|
|
|
CONSOLE_WINDOWED_MODE - data is displayed in a window
|
|
|
|
lpNewScreenBufferDimensions - On output, contains the new dimensions of
|
|
the screen buffer. The dimensions are in rows and columns for
|
|
textmode screen buffers.
|
|
|
|
Return value:
|
|
|
|
TRUE - The operation was successful.
|
|
|
|
FALSE/NULL - The operation failed. Extended error status is available
|
|
using GetLastError.
|
|
|
|
--*/
|
|
|
|
{
|
|
PCONSOLE_SETDISPLAYMODE_MSG a = (PCONSOLE_SETDISPLAYMODE_MSG)&m->u.ApiMessageData;
|
|
NTSTATUS Status;
|
|
PCONSOLE_INFORMATION Console;
|
|
PHANDLE_DATA HandleData;
|
|
PSCREEN_INFORMATION ScreenInfo;
|
|
UINT State;
|
|
HANDLE hEvent = NULL;
|
|
|
|
UNREFERENCED_PARAMETER(ReplyStatus);
|
|
|
|
Status = ApiPreamble(a->ConsoleHandle,
|
|
&Console
|
|
);
|
|
if (!NT_SUCCESS(Status)) {
|
|
return Status;
|
|
}
|
|
|
|
Status = NtDuplicateObject(CONSOLE_CLIENTPROCESSHANDLE(),
|
|
a->hEvent,
|
|
NtCurrentProcess(),
|
|
&hEvent,
|
|
0,
|
|
FALSE,
|
|
DUPLICATE_SAME_ACCESS
|
|
);
|
|
if (!NT_SUCCESS(Status)) {
|
|
goto SrvSetConsoleDisplayModeFailure;
|
|
}
|
|
Status = DereferenceIoHandle(CONSOLE_PERPROCESSDATA(),
|
|
a->OutputHandle,
|
|
CONSOLE_OUTPUT_HANDLE | CONSOLE_GRAPHICS_OUTPUT_HANDLE,
|
|
GENERIC_WRITE,
|
|
&HandleData
|
|
);
|
|
if (NT_SUCCESS(Status)) {
|
|
ScreenInfo = HandleData->Buffer.ScreenBuffer;
|
|
if (!ACTIVE_SCREEN_BUFFER(ScreenInfo)) {
|
|
Status = STATUS_INVALID_PARAMETER;
|
|
goto SrvSetConsoleDisplayModeFailure;
|
|
}
|
|
if (a->dwFlags == CONSOLE_FULLSCREEN_MODE) {
|
|
#if !defined(_X86_)
|
|
if (ScreenInfo->Flags & CONSOLE_TEXTMODE_BUFFER) {
|
|
Status = STATUS_INVALID_PARAMETER;
|
|
goto SrvSetConsoleDisplayModeFailure;
|
|
}
|
|
#else
|
|
if (!FullScreenInitialized || GetSystemMetrics(SM_REMOTESESSION)) {
|
|
Status = STATUS_INVALID_PARAMETER;
|
|
goto SrvSetConsoleDisplayModeFailure;
|
|
}
|
|
#endif
|
|
if (Console->FullScreenFlags & CONSOLE_FULLSCREEN) {
|
|
KdPrint(("CONSRV: VDM converting to fullscreen twice\n"));
|
|
Status = STATUS_SUCCESS;
|
|
goto SrvSetConsoleDisplayModeFailure;
|
|
}
|
|
State = FULLSCREEN;
|
|
} else {
|
|
if (Console->FullScreenFlags == 0) {
|
|
KdPrint(("CONSRV: VDM converting to windowed twice\n"));
|
|
Status = STATUS_SUCCESS;
|
|
goto SrvSetConsoleDisplayModeFailure;
|
|
}
|
|
State = WINDOWED;
|
|
}
|
|
Status = QueueConsoleMessage(Console,
|
|
CM_MODE_TRANSITION,
|
|
State,
|
|
(LPARAM)hEvent
|
|
);
|
|
if (!NT_SUCCESS(Status)) {
|
|
goto SrvSetConsoleDisplayModeFailure;
|
|
}
|
|
}
|
|
UnlockConsole(Console);
|
|
return Status;
|
|
|
|
SrvSetConsoleDisplayModeFailure:
|
|
if (hEvent) {
|
|
NtSetEvent(hEvent, NULL);
|
|
NtClose(hEvent);
|
|
}
|
|
|
|
UnlockConsole(Console);
|
|
return Status;
|
|
}
|
|
|
|
VOID
|
|
UnregisterVDM(
|
|
IN PCONSOLE_INFORMATION Console
|
|
)
|
|
{
|
|
// williamh, Feb 2 1994.
|
|
// catch multiple calls to unregister vdm. Believe it or not, this could
|
|
// happen
|
|
//UserAssert(Console->Flags & CONSOLE_VDM_REGISTERED);
|
|
if (!(Console->Flags & CONSOLE_VDM_REGISTERED))
|
|
return;
|
|
|
|
#if defined(FE_SB) && defined(i386)
|
|
// When HDOS apps exit, console screen resolution is changed to 640*400. Because HBIOS set
|
|
// Screen resolution to 640*400. So, we should replace current screen resoultion(640*480).
|
|
// 09/11/96 bklee
|
|
{
|
|
|
|
if ((Console->FullScreenFlags & CONSOLE_FULLSCREEN_HARDWARE) &&
|
|
( Console->OutputCP == KOREAN_CP ||
|
|
(Console->OutputCP == JAPAN_CP && ISNECPC98(gdwMachineId) ) )) {
|
|
|
|
ULONG Index;
|
|
DEVMODEW Devmode;
|
|
BOOL fGraphics = fFullScreenGraphics ? IsAvailableFsCodePage(Console->OutputCP) : FALSE;
|
|
|
|
Index = Console->CurrentScreenBuffer->BufferInfo.TextInfo.ModeIndex;
|
|
|
|
ZeroMemory(&Devmode, sizeof(Devmode));
|
|
|
|
Devmode.dmSize = sizeof(Devmode);
|
|
Devmode.dmDriverExtra = 0;
|
|
Devmode.dmFields = DM_BITSPERPEL |
|
|
DM_PELSWIDTH |
|
|
DM_PELSHEIGHT |
|
|
DM_DISPLAYFLAGS;
|
|
|
|
Devmode.dmBitsPerPel = 4;
|
|
|
|
Devmode.dmPelsWidth = RegModeFontPairs[Index].Resolution.X;
|
|
Devmode.dmPelsHeight = RegModeFontPairs[Index].Resolution.Y;
|
|
Devmode.dmDisplayFlags = (fGraphics && (RegModeFontPairs[Index].Mode & FS_MODE_GRAPHICS)) ? 0 : DMDISPLAYFLAGS_TEXTMODE;
|
|
|
|
GdiFullscreenControl(FullscreenControlSetMode,
|
|
&Devmode,
|
|
sizeof(Devmode),
|
|
NULL,
|
|
NULL);
|
|
}
|
|
}
|
|
#endif
|
|
#ifdef i386
|
|
if (Console->FullScreenFlags & CONSOLE_FULLSCREEN_HARDWARE &&
|
|
Console->Flags & CONSOLE_CONNECTED_TO_EMULATOR) {
|
|
NtUserConsoleControl(ConsoleSetVDMCursorBounds, NULL, 0);
|
|
// connect emulator
|
|
ConnectToEmulator(FALSE, Console);
|
|
}
|
|
|
|
if (FullScreenInitialized) {
|
|
CloseHandle(Console->VDMStartHardwareEvent);
|
|
CloseHandle(Console->VDMEndHardwareEvent);
|
|
if (Console->VDMErrorHardwareEvent) {
|
|
CloseHandle(Console->VDMErrorHardwareEvent);
|
|
Console->VDMErrorHardwareEvent = NULL;
|
|
}
|
|
NtUnmapViewOfSection(NtCurrentProcess(), Console->StateBuffer);
|
|
NtUnmapViewOfSection(Console->VDMProcessHandle, Console->StateBufferClient);
|
|
NtClose(Console->StateSectionHandle);
|
|
Console->StateLength = 0;
|
|
}
|
|
|
|
#endif
|
|
|
|
Console->Flags &= ~CONSOLE_VDM_REGISTERED;
|
|
|
|
if (Console->Flags & CONSOLE_HAS_FOCUS) {
|
|
USERTHREAD_FLAGS Flags;
|
|
|
|
Flags.dwFlags = 0;
|
|
Flags.dwMask = (TIF_VDMAPP | TIF_DOSEMULATOR);
|
|
NtUserSetInformationThread(Console->InputThreadInfo->ThreadHandle,
|
|
UserThreadFlags, &Flags, sizeof(Flags));
|
|
}
|
|
Console->Flags &= ~CONSOLE_WOW_REGISTERED;
|
|
UserAssert(Console->VDMBuffer != NULL);
|
|
if (Console->VDMBuffer != NULL) {
|
|
NtUnmapViewOfSection(Console->VDMProcessHandle, Console->VDMBufferClient);
|
|
NtUnmapViewOfSection(NtCurrentProcess(), Console->VDMBuffer);
|
|
NtClose(Console->VDMBufferSectionHandle);
|
|
Console->VDMBuffer = NULL;
|
|
}
|
|
#ifdef i386
|
|
if (Console->CurrentScreenBuffer &&
|
|
Console->CurrentScreenBuffer->Flags & CONSOLE_TEXTMODE_BUFFER) {
|
|
Console->CurrentScreenBuffer->BufferInfo.TextInfo.MousePosition.X = 0;
|
|
Console->CurrentScreenBuffer->BufferInfo.TextInfo.MousePosition.Y = 0;
|
|
}
|
|
#endif
|
|
UserAssert(Console->VDMProcessHandle);
|
|
CloseHandle(Console->VDMProcessHandle);
|
|
Console->VDMProcessHandle = NULL;
|
|
|
|
#if defined(FE_SB) && defined(FE_IME) && defined(i386)
|
|
{
|
|
if (Console->FullScreenFlags & CONSOLE_FULLSCREEN) {
|
|
Console->Flags |= CONSOLE_JUST_VDM_UNREGISTERED ;
|
|
}
|
|
else if (Console->CurrentScreenBuffer->Flags & CONSOLE_TEXTMODE_BUFFER) {
|
|
AdjustCursorPosition(Console->CurrentScreenBuffer,
|
|
Console->CurrentScreenBuffer->BufferInfo.TextInfo.CursorPosition,
|
|
TRUE,
|
|
NULL);
|
|
}
|
|
}
|
|
#endif
|
|
}
|
|
|
|
ULONG
|
|
SrvRegisterConsoleVDM(
|
|
IN OUT PCSR_API_MSG m,
|
|
IN OUT PCSR_REPLY_STATUS ReplyStatus
|
|
)
|
|
{
|
|
PCONSOLE_REGISTERVDM_MSG a = (PCONSOLE_REGISTERVDM_MSG)&m->u.ApiMessageData;
|
|
NTSTATUS Status;
|
|
PCONSOLE_INFORMATION Console;
|
|
SIZE_T ViewSize;
|
|
VDM_QUERY_VDM_PROCESS_DATA QueryVdmProcessData;
|
|
#ifdef i386
|
|
VIDEO_REGISTER_VDM RegisterVdm;
|
|
ULONG RegisterVdmSize = sizeof(RegisterVdm);
|
|
VIDEO_VDM Vdm;
|
|
#endif //i386
|
|
|
|
Status = ApiPreamble(a->ConsoleHandle,
|
|
&Console
|
|
);
|
|
if (!NT_SUCCESS(Status)) {
|
|
return Status;
|
|
}
|
|
|
|
//
|
|
// First make sure the caller is a VDM process
|
|
//
|
|
|
|
QueryVdmProcessData.ProcessHandle = CONSOLE_CLIENTPROCESSHANDLE();
|
|
Status = NtVdmControl(VdmQueryVdmProcess, &QueryVdmProcessData);
|
|
if (!NT_SUCCESS(Status) || QueryVdmProcessData.IsVdmProcess == FALSE) {
|
|
UnlockConsole(Console);
|
|
return STATUS_ACCESS_DENIED;
|
|
}
|
|
|
|
if (a->RegisterFlags & CONSOLE_REGISTER_VDM) {
|
|
ConsoleNotifyWinEvent(Console,
|
|
EVENT_CONSOLE_START_APPLICATION,
|
|
HandleToUlong(CONSOLE_CLIENTPROCESSID()),
|
|
CONSOLE_APPLICATION_16BIT);
|
|
} else if (a->RegisterFlags == 0) {
|
|
ConsoleNotifyWinEvent(Console,
|
|
EVENT_CONSOLE_END_APPLICATION,
|
|
HandleToUlong(CONSOLE_CLIENTPROCESSID()),
|
|
CONSOLE_APPLICATION_16BIT);
|
|
}
|
|
|
|
|
|
if (!a->RegisterFlags) {
|
|
// williamh, Jan 28 1994
|
|
// do not do an assert here because we may have unregistered the ntvdm
|
|
// and the ntvdm doesn't necessarily know this(and it could post another
|
|
// unregistervdm). Return error here so NTVDM knows what to do
|
|
// UserAssert(Console->Flags & CONSOLE_VDM_REGISTERED);
|
|
|
|
if (Console->Flags & CONSOLE_VDM_REGISTERED) {
|
|
UserAssert(!(Console->Flags & CONSOLE_FULLSCREEN_NOPAINT));
|
|
UnregisterVDM(Console);
|
|
#ifdef i386
|
|
if (Console->FullScreenFlags & CONSOLE_FULLSCREEN_HARDWARE &&
|
|
Console->CurrentScreenBuffer->Flags & CONSOLE_TEXTMODE_BUFFER) {
|
|
// SetVideoMode(Console->CurrentScreenBuffer);
|
|
// Set up cursor
|
|
SetCursorInformationHW(Console->CurrentScreenBuffer,
|
|
Console->CurrentScreenBuffer->BufferInfo.TextInfo.CursorSize,
|
|
Console->CurrentScreenBuffer->BufferInfo.TextInfo.CursorVisible);
|
|
SetCursorPositionHW(Console->CurrentScreenBuffer,
|
|
Console->CurrentScreenBuffer->BufferInfo.TextInfo.CursorPosition);
|
|
}
|
|
#endif // i386
|
|
Status = STATUS_SUCCESS;
|
|
} else {
|
|
Status = STATUS_ACCESS_DENIED;
|
|
}
|
|
UnlockConsole(Console);
|
|
return Status;
|
|
}
|
|
|
|
if (!CsrValidateMessageBuffer(m, &a->VDMBufferSectionName, a->VDMBufferSectionNameLength, sizeof(BYTE))) {
|
|
|
|
UnlockConsole(Console);
|
|
return STATUS_INVALID_PARAMETER;
|
|
}
|
|
|
|
// check it out. A console should have only one VDM registered.
|
|
UserAssert(!(Console->Flags & CONSOLE_VDM_REGISTERED));
|
|
|
|
if (Console->Flags & CONSOLE_VDM_REGISTERED) {
|
|
UnlockConsole(Console);
|
|
return (ULONG) STATUS_ACCESS_DENIED;
|
|
}
|
|
|
|
UserAssert(!Console->VDMProcessHandle);
|
|
|
|
Status = NtDuplicateObject(NtCurrentProcess(), CONSOLE_CLIENTPROCESSHANDLE(),
|
|
NtCurrentProcess(), &Console->VDMProcessHandle,
|
|
0, FALSE, DUPLICATE_SAME_ACCESS);
|
|
if (!NT_SUCCESS(Status)) {
|
|
UnlockConsole(Console);
|
|
return Status;
|
|
}
|
|
Console->VDMProcessId = CONSOLE_CLIENTPROCESSID();
|
|
|
|
#ifdef i386
|
|
|
|
Vdm.ProcessHandle = Console->VDMProcessHandle;
|
|
|
|
//
|
|
// Assume fullscreen initialization will fail.
|
|
// have state length set to zero so that NTVDM will know
|
|
// full screen is disabled.
|
|
//
|
|
|
|
a->StateLength = 0;
|
|
Console->StateLength = 0;
|
|
Console->StateBufferClient = NULL;
|
|
Console->VDMErrorHardwareEvent = NULL;
|
|
|
|
if (FullScreenInitialized) {
|
|
|
|
Status = NtDuplicateObject(CONSOLE_CLIENTPROCESSHANDLE(),
|
|
a->StartEvent,
|
|
NtCurrentProcess(),
|
|
&Console->VDMStartHardwareEvent,
|
|
0,
|
|
FALSE,
|
|
DUPLICATE_SAME_ACCESS
|
|
);
|
|
if (NT_SUCCESS(Status)) {
|
|
Status = NtDuplicateObject(CONSOLE_CLIENTPROCESSHANDLE(),
|
|
a->EndEvent,
|
|
NtCurrentProcess(),
|
|
&Console->VDMEndHardwareEvent,
|
|
0,
|
|
FALSE,
|
|
DUPLICATE_SAME_ACCESS
|
|
);
|
|
if (NT_SUCCESS(Status)) {
|
|
if (a->ErrorEvent) {
|
|
Status = NtDuplicateObject(CONSOLE_CLIENTPROCESSHANDLE(),
|
|
a->ErrorEvent,
|
|
NtCurrentProcess(),
|
|
&Console->VDMErrorHardwareEvent,
|
|
0,
|
|
FALSE,
|
|
DUPLICATE_SAME_ACCESS
|
|
);
|
|
}
|
|
if (NT_SUCCESS(Status)) {
|
|
Status = GdiFullscreenControl(FullscreenControlRegisterVdm,
|
|
&Vdm,
|
|
sizeof(Vdm),
|
|
&RegisterVdm,
|
|
&RegisterVdmSize
|
|
);
|
|
|
|
if (NT_SUCCESS(Status)) {
|
|
|
|
//
|
|
// create state section and map a view of it into server and vdm.
|
|
// this section is used to get/set video hardware state during
|
|
// the fullscreen<->windowed transition. we create the section
|
|
// instead of the vdm for security purposes.
|
|
//
|
|
|
|
Status = MapViewOfSection(&Console->StateSectionHandle,
|
|
RegisterVdm.MinimumStateSize,
|
|
&Console->StateBuffer,
|
|
&ViewSize,
|
|
Console->VDMProcessHandle,
|
|
&a->StateBuffer
|
|
);
|
|
|
|
if (NT_SUCCESS(Status)) {
|
|
a->StateLength = RegisterVdm.MinimumStateSize;
|
|
Console->StateLength = RegisterVdm.MinimumStateSize;
|
|
Console->StateBufferClient = a->StateBuffer;
|
|
} else {
|
|
CloseHandle(Console->VDMStartHardwareEvent);
|
|
CloseHandle(Console->VDMEndHardwareEvent);
|
|
if (Console->VDMErrorHardwareEvent) {
|
|
CloseHandle(Console->VDMErrorHardwareEvent);
|
|
}
|
|
}
|
|
|
|
} else {
|
|
|
|
CloseHandle(Console->VDMStartHardwareEvent);
|
|
CloseHandle(Console->VDMEndHardwareEvent);
|
|
if (Console->VDMErrorHardwareEvent) {
|
|
CloseHandle(Console->VDMErrorHardwareEvent);
|
|
}
|
|
}
|
|
} else {
|
|
CloseHandle(Console->VDMStartHardwareEvent);
|
|
CloseHandle(Console->VDMEndHardwareEvent);
|
|
}
|
|
|
|
} else {
|
|
|
|
CloseHandle(Console->VDMStartHardwareEvent);
|
|
}
|
|
|
|
}
|
|
|
|
//
|
|
// If we failed to duplicate screen switch events or map
|
|
// view to video state shared buffer, fail this API.
|
|
//
|
|
if (!NT_SUCCESS(Status)) {
|
|
UnlockConsole(Console);
|
|
return Status;
|
|
}
|
|
}
|
|
|
|
#endif // i386
|
|
|
|
//
|
|
// Create vdm char section and map a view of it into server and vdm.
|
|
// This section is used by the vdm to update the screen when in a
|
|
// charmode window. This is a performance optimization. We create
|
|
// the section instead of the vdm for security purposes.
|
|
//
|
|
|
|
Status = MapViewOfSection(&Console->VDMBufferSectionHandle,
|
|
#ifdef i386
|
|
a->VDMBufferSize.X * a->VDMBufferSize.Y * 2,
|
|
#else
|
|
a->VDMBufferSize.X * a->VDMBufferSize.Y * 4,
|
|
#endif // i386
|
|
&Console->VDMBuffer,
|
|
&ViewSize,
|
|
Console->VDMProcessHandle,
|
|
&a->VDMBuffer
|
|
);
|
|
if (!NT_SUCCESS(Status)) {
|
|
|
|
Console->VDMBuffer = NULL;
|
|
|
|
#ifdef i386
|
|
|
|
if (FullScreenInitialized) {
|
|
|
|
NtUnmapViewOfSection(NtCurrentProcess(), Console->StateBuffer);
|
|
NtUnmapViewOfSection(Console->VDMProcessHandle, Console->StateBufferClient);
|
|
NtClose(Console->StateSectionHandle);
|
|
CloseHandle(Console->VDMStartHardwareEvent);
|
|
CloseHandle(Console->VDMEndHardwareEvent);
|
|
if (Console->VDMErrorHardwareEvent) {
|
|
CloseHandle(Console->VDMErrorHardwareEvent);
|
|
}
|
|
}
|
|
|
|
#endif // i386
|
|
CloseHandle(Console->VDMProcessHandle);
|
|
Console->VDMProcessHandle = NULL;
|
|
UnlockConsole(Console);
|
|
return Status;
|
|
}
|
|
Console->VDMBufferClient = a->VDMBuffer;
|
|
|
|
Console->Flags |= CONSOLE_VDM_REGISTERED;
|
|
|
|
if (Console->Flags & CONSOLE_HAS_FOCUS) {
|
|
USERTHREAD_FLAGS Flags;
|
|
|
|
Flags.dwFlags = TIF_VDMAPP;
|
|
Flags.dwMask = TIF_VDMAPP;
|
|
NtUserSetInformationThread(Console->InputThreadInfo->ThreadHandle,
|
|
UserThreadFlags,
|
|
&Flags,
|
|
sizeof(Flags));
|
|
}
|
|
Console->VDMBufferSize = a->VDMBufferSize;
|
|
|
|
if (a->RegisterFlags & CONSOLE_REGISTER_WOW) {
|
|
Console->Flags |= CONSOLE_WOW_REGISTERED;
|
|
} else {
|
|
Console->Flags &= ~CONSOLE_WOW_REGISTERED;
|
|
}
|
|
|
|
//
|
|
// If we're already in fullscreen and we run a DOS app for
|
|
// the first time, connect the emulator.
|
|
//
|
|
|
|
#ifdef i386
|
|
if (Console->FullScreenFlags & CONSOLE_FULLSCREEN_HARDWARE) {
|
|
RECT CursorRect = {-32767, -32767, 32767, 32767};
|
|
NtUserConsoleControl(ConsoleSetVDMCursorBounds, &CursorRect, sizeof(RECT));
|
|
|
|
UserAssert(!(Console->Flags & CONSOLE_CONNECTED_TO_EMULATOR));
|
|
ConnectToEmulator(TRUE, Console);
|
|
}
|
|
#endif
|
|
|
|
UnlockConsole(Console);
|
|
return Status;
|
|
UNREFERENCED_PARAMETER(ReplyStatus);
|
|
}
|
|
|
|
NTSTATUS
|
|
SrvConsoleNotifyLastClose(
|
|
IN OUT PCSR_API_MSG m,
|
|
IN OUT PCSR_REPLY_STATUS ReplyStatus
|
|
)
|
|
{
|
|
PCONSOLE_NOTIFYLASTCLOSE_MSG a = (PCONSOLE_NOTIFYLASTCLOSE_MSG)&m->u.ApiMessageData;
|
|
NTSTATUS Status;
|
|
PCONSOLE_INFORMATION Console;
|
|
|
|
UNREFERENCED_PARAMETER(ReplyStatus);
|
|
|
|
Status = ApiPreamble(a->ConsoleHandle,
|
|
&Console
|
|
);
|
|
if (!NT_SUCCESS(Status)) {
|
|
return Status;
|
|
}
|
|
|
|
//
|
|
// Don't allow two or more processes to have last-close notify on
|
|
// the same console.
|
|
//
|
|
if (Console->Flags & CONSOLE_NOTIFY_LAST_CLOSE) {
|
|
UnlockConsole(Console);
|
|
return STATUS_ACCESS_DENIED;
|
|
}
|
|
|
|
Status = NtDuplicateObject(NtCurrentProcess(), CONSOLE_CLIENTPROCESSHANDLE(),
|
|
NtCurrentProcess(),
|
|
&Console->hProcessLastNotifyClose,
|
|
0, FALSE, DUPLICATE_SAME_ACCESS
|
|
);
|
|
if (!NT_SUCCESS(Status)) {
|
|
UnlockConsole(Console);
|
|
return Status;
|
|
}
|
|
|
|
Console->Flags |= CONSOLE_NOTIFY_LAST_CLOSE;
|
|
Console->ProcessIdLastNotifyClose = CONSOLE_CLIENTPROCESSID();
|
|
UnlockConsole(Console);
|
|
return Status;
|
|
}
|
|
|
|
NTSTATUS
|
|
MapViewOfSection(
|
|
PHANDLE SectionHandle,
|
|
ULONG CommitSize,
|
|
PVOID *BaseAddress,
|
|
PSIZE_T ViewSize,
|
|
HANDLE ClientHandle,
|
|
PVOID *BaseClientAddress
|
|
)
|
|
{
|
|
|
|
OBJECT_ATTRIBUTES Obja;
|
|
NTSTATUS Status;
|
|
LARGE_INTEGER secSize;
|
|
|
|
//
|
|
// open section and map a view of it.
|
|
//
|
|
InitializeObjectAttributes(
|
|
&Obja,
|
|
NULL,
|
|
OBJ_CASE_INSENSITIVE,
|
|
NULL,
|
|
NULL
|
|
);
|
|
|
|
secSize.QuadPart = CommitSize;
|
|
Status = NtCreateSection (SectionHandle,
|
|
SECTION_ALL_ACCESS,
|
|
&Obja,
|
|
&secSize,
|
|
PAGE_READWRITE,
|
|
SEC_RESERVE,
|
|
NULL
|
|
);
|
|
if (!NT_SUCCESS(Status)) {
|
|
return((ULONG) Status);
|
|
}
|
|
|
|
*BaseAddress = 0;
|
|
*ViewSize = 0;
|
|
|
|
Status = NtMapViewOfSection(*SectionHandle,
|
|
NtCurrentProcess(),
|
|
BaseAddress, // Receives the base
|
|
// address of the section.
|
|
|
|
0, // No specific type of
|
|
// address required.
|
|
|
|
CommitSize, // Commit size. It was
|
|
// passed by the caller.
|
|
// NULL for a save, and
|
|
// size of the section
|
|
// for a set.
|
|
|
|
NULL, // Section offset it NULL;
|
|
// Map from the start.
|
|
|
|
ViewSize, // View Size is NULL since
|
|
// we want to map the
|
|
// entire section.
|
|
|
|
ViewUnmap,
|
|
0L,
|
|
PAGE_READWRITE
|
|
);
|
|
if (!NT_SUCCESS(Status)) {
|
|
NtClose(*SectionHandle);
|
|
return Status;
|
|
}
|
|
|
|
*BaseClientAddress = 0;
|
|
*ViewSize = 0;
|
|
Status = NtMapViewOfSection(*SectionHandle,
|
|
ClientHandle,
|
|
BaseClientAddress, // Receives the base
|
|
// address of the section.
|
|
|
|
0, // No specific type of
|
|
// address required.
|
|
|
|
CommitSize, // Commit size. It was
|
|
// passed by the caller.
|
|
// NULL for a save, and
|
|
// size of the section
|
|
// for a set.
|
|
|
|
NULL, // Section offset it NULL;
|
|
// Map from the start.
|
|
|
|
ViewSize, // View Size is NULL since
|
|
// we want to map the
|
|
// entire section.
|
|
|
|
ViewUnmap,
|
|
// williamh, Jan 28 1994
|
|
// This MEM_TOP_DOWN is necessary.
|
|
// if the console has VDM registered, ntvdm would have released its video memory
|
|
// address space(0xA0000 ~ 0xBFFFF). Without the MEM_TOP_DOWN, the
|
|
// NtMapViewOfSection can grab the address space and we will have trouble of
|
|
// mapping the address space to the physical video ram. We don't do a test
|
|
// for VDM because there is no harm of doing this for non-vdm application.
|
|
MEM_TOP_DOWN,
|
|
PAGE_READWRITE
|
|
);
|
|
if (!NT_SUCCESS(Status)) {
|
|
NtUnmapViewOfSection(NtCurrentProcess(), *BaseAddress);
|
|
NtClose(*SectionHandle);
|
|
}
|
|
return((ULONG) Status);
|
|
}
|
|
|
|
NTSTATUS
|
|
ConnectToEmulator(
|
|
IN BOOL Connect,
|
|
IN PCONSOLE_INFORMATION Console
|
|
)
|
|
{
|
|
NTSTATUS Status;
|
|
FULLSCREENCONTROL fsctl;
|
|
VIDEO_VDM ConnectInfo;
|
|
HANDLE ProcessHandle = Console->VDMProcessHandle;
|
|
USERTHREAD_FLAGS Flags;
|
|
|
|
DBGFULLSCR(("ConnectToEmulator : %s - entering\n", Connect ? "CONNECT" : "DISCONNECT"));
|
|
|
|
Flags.dwMask = TIF_DOSEMULATOR;
|
|
if (Connect) {
|
|
fsctl = FullscreenControlEnable;
|
|
UserAssert((Console->Flags & CONSOLE_CONNECTED_TO_EMULATOR) == 0);
|
|
Console->Flags |= CONSOLE_CONNECTED_TO_EMULATOR;
|
|
Flags.dwFlags = TIF_DOSEMULATOR;
|
|
#if DBG
|
|
RtlWalkFrameChain(Console->ConnectStack,
|
|
ARRAY_SIZE(Console->ConnectStack),
|
|
0);
|
|
#endif
|
|
} else {
|
|
fsctl = FullscreenControlDisable;
|
|
UserAssert((Console->Flags & CONSOLE_CONNECTED_TO_EMULATOR) != 0);
|
|
Console->Flags &= ~CONSOLE_CONNECTED_TO_EMULATOR;
|
|
Flags.dwFlags = 0;
|
|
#if DBG
|
|
RtlWalkFrameChain(Console->DisconnectStack,
|
|
ARRAY_SIZE(Console->DisconnectStack),
|
|
0);
|
|
#endif
|
|
}
|
|
|
|
if (Console->Flags & CONSOLE_HAS_FOCUS) {
|
|
NtUserSetInformationThread(Console->InputThreadInfo->ThreadHandle,
|
|
UserThreadFlags, &Flags, sizeof(Flags));
|
|
}
|
|
|
|
ConnectInfo.ProcessHandle = ProcessHandle;
|
|
|
|
|
|
Status = GdiFullscreenControl(fsctl,
|
|
&ConnectInfo,
|
|
sizeof(ConnectInfo),
|
|
NULL,
|
|
NULL);
|
|
|
|
UserAssert(Status == STATUS_SUCCESS ||
|
|
Status == STATUS_CONFLICTING_ADDRESSES ||
|
|
Status == STATUS_PROCESS_IS_TERMINATING);
|
|
|
|
DBGFULLSCR(("ConnectToEmulator : leaving, staus = %08lx\n", Status));
|
|
|
|
return Status;
|
|
}
|
|
|
|
#define CONSOLE_VDM_TIMEOUT 20000
|
|
#ifdef i386
|
|
VOID
|
|
LockConsoleForCurrentThread(
|
|
IN PCONSOLE_INFORMATION Console,
|
|
IN DWORD Count)
|
|
{
|
|
while (Console->ConsoleLock.RecursionCount != Count) {
|
|
LockConsole(Console);
|
|
}
|
|
}
|
|
DWORD
|
|
UnlockConsoleForCurrentThread(
|
|
IN PCONSOLE_INFORMATION Console)
|
|
{
|
|
DWORD Count, i;
|
|
|
|
Count = Console->ConsoleLock.RecursionCount;
|
|
ASSERT(Count > 0);
|
|
for (i = 0; i < Count; i++) {
|
|
UnlockConsole(Console);
|
|
}
|
|
return Count;
|
|
}
|
|
#endif
|
|
|
|
NTSTATUS
|
|
DisplayModeTransition(
|
|
IN BOOL bForeground,
|
|
IN PCONSOLE_INFORMATION Console,
|
|
IN PSCREEN_INFORMATION ScreenInfo
|
|
)
|
|
{
|
|
#ifdef i386
|
|
NTSTATUS Status;
|
|
LARGE_INTEGER li;
|
|
DWORD WaitCount;
|
|
HANDLE WaitHandles[2];
|
|
|
|
if (!FullScreenInitialized || GetSystemMetrics(SM_REMOTESESSION)) {
|
|
return STATUS_SUCCESS;
|
|
}
|
|
|
|
WaitCount = 1;
|
|
WaitHandles[0] = Console->VDMEndHardwareEvent;
|
|
if (Console->VDMErrorHardwareEvent) {
|
|
WaitHandles[1] = Console->VDMErrorHardwareEvent;
|
|
WaitCount++;
|
|
}
|
|
|
|
|
|
if (bForeground) {
|
|
if (!(Console->FullScreenFlags & CONSOLE_FULLSCREEN)) {
|
|
KdPrint(("CONSRV: received fullscreen message too early\n"));
|
|
return STATUS_UNSUCCESSFUL;
|
|
}
|
|
|
|
} else {
|
|
|
|
//
|
|
// Check first to see if we're not already fullscreen. If we aren't,
|
|
// don't allow this. Temporary BETA fix till USER gets fixed.
|
|
//
|
|
if (!(Console->FullScreenFlags & CONSOLE_FULLSCREEN_HARDWARE)) {
|
|
KdPrint(("CONSRV: received multiple windowed messages\n"));
|
|
return STATUS_SUCCESS;
|
|
}
|
|
}
|
|
|
|
//
|
|
// Before actually doing any mode switching work, notify ntvdm to give it
|
|
// a chance to bring all threads to a stoppable state.
|
|
//
|
|
|
|
if (Console->Flags & CONSOLE_VDM_REGISTERED) {
|
|
|
|
//
|
|
// make a special case for ntvdm during switching because
|
|
// ntvdm has to suspend its main thread which may be making
|
|
// console api calls before it can be suspended.
|
|
//
|
|
HANDLE ConsoleHandle = Console->ConsoleHandle;
|
|
NTSTATUS Statusx;
|
|
|
|
//
|
|
// Make timeout longer than usual. It will take long to
|
|
// stop ntvdm threads under stress.
|
|
//
|
|
li.QuadPart = (LONGLONG)-10000 * CONSOLE_VDM_TIMEOUT * 3;
|
|
Status = NtSetEvent(Console->VDMStartHardwareEvent, NULL);
|
|
if (NT_SUCCESS(Status)) {
|
|
HANDLE OwningThread;
|
|
DWORD RecursionCount;
|
|
|
|
//
|
|
// After notifying ntvdm to do screen switch, we must release
|
|
// console locks to avoid deadlock. Before doing so, we need to
|
|
// remember how many locks are released and who releases them, in
|
|
// case that we fail to get the locks back.
|
|
//
|
|
OwningThread = Console->ConsoleLock.OwningThread;
|
|
Console->UnlockConsoleOwningThread = OwningThread;
|
|
RecursionCount = UnlockConsoleForCurrentThread(Console);
|
|
Status = NtWaitForMultipleObjects(WaitCount, WaitHandles, WaitAny, FALSE, &li);
|
|
Statusx = RevalidateConsole(ConsoleHandle, &Console);
|
|
if (!NT_SUCCESS(Statusx)) {
|
|
Console->UnlockConsoleSkipCount = RecursionCount;
|
|
NtSetEvent(Console->VDMErrorHardwareEvent, NULL);
|
|
return Statusx;
|
|
}
|
|
|
|
//
|
|
// Get back the original number of locks. Since the lock is owned
|
|
// by us already (by calling RevalidateConsole), the following call
|
|
// simply increases the recursion count of the lock.
|
|
//
|
|
LockConsoleForCurrentThread(Console, RecursionCount);
|
|
Console->UnlockConsoleOwningThread = NULL;
|
|
}
|
|
|
|
if (Status != 0) {
|
|
Console->Flags &= ~CONSOLE_FULLSCREEN_NOPAINT;
|
|
NtSetEvent(Console->VDMErrorHardwareEvent, NULL);
|
|
UnregisterVDM(Console);
|
|
KdPrint(("CONSRV: VDM not responding to initial request.\n"));
|
|
}
|
|
}
|
|
|
|
if (bForeground) {
|
|
PSCREEN_INFORMATION ScreenInfo = Console->CurrentScreenBuffer;
|
|
NTSTATUS Status;
|
|
|
|
KdPrint((" CONSRV - Display Mode transition to fullscreen \n"));
|
|
|
|
Console->FullScreenFlags |= CONSOLE_FULLSCREEN_HARDWARE;
|
|
|
|
//
|
|
// Fire off an event to let accessibility apps know the layout has
|
|
// changed. We only want to do this while going *to* full-screen;
|
|
// on the way back we'll send this event from InternalUpdateScrollBars.
|
|
//
|
|
ConsoleNotifyWinEvent(ScreenInfo->Console,
|
|
EVENT_CONSOLE_LAYOUT,
|
|
0,
|
|
0);
|
|
if (!(ScreenInfo->Flags & CONSOLE_GRAPHICS_BUFFER)) {
|
|
#if defined(FE_SB)
|
|
BOOL fGraphics = fFullScreenGraphics ? IsAvailableFsCodePage(Console->OutputCP) : FALSE;
|
|
#endif
|
|
#if 1
|
|
DEVMODEW Devmode;
|
|
ULONG Index;
|
|
|
|
Index = Console->CurrentScreenBuffer->BufferInfo.TextInfo.ModeIndex;
|
|
|
|
//
|
|
// set mode to go to full screen
|
|
//
|
|
|
|
ZeroMemory(&Devmode, sizeof(Devmode));
|
|
|
|
Devmode.dmSize = sizeof(Devmode);
|
|
Devmode.dmDriverExtra = 0;
|
|
Devmode.dmFields = DM_BITSPERPEL |
|
|
DM_PELSWIDTH |
|
|
DM_PELSHEIGHT |
|
|
DM_DISPLAYFLAGS;
|
|
|
|
Devmode.dmBitsPerPel = 4;
|
|
#if defined(FE_SB)
|
|
Devmode.dmPelsWidth = RegModeFontPairs[Index].Resolution.X;
|
|
Devmode.dmPelsHeight = RegModeFontPairs[Index].Resolution.Y;
|
|
Devmode.dmDisplayFlags = (fGraphics && (RegModeFontPairs[Index].Mode & FS_MODE_GRAPHICS)) ? 0 : DMDISPLAYFLAGS_TEXTMODE;
|
|
#else
|
|
Devmode.dmPelsWidth = ModeFontPairs[Index].Resolution.X;
|
|
Devmode.dmPelsHeight = ModeFontPairs[Index].Resolution.Y;
|
|
Devmode.dmDisplayFlags = DMDISPLAYFLAGS_TEXTMODE;
|
|
#endif
|
|
|
|
if (NT_SUCCESS(GdiFullscreenControl(FullscreenControlSetMode,
|
|
&Devmode,
|
|
sizeof(Devmode),
|
|
NULL,
|
|
NULL)))
|
|
{
|
|
#endif
|
|
// set video mode and font
|
|
if (SetVideoMode(ScreenInfo)) {
|
|
|
|
#if defined(FE_SB)
|
|
if (!(Console->Flags & CONSOLE_VDM_REGISTERED)) {
|
|
int i ;
|
|
for (i = 0 ; i < ScreenInfo->ScreenBufferSize.Y; i++) {
|
|
ScreenInfo->BufferInfo.TextInfo.Rows[i].CharRow.OldLeft = INVALID_OLD_LENGTH ;
|
|
ScreenInfo->BufferInfo.TextInfo.Rows[i].CharRow.OldRight = INVALID_OLD_LENGTH ;
|
|
}
|
|
}
|
|
#endif
|
|
//set up cursor
|
|
|
|
SetCursorInformationHW(ScreenInfo,
|
|
ScreenInfo->BufferInfo.TextInfo.CursorSize,
|
|
ScreenInfo->BufferInfo.TextInfo.CursorVisible);
|
|
SetCursorPositionHW(ScreenInfo,
|
|
ScreenInfo->BufferInfo.TextInfo.CursorPosition);
|
|
}
|
|
}
|
|
}
|
|
|
|
// tell VDM to unmap memory
|
|
|
|
if (Console->Flags & CONSOLE_VDM_REGISTERED) {
|
|
|
|
//
|
|
// Increase the regular timeout value, since ntvdm needs to
|
|
// do A LOT of transitions between kernel mode and user mode
|
|
// to collect the video state information. (During stress, it may
|
|
// take a long time.)
|
|
//
|
|
|
|
li.QuadPart = (LONGLONG)-10000 * CONSOLE_VDM_TIMEOUT * 3;
|
|
Status = NtSetEvent(Console->VDMStartHardwareEvent, NULL);
|
|
if (NT_SUCCESS(Status)) {
|
|
Status = NtWaitForMultipleObjects(WaitCount, WaitHandles, WaitAny, FALSE, &li);
|
|
}
|
|
if (Status != 0) {
|
|
Console->Flags &= ~CONSOLE_FULLSCREEN_NOPAINT;
|
|
NtSetEvent(Console->VDMErrorHardwareEvent, NULL);
|
|
UnregisterVDM(Console);
|
|
KdPrint(("CONSRV: VDM not responding or in error state.\n"));
|
|
}
|
|
}
|
|
|
|
if (!(ScreenInfo->Flags & CONSOLE_GRAPHICS_BUFFER)) {
|
|
|
|
WriteRegionToScreen(ScreenInfo, &ScreenInfo->Window);
|
|
}
|
|
|
|
if (Console->Flags & CONSOLE_VDM_REGISTERED) {
|
|
|
|
// connect emulator and map memory into the VDMs address space.
|
|
UserAssert(!(Console->Flags & CONSOLE_CONNECTED_TO_EMULATOR));
|
|
|
|
Status = ConnectToEmulator(TRUE, Console);
|
|
|
|
if (NT_SUCCESS(Status)) {
|
|
|
|
VIDEO_HARDWARE_STATE State;
|
|
ULONG StateSize = sizeof(State);
|
|
|
|
State.StateHeader = Console->StateBuffer;
|
|
State.StateLength = Console->StateLength;
|
|
|
|
|
|
Status = GdiFullscreenControl(FullscreenControlRestoreHardwareState,
|
|
&State,
|
|
StateSize,
|
|
&State,
|
|
&StateSize);
|
|
}
|
|
|
|
if (Status != STATUS_SUCCESS) {
|
|
Console->Flags &= ~CONSOLE_FULLSCREEN_NOPAINT;
|
|
NtSetEvent(Console->VDMErrorHardwareEvent, NULL);
|
|
UnregisterVDM(Console);
|
|
KdPrint(("CONSRV: set hardware state failed.\n"));
|
|
} else {
|
|
|
|
//
|
|
// tell VDM that it's getting the hardware.
|
|
//
|
|
|
|
RECT CursorRect;
|
|
CursorRect.left = -32767;
|
|
CursorRect.top = -32767;
|
|
CursorRect.right = 32767;
|
|
CursorRect.bottom = 32767;
|
|
NtUserConsoleControl(ConsoleSetVDMCursorBounds,
|
|
&CursorRect, sizeof(RECT));
|
|
|
|
//
|
|
// wait for vdm to say ok. We could initiate another switch
|
|
// (set hStartHardwareEvent which vdm is now waiting for to
|
|
// complete the handshaking) when we return (WM_FULLSCREEN
|
|
// could be in the message queue already). If we don't wait
|
|
// for vdm to get signaled here, the hStartHardwareEvent
|
|
// can get set twice and signaled once so the vdm will never
|
|
// gets the newly switch request we may post after return.
|
|
//
|
|
// Also, we can not simply continue if timeout or error occurs.
|
|
// We could set hStartHardwareEvent twice if a new mode switch
|
|
// request is in our queue. This will cause us and ntvdm
|
|
// out of sync.
|
|
//
|
|
|
|
li.QuadPart = (LONGLONG)-10000 * CONSOLE_VDM_TIMEOUT;
|
|
NtSetEvent(Console->VDMStartHardwareEvent, NULL);
|
|
Status = NtWaitForMultipleObjects(WaitCount, WaitHandles, WaitAny, FALSE, &li);
|
|
if (Status != 0) {
|
|
Console->Flags &= ~CONSOLE_FULLSCREEN_NOPAINT;
|
|
NtSetEvent(Console->VDMErrorHardwareEvent, NULL);
|
|
UnregisterVDM(Console);
|
|
KdPrint(("CONSRV: VDM not responding or in error state.\n"));
|
|
}
|
|
}
|
|
}
|
|
|
|
//
|
|
// let the app know that it has the focus.
|
|
//
|
|
|
|
HandleFocusEvent(Console, TRUE);
|
|
|
|
// unset palette
|
|
|
|
if (ScreenInfo->hPalette != NULL) {
|
|
SelectPalette(ScreenInfo->Console->hDC,
|
|
ScreenInfo->Console->hSysPalette,
|
|
FALSE);
|
|
UnsetActivePalette(ScreenInfo);
|
|
}
|
|
SetConsoleReserveKeys(Console->hWnd, Console->ReserveKeys);
|
|
HandleFocusEvent(Console, TRUE);
|
|
|
|
} else {
|
|
|
|
KdPrint((" CONSRV - Display Mode transition to windowed \n"));
|
|
|
|
// turn off mouse pointer so VDM doesn't see it when saving
|
|
// hardware
|
|
if (!(ScreenInfo->Flags & CONSOLE_GRAPHICS_BUFFER)) {
|
|
ReverseMousePointer(ScreenInfo, &ScreenInfo->Window);
|
|
}
|
|
|
|
|
|
Console->FullScreenFlags &= ~CONSOLE_FULLSCREEN_HARDWARE;
|
|
if (Console->Flags & CONSOLE_VDM_REGISTERED) {
|
|
|
|
//
|
|
// tell vdm that it's losing the hardware
|
|
//
|
|
|
|
li.QuadPart = (LONGLONG)-10000 * CONSOLE_VDM_TIMEOUT;
|
|
Status = NtSetEvent(Console->VDMStartHardwareEvent, NULL);
|
|
if (NT_SUCCESS(Status)) {
|
|
Status = NtWaitForMultipleObjects(WaitCount, WaitHandles, WaitAny, FALSE, &li);
|
|
if (Status == STATUS_TIMEOUT || Status == 1) {
|
|
Status = STATUS_UNSUCCESSFUL;
|
|
}
|
|
}
|
|
|
|
// if ntvdm didn't respond or we failed to save the video hardware
|
|
// states, kick ntvdm out of our world. The ntvdm process eventually
|
|
// would die but what choice do have here?
|
|
|
|
if (NT_SUCCESS(Status)) {
|
|
VIDEO_HARDWARE_STATE State;
|
|
ULONG StateSize = sizeof(State);
|
|
|
|
State.StateHeader = Console->StateBuffer;
|
|
State.StateLength = Console->StateLength;
|
|
|
|
|
|
Status = GdiFullscreenControl(FullscreenControlSaveHardwareState,
|
|
&State,
|
|
StateSize,
|
|
&State,
|
|
&StateSize);
|
|
}
|
|
|
|
if (NT_SUCCESS(Status)) {
|
|
NtUserConsoleControl(ConsoleSetVDMCursorBounds, NULL, 0);
|
|
|
|
// disconnect emulator and unmap video memory
|
|
|
|
UserAssert(Console->Flags & CONSOLE_CONNECTED_TO_EMULATOR);
|
|
ConnectToEmulator(FALSE, Console);
|
|
|
|
} else {
|
|
|
|
Console->Flags &= ~CONSOLE_FULLSCREEN_NOPAINT;
|
|
NtSetEvent(Console->VDMErrorHardwareEvent, NULL);
|
|
UnregisterVDM(Console);
|
|
if (Status != 0) {
|
|
KdPrint(("CONSRV: VDM not responding.\n"));
|
|
} else {
|
|
KdPrint(("CONSRV: Save Video States Failed\n"));
|
|
}
|
|
}
|
|
}
|
|
|
|
// tell VDM to map memory
|
|
|
|
if (Console->Flags & CONSOLE_VDM_REGISTERED) {
|
|
|
|
// make a special case for ntvdm during switching because
|
|
// ntvdm has to make console api calls. We don't want to
|
|
// unlock the console at this moment because as soon as
|
|
// we release the lock, other theads which are waiting
|
|
// for the lock will claim the lock and the ntvdm thread doing
|
|
// the screen switch will have to wait for the lock. In an
|
|
// extreme case, the following NtWaitForSingleObject will time
|
|
// out because the ntvdm may be still waiting for the lock.
|
|
// We keep this thing in a single global variable because
|
|
// there is only one process who can own the screen at any moment.
|
|
|
|
RtlEnterCriticalSection(&ConsoleVDMCriticalSection);
|
|
ConsoleVDMOnSwitching = Console;
|
|
RtlLeaveCriticalSection(&ConsoleVDMCriticalSection);
|
|
|
|
//
|
|
// Increase the regular timeout value, since ntvdm needs to
|
|
// do A LOT of transitions between kernel mode and user mode
|
|
// before it can signal console back. (During stress, it may
|
|
// take a long time.)
|
|
//
|
|
|
|
li.QuadPart = (LONGLONG)-10000 * CONSOLE_VDM_TIMEOUT * 3;
|
|
Status = NtSetEvent(Console->VDMStartHardwareEvent, NULL);
|
|
if (NT_SUCCESS(Status)) {
|
|
Status = NtWaitForMultipleObjects(WaitCount, WaitHandles, WaitAny, FALSE, &li);
|
|
}
|
|
|
|
// time to go back to normal
|
|
RtlEnterCriticalSection(&ConsoleVDMCriticalSection);
|
|
ConsoleVDMOnSwitching = NULL;
|
|
RtlLeaveCriticalSection(&ConsoleVDMCriticalSection);
|
|
|
|
if (Status != 0) {
|
|
Console->Flags &= ~CONSOLE_FULLSCREEN_NOPAINT;
|
|
NtSetEvent(Console->VDMErrorHardwareEvent, NULL);
|
|
UnregisterVDM(Console);
|
|
KdPrint(("CONSRV: VDM not responding. - second wait\n"));
|
|
return Status;
|
|
}
|
|
ScreenInfo = Console->CurrentScreenBuffer;
|
|
}
|
|
|
|
// set palette
|
|
|
|
if (ScreenInfo->hPalette != NULL) {
|
|
SelectPalette(ScreenInfo->Console->hDC,
|
|
ScreenInfo->hPalette,
|
|
FALSE);
|
|
SetActivePalette(ScreenInfo);
|
|
}
|
|
SetConsoleReserveKeys(Console->hWnd, CONSOLE_NOSHORTCUTKEY);
|
|
HandleFocusEvent(Console, FALSE);
|
|
|
|
}
|
|
|
|
/*
|
|
* Boost or lower the priority if we are going fullscreen or away.
|
|
*
|
|
* Note that console usually boosts and lowers its priority based on
|
|
* WM_FOCUS and WM_KILLFOCUS but when you switch to full screen the
|
|
* implementation actually sends a WM_KILLFOCUS so we reboost the
|
|
* correct console here.
|
|
*/
|
|
ModifyConsoleProcessFocus(Console, bForeground);
|
|
#else
|
|
UNREFERENCED_PARAMETER(bForeground);
|
|
UNREFERENCED_PARAMETER(Console);
|
|
UNREFERENCED_PARAMETER(ScreenInfo);
|
|
#endif
|
|
|
|
return STATUS_SUCCESS;
|
|
}
|
|
#if defined(_X86_)
|
|
|
|
BOOL
|
|
SetVideoMode(
|
|
IN PSCREEN_INFORMATION ScreenInfo
|
|
)
|
|
{
|
|
NTSTATUS Status;
|
|
UINT i, j;
|
|
|
|
#if defined(FE_SB)
|
|
//
|
|
// load RAM font
|
|
//
|
|
|
|
Status = SetRAMFontCodePage(ScreenInfo);
|
|
#endif
|
|
|
|
//
|
|
// load ROM font
|
|
//
|
|
|
|
Status = SetROMFontCodePage(ScreenInfo->Console->OutputCP,
|
|
ScreenInfo->BufferInfo.TextInfo.ModeIndex);
|
|
|
|
if (Status == STATUS_INVALID_PARAMETER) {
|
|
Status = SetROMFontCodePage(GetOEMCP(),
|
|
ScreenInfo->BufferInfo.TextInfo.ModeIndex);
|
|
|
|
if (Status == STATUS_INVALID_PARAMETER) {
|
|
Status = SetROMFontCodePage(CONSOLE_DEFAULT_ROM_FONT,
|
|
ScreenInfo->BufferInfo.TextInfo.ModeIndex);
|
|
}
|
|
}
|
|
|
|
//
|
|
// initialize palette
|
|
//
|
|
|
|
#if defined(FE_SB)
|
|
Status = GdiFullscreenControl(FullscreenControlSetPalette,
|
|
(PVOID) RegInitialPalette,
|
|
RegInitialPalette[0] * sizeof(USHORT) + sizeof(DWORD),
|
|
NULL,
|
|
NULL);
|
|
#else
|
|
Status = GdiFullscreenControl(FullscreenControlSetPalette,
|
|
(PVOID) &InitialPalette,
|
|
sizeof (InitialPalette),
|
|
NULL,
|
|
NULL);
|
|
#endif
|
|
|
|
if (Status != STATUS_SUCCESS) {
|
|
RIPMSG1(RIP_WARNING, "FullscreenControlSetPalette failed - Status = 0x%x",
|
|
Status);
|
|
return FALSE;
|
|
}
|
|
|
|
//
|
|
// initialize color table
|
|
//
|
|
|
|
#if defined(FE_SB)
|
|
if (RegColorBufferNoTranslate)
|
|
{
|
|
Status = GdiFullscreenControl(FullscreenControlSetColors,
|
|
(PVOID) RegColorBufferNoTranslate,
|
|
RegColorBufferNoTranslate[0] * sizeof(DWORD) + sizeof(DWORD),
|
|
NULL,
|
|
NULL);
|
|
}
|
|
else
|
|
{
|
|
for (i = 0, j = 4; i < 16; i++) {
|
|
RegColorBuffer[j++] = ((((GetRValue(ScreenInfo->Console->ColorTable[i]) +
|
|
0x2A) * 0x02) / 0x55) * 0x15) / 0x02;
|
|
RegColorBuffer[j++] = ((((GetGValue(ScreenInfo->Console->ColorTable[i]) +
|
|
0x2A) * 0x02) / 0x55) * 0x15) / 0x02;
|
|
RegColorBuffer[j++] = ((((GetBValue(ScreenInfo->Console->ColorTable[i]) +
|
|
0x2A) * 0x02) / 0x55) * 0x15) / 0x02;
|
|
RegColorBuffer[j++] = 0;
|
|
}
|
|
|
|
Status = GdiFullscreenControl(FullscreenControlSetColors,
|
|
(PVOID) RegColorBuffer,
|
|
RegColorBuffer[0] * sizeof(DWORD) + sizeof(DWORD),
|
|
NULL,
|
|
NULL);
|
|
}
|
|
#else
|
|
for (i = 0, j = 4; i < 16; i++) {
|
|
ColorBuffer[j++] = ((((GetRValue(ScreenInfo->Console->ColorTable[i]) +
|
|
0x2A) * 0x02) / 0x55) * 0x15) / 0x02;
|
|
ColorBuffer[j++] = ((((GetGValue(ScreenInfo->Console->ColorTable[i]) +
|
|
0x2A) * 0x02) / 0x55) * 0x15) / 0x02;
|
|
ColorBuffer[j++] = ((((GetBValue(ScreenInfo->Console->ColorTable[i]) +
|
|
0x2A) * 0x02) / 0x55) * 0x15) / 0x02;
|
|
ColorBuffer[j++] = 0;
|
|
}
|
|
|
|
Status = GdiFullscreenControl(FullscreenControlSetColors,
|
|
(PVOID) &ColorBuffer,
|
|
sizeof (ColorBuffer),
|
|
NULL,
|
|
NULL);
|
|
#endif
|
|
|
|
if (Status != STATUS_SUCCESS) {
|
|
RIPMSG1(RIP_WARNING, "FullscreenControlSetColors failed - Status = 0x%x",
|
|
Status);
|
|
return FALSE;
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
#endif
|
|
|
|
|
|
#if defined(_X86_)
|
|
|
|
NTSTATUS
|
|
ChangeDispSettings(
|
|
PCONSOLE_INFORMATION Console,
|
|
HWND hwnd,
|
|
DWORD dwFlags)
|
|
{
|
|
DEVMODEW Devmode;
|
|
ULONG Index;
|
|
CONSOLE_FULLSCREEN_SWITCH switchBlock;
|
|
|
|
if (dwFlags == CDS_FULLSCREEN) {
|
|
#if defined(FE_SB)
|
|
BOOL fGraphics = fFullScreenGraphics ? IsAvailableFsCodePage(Console->OutputCP) : FALSE;
|
|
#endif
|
|
|
|
Index = Console->CurrentScreenBuffer->BufferInfo.TextInfo.ModeIndex;
|
|
|
|
//
|
|
// set mode to go to full screen
|
|
//
|
|
|
|
ZeroMemory(&Devmode, sizeof(Devmode));
|
|
|
|
Devmode.dmSize = sizeof(Devmode);
|
|
Devmode.dmDriverExtra = 0;
|
|
Devmode.dmFields = DM_BITSPERPEL |
|
|
DM_PELSWIDTH |
|
|
DM_PELSHEIGHT |
|
|
DM_DISPLAYFLAGS;
|
|
|
|
Devmode.dmBitsPerPel = 4;
|
|
#if defined(FE_SB)
|
|
Devmode.dmPelsWidth = RegModeFontPairs[Index].Resolution.X;
|
|
Devmode.dmPelsHeight = RegModeFontPairs[Index].Resolution.Y;
|
|
Devmode.dmDisplayFlags = (fGraphics && (RegModeFontPairs[Index].Mode & FS_MODE_GRAPHICS)) ? 0 : DMDISPLAYFLAGS_TEXTMODE;
|
|
#else
|
|
Devmode.dmPelsWidth = ModeFontPairs[Index].Resolution.X;
|
|
Devmode.dmPelsHeight = ModeFontPairs[Index].Resolution.Y;
|
|
Devmode.dmDisplayFlags = DMDISPLAYFLAGS_TEXTMODE;
|
|
#endif
|
|
|
|
switchBlock.bFullscreenSwitch = TRUE;
|
|
switchBlock.hwnd = hwnd;
|
|
switchBlock.pNewMode = &Devmode;
|
|
|
|
} else {
|
|
switchBlock.bFullscreenSwitch = FALSE;
|
|
switchBlock.hwnd = hwnd;
|
|
switchBlock.pNewMode = NULL;
|
|
}
|
|
|
|
return NtUserConsoleControl(ConsoleFullscreenSwitch,
|
|
&switchBlock,
|
|
sizeof(CONSOLE_FULLSCREEN_SWITCH));
|
|
}
|
|
|
|
#endif
|
|
|
|
BOOL
|
|
InitializeFullScreen( VOID )
|
|
{
|
|
UNICODE_STRING vgaString;
|
|
DEVMODEW devmode;
|
|
ULONG i;
|
|
#ifdef FE_SB
|
|
DWORD mode1 = 0;
|
|
DWORD mode2 = 0;
|
|
#else
|
|
BOOLEAN mode1 = FALSE;
|
|
BOOLEAN mode2 = FALSE;
|
|
#endif
|
|
|
|
CHAR WindowsDir[CONSOLE_WINDOWS_DIR_LENGTH + CONSOLE_EGACPI_LENGTH];
|
|
UINT WindowsDirLength;
|
|
|
|
//
|
|
// query number of available modes
|
|
//
|
|
|
|
ZeroMemory(&devmode, sizeof(DEVMODEW));
|
|
devmode.dmSize = sizeof(DEVMODEW);
|
|
|
|
RtlInitUnicodeString(&vgaString, L"VGACOMPATIBLE");
|
|
|
|
DBGCHARS(("Number of modes = %d\n", NUMBER_OF_MODE_FONT_PAIRS));
|
|
|
|
for (i=0; ; i++)
|
|
{
|
|
DBGCHARS(("EnumDisplaySettings %d\n", i));
|
|
|
|
if (!(NT_SUCCESS(NtUserEnumDisplaySettings(&vgaString,
|
|
i,
|
|
&devmode,
|
|
0))))
|
|
{
|
|
break;
|
|
}
|
|
|
|
#if defined(FE_SB)
|
|
{
|
|
ULONG Index;
|
|
|
|
DBGCHARS(("Mode X = %d, Y = %d\n",
|
|
devmode.dmPelsWidth, devmode.dmPelsHeight));
|
|
|
|
for (Index=0;Index<NUMBER_OF_MODE_FONT_PAIRS;Index++)
|
|
{
|
|
if ((SHORT)devmode.dmPelsWidth == RegModeFontPairs[Index].Resolution.X &&
|
|
(SHORT)devmode.dmPelsHeight == RegModeFontPairs[Index].Resolution.Y )
|
|
{
|
|
if (devmode.dmDisplayFlags & DMDISPLAYFLAGS_TEXTMODE)
|
|
{
|
|
if (RegModeFontPairs[Index].Mode & FS_MODE_TEXT)
|
|
{
|
|
RegModeFontPairs[Index].Mode |= FS_MODE_FIND;
|
|
mode1++;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if (RegModeFontPairs[Index].Mode & FS_MODE_GRAPHICS)
|
|
{
|
|
RegModeFontPairs[Index].Mode |= FS_MODE_FIND;
|
|
mode2++;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
DBGCHARS(("mode1 = %d, mode2 = %d\n", mode1, mode2));
|
|
}
|
|
#else
|
|
|
|
if (devmode.dmPelsWidth == 720 &&
|
|
devmode.dmPelsHeight == 400)
|
|
{
|
|
mode1 = TRUE;
|
|
}
|
|
if (devmode.dmPelsWidth == 640 &&
|
|
devmode.dmPelsHeight == 350)
|
|
{
|
|
mode2 = TRUE;
|
|
}
|
|
#endif
|
|
}
|
|
|
|
#if !defined(FE_SB)
|
|
if (!(mode1 && mode2))
|
|
#else
|
|
if (mode1 < 2)
|
|
#endif
|
|
{
|
|
//
|
|
// One of the modes we expected to get was not returned.
|
|
// lets just fail fullscreen initialization.
|
|
//
|
|
|
|
KdPrint(("CONSRV: InitializeFullScreen Missing text mode\n"));
|
|
return FALSE;
|
|
}
|
|
|
|
#if defined(FE_SB)
|
|
if (mode2 > 0)
|
|
{
|
|
// Can do trun graphics mode.
|
|
fFullScreenGraphics = TRUE;
|
|
}
|
|
#endif
|
|
|
|
//
|
|
// open ega.cpi
|
|
//
|
|
|
|
WindowsDirLength = GetSystemDirectoryA(WindowsDir,
|
|
CONSOLE_WINDOWS_DIR_LENGTH);
|
|
if (WindowsDirLength == 0)
|
|
{
|
|
KdPrint(("CONSRV: InitializeFullScreen Finding Font file failed\n"));
|
|
return FALSE;
|
|
}
|
|
|
|
RtlCopyMemory(&WindowsDir[WindowsDirLength],
|
|
CONSOLE_EGACPI,
|
|
CONSOLE_EGACPI_LENGTH);
|
|
|
|
if ((hCPIFile = CreateFileA(WindowsDir,
|
|
GENERIC_READ,
|
|
FILE_SHARE_READ,
|
|
NULL,
|
|
OPEN_EXISTING,
|
|
0,
|
|
NULL)) == INVALID_HANDLE_VALUE)
|
|
{
|
|
KdPrint(("CONSRV: InitializeFullScreen Opening Font file failed\n"));
|
|
return FALSE;
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
|
|
ULONG
|
|
SrvGetConsoleHardwareState(
|
|
IN OUT PCSR_API_MSG m,
|
|
IN OUT PCSR_REPLY_STATUS ReplyStatus
|
|
)
|
|
{
|
|
#ifdef i386
|
|
PCONSOLE_GETHARDWARESTATE_MSG a = (PCONSOLE_GETHARDWARESTATE_MSG)&m->u.ApiMessageData;
|
|
NTSTATUS Status;
|
|
PCONSOLE_INFORMATION Console;
|
|
PHANDLE_DATA HandleData;
|
|
PSCREEN_INFORMATION ScreenInfo;
|
|
|
|
Status = ApiPreamble(a->ConsoleHandle,
|
|
&Console
|
|
);
|
|
if (!NT_SUCCESS(Status)) {
|
|
return Status;
|
|
}
|
|
Status = DereferenceIoHandle(CONSOLE_PERPROCESSDATA(),
|
|
a->OutputHandle,
|
|
CONSOLE_OUTPUT_HANDLE,
|
|
GENERIC_READ,
|
|
&HandleData
|
|
);
|
|
if (NT_SUCCESS(Status)) {
|
|
ScreenInfo = HandleData->Buffer.ScreenBuffer;
|
|
if (ScreenInfo->BufferInfo.TextInfo.ModeIndex == -1) {
|
|
UnlockConsole(Console);
|
|
return STATUS_UNSUCCESSFUL;
|
|
}
|
|
#if defined(FE_SB)
|
|
a->Resolution = RegModeFontPairs[ScreenInfo->BufferInfo.TextInfo.ModeIndex].Resolution;
|
|
a->FontSize = RegModeFontPairs[ScreenInfo->BufferInfo.TextInfo.ModeIndex].FontSize;
|
|
#else
|
|
a->Resolution = ModeFontPairs[ScreenInfo->BufferInfo.TextInfo.ModeIndex].Resolution;
|
|
a->FontSize = ModeFontPairs[ScreenInfo->BufferInfo.TextInfo.ModeIndex].FontSize;
|
|
#endif
|
|
}
|
|
UnlockConsole(Console);
|
|
return Status;
|
|
#else
|
|
return STATUS_UNSUCCESSFUL;
|
|
UNREFERENCED_PARAMETER(m);
|
|
#endif
|
|
UNREFERENCED_PARAMETER(ReplyStatus);
|
|
}
|
|
|
|
ULONG
|
|
SrvSetConsoleHardwareState(
|
|
IN OUT PCSR_API_MSG m,
|
|
IN OUT PCSR_REPLY_STATUS ReplyStatus
|
|
)
|
|
{
|
|
#ifdef i386
|
|
PCONSOLE_SETHARDWARESTATE_MSG a = (PCONSOLE_SETHARDWARESTATE_MSG)&m->u.ApiMessageData;
|
|
NTSTATUS Status;
|
|
PCONSOLE_INFORMATION Console;
|
|
PHANDLE_DATA HandleData;
|
|
PSCREEN_INFORMATION ScreenInfo;
|
|
ULONG Index;
|
|
|
|
Status = ApiPreamble(a->ConsoleHandle,
|
|
&Console
|
|
);
|
|
if (!NT_SUCCESS(Status)) {
|
|
return Status;
|
|
}
|
|
if (!(Console->FullScreenFlags & CONSOLE_FULLSCREEN_HARDWARE)) {
|
|
UnlockConsole(Console);
|
|
return STATUS_UNSUCCESSFUL;
|
|
}
|
|
Status = DereferenceIoHandle(CONSOLE_PERPROCESSDATA(),
|
|
a->OutputHandle,
|
|
CONSOLE_OUTPUT_HANDLE,
|
|
GENERIC_READ,
|
|
&HandleData
|
|
);
|
|
if (NT_SUCCESS(Status)) {
|
|
#if defined(FE_SB)
|
|
BOOL fGraphics = fFullScreenGraphics ? IsAvailableFsCodePage(Console->OutputCP) : FALSE;
|
|
#endif
|
|
ScreenInfo = HandleData->Buffer.ScreenBuffer;
|
|
|
|
// match requested mode
|
|
|
|
for (Index=0;Index<NUMBER_OF_MODE_FONT_PAIRS;Index++) {
|
|
#if defined(FE_SB)
|
|
if (a->Resolution.X == RegModeFontPairs[Index].Resolution.X &&
|
|
a->Resolution.Y == RegModeFontPairs[Index].Resolution.Y &&
|
|
a->FontSize.Y == RegModeFontPairs[Index].FontSize.Y &&
|
|
a->FontSize.X == RegModeFontPairs[Index].FontSize.X &&
|
|
( ( fGraphics && (RegModeFontPairs[Index].Mode & FS_GRAPHICS)==FS_GRAPHICS) ||
|
|
(!fGraphics && (RegModeFontPairs[Index].Mode & FS_TEXT)==FS_TEXT) )
|
|
) {
|
|
break;
|
|
}
|
|
#else
|
|
if (a->Resolution.X == ModeFontPairs[Index].Resolution.X &&
|
|
a->Resolution.Y == ModeFontPairs[Index].Resolution.Y &&
|
|
a->FontSize.Y == ModeFontPairs[Index].FontSize.Y &&
|
|
a->FontSize.X == ModeFontPairs[Index].FontSize.X) {
|
|
break;
|
|
}
|
|
#endif
|
|
}
|
|
if (Index == NUMBER_OF_MODE_FONT_PAIRS) {
|
|
Status = STATUS_INVALID_PARAMETER;
|
|
} else {
|
|
// set requested mode
|
|
ScreenInfo->BufferInfo.TextInfo.ModeIndex = Index;
|
|
SetVideoMode(ScreenInfo);
|
|
}
|
|
}
|
|
UnlockConsole(Console);
|
|
return Status;
|
|
#else
|
|
return STATUS_UNSUCCESSFUL;
|
|
UNREFERENCED_PARAMETER(m);
|
|
#endif
|
|
UNREFERENCED_PARAMETER(ReplyStatus);
|
|
}
|
|
|
|
ULONG
|
|
SrvGetConsoleDisplayMode(
|
|
IN OUT PCSR_API_MSG m,
|
|
IN OUT PCSR_REPLY_STATUS ReplyStatus
|
|
)
|
|
{
|
|
PCONSOLE_GETDISPLAYMODE_MSG a = (PCONSOLE_GETDISPLAYMODE_MSG)&m->u.ApiMessageData;
|
|
NTSTATUS Status;
|
|
PCONSOLE_INFORMATION Console;
|
|
|
|
Status = ApiPreamble(a->ConsoleHandle,
|
|
&Console
|
|
);
|
|
if (NT_SUCCESS(Status)) {
|
|
a->ModeFlags = Console->FullScreenFlags;
|
|
UnlockConsole(Console);
|
|
}
|
|
return Status;
|
|
UNREFERENCED_PARAMETER(ReplyStatus);
|
|
}
|
|
|
|
ULONG
|
|
SrvSetConsoleMenuClose(
|
|
IN OUT PCSR_API_MSG m,
|
|
IN OUT PCSR_REPLY_STATUS ReplyStatus
|
|
)
|
|
{
|
|
PCONSOLE_SETMENUCLOSE_MSG a = (PCONSOLE_SETMENUCLOSE_MSG)&m->u.ApiMessageData;
|
|
NTSTATUS Status;
|
|
PCONSOLE_INFORMATION Console;
|
|
|
|
UNREFERENCED_PARAMETER(ReplyStatus);
|
|
|
|
Status = ApiPreamble(a->ConsoleHandle,
|
|
&Console
|
|
);
|
|
if (!NT_SUCCESS(Status)) {
|
|
return Status;
|
|
}
|
|
|
|
if (a->Enable) {
|
|
Console->Flags &= ~CONSOLE_DISABLE_CLOSE;
|
|
} else {
|
|
Console->Flags |= CONSOLE_DISABLE_CLOSE;
|
|
}
|
|
|
|
UnlockConsole(Console);
|
|
return Status;
|
|
}
|
|
|
|
|
|
DWORD
|
|
ConvertHotKey(
|
|
IN LPAPPKEY UserAppKey
|
|
)
|
|
{
|
|
DWORD wParam;
|
|
|
|
wParam = MapVirtualKey(UserAppKey->ScanCode,1);
|
|
if (UserAppKey->Modifier & CONSOLE_MODIFIER_SHIFT) {
|
|
wParam |= 0x0100;
|
|
}
|
|
if (UserAppKey->Modifier & CONSOLE_MODIFIER_CONTROL) {
|
|
wParam |= 0x0200;
|
|
}
|
|
if (UserAppKey->Modifier & CONSOLE_MODIFIER_ALT) {
|
|
wParam |= 0x0400;
|
|
}
|
|
return wParam;
|
|
}
|
|
|
|
ULONG
|
|
SrvSetConsoleKeyShortcuts(
|
|
IN OUT PCSR_API_MSG m,
|
|
IN OUT PCSR_REPLY_STATUS ReplyStatus
|
|
)
|
|
{
|
|
PCONSOLE_SETKEYSHORTCUTS_MSG a = (PCONSOLE_SETKEYSHORTCUTS_MSG)&m->u.ApiMessageData;
|
|
NTSTATUS Status;
|
|
PCONSOLE_INFORMATION Console;
|
|
|
|
UNREFERENCED_PARAMETER(ReplyStatus);
|
|
|
|
Status = ApiPreamble(a->ConsoleHandle,
|
|
&Console
|
|
);
|
|
if (!NT_SUCCESS(Status)) {
|
|
return Status;
|
|
}
|
|
|
|
/*
|
|
* We only call CsrCaptureMessageBuffer() in SetConsoleKeyShortcuts if
|
|
* a->NumAppKeys != 0, so we only want to call CsrValidateMessageBuffer when
|
|
* that condition's true.
|
|
*/
|
|
if (a->NumAppKeys != 0 && !CsrValidateMessageBuffer(m, &a->AppKeys, a->NumAppKeys, sizeof(*a->AppKeys))) {
|
|
UnlockConsole(Console);
|
|
return STATUS_INVALID_PARAMETER;
|
|
}
|
|
|
|
if (a->NumAppKeys <= CONSOLE_MAX_APP_SHORTCUTS) {
|
|
Console->ReserveKeys = a->ReserveKeys;
|
|
if (Console->Flags & CONSOLE_HAS_FOCUS) {
|
|
if (!(SetConsoleReserveKeys(Console->hWnd, a->ReserveKeys))) {
|
|
Status = STATUS_INVALID_PARAMETER;
|
|
}
|
|
}
|
|
if (a->NumAppKeys) {
|
|
PostMessage(Console->hWnd,
|
|
WM_SETHOTKEY,
|
|
ConvertHotKey(a->AppKeys),
|
|
0
|
|
);
|
|
}
|
|
} else {
|
|
Status = STATUS_INVALID_PARAMETER;
|
|
}
|
|
|
|
UnlockConsole(Console);
|
|
return Status;
|
|
}
|
|
|
|
#ifdef i386
|
|
ULONG
|
|
MatchWindowSize(
|
|
#if defined(FE_SB)
|
|
IN UINT CodePage,
|
|
#endif
|
|
IN COORD WindowSize,
|
|
OUT PCOORD pWindowSize
|
|
)
|
|
|
|
/*++
|
|
|
|
find the best match font. it's the one that's the same size
|
|
or slightly larger than the window size.
|
|
|
|
--*/
|
|
{
|
|
ULONG i;
|
|
#if defined(FE_SB)
|
|
BOOL fGraphics = fFullScreenGraphics ? IsAvailableFsCodePage(CodePage) : FALSE;
|
|
#endif
|
|
|
|
for (i=0;i<NUMBER_OF_MODE_FONT_PAIRS;i++) {
|
|
#if defined(FE_SB)
|
|
if (WindowSize.Y <= RegModeFontPairs[i].ScreenSize.Y &&
|
|
( ( fGraphics && (RegModeFontPairs[i].Mode & FS_GRAPHICS)==FS_GRAPHICS) ||
|
|
(!fGraphics && (RegModeFontPairs[i].Mode & FS_TEXT)==FS_TEXT) )
|
|
)
|
|
#else
|
|
if (WindowSize.Y <= (SHORT)ModeFontPairs[i].Height)
|
|
#endif
|
|
{
|
|
break;
|
|
}
|
|
}
|
|
if (i == NUMBER_OF_MODE_FONT_PAIRS)
|
|
#if defined(FE_SB)
|
|
{
|
|
DWORD Find;
|
|
ULONG FindIndex;
|
|
COORD WindowSizeDelta;
|
|
|
|
FindIndex = 0;
|
|
Find = (DWORD)-1;
|
|
for (i=0; i<NUMBER_OF_MODE_FONT_PAIRS;i++) {
|
|
if ( ( fGraphics && (RegModeFontPairs[i].Mode & FS_GRAPHICS)==FS_GRAPHICS) ||
|
|
(!fGraphics && (RegModeFontPairs[i].Mode & FS_TEXT)==FS_TEXT) )
|
|
{
|
|
WindowSizeDelta.Y = (SHORT) abs(WindowSize.Y - RegModeFontPairs[i].ScreenSize.Y);
|
|
if (Find > (DWORD)(WindowSizeDelta.Y))
|
|
{
|
|
Find = (DWORD)(WindowSizeDelta.Y);
|
|
FindIndex = i;
|
|
}
|
|
}
|
|
}
|
|
|
|
i = FindIndex;
|
|
}
|
|
#else
|
|
i-=1;
|
|
#endif
|
|
#if defined(FE_SB)
|
|
*pWindowSize = RegModeFontPairs[i].ScreenSize;
|
|
#else
|
|
pWindowSize->X = 80;
|
|
pWindowSize->Y = (SHORT)ModeFontPairs[i].Height;
|
|
#endif
|
|
return i;
|
|
}
|
|
|
|
VOID
|
|
ReadRegionFromScreenHW(
|
|
IN PSCREEN_INFORMATION ScreenInfo,
|
|
IN PSMALL_RECT Region,
|
|
IN PCHAR_INFO ReadBufPtr
|
|
)
|
|
{
|
|
ULONG CurFrameBufPtr; // offset in frame buffer
|
|
SHORT FrameY;
|
|
SHORT WindowY, WindowX, WindowSizeX;
|
|
|
|
//
|
|
// get pointer to start of region in frame buffer
|
|
//
|
|
|
|
WindowY = Region->Top - ScreenInfo->Window.Top;
|
|
WindowX = Region->Left - ScreenInfo->Window.Left;
|
|
WindowSizeX = CONSOLE_WINDOW_SIZE_X(ScreenInfo);
|
|
|
|
//
|
|
// copy the chars and attrs from the frame buffer
|
|
//
|
|
|
|
for (FrameY = Region->Top;
|
|
FrameY <= Region->Bottom;
|
|
FrameY++, WindowY++) {
|
|
|
|
CurFrameBufPtr = SCREEN_BUFFER_POINTER(WindowX,
|
|
WindowY,
|
|
WindowSizeX,
|
|
sizeof(VGA_CHAR));
|
|
|
|
GdiFullscreenControl(FullscreenControlReadFromFrameBuffer,
|
|
(PULONG) CurFrameBufPtr,
|
|
(Region->Right - Region->Left + 1) *
|
|
sizeof(VGA_CHAR),
|
|
ReadBufPtr, NULL);
|
|
ReadBufPtr += (Region->Right - Region->Left + 1);
|
|
}
|
|
}
|
|
|
|
VOID
|
|
ReverseMousePointer(
|
|
IN PSCREEN_INFORMATION ScreenInfo,
|
|
IN PSMALL_RECT Region
|
|
)
|
|
{
|
|
ULONG CurFrameBufPtr; // offset in frame buffer
|
|
SHORT WindowSizeX;
|
|
|
|
#ifdef FE_SB
|
|
// fail safe
|
|
UserAssert(ScreenInfo->Flags & CONSOLE_TEXTMODE_BUFFER);
|
|
if (!(ScreenInfo->Flags & CONSOLE_TEXTMODE_BUFFER)) {
|
|
return;
|
|
}
|
|
#endif
|
|
|
|
WindowSizeX = CONSOLE_WINDOW_SIZE_X(ScreenInfo);
|
|
|
|
if (ScreenInfo->BufferInfo.TextInfo.MousePosition.X < Region->Left ||
|
|
ScreenInfo->BufferInfo.TextInfo.MousePosition.X > Region->Right ||
|
|
ScreenInfo->BufferInfo.TextInfo.MousePosition.Y < Region->Top ||
|
|
ScreenInfo->BufferInfo.TextInfo.MousePosition.Y > Region->Bottom ||
|
|
ScreenInfo->CursorDisplayCount < 0 ||
|
|
!(ScreenInfo->Console->InputBuffer.InputMode & ENABLE_MOUSE_INPUT) ||
|
|
ScreenInfo->Console->Flags & CONSOLE_VDM_REGISTERED) {
|
|
return;
|
|
}
|
|
|
|
#if defined(FE_SB)
|
|
{
|
|
FSVIDEO_REVERSE_MOUSE_POINTER MousePointer;
|
|
SHORT RowIndex;
|
|
PROW Row;
|
|
COORD TargetPoint;
|
|
|
|
TargetPoint.X = ScreenInfo->BufferInfo.TextInfo.MousePosition.X;
|
|
TargetPoint.Y = ScreenInfo->BufferInfo.TextInfo.MousePosition.Y;
|
|
|
|
RowIndex = (ScreenInfo->BufferInfo.TextInfo.FirstRow+TargetPoint.Y) % ScreenInfo->ScreenBufferSize.Y;
|
|
Row = &ScreenInfo->BufferInfo.TextInfo.Rows[RowIndex];
|
|
if (!CONSOLE_IS_DBCS_CP(ScreenInfo->Console))
|
|
MousePointer.dwType = CHAR_TYPE_SBCS;
|
|
else if (Row->CharRow.KAttrs[TargetPoint.X] & ATTR_TRAILING_BYTE)
|
|
MousePointer.dwType = CHAR_TYPE_TRAILING;
|
|
else if (Row->CharRow.KAttrs[TargetPoint.X] & ATTR_LEADING_BYTE)
|
|
MousePointer.dwType = CHAR_TYPE_LEADING;
|
|
else
|
|
MousePointer.dwType = CHAR_TYPE_SBCS;
|
|
|
|
MousePointer.Screen.Position.X = TargetPoint.X - ScreenInfo->Window.Left;
|
|
MousePointer.Screen.Position.Y = TargetPoint.Y - ScreenInfo->Window.Top;
|
|
MousePointer.Screen.ScreenSize.X = WindowSizeX;
|
|
MousePointer.Screen.ScreenSize.Y = CONSOLE_WINDOW_SIZE_Y(ScreenInfo);
|
|
MousePointer.Screen.nNumberOfChars = 0;
|
|
|
|
GdiFullscreenControl(FullscreenControlReverseMousePointerDB,
|
|
&MousePointer,
|
|
sizeof(MousePointer),
|
|
NULL,
|
|
NULL);
|
|
|
|
UNREFERENCED_PARAMETER(CurFrameBufPtr);
|
|
}
|
|
#else
|
|
CurFrameBufPtr = SCREEN_BUFFER_POINTER(ScreenInfo->BufferInfo.TextInfo.MousePosition.X - ScreenInfo->Window.Left,
|
|
ScreenInfo->BufferInfo.TextInfo.MousePosition.Y - ScreenInfo->Window.Top,
|
|
WindowSizeX,
|
|
sizeof(VGA_CHAR));
|
|
|
|
GdiFullscreenControl(FullscreenControlReverseMousePointer,
|
|
(PULONG)CurFrameBufPtr,
|
|
0,
|
|
NULL,
|
|
NULL);
|
|
#endif
|
|
}
|
|
|
|
VOID
|
|
CopyVideoMemory(
|
|
SHORT SourceY,
|
|
SHORT TargetY,
|
|
SHORT Length,
|
|
IN PSCREEN_INFORMATION ScreenInfo
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
This routine copies rows of characters in video memory. It only copies
|
|
complete rows.
|
|
|
|
Arguments:
|
|
|
|
SourceY - Row to copy from.
|
|
|
|
TargetY - Row to copy to.
|
|
|
|
Length - Number of rows to copy.
|
|
|
|
Return Value:
|
|
|
|
--*/
|
|
|
|
{
|
|
ULONG SourcePtr, TargetPtr;
|
|
SHORT WindowSizeX, WindowSizeY;
|
|
|
|
WindowSizeX = CONSOLE_WINDOW_SIZE_X(ScreenInfo);
|
|
WindowSizeY = CONSOLE_WINDOW_SIZE_Y(ScreenInfo);
|
|
|
|
if (max(SourceY, TargetY) + Length > WindowSizeY) {
|
|
Length = WindowSizeY - max(SourceY, TargetY);
|
|
if (Length <= 0 ) {
|
|
return;
|
|
}
|
|
}
|
|
|
|
#if defined(FE_SB)
|
|
{
|
|
FSCNTL_SCREEN_INFO FsCntlSrc;
|
|
FSCNTL_SCREEN_INFO FsCntlDest;
|
|
|
|
FsCntlSrc.Position.X = 0;
|
|
FsCntlSrc.Position.Y = SourceY;
|
|
FsCntlSrc.ScreenSize.X = WindowSizeX;
|
|
FsCntlSrc.ScreenSize.Y = CONSOLE_WINDOW_SIZE_Y(ScreenInfo);
|
|
FsCntlSrc.nNumberOfChars = Length * WindowSizeX;
|
|
|
|
FsCntlDest.Position.X = 0;
|
|
FsCntlDest.Position.Y = TargetY;
|
|
FsCntlDest.ScreenSize.X = WindowSizeX;
|
|
FsCntlDest.ScreenSize.Y = CONSOLE_WINDOW_SIZE_Y(ScreenInfo);
|
|
FsCntlDest.nNumberOfChars = Length * WindowSizeX;
|
|
|
|
GdiFullscreenControl(FullscreenControlCopyFrameBufferDB,
|
|
&FsCntlSrc,
|
|
sizeof(FsCntlSrc),
|
|
&FsCntlDest,
|
|
(PULONG)sizeof(FsCntlDest));
|
|
|
|
UNREFERENCED_PARAMETER(SourcePtr);
|
|
UNREFERENCED_PARAMETER(TargetPtr);
|
|
}
|
|
#else
|
|
SourcePtr = SCREEN_BUFFER_POINTER(0,
|
|
SourceY,
|
|
WindowSizeX,
|
|
sizeof(VGA_CHAR));
|
|
|
|
TargetPtr = SCREEN_BUFFER_POINTER(0,
|
|
TargetY,
|
|
WindowSizeX,
|
|
sizeof(VGA_CHAR));
|
|
|
|
GdiFullscreenControl(FullscreenControlCopyFrameBuffer,
|
|
(PULONG) SourcePtr,
|
|
Length * WindowSizeX * sizeof(VGA_CHAR),
|
|
(PULONG) TargetPtr,
|
|
(PULONG) (Length * WindowSizeX * sizeof(VGA_CHAR)));
|
|
#endif
|
|
}
|
|
|
|
VOID
|
|
ScrollHW(
|
|
IN PSCREEN_INFORMATION ScreenInfo,
|
|
IN PSMALL_RECT ScrollRect,
|
|
IN PSMALL_RECT MergeRect,
|
|
IN COORD TargetPoint
|
|
)
|
|
{
|
|
SMALL_RECT TargetRectangle;
|
|
if (ScreenInfo->Console->Flags & CONSOLE_VDM_REGISTERED)
|
|
return;
|
|
|
|
TargetRectangle.Left = TargetPoint.X;
|
|
TargetRectangle.Top = TargetPoint.Y;
|
|
TargetRectangle.Right = TargetPoint.X + ScrollRect->Right - ScrollRect->Left;
|
|
TargetRectangle.Bottom = TargetPoint.Y + ScrollRect->Bottom - ScrollRect->Top;
|
|
|
|
//
|
|
// if the scroll region is as wide as the screen, we can update
|
|
// the screen by copying the video memory. if we scroll this
|
|
// way, we then must clip and update the fill region.
|
|
//
|
|
|
|
if (ScrollRect->Left == ScreenInfo->Window.Left &&
|
|
TargetRectangle.Left == ScreenInfo->Window.Left &&
|
|
ScrollRect->Right == ScreenInfo->Window.Right &&
|
|
TargetRectangle.Right == ScreenInfo->Window.Right &&
|
|
ScrollRect->Top >= ScreenInfo->Window.Top &&
|
|
TargetRectangle.Top >= ScreenInfo->Window.Top &&
|
|
ScrollRect->Bottom <= ScreenInfo->Window.Bottom &&
|
|
TargetRectangle.Bottom <= ScreenInfo->Window.Bottom) {
|
|
|
|
//
|
|
// we must first make the mouse pointer invisible because
|
|
// otherwise it would get copied to another place on the
|
|
// screen if it were part of the scroll region.
|
|
//
|
|
|
|
ReverseMousePointer(ScreenInfo, &ScreenInfo->Window);
|
|
|
|
CopyVideoMemory((SHORT) (ScrollRect->Top - ScreenInfo->Window.Top),
|
|
(SHORT) (TargetRectangle.Top - ScreenInfo->Window.Top),
|
|
(SHORT) (TargetRectangle.Bottom - TargetRectangle.Top + 1),
|
|
ScreenInfo);
|
|
|
|
//
|
|
// update the fill region. first we ensure that the scroll and
|
|
// target regions aren't the same. if they are, we don't fill.
|
|
//
|
|
|
|
if (TargetRectangle.Top != ScrollRect->Top) {
|
|
|
|
//
|
|
// if scroll and target regions overlap, with scroll
|
|
// region above target region, clip scroll region.
|
|
//
|
|
|
|
if (TargetRectangle.Top <= ScrollRect->Bottom &&
|
|
TargetRectangle.Bottom >= ScrollRect->Bottom) {
|
|
ScrollRect->Bottom = (SHORT)(TargetRectangle.Top-1);
|
|
}
|
|
else if (TargetRectangle.Top <= ScrollRect->Top &&
|
|
TargetRectangle.Bottom >= ScrollRect->Top) {
|
|
ScrollRect->Top = (SHORT)(TargetRectangle.Bottom+1);
|
|
}
|
|
WriteToScreen(ScreenInfo, ScrollRect);
|
|
|
|
//
|
|
// WriteToScreen should take care of writing the mouse pointer.
|
|
// however, the update region may be clipped so that the
|
|
// mouse pointer is not written. in that case, we draw the
|
|
// mouse pointer here.
|
|
//
|
|
|
|
if (ScreenInfo->BufferInfo.TextInfo.MousePosition.Y < ScrollRect->Top ||
|
|
ScreenInfo->BufferInfo.TextInfo.MousePosition.Y > ScrollRect->Bottom) {
|
|
ReverseMousePointer(ScreenInfo, &ScreenInfo->Window);
|
|
}
|
|
}
|
|
if (MergeRect) {
|
|
WriteToScreen(ScreenInfo, MergeRect);
|
|
}
|
|
}
|
|
else {
|
|
if (MergeRect) {
|
|
WriteToScreen(ScreenInfo, MergeRect);
|
|
}
|
|
WriteToScreen(ScreenInfo, ScrollRect);
|
|
WriteToScreen(ScreenInfo, &TargetRectangle);
|
|
}
|
|
}
|
|
|
|
VOID
|
|
UpdateMousePosition(
|
|
PSCREEN_INFORMATION ScreenInfo,
|
|
COORD Position
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
This routine moves the mouse pointer.
|
|
|
|
Arguments:
|
|
|
|
ScreenInfo - Pointer to screen buffer information.
|
|
|
|
Position - Contains the new position of the mouse in screen buffer
|
|
coordinates.
|
|
|
|
Return Value:
|
|
|
|
none.
|
|
|
|
--*/
|
|
|
|
// Note: CurrentConsole lock must be held in share mode when calling this routine
|
|
{
|
|
SMALL_RECT CursorRegion;
|
|
#ifdef FE_SB
|
|
SHORT RowIndex;
|
|
PROW Row;
|
|
BOOL fOneMore = FALSE;
|
|
#endif
|
|
|
|
if ((ScreenInfo->Console->Flags & CONSOLE_VDM_REGISTERED) ||
|
|
(ScreenInfo->Flags & CONSOLE_GRAPHICS_BUFFER)) {
|
|
return;
|
|
}
|
|
|
|
if (Position.X < ScreenInfo->Window.Left ||
|
|
Position.X > ScreenInfo->Window.Right ||
|
|
Position.Y < ScreenInfo->Window.Top ||
|
|
Position.Y > ScreenInfo->Window.Bottom) {
|
|
return;
|
|
}
|
|
|
|
if (Position.X == ScreenInfo->BufferInfo.TextInfo.MousePosition.X &&
|
|
Position.Y == ScreenInfo->BufferInfo.TextInfo.MousePosition.Y) {
|
|
return;
|
|
}
|
|
|
|
#ifdef FE_SB
|
|
if (CONSOLE_IS_DBCS_CP(ScreenInfo->Console)) {
|
|
RowIndex = (ScreenInfo->BufferInfo.TextInfo.FirstRow+Position.Y) % ScreenInfo->ScreenBufferSize.Y;
|
|
Row = &ScreenInfo->BufferInfo.TextInfo.Rows[RowIndex];
|
|
if (Row->CharRow.KAttrs[Position.X] & ATTR_LEADING_BYTE) {
|
|
if (Position.X != ScreenInfo->ScreenBufferSize.X - 1) {
|
|
fOneMore = TRUE;
|
|
}
|
|
} else if (Row->CharRow.KAttrs[Position.X] & ATTR_TRAILING_BYTE) {
|
|
if (Position.X != 0) {
|
|
fOneMore = TRUE;
|
|
Position.X--;
|
|
}
|
|
}
|
|
|
|
}
|
|
#endif
|
|
|
|
if (ScreenInfo->CursorDisplayCount < 0 || !(ScreenInfo->Console->InputBuffer.InputMode & ENABLE_MOUSE_INPUT)) {
|
|
ScreenInfo->BufferInfo.TextInfo.MousePosition = Position;
|
|
return;
|
|
}
|
|
|
|
|
|
// turn off old mouse position.
|
|
|
|
CursorRegion.Left = CursorRegion.Right = ScreenInfo->BufferInfo.TextInfo.MousePosition.X;
|
|
CursorRegion.Top = CursorRegion.Bottom = ScreenInfo->BufferInfo.TextInfo.MousePosition.Y;
|
|
|
|
#ifdef FE_SB
|
|
if (CONSOLE_IS_DBCS_CP(ScreenInfo->Console)) {
|
|
RowIndex = (ScreenInfo->BufferInfo.TextInfo.FirstRow+CursorRegion.Top) % ScreenInfo->ScreenBufferSize.Y;
|
|
Row = &ScreenInfo->BufferInfo.TextInfo.Rows[RowIndex];
|
|
if (Row->CharRow.KAttrs[CursorRegion.Left] & ATTR_LEADING_BYTE) {
|
|
if (CursorRegion.Left != ScreenInfo->ScreenBufferSize.X - 1) {
|
|
CursorRegion.Right++;
|
|
}
|
|
}
|
|
}
|
|
#endif
|
|
|
|
// store new mouse position
|
|
|
|
ScreenInfo->BufferInfo.TextInfo.MousePosition.X = Position.X;
|
|
ScreenInfo->BufferInfo.TextInfo.MousePosition.Y = Position.Y;
|
|
WriteToScreen(ScreenInfo, &CursorRegion);
|
|
|
|
// turn on new mouse position
|
|
|
|
CursorRegion.Left = CursorRegion.Right = Position.X;
|
|
CursorRegion.Top = CursorRegion.Bottom = Position.Y;
|
|
#ifdef FE_SB
|
|
if (fOneMore)
|
|
CursorRegion.Right++;
|
|
#endif
|
|
WriteToScreen(ScreenInfo, &CursorRegion);
|
|
}
|
|
|
|
NTSTATUS
|
|
SetROMFontCodePage(
|
|
IN UINT wCodePage,
|
|
IN ULONG ModeIndex
|
|
)
|
|
|
|
/*
|
|
|
|
this function opens ega.cpi and looks for the desired font in the
|
|
specified codepage. if found, it loads it into the video ROM.
|
|
|
|
*/
|
|
|
|
{
|
|
BYTE Buffer[CONSOLE_FONT_BUFFER_LENGTH];
|
|
DWORD dwBytesRead;
|
|
LPFONTFILEHEADER lpFontFileHeader=(LPFONTFILEHEADER)Buffer;
|
|
LPFONTINFOHEADER lpFontInfoHeader=(LPFONTINFOHEADER)Buffer;
|
|
LPFONTDATAHEADER lpFontDataHeader=(LPFONTDATAHEADER)Buffer;
|
|
LPCPENTRYHEADER lpCPEntryHeader=(LPCPENTRYHEADER)Buffer;
|
|
LPSCREENFONTHEADER lpScreenFontHeader=(LPSCREENFONTHEADER)Buffer;
|
|
WORD NumEntries;
|
|
COORD FontDimensions;
|
|
NTSTATUS Status;
|
|
BOOL Found;
|
|
LONG FilePtr;
|
|
BOOL bDOS = FALSE;
|
|
|
|
FontDimensions = ModeFontPairs[ModeIndex].FontSize;
|
|
|
|
//
|
|
// read FONTINFOHEADER
|
|
//
|
|
// do {
|
|
// read CPENTRYHEADER
|
|
// if (correct codepage)
|
|
// break;
|
|
// } while (codepages)
|
|
// if (codepage found)
|
|
// read FONTDATAHEADER
|
|
//
|
|
|
|
// read FONTFILEHEADER
|
|
|
|
FilePtr = 0;
|
|
if (SetFilePointer(hCPIFile, FilePtr, NULL, FILE_BEGIN) == -1) {
|
|
Status = STATUS_INVALID_PARAMETER;
|
|
goto DoExit;
|
|
}
|
|
|
|
if (!ReadFile(hCPIFile, Buffer, sizeof(FONTFILEHEADER), &dwBytesRead, NULL) ||
|
|
dwBytesRead != sizeof(FONTFILEHEADER)) {
|
|
Status = STATUS_INVALID_PARAMETER;
|
|
goto DoExit;
|
|
}
|
|
|
|
// verify signature
|
|
|
|
if (memcmp(lpFontFileHeader->ffhFileTag, "\xFF""FONT.NT",8) ) {
|
|
if (memcmp(lpFontFileHeader->ffhFileTag, "\xFF""FONT ",8) ) {
|
|
Status = STATUS_INVALID_PARAMETER;
|
|
goto DoExit;
|
|
} else {
|
|
bDOS = TRUE;
|
|
}
|
|
}
|
|
|
|
// seek to FONTINFOHEADER. jump through hoops to get the offset value.
|
|
|
|
FilePtr = lpFontFileHeader->ffhOffset1;
|
|
FilePtr |= (lpFontFileHeader->ffhOffset2 << 8);
|
|
FilePtr |= (lpFontFileHeader->ffhOffset3 << 24);
|
|
|
|
if (SetFilePointer(hCPIFile, FilePtr, NULL, FILE_BEGIN) == -1) {
|
|
Status = STATUS_INVALID_PARAMETER;
|
|
goto DoExit;
|
|
}
|
|
|
|
// read FONTINFOHEADER
|
|
|
|
if (!ReadFile(hCPIFile, Buffer, sizeof(FONTINFOHEADER), &dwBytesRead, NULL) ||
|
|
dwBytesRead != sizeof(FONTINFOHEADER)) {
|
|
Status = STATUS_INVALID_PARAMETER;
|
|
goto DoExit;
|
|
}
|
|
FilePtr += dwBytesRead;
|
|
NumEntries = lpFontInfoHeader->fihCodePages;
|
|
|
|
Found = FALSE;
|
|
while (NumEntries &&
|
|
ReadFile(hCPIFile, Buffer, sizeof(CPENTRYHEADER), &dwBytesRead, NULL) &&
|
|
dwBytesRead == sizeof(CPENTRYHEADER)) {
|
|
if (lpCPEntryHeader->cpeCodepageID == wCodePage) {
|
|
Found = TRUE;
|
|
break;
|
|
}
|
|
// seek to next CPEENTRYHEADER
|
|
|
|
if (bDOS) {
|
|
FilePtr = MAKELONG(lpCPEntryHeader->cpeNext1, lpCPEntryHeader->cpeNext2);
|
|
} else {
|
|
FilePtr += MAKELONG(lpCPEntryHeader->cpeNext1, lpCPEntryHeader->cpeNext2);
|
|
}
|
|
if (SetFilePointer(hCPIFile, FilePtr, NULL, FILE_BEGIN) == -1) {
|
|
Status = STATUS_INVALID_PARAMETER;
|
|
goto DoExit;
|
|
}
|
|
NumEntries -= 1;
|
|
}
|
|
if (!Found) {
|
|
Status = STATUS_INVALID_PARAMETER;
|
|
goto DoExit;
|
|
}
|
|
|
|
// seek to FONTDATAHEADER
|
|
|
|
if (bDOS) {
|
|
FilePtr = lpCPEntryHeader->cpeOffset;
|
|
} else {
|
|
FilePtr += lpCPEntryHeader->cpeOffset;
|
|
}
|
|
if (SetFilePointer(hCPIFile, FilePtr, NULL, FILE_BEGIN) == -1) {
|
|
Status = STATUS_INVALID_PARAMETER;
|
|
goto DoExit;
|
|
}
|
|
|
|
// read FONTDATAHEADER
|
|
|
|
if (!ReadFile(hCPIFile, Buffer, sizeof(FONTDATAHEADER), &dwBytesRead, NULL) ||
|
|
dwBytesRead != sizeof(FONTDATAHEADER)) {
|
|
Status = STATUS_INVALID_PARAMETER;
|
|
goto DoExit;
|
|
}
|
|
FilePtr += dwBytesRead;
|
|
|
|
NumEntries = lpFontDataHeader->fdhFonts;
|
|
|
|
while (NumEntries) {
|
|
if (!ReadFile(hCPIFile, Buffer, sizeof(SCREENFONTHEADER), &dwBytesRead, NULL) ||
|
|
dwBytesRead != sizeof(SCREENFONTHEADER)) {
|
|
Status = STATUS_INVALID_PARAMETER;
|
|
goto DoExit;
|
|
}
|
|
|
|
if (lpScreenFontHeader->sfhHeight == (BYTE)FontDimensions.Y &&
|
|
lpScreenFontHeader->sfhWidth == (BYTE)FontDimensions.X) {
|
|
PVIDEO_LOAD_FONT_INFORMATION FontInformation;
|
|
|
|
FontInformation = ConsoleHeapAlloc(TMP_TAG,
|
|
lpScreenFontHeader->sfhCharacters *
|
|
lpScreenFontHeader->sfhHeight +
|
|
sizeof(VIDEO_LOAD_FONT_INFORMATION));
|
|
if (FontInformation == NULL) {
|
|
RIPMSG1(RIP_WARNING, "SetROMFontCodePage: failed to memory allocation %d bytes",
|
|
lpScreenFontHeader->sfhCharacters * lpScreenFontHeader->sfhHeight +
|
|
sizeof(VIDEO_LOAD_FONT_INFORMATION));
|
|
return STATUS_NO_MEMORY;
|
|
}
|
|
if (!ReadFile(hCPIFile, FontInformation->Font,
|
|
lpScreenFontHeader->sfhCharacters*lpScreenFontHeader->sfhHeight,
|
|
&dwBytesRead, NULL) ||
|
|
dwBytesRead != (DWORD)(lpScreenFontHeader->sfhCharacters*lpScreenFontHeader->sfhHeight)) {
|
|
ConsoleHeapFree(FontInformation);
|
|
return STATUS_INVALID_PARAMETER;
|
|
}
|
|
FontInformation->WidthInPixels = FontDimensions.X;
|
|
FontInformation->HeightInPixels = FontDimensions.Y;
|
|
FontInformation->FontSize = lpScreenFontHeader->sfhCharacters*lpScreenFontHeader->sfhHeight;
|
|
|
|
Status = GdiFullscreenControl(FullscreenControlLoadFont,
|
|
FontInformation,
|
|
lpScreenFontHeader->sfhCharacters*lpScreenFontHeader->sfhHeight + sizeof(VIDEO_LOAD_FONT_INFORMATION),
|
|
NULL,
|
|
NULL);
|
|
|
|
ConsoleHeapFree(FontInformation);
|
|
return Status;
|
|
} else {
|
|
FilePtr = lpScreenFontHeader->sfhCharacters*lpScreenFontHeader->sfhHeight;
|
|
if (SetFilePointer(hCPIFile, FilePtr, NULL, FILE_CURRENT) == -1) {
|
|
Status = STATUS_INVALID_PARAMETER;
|
|
goto DoExit;
|
|
}
|
|
}
|
|
NumEntries -= 1;
|
|
}
|
|
DoExit:
|
|
return Status;
|
|
}
|
|
#endif
|
|
|
|
NTSTATUS
|
|
GetThreadConsoleDesktop(
|
|
DWORD dwThreadId,
|
|
HDESK *phdeskConsole)
|
|
{
|
|
PCSR_THREAD pcsrt;
|
|
PCONSOLE_PER_PROCESS_DATA ProcessData;
|
|
PCONSOLE_INFORMATION Console;
|
|
NTSTATUS Status;
|
|
HANDLE ConsoleHandle = NULL;
|
|
|
|
*phdeskConsole = NULL;
|
|
Status = CsrLockThreadByClientId(LongToHandle(dwThreadId), &pcsrt);
|
|
if (NT_SUCCESS(Status)) {
|
|
ProcessData = CONSOLE_FROMTHREADPERPROCESSDATA(pcsrt);
|
|
ConsoleHandle = ProcessData->ConsoleHandle;
|
|
CsrUnlockThread(pcsrt);
|
|
}
|
|
|
|
//
|
|
// If this process is a console app, return the
|
|
// handle to its desktop. Otherwise, return NULL.
|
|
//
|
|
|
|
if (ConsoleHandle != NULL) {
|
|
Status = RevalidateConsole(ConsoleHandle, &Console);
|
|
if (NT_SUCCESS(Status)) {
|
|
*phdeskConsole = Console->hDesk;
|
|
}
|
|
UnlockConsole(Console);
|
|
}
|
|
|
|
return STATUS_SUCCESS;
|
|
}
|
|
|
|
|
|
NTSTATUS
|
|
SetRAMFontCodePage(
|
|
IN PSCREEN_INFORMATION ScreenInfo
|
|
)
|
|
{
|
|
FSVIDEO_SCREEN_INFORMATION ScreenInformation;
|
|
ULONG ModeIndex = ScreenInfo->BufferInfo.TextInfo.ModeIndex;
|
|
COORD FontSize;
|
|
WCHAR wChar;
|
|
WCHAR wCharBuf[2];
|
|
LPSTRINGBITMAP StringBitmap;
|
|
DWORD BufferSize;
|
|
PWORD FontImage;
|
|
PFONT_CACHE_INFORMATION FontCache;
|
|
WCHAR AltFaceName[LF_FACESIZE];
|
|
COORD AltFontSize;
|
|
BYTE AltFontFamily;
|
|
ULONG AltFontIndex = 0;
|
|
HFONT hOldFont;
|
|
NTSTATUS Status;
|
|
|
|
ScreenInformation.ScreenSize = RegModeFontPairs[ModeIndex].ScreenSize;
|
|
ScreenInformation.FontSize = RegModeFontPairs[ModeIndex].FontSize;
|
|
if (ScreenInfo->Console->FontCacheInformation == NULL)
|
|
{
|
|
Status = CreateFontCache(&FontCache);
|
|
if (!NT_SUCCESS(Status)) {
|
|
RIPMSG1(RIP_WARNING, "SetRAMFontCodePage: failed in CreateFontCache. Status=%08x", Status);
|
|
return STATUS_UNSUCCESSFUL;
|
|
}
|
|
(PFONT_CACHE_INFORMATION)ScreenInfo->Console->FontCacheInformation = FontCache;
|
|
|
|
MakeAltRasterFont(SCR_FONTCODEPAGE(ScreenInfo),
|
|
RegModeFontPairs[ModeIndex].FontSize,
|
|
&AltFontSize, &AltFontFamily, &AltFontIndex, AltFaceName);
|
|
FontCache->FullScreenFontIndex = AltFontIndex;
|
|
FontCache->FullScreenFontSize = AltFontSize;
|
|
|
|
BufferSize = CalcBitmapBufferSize(FontCache->FullScreenFontSize, BYTE_ALIGN);
|
|
StringBitmap = ConsoleHeapAlloc(TMP_DBCS_TAG,
|
|
sizeof(STRINGBITMAP) + sizeof(StringBitmap->ajBits) * BufferSize);
|
|
if (StringBitmap==NULL) {
|
|
RIPMSG0(RIP_WARNING, "SetRAMFontCodePage: failed to allocate StringBitmap");
|
|
return STATUS_UNSUCCESSFUL;
|
|
}
|
|
|
|
|
|
/*
|
|
* Change GDI font to full screen font that best matched.
|
|
*/
|
|
hOldFont = SelectObject(ScreenInfo->Console->hDC, FontInfo[FontCache->FullScreenFontIndex].hFont);
|
|
|
|
|
|
for (wChar=0x00; wChar < 0x80; wChar++) {
|
|
wCharBuf[0] = wChar;
|
|
wCharBuf[1] = TEXT('\0');
|
|
GetStringBitmapW(ScreenInfo->Console->hDC,
|
|
wCharBuf,
|
|
1,
|
|
(ULONG)ConsoleHeapSize(StringBitmap),
|
|
(BYTE*)StringBitmap
|
|
);
|
|
|
|
FontSize.X = (SHORT)StringBitmap->uiWidth;
|
|
FontSize.Y = (SHORT)StringBitmap->uiHeight;
|
|
|
|
#if defined(LATER_DBCS_FOR_GRID_CHAR) // by kazum
|
|
BufferSize = CalcBitmapBufferSize(FontSize, BYTE_ALIGN);
|
|
*(StringBitmap->ajBits + BufferSize) = 0;
|
|
*(StringBitmap->ajBits + BufferSize + 1) = 0;
|
|
|
|
if (gpGridCharacter) {
|
|
PGRID_CHARACTER_INFORMATION GridCharacter;
|
|
PWCHAR CodePoint;
|
|
|
|
GridCharacter = gpGridCharacter;
|
|
do {
|
|
if (GridCharacter->CodePage == OEMCP) {
|
|
CodePoint = GridCharacter->CodePoint;
|
|
while (*CodePoint) {
|
|
if (*CodePoint == wChar) {
|
|
if (FontSize.X <= 8)
|
|
*(StringBitmap->ajBits + BufferSize) = *(StringBitmap->ajBits + BufferSize - 1);
|
|
else {
|
|
*(StringBitmap->ajBits + BufferSize) = *(StringBitmap->ajBits + BufferSize - 2);
|
|
*(StringBitmap->ajBits + BufferSize + 1) = *(StringBitmap->ajBits + BufferSize - 1);
|
|
}
|
|
break;
|
|
}
|
|
else
|
|
CodePoint++;
|
|
}
|
|
break;
|
|
}
|
|
} while (GridCharacter = GridCharacter->pNext);
|
|
}
|
|
#endif // LATER_DBCS_FOR_GRID_CHAR // by kazum
|
|
|
|
Status = SetFontImage(ScreenInfo->Console->FontCacheInformation,
|
|
wChar,
|
|
FontSize,
|
|
BYTE_ALIGN,
|
|
StringBitmap->ajBits
|
|
);
|
|
if (!NT_SUCCESS(Status)) {
|
|
RIPMSG3(RIP_WARNING, "SetRAMFontCodePage: failed to set font image. wc=%04x sz=(%x, %x).",
|
|
wChar, FontSize.X, FontSize.Y);
|
|
}
|
|
|
|
if (FontSize.X != ScreenInformation.FontSize.X ||
|
|
FontSize.Y != ScreenInformation.FontSize.Y) {
|
|
BufferSize = CalcBitmapBufferSize(ScreenInformation.FontSize, WORD_ALIGN);
|
|
FontImage = ConsoleHeapAlloc(TMP_DBCS_TAG, BufferSize);
|
|
if (FontImage!=NULL) {
|
|
GetExpandFontImage(ScreenInfo->Console->FontCacheInformation,
|
|
wChar,
|
|
FontSize,
|
|
ScreenInformation.FontSize,
|
|
FontImage);
|
|
|
|
Status = SetFontImage(ScreenInfo->Console->FontCacheInformation,
|
|
wChar,
|
|
ScreenInformation.FontSize,
|
|
WORD_ALIGN,
|
|
FontImage);
|
|
if (!NT_SUCCESS(Status)) {
|
|
RIPMSG3(RIP_WARNING, "SetRAMFontCodePage: failed to set font image. wc=%04x, sz=(%x,%x)",
|
|
wChar, ScreenInformation.FontSize.X, ScreenInformation.FontSize.Y);
|
|
}
|
|
|
|
ConsoleHeapFree(FontImage);
|
|
} else {
|
|
RIPMSG0(RIP_WARNING, "SetRAMFontCodePage: failed to allocate FontImage.");
|
|
}
|
|
}
|
|
}
|
|
|
|
ConsoleHeapFree(StringBitmap);
|
|
|
|
/*
|
|
* Back to GDI font
|
|
*/
|
|
SelectObject(ScreenInfo->Console->hDC, hOldFont);
|
|
}
|
|
|
|
Status = GdiFullscreenControl(FullscreenControlSetScreenInformation,
|
|
&ScreenInformation,
|
|
sizeof(ScreenInformation),
|
|
NULL,
|
|
NULL);
|
|
|
|
return Status;
|
|
}
|
|
|
|
NTSTATUS
|
|
SetRAMFont(
|
|
IN PSCREEN_INFORMATION ScreenInfo,
|
|
IN PCHAR_INFO ScreenBufPtr,
|
|
IN DWORD Length
|
|
)
|
|
{
|
|
ULONG ModeIndex = ScreenInfo->BufferInfo.TextInfo.ModeIndex;
|
|
COORD FsFontSize1 = RegModeFontPairs[ModeIndex].FontSize;
|
|
COORD FsFontSize2 = FsFontSize1;
|
|
COORD GdiFontSize1;
|
|
COORD GdiFontSize2;
|
|
COORD RetFontSize;
|
|
WCHAR wCharBuf[2];
|
|
LPSTRINGBITMAP StringBitmap;
|
|
DWORD BufferSize;
|
|
PWORD FontImage;
|
|
PFONT_CACHE_INFORMATION FontCache;
|
|
HFONT hOldFont;
|
|
NTSTATUS Status;
|
|
|
|
FontCache = (PFONT_CACHE_INFORMATION)ScreenInfo->Console->FontCacheInformation;
|
|
if (FontCache==NULL)
|
|
{
|
|
RIPMSG0(RIP_ERROR, "SetRAMFont: ScreenInfo->Console->FontCacheInformation == NULL.");
|
|
return STATUS_UNSUCCESSFUL;
|
|
}
|
|
|
|
GdiFontSize1 = FontCache->FullScreenFontSize;
|
|
GdiFontSize2 = GdiFontSize1;
|
|
GdiFontSize2.X *= 2;
|
|
FsFontSize2.X *= 2;
|
|
|
|
BufferSize = CalcBitmapBufferSize(GdiFontSize2, BYTE_ALIGN);
|
|
StringBitmap = ConsoleHeapAlloc(TMP_DBCS_TAG,
|
|
sizeof(STRINGBITMAP) + sizeof(StringBitmap->ajBits) * BufferSize);
|
|
if (StringBitmap == NULL) {
|
|
RIPMSG0(RIP_WARNING, "SetRAMFont: failed to allocate StringBitmap");
|
|
return STATUS_UNSUCCESSFUL;
|
|
}
|
|
|
|
/*
|
|
* Change GDI font to full screen font that best matched.
|
|
*/
|
|
hOldFont = SelectObject(ScreenInfo->Console->hDC, FontInfo[FontCache->FullScreenFontIndex].hFont);
|
|
|
|
while (Length--) {
|
|
Status = GetFontImage(ScreenInfo->Console->FontCacheInformation,
|
|
ScreenBufPtr->Char.UnicodeChar,
|
|
(ScreenBufPtr->Attributes & COMMON_LVB_SBCSDBCS) ? FsFontSize2 : FsFontSize1,
|
|
0,
|
|
NULL);
|
|
if (!NT_SUCCESS(Status)) {
|
|
wCharBuf[0] = ScreenBufPtr->Char.UnicodeChar;
|
|
wCharBuf[1] = TEXT('\0');
|
|
GetStringBitmapW(ScreenInfo->Console->hDC,
|
|
wCharBuf,
|
|
1,
|
|
(ULONG)ConsoleHeapSize(StringBitmap),
|
|
(BYTE*)StringBitmap
|
|
);
|
|
|
|
RetFontSize.X = (SHORT)StringBitmap->uiWidth;
|
|
RetFontSize.Y = (SHORT)StringBitmap->uiHeight;
|
|
|
|
Status = SetFontImage(ScreenInfo->Console->FontCacheInformation,
|
|
ScreenBufPtr->Char.UnicodeChar,
|
|
RetFontSize,
|
|
BYTE_ALIGN,
|
|
StringBitmap->ajBits
|
|
);
|
|
if (!NT_SUCCESS(Status)) {
|
|
RIPMSG3(RIP_WARNING, "SetRAMFont: failed to set font image. wc=%04x sz=(%x,%x)",
|
|
ScreenBufPtr->Char.UnicodeChar, RetFontSize.X, RetFontSize.Y);
|
|
}
|
|
|
|
if (((ScreenBufPtr->Attributes & COMMON_LVB_SBCSDBCS) &&
|
|
(GdiFontSize2.X != FsFontSize2.X || GdiFontSize2.Y != FsFontSize2.Y)) ||
|
|
(!(ScreenBufPtr->Attributes & COMMON_LVB_SBCSDBCS) &&
|
|
(GdiFontSize1.X != FsFontSize1.X || GdiFontSize1.Y != FsFontSize1.Y))) {
|
|
BufferSize = CalcBitmapBufferSize(FsFontSize2, WORD_ALIGN);
|
|
FontImage = ConsoleHeapAlloc(TMP_DBCS_TAG, BufferSize);
|
|
if (FontImage != NULL) {
|
|
|
|
GetExpandFontImage(ScreenInfo->Console->FontCacheInformation,
|
|
ScreenBufPtr->Char.UnicodeChar,
|
|
(ScreenBufPtr->Attributes & COMMON_LVB_SBCSDBCS) ? GdiFontSize2 : GdiFontSize1,
|
|
(ScreenBufPtr->Attributes & COMMON_LVB_SBCSDBCS) ? FsFontSize2 : FsFontSize1,
|
|
FontImage
|
|
);
|
|
|
|
Status = SetFontImage(ScreenInfo->Console->FontCacheInformation,
|
|
ScreenBufPtr->Char.UnicodeChar,
|
|
(ScreenBufPtr->Attributes & COMMON_LVB_SBCSDBCS) ? FsFontSize2 : FsFontSize1,
|
|
WORD_ALIGN,
|
|
FontImage
|
|
);
|
|
if (!NT_SUCCESS(Status)) {
|
|
RIPMSG3(RIP_WARNING, "SetRAMFont: failed to set font image. wc=%04x sz=(%x,%x)",
|
|
ScreenBufPtr->Char.UnicodeChar,
|
|
((ScreenBufPtr->Attributes & COMMON_LVB_SBCSDBCS) ? FsFontSize2 : FsFontSize1).X,
|
|
((ScreenBufPtr->Attributes & COMMON_LVB_SBCSDBCS) ? FsFontSize2 : FsFontSize1).Y);
|
|
}
|
|
|
|
ConsoleHeapFree(FontImage);
|
|
} else {
|
|
RIPMSG0(RIP_WARNING, "SetRAMFont: failed to allocate FontImage.");
|
|
}
|
|
}
|
|
}
|
|
|
|
if (ScreenBufPtr->Attributes & COMMON_LVB_SBCSDBCS) {
|
|
ScreenBufPtr += 2;
|
|
if (Length >= 1) {
|
|
Length -= 1;
|
|
} else {
|
|
break;
|
|
}
|
|
} else {
|
|
ScreenBufPtr++;
|
|
}
|
|
}
|
|
|
|
ConsoleHeapFree(StringBitmap);
|
|
|
|
/*
|
|
* Back to GDI font
|
|
*/
|
|
SelectObject(ScreenInfo->Console->hDC, hOldFont);
|
|
|
|
return Status;
|
|
}
|
|
|
|
#ifdef i386
|
|
#if defined(FE_SB)
|
|
|
|
#define WWSB_NOFE
|
|
#include "_priv.h"
|
|
#undef WWSB_NOFE
|
|
#define WWSB_FE
|
|
#include "_priv.h"
|
|
#undef WWSB_FE
|
|
|
|
#endif // FE_SB
|
|
#endif // i386
|