/*++ 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(MAKE_TAG( 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;jFlags & 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 = (PWCHAR)ConsoleHeapAlloc(MAKE_TAG( 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 = (PWCHAR)ConsoleHeapAlloc(MAKE_TAG( 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, 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 = (PWCHAR)ConsoleHeapAlloc(MAKE_TAG( TMP_DBCS_TAG ),NumCharsToWrite * 2 * sizeof(WCHAR)); if (TransBuffer == NULL) { return (ULONG)STATUS_NO_MEMORY; } TransBufferA = (PCHAR)ConsoleHeapAlloc(MAKE_TAG( 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 = (PWCHAR)ConsoleHeapAlloc(MAKE_TAG( 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 = (PWCHAR)ConsoleHeapAlloc(MAKE_TAG( 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; }