Windows2000/private/ntos/w32/ntcon/server/bitmap.c
2020-09-30 17:12:32 +02:00

337 lines
12 KiB
C

/*++
Copyright (c) 1985 - 1999, Microsoft Corporation
Module Name:
bitmap.c
Abstract:
This file implements bitmap video buffer management.
Author:
Therese Stowell (thereses) 4-Sept-1991
--*/
#include "precomp.h"
#pragma hdrstop
NTSTATUS CreateConsoleBitmap(IN OUT PCONSOLE_GRAPHICS_BUFFER_INFO GraphicsInfo, IN OUT PSCREEN_INFORMATION ScreenInfo, OUT PVOID *lpBitmap, OUT HANDLE *hMutex)
{
NTSTATUS Status;
LARGE_INTEGER MaximumSize;
SIZE_T ViewSize;
// adjust bitmap info
if (GraphicsInfo->lpBitMapInfo->bmiHeader.biHeight > 0)
{
#if DBG
DbgPrint("*************** Negating biHeight\n");
#endif
GraphicsInfo->lpBitMapInfo->bmiHeader.biHeight = -GraphicsInfo->lpBitMapInfo->bmiHeader.biHeight;
}
if (GraphicsInfo->lpBitMapInfo->bmiHeader.biCompression != BI_RGB)
{
#if DBG
DbgPrint("*************** setting Compression to BI_RGB)\n");
#endif
GraphicsInfo->lpBitMapInfo->bmiHeader.biCompression = BI_RGB;
}
// allocate screeninfo buffer data and copy it
ScreenInfo->BufferInfo.GraphicsInfo.lpBitMapInfo = (LPBITMAPINFO) ConsoleHeapAlloc(MAKE_TAG(BMP_TAG), GraphicsInfo->dwBitMapInfoLength);
if (ScreenInfo->BufferInfo.GraphicsInfo.lpBitMapInfo == NULL)
{
return STATUS_NO_MEMORY;
}
ScreenInfo->BufferInfo.GraphicsInfo.BitMapInfoLength = GraphicsInfo->dwBitMapInfoLength;
RtlCopyMemory(ScreenInfo->BufferInfo.GraphicsInfo.lpBitMapInfo, GraphicsInfo->lpBitMapInfo, GraphicsInfo->dwBitMapInfoLength);
ASSERT((GraphicsInfo->lpBitMapInfo->bmiHeader.biWidth * -GraphicsInfo->lpBitMapInfo->bmiHeader.biHeight / 8 * GraphicsInfo->lpBitMapInfo->bmiHeader.biBitCount) == (LONG) GraphicsInfo->lpBitMapInfo->bmiHeader.biSizeImage);
// create bitmap section
MaximumSize.QuadPart = GraphicsInfo->lpBitMapInfo->bmiHeader.biSizeImage;
Status = NtCreateSection(&ScreenInfo->BufferInfo.GraphicsInfo.hSection,
SECTION_ALL_ACCESS,
NULL,
&MaximumSize,
PAGE_READWRITE,
SEC_COMMIT,
NULL
);
if (!NT_SUCCESS(Status)) {
ConsoleHeapFree(ScreenInfo->BufferInfo.GraphicsInfo.lpBitMapInfo);
return Status;
}
// map server view of section
ViewSize = GraphicsInfo->lpBitMapInfo->bmiHeader.biSizeImage;
ScreenInfo->BufferInfo.GraphicsInfo.BitMap = 0;
Status = NtMapViewOfSection(ScreenInfo->BufferInfo.GraphicsInfo.hSection,
NtCurrentProcess(),
&ScreenInfo->BufferInfo.GraphicsInfo.BitMap,
0L,
GraphicsInfo->lpBitMapInfo->bmiHeader.biSizeImage,
NULL,
&ViewSize,
ViewUnmap,
0L,
PAGE_READWRITE
);
if (!NT_SUCCESS(Status)) {
ConsoleHeapFree(ScreenInfo->BufferInfo.GraphicsInfo.lpBitMapInfo);
NtClose(ScreenInfo->BufferInfo.GraphicsInfo.hSection);
return Status;
}
// map client view of section
ViewSize = GraphicsInfo->lpBitMapInfo->bmiHeader.biSizeImage;
*lpBitmap = 0;
Status = NtMapViewOfSection(ScreenInfo->BufferInfo.GraphicsInfo.hSection,
CONSOLE_CLIENTPROCESSHANDLE(),
lpBitmap,
0L,
GraphicsInfo->lpBitMapInfo->bmiHeader.biSizeImage,
NULL,
&ViewSize,
ViewUnmap,
0L,
PAGE_READWRITE
);
if (!NT_SUCCESS(Status)) {
ConsoleHeapFree(ScreenInfo->BufferInfo.GraphicsInfo.lpBitMapInfo);
NtUnmapViewOfSection(NtCurrentProcess(),ScreenInfo->BufferInfo.GraphicsInfo.BitMap);
NtClose(ScreenInfo->BufferInfo.GraphicsInfo.hSection);
return Status;
}
ScreenInfo->BufferInfo.GraphicsInfo.ClientProcess = CONSOLE_CLIENTPROCESSHANDLE();
ScreenInfo->BufferInfo.GraphicsInfo.ClientBitMap = *lpBitmap;
// create mutex to serialize access to bitmap, then map handle to mutex to client side
NtCreateMutant(&ScreenInfo->BufferInfo.GraphicsInfo.hMutex,
MUTANT_ALL_ACCESS, NULL, FALSE);
MapHandle(CONSOLE_CLIENTPROCESSHANDLE(),
ScreenInfo->BufferInfo.GraphicsInfo.hMutex,
hMutex
);
ScreenInfo->BufferInfo.GraphicsInfo.dwUsage = GraphicsInfo->dwUsage;
ScreenInfo->ScreenBufferSize.X = (WORD)GraphicsInfo->lpBitMapInfo->bmiHeader.biWidth;
ScreenInfo->ScreenBufferSize.Y = (WORD)-GraphicsInfo->lpBitMapInfo->bmiHeader.biHeight;
ScreenInfo->Window.Left = 0;
ScreenInfo->Window.Top = 0;
ScreenInfo->Window.Right = (SHORT)(ScreenInfo->Window.Left+ScreenInfo->ScreenBufferSize.X-1);
ScreenInfo->Window.Bottom = (SHORT)(ScreenInfo->Window.Top+ScreenInfo->ScreenBufferSize.Y-1);
return STATUS_SUCCESS;
}
ULONG
SrvInvalidateBitMapRect(
IN OUT PCSR_API_MSG m,
IN OUT PCSR_REPLY_STATUS ReplyStatus
)
/*++
Routine Description:
This routine is called to indicate that the application has modified a region
in the bitmap. We update the region to the screen.
Arguments:
m - message containing api parameters
ReplyStatus - Indicates whether to reply to the dll port.
Return Value:
--*/
{
PCONSOLE_INVALIDATERECT_MSG a = (PCONSOLE_INVALIDATERECT_MSG)&m->u.ApiMessageData;
PCONSOLE_INFORMATION Console;
PHANDLE_DATA HandleData;
NTSTATUS Status;
UINT Codepage;
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)) {
UnlockConsole(Console);
return((ULONG) Status);
}
if (HandleData->HandleType & CONSOLE_OUTPUT_HANDLE) {
//ASSERT(Console->Flags & CONSOLE_VDM_REGISTERED);
//ASSERT(!(Console->FullScreenFlags & CONSOLE_FULLSCREEN_HARDWARE));
ASSERT(Console->VDMBuffer != NULL);
if (Console->VDMBuffer != NULL) {
//ASSERT(HandleData->Buffer.ScreenBuffer->ScreenBufferSize.X <= Console->VDMBufferSize.X);
//ASSERT(HandleData->Buffer.ScreenBuffer->ScreenBufferSize.Y <= Console->VDMBufferSize.Y);
if (HandleData->Buffer.ScreenBuffer->ScreenBufferSize.X <= Console->VDMBufferSize.X &&
HandleData->Buffer.ScreenBuffer->ScreenBufferSize.Y <= Console->VDMBufferSize.Y) {
COORD TargetPoint;
TargetPoint.X = a->Rect.Left;
TargetPoint.Y = a->Rect.Top;
// VDM can sometimes get out of sync with window size
//ASSERT(a->Rect.Left >= 0);
//ASSERT(a->Rect.Top >= 0);
//ASSERT(a->Rect.Right < HandleData->Buffer.ScreenBuffer->ScreenBufferSize.X);
//ASSERT(a->Rect.Bottom < HandleData->Buffer.ScreenBuffer->ScreenBufferSize.Y);
//ASSERT(a->Rect.Left <= a->Rect.Right);
//ASSERT(a->Rect.Top <= a->Rect.Bottom);
if ((a->Rect.Left >= 0) &&
(a->Rect.Top >= 0) &&
(a->Rect.Right < HandleData->Buffer.ScreenBuffer->ScreenBufferSize.X) &&
(a->Rect.Bottom < HandleData->Buffer.ScreenBuffer->ScreenBufferSize.Y) &&
(a->Rect.Left <= a->Rect.Right) &&
(a->Rect.Top <= a->Rect.Bottom) ) {
if ((Console->CurrentScreenBuffer->Flags & CONSOLE_OEMFONT_DISPLAY) && ((Console->FullScreenFlags & CONSOLE_FULLSCREEN) == 0)) {
#if defined(FE_SB)
if (CONSOLE_IS_DBCS_ENABLED() &&
Console->OutputCP != WINDOWSCP )
{
Codepage = USACP;
}
else
#endif
// we want UnicodeOem characters
Codepage = WINDOWSCP;
} else {
#if defined(FE_SB)
if (CONSOLE_IS_DBCS_ENABLED()) {
Codepage = Console->OutputCP;
}
else
#endif
// we want real Unicode characters
Codepage = Console->CP;
}
WriteRectToScreenBuffer((PBYTE)Console->VDMBuffer,
Console->VDMBufferSize, &a->Rect,
HandleData->Buffer.ScreenBuffer, TargetPoint,
Codepage);
WriteToScreen(HandleData->Buffer.ScreenBuffer,&a->Rect);
} else {
Status = STATUS_INVALID_PARAMETER;
}
} else {
Status = STATUS_INVALID_PARAMETER;
}
} else {
Status = STATUS_INVALID_PARAMETER;
}
} else {
// write data to screen
WriteToScreen(HandleData->Buffer.ScreenBuffer,&a->Rect);
}
UnlockConsole(Console);
return Status;
UNREFERENCED_PARAMETER(ReplyStatus);
}
NTSTATUS
WriteRegionToScreenBitMap(
IN PSCREEN_INFORMATION ScreenInfo,
IN PSMALL_RECT Region
)
{
DWORD NumScanLines;
int Height;
// if we have a selection, turn it off.
InvertSelection(ScreenInfo->Console,TRUE);
NtWaitForSingleObject(ScreenInfo->BufferInfo.GraphicsInfo.hMutex,
FALSE, NULL);
// The origin of (xSrc, ySrc) passed to SetDIBitsToDevice is located
// at the DIB's bottom-left corner no matter if the DIB is
// a top-down or bottom-up. Thus, if the DIB is a top-down, we have
// to translate ySrc accordingly:
// if (height < 0) { // top-down
// ySrc = abs(height) - rect.Bottom -1;
// else
// ySrc = rect.Bottom;
Height = ScreenInfo->BufferInfo.GraphicsInfo.lpBitMapInfo->bmiHeader.biHeight;
NumScanLines = SetDIBitsToDevice(ScreenInfo->Console->hDC,
Region->Left - ScreenInfo->Window.Left,
Region->Top - ScreenInfo->Window.Top,
Region->Right - Region->Left + 1,
Region->Bottom - Region->Top + 1,
Region->Left,
Height < 0 ? -Height - Region->Bottom - 1 : Region->Bottom,
0,
ScreenInfo->ScreenBufferSize.Y,
ScreenInfo->BufferInfo.GraphicsInfo.BitMap,
ScreenInfo->BufferInfo.GraphicsInfo.lpBitMapInfo,
ScreenInfo->BufferInfo.GraphicsInfo.dwUsage
);
NtReleaseMutant(ScreenInfo->BufferInfo.GraphicsInfo.hMutex, NULL);
// if we have a selection, turn it on.
InvertSelection(ScreenInfo->Console,FALSE);
if (NumScanLines == 0) {
return STATUS_UNSUCCESSFUL;
}
return STATUS_SUCCESS;
}
VOID
FreeConsoleBitmap(
IN PSCREEN_INFORMATION ScreenInfo
)
{
NtUnmapViewOfSection(NtCurrentProcess(),
ScreenInfo->BufferInfo.GraphicsInfo.BitMap);
NtUnmapViewOfSection(ScreenInfo->BufferInfo.GraphicsInfo.ClientProcess,
ScreenInfo->BufferInfo.GraphicsInfo.ClientBitMap);
NtClose(ScreenInfo->BufferInfo.GraphicsInfo.hSection);
NtClose(ScreenInfo->BufferInfo.GraphicsInfo.hMutex);
ConsoleHeapFree(ScreenInfo->BufferInfo.GraphicsInfo.lpBitMapInfo);
}