|
|
/*++
Copyright (c) 1985 - 1999, Microsoft Corporation
Module Name:
_stream.h
Abstract:
Performance critical routine for Single Binary
Each function will be created with two flavors FE and non FE
Author:
KazuM Jun.09.1997
Revision History:
--*/
#define WWSB_NEUTRAL_FILE 1
#if !defined(FE_SB)
#error This header file should be included with FE_SB
#endif
#if !defined(WWSB_FE) && !defined(WWSB_NOFE)
#error Either WWSB_FE and WWSB_NOFE must be defined.
#endif
#if defined(WWSB_FE) && defined(WWSB_NOFE)
#error Both WWSB_FE and WWSB_NOFE defined.
#endif
#ifdef WWSB_FE
#pragma alloc_text(FE_TEXT, FE_AdjustCursorPosition)
#pragma alloc_text(FE_TEXT, FE_WriteChars)
#pragma alloc_text(FE_TEXT, FE_DoWriteConsole)
#pragma alloc_text(FE_TEXT, FE_DoSrvWriteConsole)
#endif
NTSTATUS WWSB_AdjustCursorPosition( IN PSCREEN_INFORMATION ScreenInfo, IN COORD CursorPosition, IN BOOL KeepCursorVisible, OUT PSHORT ScrollY OPTIONAL )
/*++
Routine Description:
This routine updates the cursor position. Its input is the non-special cased new location of the cursor. For example, if the cursor were being moved one space backwards from the left edge of the screen, the X coordinate would be -1. This routine would set the X coordinate to the right edge of the screen and decrement the Y coordinate by one.
Arguments:
ScreenInfo - Pointer to screen buffer information structure.
CursorPosition - New location of cursor.
KeepCursorVisible - TRUE if changing window origin desirable when hit right edge
Return Value:
--*/
{ COORD WindowOrigin; NTSTATUS Status; #ifdef WWSB_FE
PCONSOLE_INFORMATION Console = ScreenInfo->Console;
if (!(ScreenInfo->Flags & CONSOLE_TEXTMODE_BUFFER)) return STATUS_SUCCESS; #endif
if (CursorPosition.X < 0) { if (CursorPosition.Y > 0) { CursorPosition.X = (SHORT)(ScreenInfo->ScreenBufferSize.X+CursorPosition.X); CursorPosition.Y = (SHORT)(CursorPosition.Y-1); } else { CursorPosition.X = 0; } } else if (CursorPosition.X >= ScreenInfo->ScreenBufferSize.X) {
//
// at end of line. if wrap mode, wrap cursor. otherwise leave it
// where it is.
//
if (ScreenInfo->OutputMode & ENABLE_WRAP_AT_EOL_OUTPUT) { CursorPosition.Y += CursorPosition.X / ScreenInfo->ScreenBufferSize.X; CursorPosition.X = CursorPosition.X % ScreenInfo->ScreenBufferSize.X; } else { CursorPosition.X = ScreenInfo->BufferInfo.TextInfo.CursorPosition.X; } } #ifdef WWSB_FE
if (CursorPosition.Y >= ScreenInfo->ScreenBufferSize.Y && !(Console->InputBuffer.ImeMode.Open) ) #else
if (CursorPosition.Y >= ScreenInfo->ScreenBufferSize.Y) #endif
{
//
// at end of buffer. scroll contents of screen buffer so new
// position is visible.
//
ASSERT (CursorPosition.Y == ScreenInfo->ScreenBufferSize.Y); StreamScrollRegion(ScreenInfo);
if (ARGUMENT_PRESENT(ScrollY)) { *ScrollY += (SHORT)(ScreenInfo->ScreenBufferSize.Y - CursorPosition.Y - 1); } CursorPosition.Y += (SHORT)(ScreenInfo->ScreenBufferSize.Y - CursorPosition.Y - 1); } #ifdef WWSB_FE
else if (!(Console->InputBuffer.ImeMode.Disable) && Console->InputBuffer.ImeMode.Open) { if (CursorPosition.Y == (ScreenInfo->ScreenBufferSize.Y-1)) { ConsoleImeBottomLineUse(ScreenInfo,2); if (ARGUMENT_PRESENT(ScrollY)) { *ScrollY += (SHORT)(ScreenInfo->ScreenBufferSize.Y - CursorPosition.Y - 2); } CursorPosition.Y += (SHORT)(ScreenInfo->ScreenBufferSize.Y - CursorPosition.Y - 2); if (!ARGUMENT_PRESENT(ScrollY) && Console->lpCookedReadData) { ((PCOOKED_READ_DATA)(Console->lpCookedReadData))->OriginalCursorPosition.Y--; } } else if (CursorPosition.Y == ScreenInfo->Window.Bottom) { ; } } #endif
//
// if at right or bottom edge of window, scroll right or down one char.
//
#ifdef WWSB_FE
if (CursorPosition.Y > ScreenInfo->Window.Bottom && !(Console->InputBuffer.ImeMode.Open) ) #else
if (CursorPosition.Y > ScreenInfo->Window.Bottom) #endif
{ WindowOrigin.X = 0; WindowOrigin.Y = CursorPosition.Y - ScreenInfo->Window.Bottom; Status = SetWindowOrigin(ScreenInfo, FALSE, WindowOrigin ); if (!NT_SUCCESS(Status)) { return Status; } } #ifdef WWSB_FE
else if (Console->InputBuffer.ImeMode.Open) { if (CursorPosition.Y >= ScreenInfo->Window.Bottom && CONSOLE_WINDOW_SIZE_Y(ScreenInfo) > 1 ) { WindowOrigin.X = 0; WindowOrigin.Y = CursorPosition.Y - ScreenInfo->Window.Bottom + 1; Status = SetWindowOrigin(ScreenInfo, FALSE, WindowOrigin ); if (!NT_SUCCESS(Status)) { return Status; } } } #endif
if (KeepCursorVisible) { MakeCursorVisible(ScreenInfo,CursorPosition); } Status = SetCursorPosition(ScreenInfo, CursorPosition, KeepCursorVisible ); return Status; }
#define LOCAL_BUFFER_SIZE 100
NTSTATUS WWSB_WriteChars( IN PSCREEN_INFORMATION ScreenInfo, IN PWCHAR lpBufferBackupLimit, IN PWCHAR lpBuffer, IN PWCHAR lpRealUnicodeString, IN OUT PDWORD NumBytes, OUT PLONG NumSpaces OPTIONAL, IN SHORT OriginalXPosition, IN DWORD dwFlags, OUT PSHORT ScrollY OPTIONAL )
/*++
Routine Description:
This routine writes a string to the screen, processing any embedded unicode characters. The string is also copied to the input buffer, if the output mode is line mode.
Arguments:
ScreenInfo - Pointer to screen buffer information structure.
lpBufferBackupLimit - Pointer to beginning of buffer.
lpBuffer - Pointer to buffer to copy string to. assumed to be at least as long as lpRealUnicodeString. This pointer is updated to point to the next position in the buffer.
lpRealUnicodeString - Pointer to string to write.
NumBytes - On input, number of bytes to write. On output, number of bytes written.
NumSpaces - On output, the number of spaces consumed by the written characters.
dwFlags - WC_DESTRUCTIVE_BACKSPACE backspace overwrites characters. WC_KEEP_CURSOR_VISIBLE change window origin desirable when hit rt. edge WC_ECHO if called by Read (echoing characters) WC_FALSIFY_UNICODE if RealUnicodeToFalseUnicode need be called.
Return Value:
Note:
This routine does not process tabs and backspace properly. That code will be implemented as part of the line editing services.
--*/
{ DWORD BufferSize; COORD CursorPosition; NTSTATUS Status; ULONG NumChars; static WCHAR Blanks[TAB_SIZE] = { UNICODE_SPACE, UNICODE_SPACE, UNICODE_SPACE, UNICODE_SPACE, UNICODE_SPACE, UNICODE_SPACE, UNICODE_SPACE, UNICODE_SPACE }; SHORT XPosition; WCHAR LocalBuffer[LOCAL_BUFFER_SIZE]; PWCHAR LocalBufPtr; ULONG i,j; SMALL_RECT Region; ULONG TabSize; DWORD TempNumSpaces; WCHAR Char; WCHAR RealUnicodeChar; WORD Attributes; PWCHAR lpString; PWCHAR lpAllocatedString; BOOL fUnprocessed = ((ScreenInfo->OutputMode & ENABLE_PROCESSED_OUTPUT) == 0); #ifdef WWSB_FE
CHAR LocalBufferA[LOCAL_BUFFER_SIZE]; PCHAR LocalBufPtrA; #endif
ConsoleHideCursor(ScreenInfo);
Attributes = ScreenInfo->Attributes; BufferSize = *NumBytes; *NumBytes = 0; TempNumSpaces = 0;
lpAllocatedString = NULL; if (dwFlags & WC_FALSIFY_UNICODE) { // translation from OEM -> ANSI -> OEM doesn't
// necessarily yield the same value, so do
// translation in a separate buffer.
lpString = ConsoleHeapAlloc(TMP_TAG, BufferSize); if (lpString == NULL) { Status = STATUS_NO_MEMORY; goto ExitWriteChars; }
lpAllocatedString = lpString; RtlCopyMemory(lpString, lpRealUnicodeString, BufferSize); Status = RealUnicodeToFalseUnicode(lpString, BufferSize / sizeof(WCHAR), ScreenInfo->Console->OutputCP ); if (!NT_SUCCESS(Status)) { goto ExitWriteChars; } } else { lpString = lpRealUnicodeString; }
while (*NumBytes < BufferSize) {
//
// as an optimization, collect characters in buffer and
// print out all at once.
//
XPosition = ScreenInfo->BufferInfo.TextInfo.CursorPosition.X; i=0; LocalBufPtr = LocalBuffer; #ifdef WWSB_FE
LocalBufPtrA = LocalBufferA; #endif
while (*NumBytes < BufferSize && i < LOCAL_BUFFER_SIZE && XPosition < ScreenInfo->ScreenBufferSize.X) { Char = *lpString; RealUnicodeChar = *lpRealUnicodeString; if (!IS_GLYPH_CHAR(RealUnicodeChar) || fUnprocessed) { #ifdef WWSB_FE
if (IsConsoleFullWidth(ScreenInfo->Console->hDC, ScreenInfo->Console->OutputCP,Char)) { if (i < (LOCAL_BUFFER_SIZE-1) && XPosition < (ScreenInfo->ScreenBufferSize.X-1)) { *LocalBufPtr++ = Char; *LocalBufPtrA++ = ATTR_LEADING_BYTE; *LocalBufPtr++ = Char; *LocalBufPtrA++ = ATTR_TRAILING_BYTE; XPosition+=2; i+=2; lpBuffer++; } else goto EndWhile; } else { #endif
*LocalBufPtr = Char; LocalBufPtr++; XPosition++; i++; lpBuffer++; #ifdef WWSB_FE
*LocalBufPtrA++ = 0; } #endif
} else { ASSERT(ScreenInfo->OutputMode & ENABLE_PROCESSED_OUTPUT); switch (RealUnicodeChar) { case UNICODE_BELL: if (dwFlags & WC_ECHO) { goto CtrlChar; } else { SendNotifyMessage(ScreenInfo->Console->hWnd, CM_BEEP, 0, 0x47474747); } break; case UNICODE_BACKSPACE:
// automatically go to EndWhile. this is because
// backspace is not destructive, so "aBkSp" prints
// a with the cursor on the "a". we could achieve
// this behavior staying in this loop and figuring out
// the string that needs to be printed, but it would
// be expensive and it's the exceptional case.
goto EndWhile; break; case UNICODE_TAB: TabSize = NUMBER_OF_SPACES_IN_TAB(XPosition); XPosition = (SHORT)(XPosition + TabSize); if (XPosition >= ScreenInfo->ScreenBufferSize.X) { goto EndWhile; } for (j=0;j<TabSize && i<LOCAL_BUFFER_SIZE;j++,i++) { *LocalBufPtr = (WCHAR)' '; LocalBufPtr++; #ifdef WWSB_FE
*LocalBufPtrA++ = 0; #endif
} lpBuffer++; break; case UNICODE_LINEFEED: case UNICODE_CARRIAGERETURN: goto EndWhile; default:
//
// if char is ctrl char, write ^char.
//
if ((dwFlags & WC_ECHO) && (IS_CONTROL_CHAR(RealUnicodeChar))) {
CtrlChar: if (i < (LOCAL_BUFFER_SIZE-1)) { *LocalBufPtr = (WCHAR)'^'; LocalBufPtr++; XPosition++; i++; *LocalBufPtr = (WCHAR)(RealUnicodeChar+(WCHAR)'@'); LocalBufPtr++; XPosition++; i++; lpBuffer++; #ifdef WWSB_FE
*LocalBufPtrA++ = 0; *LocalBufPtrA++ = 0; #endif
} else { goto EndWhile; } } else { if (!(ScreenInfo->Flags & CONSOLE_OEMFONT_DISPLAY) || (ScreenInfo->Console->FullScreenFlags & CONSOLE_FULLSCREEN)) { /*
* As a special favor to incompetent apps * that attempt to display control chars, * convert to corresponding OEM Glyph Chars */ #ifdef WWSB_FE
WORD CharType;
GetStringTypeW(CT_CTYPE1,&RealUnicodeChar,1,&CharType); if (CharType == C1_CNTRL) ConvertOutputToUnicode(ScreenInfo->Console->OutputCP, &(char)RealUnicodeChar, 1, LocalBufPtr, 1); else *LocalBufPtr = Char; #else
*LocalBufPtr = SB_CharToWcharGlyph( ScreenInfo->Console->OutputCP, (char)RealUnicodeChar); #endif
} else { *LocalBufPtr = Char; } LocalBufPtr++; XPosition++; i++; lpBuffer++; #ifdef WWSB_FE
*LocalBufPtrA++ = 0; #endif
} } } lpString++; lpRealUnicodeString++; *NumBytes += sizeof(WCHAR); } EndWhile: if (i != 0) {
//
// Make sure we don't write past the end of the buffer.
//
if (i > (ULONG)ScreenInfo->ScreenBufferSize.X - ScreenInfo->BufferInfo.TextInfo.CursorPosition.X) { i = (ULONG)ScreenInfo->ScreenBufferSize.X - ScreenInfo->BufferInfo.TextInfo.CursorPosition.X; }
#ifdef WWSB_FE
FE_StreamWriteToScreenBuffer(LocalBuffer, (SHORT)i, ScreenInfo, LocalBufferA ); #else
SB_StreamWriteToScreenBuffer(LocalBuffer, (SHORT)i, ScreenInfo ); #endif
Region.Left = ScreenInfo->BufferInfo.TextInfo.CursorPosition.X; Region.Right = (SHORT)(ScreenInfo->BufferInfo.TextInfo.CursorPosition.X + i - 1); Region.Top = ScreenInfo->BufferInfo.TextInfo.CursorPosition.Y; Region.Bottom = ScreenInfo->BufferInfo.TextInfo.CursorPosition.Y; WWSB_WriteToScreen(ScreenInfo,&Region); TempNumSpaces += i; CursorPosition.X = (SHORT)(ScreenInfo->BufferInfo.TextInfo.CursorPosition.X + i); CursorPosition.Y = ScreenInfo->BufferInfo.TextInfo.CursorPosition.Y; Status = WWSB_AdjustCursorPosition(ScreenInfo,CursorPosition, dwFlags & WC_KEEP_CURSOR_VISIBLE,ScrollY); if (*NumBytes == BufferSize) { ConsoleShowCursor(ScreenInfo); if (ARGUMENT_PRESENT(NumSpaces)) { *NumSpaces = TempNumSpaces; } Status = STATUS_SUCCESS; goto ExitWriteChars; } continue; } else if (*NumBytes == BufferSize) {
ASSERT(ScreenInfo->OutputMode & ENABLE_PROCESSED_OUTPUT); // this catches the case where the number of backspaces ==
// the number of characters.
if (ARGUMENT_PRESENT(NumSpaces)) { *NumSpaces = TempNumSpaces; } ConsoleShowCursor(ScreenInfo); Status = STATUS_SUCCESS; goto ExitWriteChars; }
ASSERT(ScreenInfo->OutputMode & ENABLE_PROCESSED_OUTPUT); switch (*lpString) { case UNICODE_BACKSPACE:
//
// move cursor backwards one space. overwrite current char with blank.
//
// we get here because we have to backspace from the beginning of the line
CursorPosition = ScreenInfo->BufferInfo.TextInfo.CursorPosition; TempNumSpaces -= 1; if (lpBuffer == lpBufferBackupLimit) { CursorPosition.X-=1; } else { PWCHAR pBuffer; WCHAR TmpBuffer[LOCAL_BUFFER_SIZE]; PWCHAR Tmp,Tmp2; WCHAR LastChar; ULONG i;
if (lpBuffer-lpBufferBackupLimit > LOCAL_BUFFER_SIZE) { pBuffer = ConsoleHeapAlloc(TMP_TAG, (ULONG)(lpBuffer-lpBufferBackupLimit) * sizeof(WCHAR)); if (pBuffer == NULL) { Status = STATUS_NO_MEMORY; goto ExitWriteChars; } } else { pBuffer = TmpBuffer; }
for (i=0,Tmp2=pBuffer,Tmp=lpBufferBackupLimit; i<(ULONG)(lpBuffer-lpBufferBackupLimit); i++,Tmp++) { if (*Tmp == UNICODE_BACKSPACE) { if (Tmp2 > pBuffer) { Tmp2--; } } else { ASSERT(Tmp2 >= pBuffer); *Tmp2++ = *Tmp; }
} if (Tmp2 == pBuffer) { LastChar = (WCHAR)' '; } else { LastChar = *(Tmp2-1); } if (pBuffer != TmpBuffer) { ConsoleHeapFree(pBuffer); }
if (LastChar == UNICODE_TAB) { CursorPosition.X -= (SHORT)(RetrieveNumberOfSpaces(OriginalXPosition, lpBufferBackupLimit, (ULONG)(lpBuffer - lpBufferBackupLimit - 1), ScreenInfo->Console, ScreenInfo->Console->OutputCP )); if (CursorPosition.X < 0) { CursorPosition.X = (ScreenInfo->ScreenBufferSize.X - 1)/TAB_SIZE; CursorPosition.X *= TAB_SIZE; CursorPosition.X += 1; CursorPosition.Y -= 1; } } else if (IS_CONTROL_CHAR(LastChar)) { CursorPosition.X-=1; TempNumSpaces -= 1;
//
// overwrite second character of ^x sequence.
//
if (dwFlags & WC_DESTRUCTIVE_BACKSPACE) { NumChars = 1; Status = WWSB_WriteOutputString(ScreenInfo, Blanks, CursorPosition, CONSOLE_FALSE_UNICODE, // faster than real unicode
&NumChars, NULL); Status = WWSB_FillOutput(ScreenInfo, Attributes, CursorPosition, CONSOLE_ATTRIBUTE, &NumChars); } CursorPosition.X-=1; } #ifdef WWSB_FE
else if (IsConsoleFullWidth(ScreenInfo->Console->hDC, ScreenInfo->Console->OutputCP,LastChar)) { CursorPosition.X-=1; TempNumSpaces -= 1;
Status = WWSB_AdjustCursorPosition(ScreenInfo,CursorPosition, dwFlags & WC_KEEP_CURSOR_VISIBLE,ScrollY); if (dwFlags & WC_DESTRUCTIVE_BACKSPACE) { // bug 7672
NumChars = 1; Status = WWSB_WriteOutputString(ScreenInfo, Blanks, ScreenInfo->BufferInfo.TextInfo.CursorPosition, CONSOLE_FALSE_UNICODE, // faster than real unicode
&NumChars, NULL); Status = WWSB_FillOutput(ScreenInfo, Attributes, ScreenInfo->BufferInfo.TextInfo.CursorPosition, CONSOLE_ATTRIBUTE, &NumChars); } CursorPosition.X-=1; } #endif
else { CursorPosition.X--; } } if ((dwFlags & WC_LIMIT_BACKSPACE) && (CursorPosition.X < 0)) { CursorPosition.X = 0; KdPrint(("CONSRV: Ignoring backspace to previous line\n")); } Status = WWSB_AdjustCursorPosition(ScreenInfo,CursorPosition, (dwFlags & WC_KEEP_CURSOR_VISIBLE) != 0,ScrollY); if (dwFlags & WC_DESTRUCTIVE_BACKSPACE) { NumChars = 1; Status = WWSB_WriteOutputString(ScreenInfo, Blanks, ScreenInfo->BufferInfo.TextInfo.CursorPosition, CONSOLE_FALSE_UNICODE, //faster than real unicode
&NumChars, NULL); Status = WWSB_FillOutput(ScreenInfo, Attributes, ScreenInfo->BufferInfo.TextInfo.CursorPosition, CONSOLE_ATTRIBUTE, &NumChars); } #ifdef WWSB_FE
if (ScreenInfo->BufferInfo.TextInfo.CursorPosition.X == 0 && (ScreenInfo->OutputMode & ENABLE_WRAP_AT_EOL_OUTPUT) && lpBuffer > lpBufferBackupLimit) { if (CheckBisectProcessW(ScreenInfo, ScreenInfo->Console->OutputCP, lpBufferBackupLimit, (ULONG)(lpBuffer+1-lpBufferBackupLimit), ScreenInfo->ScreenBufferSize.X-OriginalXPosition, OriginalXPosition, dwFlags & WC_ECHO)) { CursorPosition.X = ScreenInfo->ScreenBufferSize.X-1; CursorPosition.Y = (SHORT)(ScreenInfo->BufferInfo.TextInfo.CursorPosition.Y-1); Status = WWSB_AdjustCursorPosition(ScreenInfo,CursorPosition, dwFlags & WC_KEEP_CURSOR_VISIBLE,ScrollY); } } #endif
break; case UNICODE_TAB: TabSize = NUMBER_OF_SPACES_IN_TAB(ScreenInfo->BufferInfo.TextInfo.CursorPosition.X); CursorPosition.X = (SHORT)(ScreenInfo->BufferInfo.TextInfo.CursorPosition.X + TabSize);
//
// move cursor forward to next tab stop. fill space with blanks.
// we get here when the tab extends beyond the right edge of the
// window. if the tab goes wraps the line, set the cursor to the first
// position in the next line.
//
lpBuffer++;
TempNumSpaces += TabSize; if (CursorPosition.X >= ScreenInfo->ScreenBufferSize.X) { NumChars = ScreenInfo->ScreenBufferSize.X - ScreenInfo->BufferInfo.TextInfo.CursorPosition.X; CursorPosition.X = 0; CursorPosition.Y = ScreenInfo->BufferInfo.TextInfo.CursorPosition.Y+1; } else { NumChars = CursorPosition.X - ScreenInfo->BufferInfo.TextInfo.CursorPosition.X; CursorPosition.Y = ScreenInfo->BufferInfo.TextInfo.CursorPosition.Y; } Status = WWSB_WriteOutputString(ScreenInfo, Blanks, ScreenInfo->BufferInfo.TextInfo.CursorPosition, CONSOLE_FALSE_UNICODE, // faster than real unicode
&NumChars, NULL); Status = WWSB_FillOutput(ScreenInfo, Attributes, ScreenInfo->BufferInfo.TextInfo.CursorPosition, CONSOLE_ATTRIBUTE, &NumChars); Status = WWSB_AdjustCursorPosition(ScreenInfo,CursorPosition, (dwFlags & WC_KEEP_CURSOR_VISIBLE) != 0,ScrollY); break; case UNICODE_CARRIAGERETURN:
//
// Carriage return moves the cursor to the beginning of the line.
// We don't need to worry about handling cr or lf for
// backspace because input is sent to the user on cr or lf.
//
lpBuffer++; CursorPosition.X = 0; CursorPosition.Y = ScreenInfo->BufferInfo.TextInfo.CursorPosition.Y; Status = WWSB_AdjustCursorPosition(ScreenInfo,CursorPosition, (dwFlags & WC_KEEP_CURSOR_VISIBLE) != 0,ScrollY); break; case UNICODE_LINEFEED:
//
// move cursor to the beginning of the next line.
//
lpBuffer++; CursorPosition.X = 0; CursorPosition.Y = (SHORT)(ScreenInfo->BufferInfo.TextInfo.CursorPosition.Y+1); Status = WWSB_AdjustCursorPosition(ScreenInfo,CursorPosition, (dwFlags & WC_KEEP_CURSOR_VISIBLE) != 0,ScrollY); break; default: #ifdef WWSB_FE
Char = *lpString; if (Char >= (WCHAR)' ' && IsConsoleFullWidth(ScreenInfo->Console->hDC, ScreenInfo->Console->OutputCP,Char) && XPosition >= (ScreenInfo->ScreenBufferSize.X-1) && (ScreenInfo->OutputMode & ENABLE_WRAP_AT_EOL_OUTPUT)) {
SHORT RowIndex; PROW Row; PWCHAR Char; COORD TargetPoint; PCHAR AttrP;
TargetPoint = ScreenInfo->BufferInfo.TextInfo.CursorPosition; RowIndex = (ScreenInfo->BufferInfo.TextInfo.FirstRow+TargetPoint.Y) % ScreenInfo->ScreenBufferSize.Y; Row = &ScreenInfo->BufferInfo.TextInfo.Rows[RowIndex]; Char = &Row->CharRow.Chars[TargetPoint.X]; AttrP = &Row->CharRow.KAttrs[TargetPoint.X];
if (*AttrP & ATTR_TRAILING_BYTE) { *(Char-1) = UNICODE_SPACE; *Char = UNICODE_SPACE; *AttrP = 0; *(AttrP-1) = 0;
Region.Left = ScreenInfo->BufferInfo.TextInfo.CursorPosition.X-1; Region.Right = (SHORT)(ScreenInfo->BufferInfo.TextInfo.CursorPosition.X); Region.Top = ScreenInfo->BufferInfo.TextInfo.CursorPosition.Y; Region.Bottom = ScreenInfo->BufferInfo.TextInfo.CursorPosition.Y; WWSB_WriteToScreen(ScreenInfo,&Region); }
CursorPosition.X = 0; CursorPosition.Y = (SHORT)(ScreenInfo->BufferInfo.TextInfo.CursorPosition.Y+1); Status = WWSB_AdjustCursorPosition(ScreenInfo,CursorPosition, dwFlags & WC_KEEP_CURSOR_VISIBLE,ScrollY); continue; } #endif
break; } if (!NT_SUCCESS(Status)) { ConsoleShowCursor(ScreenInfo); goto ExitWriteChars; }
*NumBytes += sizeof(WCHAR); lpString++; lpRealUnicodeString++; }
if (ARGUMENT_PRESENT(NumSpaces)) { *NumSpaces = TempNumSpaces; } ConsoleShowCursor(ScreenInfo);
Status = STATUS_SUCCESS;
ExitWriteChars: if (lpAllocatedString) { ConsoleHeapFree(lpAllocatedString); } return Status; }
ULONG WWSB_DoWriteConsole( IN OUT PCSR_API_MSG m, IN PCONSOLE_INFORMATION Console, IN PCSR_THREAD Thread )
//
// NOTE: console lock must be held when calling this routine
//
// string has been translated to unicode at this point
//
{ PCONSOLE_WRITECONSOLE_MSG a = (PCONSOLE_WRITECONSOLE_MSG)&m->u.ApiMessageData; PHANDLE_DATA HandleData; NTSTATUS Status; PSCREEN_INFORMATION ScreenInfo; DWORD NumCharsToWrite; #ifdef WWSB_FE
DWORD i; SHORT j; #endif
if (Console->Flags & (CONSOLE_SUSPENDED | CONSOLE_SELECTING | CONSOLE_SCROLLBAR_TRACKING)) { PWCHAR TransBuffer;
TransBuffer = ConsoleHeapAlloc(TMP_TAG, a->NumBytes); if (TransBuffer == NULL) { return (ULONG)STATUS_NO_MEMORY; } RtlCopyMemory(TransBuffer,a->TransBuffer,a->NumBytes); a->TransBuffer = TransBuffer; a->StackBuffer = FALSE; if (!CsrCreateWait(&Console->OutputQueue, WriteConsoleWaitRoutine, Thread, m, NULL)) { ConsoleHeapFree(TransBuffer); return (ULONG)STATUS_NO_MEMORY; } return (ULONG)CONSOLE_STATUS_WAIT; }
Status = DereferenceIoHandle(CONSOLE_FROMTHREADPERPROCESSDATA(Thread), a->OutputHandle, CONSOLE_OUTPUT_HANDLE, GENERIC_WRITE, &HandleData ); if (!NT_SUCCESS(Status)) { a->NumBytes = 0; return((ULONG) Status); }
ScreenInfo = HandleData->Buffer.ScreenBuffer;
//
// see if we're the typical case - a string containing no special
// characters, optionally terminated with CRLF. if so, skip the
// special processing.
//
NumCharsToWrite=a->NumBytes/sizeof(WCHAR); if ((ScreenInfo->OutputMode & ENABLE_PROCESSED_OUTPUT) && ((LONG)(ScreenInfo->BufferInfo.TextInfo.CursorPosition.X + NumCharsToWrite) < ScreenInfo->ScreenBufferSize.X) ) { SMALL_RECT Region; COORD CursorPosition;
if (a->Unicode) { #ifdef WWSB_FE
a->WriteFlags = WRITE_SPECIAL_CHARS; #else
a->WriteFlags = FastStreamWrite(a->TransBuffer,NumCharsToWrite); #endif
} if (a->WriteFlags == WRITE_SPECIAL_CHARS) { goto ProcessedWrite; }
ConsoleHideCursor(ScreenInfo);
//
// WriteFlags is designed so that the number of special characters
// is also the flag value.
//
NumCharsToWrite -= a->WriteFlags;
if (NumCharsToWrite) { #ifdef WWSB_FE
PWCHAR TransBuffer,TransBufPtr,String; PBYTE TransBufferA,TransBufPtrA; BOOL fLocalHeap = FALSE; COORD TargetPoint;
if (NumCharsToWrite > (ULONG)(ScreenInfo->ScreenBufferSize.X * ScreenInfo->ScreenBufferSize.Y)) {
TransBuffer = ConsoleHeapAlloc(TMP_DBCS_TAG, NumCharsToWrite * 2 * sizeof(WCHAR)); if (TransBuffer == NULL) { return (ULONG)STATUS_NO_MEMORY; } TransBufferA = ConsoleHeapAlloc(TMP_DBCS_TAG, NumCharsToWrite * 2 * sizeof(CHAR)); if (TransBufferA == NULL) { ConsoleHeapFree(TransBuffer); return (ULONG)STATUS_NO_MEMORY; }
fLocalHeap = TRUE; } else { TransBuffer = ScreenInfo->BufferInfo.TextInfo.DbcsScreenBuffer.TransBufferCharacter; TransBufferA = ScreenInfo->BufferInfo.TextInfo.DbcsScreenBuffer.TransBufferAttribute; }
String = a->TransBuffer; TransBufPtr = TransBuffer; TransBufPtrA = TransBufferA; for (i = 0 , j = 0 ; i < NumCharsToWrite ; i++,j++){ if (IsConsoleFullWidth(ScreenInfo->Console->hDC, ScreenInfo->Console->OutputCP,*String)){ *TransBuffer++ = *String ; *TransBufferA++ = ATTR_LEADING_BYTE; *TransBuffer++ = *String++ ; *TransBufferA++ = ATTR_TRAILING_BYTE; j++; } else{ *TransBuffer++ = *String++ ; *TransBufferA++ = 0; } } TargetPoint = ScreenInfo->BufferInfo.TextInfo.CursorPosition; BisectWrite(j,TargetPoint,ScreenInfo); if (TargetPoint.Y == ScreenInfo->ScreenBufferSize.Y-1 && TargetPoint.X+j >= ScreenInfo->ScreenBufferSize.X && *(TransBufPtrA+j) & ATTR_LEADING_BYTE){ *(TransBufPtr+ScreenInfo->ScreenBufferSize.X-TargetPoint.X-1) = UNICODE_SPACE; *(TransBufPtrA+ScreenInfo->ScreenBufferSize.X-TargetPoint.X-1) = 0; if (j > ScreenInfo->ScreenBufferSize.X-TargetPoint.X-1) { *(TransBufPtr+ScreenInfo->ScreenBufferSize.X-TargetPoint.X) = UNICODE_SPACE; *(TransBufPtrA+ScreenInfo->ScreenBufferSize.X-TargetPoint.X) = 0; } } FE_StreamWriteToScreenBuffer(TransBufPtr, (SHORT)j, ScreenInfo, TransBufPtrA ); if (fLocalHeap){ ConsoleHeapFree(TransBufPtr); ConsoleHeapFree(TransBufPtrA); } #else
SB_StreamWriteToScreenBuffer(a->TransBuffer, (SHORT)NumCharsToWrite, ScreenInfo ); #endif
Region.Left = ScreenInfo->BufferInfo.TextInfo.CursorPosition.X; #ifdef WWSB_FE
Region.Right = (SHORT)(ScreenInfo->BufferInfo.TextInfo.CursorPosition.X + j - 1); #else
Region.Right = (SHORT)(ScreenInfo->BufferInfo.TextInfo.CursorPosition.X + NumCharsToWrite - 1); #endif
Region.Top = ScreenInfo->BufferInfo.TextInfo.CursorPosition.Y; Region.Bottom = ScreenInfo->BufferInfo.TextInfo.CursorPosition.Y; ASSERT (Region.Right < ScreenInfo->ScreenBufferSize.X); if (ACTIVE_SCREEN_BUFFER(ScreenInfo) && !(ScreenInfo->Console->Flags & CONSOLE_IS_ICONIC && ScreenInfo->Console->FullScreenFlags == 0)) { WWSB_WriteRegionToScreen(ScreenInfo,&Region); } } switch (a->WriteFlags) { case WRITE_NO_CR_LF: CursorPosition.X = (SHORT)(ScreenInfo->BufferInfo.TextInfo.CursorPosition.X + NumCharsToWrite); CursorPosition.Y = ScreenInfo->BufferInfo.TextInfo.CursorPosition.Y; break; case WRITE_CR: CursorPosition.X = 0; CursorPosition.Y = ScreenInfo->BufferInfo.TextInfo.CursorPosition.Y; break; case WRITE_CR_LF: CursorPosition.X = 0; CursorPosition.Y = ScreenInfo->BufferInfo.TextInfo.CursorPosition.Y+1; break; default: ASSERT(FALSE); break; } Status = WWSB_AdjustCursorPosition(ScreenInfo,CursorPosition,FALSE,NULL); ConsoleShowCursor(ScreenInfo); return STATUS_SUCCESS; } ProcessedWrite: return WWSB_WriteChars(ScreenInfo, a->TransBuffer, a->TransBuffer, a->TransBuffer, &a->NumBytes, NULL, ScreenInfo->BufferInfo.TextInfo.CursorPosition.X, WC_LIMIT_BACKSPACE, NULL ); }
NTSTATUS WWSB_DoSrvWriteConsole( IN OUT PCSR_API_MSG m, IN OUT PCSR_REPLY_STATUS ReplyStatus, IN PCONSOLE_INFORMATION Console, IN PHANDLE_DATA HandleData ) { NTSTATUS Status = STATUS_SUCCESS; PCONSOLE_WRITECONSOLE_MSG a = (PCONSOLE_WRITECONSOLE_MSG)&m->u.ApiMessageData; PSCREEN_INFORMATION ScreenInfo; WCHAR StackBuffer[STACK_BUFFER_SIZE]; #ifdef WWSB_FE
BOOL fLocalHeap = FALSE; #endif
ScreenInfo = HandleData->Buffer.ScreenBuffer;
#ifdef WWSB_FE
// Check code for must CONSOLE_TEXTMODE_BUFFER !!
ASSERT(!(ScreenInfo->Flags & CONSOLE_GRAPHICS_BUFFER)); #endif
//
// if the string was passed in the message, rather than in
// a capture buffer, adjust the pointer.
//
if (a->BufferInMessage) { a->BufPtr = a->Buffer; }
//
// if ansi, translate string. for speed, we don't allocate a
// capture buffer if the ansi string was <= 80 chars. if it's
// greater than 80 / sizeof(WCHAR), the translated string won't
// fit into the capture buffer, so reset a->BufPtr to point to
// a heap buffer and set a->CaptureBufferSize so that we don't
// think the buffer is in the message.
//
if (!a->Unicode) { PWCHAR TransBuffer; DWORD Length; DWORD SpecialChars = 0; UINT Codepage; #ifdef WWSB_FE
PWCHAR TmpTransBuffer; ULONG NumBytes1 = 0; ULONG NumBytes2 = 0; #endif
if (a->NumBytes <= STACK_BUFFER_SIZE) { TransBuffer = StackBuffer; a->StackBuffer = TRUE; #ifdef WWSB_FE
TmpTransBuffer = TransBuffer; #endif
} #ifdef WWSB_FE
else if (a->NumBytes > (ULONG)(ScreenInfo->ScreenBufferSize.X * ScreenInfo->ScreenBufferSize.Y)) { TransBuffer = ConsoleHeapAlloc(TMP_DBCS_TAG, (a->NumBytes + 2) * sizeof(WCHAR)); if (TransBuffer == NULL) { return (ULONG)STATUS_NO_MEMORY; } TmpTransBuffer = TransBuffer; a->StackBuffer = FALSE; fLocalHeap = TRUE; } else { TransBuffer = ScreenInfo->BufferInfo.TextInfo.DbcsScreenBuffer.TransWriteConsole; TmpTransBuffer = TransBuffer; } #else
else { TransBuffer = ConsoleHeapAlloc(TMP_TAG, a->NumBytes * sizeof(WCHAR)); if (TransBuffer == NULL) { return (ULONG)STATUS_NO_MEMORY; } a->StackBuffer = FALSE; } #endif
//a->NumBytes = ConvertOutputToUnicode(Console->OutputCP,
// Buffer,
// a->NumBytes,
// TransBuffer,
// a->NumBytes);
// same as ConvertOutputToUnicode
#ifdef WWSB_FE
if (! ScreenInfo->WriteConsoleDbcsLeadByte[0]) { NumBytes1 = 0; NumBytes2 = a->NumBytes; } else { if (*(PUCHAR)a->BufPtr < (UCHAR)' ') { NumBytes1 = 0; NumBytes2 = a->NumBytes; } else if (a->NumBytes) { ScreenInfo->WriteConsoleDbcsLeadByte[1] = *(PCHAR)a->BufPtr; NumBytes1 = sizeof(ScreenInfo->WriteConsoleDbcsLeadByte); if (Console->OutputCP == OEMCP) { if ((ScreenInfo->Flags & CONSOLE_OEMFONT_DISPLAY) && ((Console->FullScreenFlags & CONSOLE_FULLSCREEN) == 0)) { /*
* Translate OEM characters into False Unicode for Window-mode * OEM font. If OutputCP != OEMCP, characters will not appear * correctly, because the OEM fonts are designed to support * only OEMCP (we can't switch fonts in Windowed mode). * Fullscreen or TT "Unicode" fonts should be used for * non-OEMCP output */ DBGCHARS(("SrvWriteConsole ACP->U %.*s\n", min(NumBytes1,10), a->BufPtr)); Status = RtlConsoleMultiByteToUnicodeN(TransBuffer, NumBytes1 * sizeof(WCHAR), &NumBytes1, ScreenInfo->WriteConsoleDbcsLeadByte, NumBytes1, &SpecialChars); } else { /*
* Good! We have Fullscreen or TT "Unicode" fonts, so convert * the OEM characters to real Unicode according to OutputCP. * First find out if any special chars are involved. */ DBGCHARS(("SrvWriteConsole %d->U %.*s\n", Console->OutputCP, min(NumBytes1,10), a->BufPtr)); NumBytes1 = sizeof(WCHAR) * MultiByteToWideChar(Console->OutputCP, 0, ScreenInfo->WriteConsoleDbcsLeadByte, NumBytes1, TransBuffer, NumBytes1); if (NumBytes1 == 0) { Status = STATUS_UNSUCCESSFUL; } } } else { if ((ScreenInfo->Flags & CONSOLE_OEMFONT_DISPLAY) && !(Console->FullScreenFlags & CONSOLE_FULLSCREEN)) { if (Console->OutputCP != WINDOWSCP) Codepage = USACP; else Codepage = WINDOWSCP; } else { Codepage = Console->OutputCP; }
if ((ScreenInfo->Flags & CONSOLE_OEMFONT_DISPLAY) && ((Console->FullScreenFlags & CONSOLE_FULLSCREEN) == 0)) { NumBytes1 = ConvertOutputToUnicode(Codepage, ScreenInfo->WriteConsoleDbcsLeadByte, NumBytes1, TransBuffer, NumBytes1); } else { NumBytes1 = MultiByteToWideChar(Console->OutputCP, 0, ScreenInfo->WriteConsoleDbcsLeadByte, NumBytes1, TransBuffer, NumBytes1); if (NumBytes1 == 0) { Status = STATUS_UNSUCCESSFUL; } } NumBytes1 *= sizeof(WCHAR); } TransBuffer++; (PCHAR)a->BufPtr += (NumBytes1 / sizeof(WCHAR)); NumBytes2 = a->NumBytes - 1; } else { NumBytes2 = 0; } ScreenInfo->WriteConsoleDbcsLeadByte[0] = 0; }
if (NumBytes2 && CheckBisectStringA(Console->OutputCP,a->BufPtr,NumBytes2,&Console->OutputCPInfo)) { ScreenInfo->WriteConsoleDbcsLeadByte[0] = *((PCHAR)a->BufPtr+NumBytes2-1); NumBytes2--; }
Length = NumBytes2; #else
Length = a->NumBytes; if (a->NumBytes >= 2 && ((PCHAR)a->BufPtr)[a->NumBytes-1] == '\n' && ((PCHAR)a->BufPtr)[a->NumBytes-2] == '\r') { Length -= 2; a->WriteFlags = WRITE_CR_LF; } else if (a->NumBytes >= 1 && ((PCHAR)a->BufPtr)[a->NumBytes-1] == '\r') { Length -= 1; a->WriteFlags = WRITE_CR; } else { a->WriteFlags = WRITE_NO_CR_LF; } #endif
if (Length != 0) { if (Console->OutputCP == OEMCP) { if ((ScreenInfo->Flags & CONSOLE_OEMFONT_DISPLAY) && ((Console->FullScreenFlags & CONSOLE_FULLSCREEN) == 0)) { /*
* Translate OEM characters into UnicodeOem for the Window-mode * OEM font. If OutputCP != OEMCP, characters will not appear * correctly, because the OEM fonts are designed to support * only OEMCP (we can't switch fonts in Windowed mode). * Fullscreen or TT "Unicode" fonts should be used for * non-OEMCP output */ DBGCHARS(("SrvWriteConsole ACP->U %.*s\n", min(Length,10), a->BufPtr)); Status = RtlConsoleMultiByteToUnicodeN(TransBuffer, Length * sizeof(WCHAR), &Length, a->BufPtr, Length, &SpecialChars); } else { /*
* Good! We have Fullscreen or TT "Unicode" fonts, so convert * the OEM characters to real Unicode according to OutputCP. * First find out if any special chars are involved. */ #ifdef WWSB_NOFE
UINT i; for (i = 0; i < Length; i++) { if (((PCHAR)a->BufPtr)[i] < 0x20) { SpecialChars = 1; break; } } #endif
DBGCHARS(("SrvWriteConsole %d->U %.*s\n", Console->OutputCP, min(Length,10), a->BufPtr)); Length = sizeof(WCHAR) * MultiByteToWideChar(Console->OutputCP, 0, a->BufPtr, Length, TransBuffer, Length); if (Length == 0) { Status = STATUS_UNSUCCESSFUL; } } } else { if ((ScreenInfo->Flags & CONSOLE_OEMFONT_DISPLAY) && !(Console->FullScreenFlags & CONSOLE_FULLSCREEN)) { if (Console->OutputCP != WINDOWSCP) Codepage = USACP; else Codepage = WINDOWSCP; } else { Codepage = Console->OutputCP; }
if ((ScreenInfo->Flags & CONSOLE_OEMFONT_DISPLAY) && ((Console->FullScreenFlags & CONSOLE_FULLSCREEN) == 0)) { Length = sizeof(WCHAR) * ConvertOutputToUnicode(Codepage, a->BufPtr, Length, TransBuffer, Length); } else { Length = sizeof(WCHAR) * MultiByteToWideChar(Console->OutputCP, 0, a->BufPtr, Length, TransBuffer, Length); if (Length == 0) { Status = STATUS_UNSUCCESSFUL; } }
#ifdef WWSB_NOFE
SpecialChars = 1; #endif
} }
#ifdef WWSB_FE
NumBytes2 = Length;
if ((NumBytes1+NumBytes2) == 0) { if (!a->StackBuffer && fLocalHeap) { ConsoleHeapFree(a->TransBuffer); } return Status; } #else
if (!NT_SUCCESS(Status)) { if (!a->StackBuffer) { ConsoleHeapFree(TransBuffer); } return Status; } #endif
#ifdef WWSB_FE
Console->WriteConOutNumBytesTemp = a->NumBytes; a->NumBytes = Console->WriteConOutNumBytesUnicode = NumBytes1 + NumBytes2; a->WriteFlags = WRITE_SPECIAL_CHARS; a->TransBuffer = TmpTransBuffer; #else
DBGOUTPUT(("TransBuffer=%lx, Length = %x(bytes), SpecialChars=%lx\n", TransBuffer, Length, SpecialChars)); a->NumBytes = Length + (a->WriteFlags * sizeof(WCHAR)); if (a->WriteFlags == WRITE_CR_LF) { TransBuffer[(Length+sizeof(WCHAR))/sizeof(WCHAR)] = UNICODE_LINEFEED; TransBuffer[Length/sizeof(WCHAR)] = UNICODE_CARRIAGERETURN; } else if (a->WriteFlags == WRITE_CR) { TransBuffer[Length/sizeof(WCHAR)] = UNICODE_CARRIAGERETURN; } if (SpecialChars) { // CRLF didn't get translated
a->WriteFlags = WRITE_SPECIAL_CHARS; } a->TransBuffer = TransBuffer; #endif
} else { if ((ScreenInfo->Flags & CONSOLE_OEMFONT_DISPLAY) && ((Console->FullScreenFlags & CONSOLE_FULLSCREEN) == 0)) { Status = RealUnicodeToFalseUnicode(a->BufPtr, a->NumBytes / sizeof(WCHAR), Console->OutputCP); if (!NT_SUCCESS(Status)) { return Status; } } a->WriteFlags = (DWORD)-1; a->TransBuffer = a->BufPtr; } Status = WWSB_DoWriteConsole(m,Console,CSR_SERVER_QUERYCLIENTTHREAD()); if (Status == CONSOLE_STATUS_WAIT) { *ReplyStatus = CsrReplyPending; return (ULONG)STATUS_SUCCESS; } else { if (!a->Unicode) { #ifdef WWSB_FE
if (a->NumBytes == Console->WriteConOutNumBytesUnicode) a->NumBytes = Console->WriteConOutNumBytesTemp; else a->NumBytes /= sizeof(WCHAR); if (!a->StackBuffer && fLocalHeap) { ConsoleHeapFree(a->TransBuffer); } #else
a->NumBytes /= sizeof(WCHAR); if (!a->StackBuffer) { ConsoleHeapFree(a->TransBuffer); } #endif
} } return Status; }
|