Windows2000/private/ntos/w32/ntcon/server/_stream.h

1359 lines
51 KiB
C
Raw Normal View History

2001-01-01 00:00:00 +01:00
/*++
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;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 = (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;
}