1 /* 2 * COPYRIGHT: See COPYING in the top level directory 3 * PROJECT: ReactOS Console Server DLL 4 * FILE: win32ss/user/winsrv/consrv/conoutput.c 5 * PURPOSE: General Console Output Functions 6 * PROGRAMMERS: Jeffrey Morlan 7 * Hermes Belusca-Maito (hermes.belusca@sfr.fr) 8 */ 9 10 /* INCLUDES *******************************************************************/ 11 12 #include "consrv.h" 13 14 #define NDEBUG 15 #include <debug.h> 16 17 /* PUBLIC SERVER APIS *********************************************************/ 18 19 /* 20 * FIXME: This function MUST be moved from condrv/conoutput.c because only 21 * consrv knows how to manipulate VDM screenbuffers. 22 */ 23 NTSTATUS NTAPI 24 ConDrvWriteConsoleOutputVDM(IN PCONSOLE Console, 25 IN PTEXTMODE_SCREEN_BUFFER Buffer, 26 IN PCHAR_CELL CharInfo/*Buffer*/, 27 IN COORD CharInfoSize, 28 IN PSMALL_RECT WriteRegion); 29 NTSTATUS NTAPI 30 ConDrvInvalidateBitMapRect(IN PCONSOLE Console, 31 IN PCONSOLE_SCREEN_BUFFER Buffer, 32 IN PSMALL_RECT Region); 33 CSR_API(SrvInvalidateBitMapRect) 34 { 35 NTSTATUS Status; 36 PCONSOLE_INVALIDATEDIBITS InvalidateDIBitsRequest = &((PCONSOLE_API_MESSAGE)ApiMessage)->Data.InvalidateDIBitsRequest; 37 PCONSOLE_SCREEN_BUFFER Buffer; 38 39 DPRINT("SrvInvalidateBitMapRect\n"); 40 41 Status = ConSrvGetScreenBuffer(ConsoleGetPerProcessData(CsrGetClientThread()->Process), 42 InvalidateDIBitsRequest->OutputHandle, 43 &Buffer, GENERIC_READ, TRUE); 44 if (!NT_SUCCESS(Status)) return Status; 45 46 /* In text-mode only, draw the VDM buffer if present */ 47 if (GetType(Buffer) == TEXTMODE_BUFFER && Buffer->Header.Console->VDMBuffer) 48 { 49 PTEXTMODE_SCREEN_BUFFER TextBuffer = (PTEXTMODE_SCREEN_BUFFER)Buffer; 50 51 /*Status =*/ ConDrvWriteConsoleOutputVDM(Buffer->Header.Console, 52 TextBuffer, 53 Buffer->Header.Console->VDMBuffer, 54 Buffer->Header.Console->VDMBufferSize, 55 &InvalidateDIBitsRequest->Region); 56 } 57 58 Status = ConDrvInvalidateBitMapRect(Buffer->Header.Console, 59 Buffer, 60 &InvalidateDIBitsRequest->Region); 61 62 ConSrvReleaseScreenBuffer(Buffer, TRUE); 63 return Status; 64 } 65 66 NTSTATUS NTAPI 67 ConDrvSetConsolePalette(IN PCONSOLE Console, 68 // IN PGRAPHICS_SCREEN_BUFFER Buffer, 69 IN PCONSOLE_SCREEN_BUFFER Buffer, 70 IN HPALETTE PaletteHandle, 71 IN UINT PaletteUsage); 72 CSR_API(SrvSetConsolePalette) 73 { 74 NTSTATUS Status; 75 PCONSOLE_SETPALETTE SetPaletteRequest = &((PCONSOLE_API_MESSAGE)ApiMessage)->Data.SetPaletteRequest; 76 // PGRAPHICS_SCREEN_BUFFER Buffer; 77 PCONSOLE_SCREEN_BUFFER Buffer; 78 79 DPRINT("SrvSetConsolePalette\n"); 80 81 // NOTE: Tests show that this function is used only for graphics screen buffers 82 // and otherwise it returns FALSE + sets last error to invalid handle. 83 // I think it's ridiculous, because if you are in text mode, simulating 84 // a change of VGA palette via DAC registers (done by a call to SetConsolePalette) 85 // cannot be done... So I allow it in ReactOS ! 86 /* 87 Status = ConSrvGetGraphicsBuffer(ConsoleGetPerProcessData(CsrGetClientThread()->Process), 88 SetPaletteRequest->OutputHandle, 89 &Buffer, GENERIC_WRITE, TRUE); 90 */ 91 Status = ConSrvGetScreenBuffer(ConsoleGetPerProcessData(CsrGetClientThread()->Process), 92 SetPaletteRequest->OutputHandle, 93 &Buffer, GENERIC_WRITE, TRUE); 94 if (!NT_SUCCESS(Status)) return Status; 95 96 /* 97 * Make the palette handle public, so that it can be 98 * used by other threads calling GDI functions on it. 99 * Indeed, the palette handle comes from a console app 100 * calling ourselves, running in CSRSS. 101 */ 102 NtUserConsoleControl(ConsoleMakePalettePublic, 103 &SetPaletteRequest->PaletteHandle, 104 sizeof(SetPaletteRequest->PaletteHandle)); 105 106 Status = ConDrvSetConsolePalette(Buffer->Header.Console, 107 Buffer, 108 SetPaletteRequest->PaletteHandle, 109 SetPaletteRequest->Usage); 110 111 ConSrvReleaseScreenBuffer(Buffer, TRUE); 112 return Status; 113 } 114 115 NTSTATUS NTAPI 116 ConDrvGetConsoleCursorInfo(IN PCONSOLE Console, 117 IN PTEXTMODE_SCREEN_BUFFER Buffer, 118 OUT PCONSOLE_CURSOR_INFO CursorInfo); 119 CSR_API(SrvGetConsoleCursorInfo) 120 { 121 NTSTATUS Status; 122 PCONSOLE_GETSETCURSORINFO CursorInfoRequest = &((PCONSOLE_API_MESSAGE)ApiMessage)->Data.CursorInfoRequest; 123 PTEXTMODE_SCREEN_BUFFER Buffer; 124 125 DPRINT("SrvGetConsoleCursorInfo\n"); 126 127 Status = ConSrvGetTextModeBuffer(ConsoleGetPerProcessData(CsrGetClientThread()->Process), 128 CursorInfoRequest->OutputHandle, 129 &Buffer, GENERIC_READ, TRUE); 130 if (!NT_SUCCESS(Status)) return Status; 131 132 Status = ConDrvGetConsoleCursorInfo(Buffer->Header.Console, 133 Buffer, 134 &CursorInfoRequest->Info); 135 136 ConSrvReleaseScreenBuffer(Buffer, TRUE); 137 return Status; 138 } 139 140 NTSTATUS NTAPI 141 ConDrvSetConsoleCursorInfo(IN PCONSOLE Console, 142 IN PTEXTMODE_SCREEN_BUFFER Buffer, 143 IN PCONSOLE_CURSOR_INFO CursorInfo); 144 CSR_API(SrvSetConsoleCursorInfo) 145 { 146 NTSTATUS Status; 147 PCONSOLE_GETSETCURSORINFO CursorInfoRequest = &((PCONSOLE_API_MESSAGE)ApiMessage)->Data.CursorInfoRequest; 148 PTEXTMODE_SCREEN_BUFFER Buffer; 149 150 DPRINT("SrvSetConsoleCursorInfo\n"); 151 152 Status = ConSrvGetTextModeBuffer(ConsoleGetPerProcessData(CsrGetClientThread()->Process), 153 CursorInfoRequest->OutputHandle, 154 &Buffer, GENERIC_WRITE, TRUE); 155 if (!NT_SUCCESS(Status)) return Status; 156 157 Status = ConDrvSetConsoleCursorInfo(Buffer->Header.Console, 158 Buffer, 159 &CursorInfoRequest->Info); 160 161 ConSrvReleaseScreenBuffer(Buffer, TRUE); 162 return Status; 163 } 164 165 NTSTATUS NTAPI 166 ConDrvSetConsoleCursorPosition(IN PCONSOLE Console, 167 IN PTEXTMODE_SCREEN_BUFFER Buffer, 168 IN PCOORD Position); 169 CSR_API(SrvSetConsoleCursorPosition) 170 { 171 NTSTATUS Status; 172 PCONSOLE_SETCURSORPOSITION SetCursorPositionRequest = &((PCONSOLE_API_MESSAGE)ApiMessage)->Data.SetCursorPositionRequest; 173 PTEXTMODE_SCREEN_BUFFER Buffer; 174 175 DPRINT("SrvSetConsoleCursorPosition\n"); 176 177 Status = ConSrvGetTextModeBuffer(ConsoleGetPerProcessData(CsrGetClientThread()->Process), 178 SetCursorPositionRequest->OutputHandle, 179 &Buffer, GENERIC_WRITE, TRUE); 180 if (!NT_SUCCESS(Status)) return Status; 181 182 Status = ConDrvSetConsoleCursorPosition(Buffer->Header.Console, 183 Buffer, 184 &SetCursorPositionRequest->Position); 185 186 ConSrvReleaseScreenBuffer(Buffer, TRUE); 187 return Status; 188 } 189 190 CSR_API(SrvCreateConsoleScreenBuffer) 191 { 192 NTSTATUS Status = STATUS_INVALID_PARAMETER; 193 PCONSOLE_CREATESCREENBUFFER CreateScreenBufferRequest = &((PCONSOLE_API_MESSAGE)ApiMessage)->Data.CreateScreenBufferRequest; 194 PCSR_PROCESS Process = CsrGetClientThread()->Process; 195 PCONSOLE_PROCESS_DATA ProcessData = ConsoleGetPerProcessData(Process); 196 PCONSRV_CONSOLE Console; 197 PCONSOLE_SCREEN_BUFFER Buff; 198 199 PVOID ScreenBufferInfo = NULL; 200 TEXTMODE_BUFFER_INFO TextModeInfo = {{80, 25}, 201 DEFAULT_SCREEN_ATTRIB, 202 DEFAULT_POPUP_ATTRIB , 203 TRUE, 204 CSR_DEFAULT_CURSOR_SIZE}; 205 GRAPHICS_BUFFER_INFO GraphicsInfo; 206 GraphicsInfo.Info = CreateScreenBufferRequest->GraphicsBufferInfo; // HACK for MSVC 207 208 DPRINT("SrvCreateConsoleScreenBuffer\n"); 209 210 Status = ConSrvGetConsole(ProcessData, &Console, TRUE); 211 if (!NT_SUCCESS(Status)) return Status; 212 213 if (CreateScreenBufferRequest->ScreenBufferType == CONSOLE_TEXTMODE_BUFFER) 214 { 215 ScreenBufferInfo = &TextModeInfo; 216 217 /* 218 if (Console->ActiveBuffer) 219 { 220 TextModeInfo.ScreenBufferSize = Console->ActiveBuffer->ScreenBufferSize; 221 if (TextModeInfo.ScreenBufferSize.X == 0) TextModeInfo.ScreenBufferSize.X = 80; 222 if (TextModeInfo.ScreenBufferSize.Y == 0) TextModeInfo.ScreenBufferSize.Y = 25; 223 224 TextModeInfo.ScreenAttrib = Console->ActiveBuffer->ScreenBuffer.TextBuffer.ScreenDefaultAttrib; 225 TextModeInfo.PopupAttrib = Console->ActiveBuffer->ScreenBuffer.TextBuffer.PopupDefaultAttrib; 226 227 TextModeInfo.IsCursorVisible = Console->ActiveBuffer->CursorInfo.bVisible; 228 TextModeInfo.CursorSize = Console->ActiveBuffer->CursorInfo.dwSize; 229 } 230 */ 231 232 /* 233 * This is Windows' behaviour 234 */ 235 236 /* Use the current console size. Regularize it if needed. */ 237 TextModeInfo.ScreenBufferSize = Console->ConsoleSize; 238 if (TextModeInfo.ScreenBufferSize.X == 0) TextModeInfo.ScreenBufferSize.X = 1; 239 if (TextModeInfo.ScreenBufferSize.Y == 0) TextModeInfo.ScreenBufferSize.Y = 1; 240 241 /* If we have an active screen buffer, use its attributes as the new ones */ 242 if (Console->ActiveBuffer && GetType(Console->ActiveBuffer) == TEXTMODE_BUFFER) 243 { 244 PTEXTMODE_SCREEN_BUFFER Buffer = (PTEXTMODE_SCREEN_BUFFER)Console->ActiveBuffer; 245 246 TextModeInfo.ScreenAttrib = Buffer->ScreenDefaultAttrib; 247 TextModeInfo.PopupAttrib = Buffer->PopupDefaultAttrib; 248 249 TextModeInfo.IsCursorVisible = Buffer->CursorInfo.bVisible; 250 TextModeInfo.CursorSize = Buffer->CursorInfo.dwSize; 251 } 252 } 253 else if (CreateScreenBufferRequest->ScreenBufferType == CONSOLE_GRAPHICS_BUFFER) 254 { 255 /* Get infos from the graphics buffer information structure */ 256 if (!CsrValidateMessageBuffer(ApiMessage, 257 (PVOID*)&CreateScreenBufferRequest->GraphicsBufferInfo.lpBitMapInfo, 258 CreateScreenBufferRequest->GraphicsBufferInfo.dwBitMapInfoLength, 259 sizeof(BYTE))) 260 { 261 Status = STATUS_INVALID_PARAMETER; 262 goto Quit; 263 } 264 265 ScreenBufferInfo = &GraphicsInfo; 266 267 /* Initialize shared variables */ 268 // CreateScreenBufferRequest->GraphicsBufferInfo.hMutex 269 CreateScreenBufferRequest->hMutex = GraphicsInfo.Info.hMutex = INVALID_HANDLE_VALUE; 270 // CreateScreenBufferRequest->GraphicsBufferInfo.lpBitMap 271 CreateScreenBufferRequest->lpBitMap = GraphicsInfo.Info.lpBitMap = NULL; 272 273 /* A graphics screen buffer is never inheritable */ 274 CreateScreenBufferRequest->InheritHandle = FALSE; 275 } 276 277 Status = ConDrvCreateScreenBuffer(&Buff, 278 (PCONSOLE)Console, 279 Process->ProcessHandle, 280 CreateScreenBufferRequest->ScreenBufferType, 281 ScreenBufferInfo); 282 if (!NT_SUCCESS(Status)) goto Quit; 283 284 /* Insert the new handle inside the process handles table */ 285 RtlEnterCriticalSection(&ProcessData->HandleTableLock); 286 287 Status = ConSrvInsertObject(ProcessData, 288 &CreateScreenBufferRequest->OutputHandle, 289 &Buff->Header, 290 CreateScreenBufferRequest->DesiredAccess, 291 CreateScreenBufferRequest->InheritHandle, 292 CreateScreenBufferRequest->ShareMode); 293 294 RtlLeaveCriticalSection(&ProcessData->HandleTableLock); 295 296 if (!NT_SUCCESS(Status)) goto Quit; 297 298 if (CreateScreenBufferRequest->ScreenBufferType == CONSOLE_GRAPHICS_BUFFER) 299 { 300 PGRAPHICS_SCREEN_BUFFER Buffer = (PGRAPHICS_SCREEN_BUFFER)Buff; 301 /* 302 * Initialize the graphics buffer information structure 303 * and give it back to the client. 304 */ 305 // CreateScreenBufferRequest->GraphicsBufferInfo.hMutex 306 CreateScreenBufferRequest->hMutex = Buffer->ClientMutex; 307 // CreateScreenBufferRequest->GraphicsBufferInfo.lpBitMap 308 CreateScreenBufferRequest->lpBitMap = Buffer->ClientBitMap; 309 } 310 311 Quit: 312 ConSrvReleaseConsole(Console, TRUE); 313 return Status; 314 } 315 316 NTSTATUS NTAPI 317 ConDrvSetConsoleActiveScreenBuffer(IN PCONSOLE Console, 318 IN PCONSOLE_SCREEN_BUFFER Buffer); 319 CSR_API(SrvSetConsoleActiveScreenBuffer) 320 { 321 NTSTATUS Status; 322 PCONSOLE_SETACTIVESCREENBUFFER SetScreenBufferRequest = &((PCONSOLE_API_MESSAGE)ApiMessage)->Data.SetScreenBufferRequest; 323 PCONSOLE_SCREEN_BUFFER Buffer; 324 325 DPRINT("SrvSetConsoleActiveScreenBuffer\n"); 326 327 Status = ConSrvGetScreenBuffer(ConsoleGetPerProcessData(CsrGetClientThread()->Process), 328 SetScreenBufferRequest->OutputHandle, 329 &Buffer, GENERIC_WRITE, TRUE); 330 if (!NT_SUCCESS(Status)) return Status; 331 332 Status = ConDrvSetConsoleActiveScreenBuffer(Buffer->Header.Console, 333 Buffer); 334 335 ConSrvReleaseScreenBuffer(Buffer, TRUE); 336 return Status; 337 } 338 339 340 /* CSR THREADS FOR WriteConsole ***********************************************/ 341 342 static NTSTATUS 343 DoWriteConsole(IN PCSR_API_MESSAGE ApiMessage, 344 IN PCSR_THREAD ClientThread, 345 IN BOOLEAN CreateWaitBlock OPTIONAL); 346 347 // Wait function CSR_WAIT_FUNCTION 348 static BOOLEAN 349 NTAPI 350 WriteConsoleThread(IN PLIST_ENTRY WaitList, 351 IN PCSR_THREAD WaitThread, 352 IN PCSR_API_MESSAGE WaitApiMessage, 353 IN PVOID WaitContext, 354 IN PVOID WaitArgument1, 355 IN PVOID WaitArgument2, 356 IN ULONG WaitFlags) 357 { 358 NTSTATUS Status; 359 360 DPRINT("WriteConsoleThread - WaitContext = 0x%p, WaitArgument1 = 0x%p, WaitArgument2 = 0x%p, WaitFlags = %lu\n", WaitContext, WaitArgument1, WaitArgument2, WaitFlags); 361 362 /* 363 * If we are notified of the process termination via a call 364 * to CsrNotifyWaitBlock triggered by CsrDestroyProcess or 365 * CsrDestroyThread, just return. 366 */ 367 if (WaitFlags & CsrProcessTerminating) 368 { 369 Status = STATUS_THREAD_IS_TERMINATING; 370 goto Quit; 371 } 372 373 Status = DoWriteConsole(WaitApiMessage, WaitThread, FALSE); 374 375 Quit: 376 if (Status != STATUS_PENDING) 377 { 378 WaitApiMessage->Status = Status; 379 } 380 381 return (Status == STATUS_PENDING ? FALSE : TRUE); 382 } 383 384 NTSTATUS NTAPI 385 ConDrvWriteConsole(IN PCONSOLE Console, 386 IN PTEXTMODE_SCREEN_BUFFER ScreenBuffer, 387 IN BOOLEAN Unicode, 388 IN PVOID StringBuffer, 389 IN ULONG NumCharsToWrite, 390 OUT PULONG NumCharsWritten OPTIONAL); 391 static NTSTATUS 392 DoWriteConsole(IN PCSR_API_MESSAGE ApiMessage, 393 IN PCSR_THREAD ClientThread, 394 IN BOOLEAN CreateWaitBlock OPTIONAL) 395 { 396 NTSTATUS Status; 397 PCONSOLE_WRITECONSOLE WriteConsoleRequest = &((PCONSOLE_API_MESSAGE)ApiMessage)->Data.WriteConsoleRequest; 398 PTEXTMODE_SCREEN_BUFFER ScreenBuffer; 399 400 PVOID Buffer; 401 ULONG NrCharactersWritten = 0; 402 ULONG CharSize = (WriteConsoleRequest->Unicode ? sizeof(WCHAR) : sizeof(CHAR)); 403 404 Status = ConSrvGetTextModeBuffer(ConsoleGetPerProcessData(ClientThread->Process), 405 WriteConsoleRequest->OutputHandle, 406 &ScreenBuffer, GENERIC_WRITE, FALSE); 407 if (!NT_SUCCESS(Status)) return Status; 408 409 /* 410 * For optimization purposes, Windows (and hence ReactOS, too, for 411 * compatibility reasons) uses a static buffer if no more than eighty 412 * bytes are written. Otherwise a new buffer is used. 413 * The client-side expects that we know this behaviour. 414 */ 415 if (WriteConsoleRequest->UsingStaticBuffer && 416 WriteConsoleRequest->NumBytes <= sizeof(WriteConsoleRequest->StaticBuffer)) 417 { 418 /* 419 * Adjust the internal pointer, because its old value points to 420 * the static buffer in the original ApiMessage structure. 421 */ 422 // WriteConsoleRequest->Buffer = WriteConsoleRequest->StaticBuffer; 423 Buffer = WriteConsoleRequest->StaticBuffer; 424 } 425 else 426 { 427 Buffer = WriteConsoleRequest->Buffer; 428 } 429 430 DPRINT("Calling ConDrvWriteConsole\n"); 431 Status = ConDrvWriteConsole(ScreenBuffer->Header.Console, 432 ScreenBuffer, 433 WriteConsoleRequest->Unicode, 434 Buffer, 435 WriteConsoleRequest->NumBytes / CharSize, // NrCharactersToWrite 436 &NrCharactersWritten); 437 DPRINT("ConDrvWriteConsole returned (%d ; Status = 0x%08x)\n", 438 NrCharactersWritten, Status); 439 440 if (Status == STATUS_PENDING) 441 { 442 if (CreateWaitBlock) 443 { 444 PCONSRV_CONSOLE Console = (PCONSRV_CONSOLE)ScreenBuffer->Header.Console; 445 446 if (!CsrCreateWait(&Console->WriteWaitQueue, 447 WriteConsoleThread, 448 ClientThread, 449 ApiMessage, 450 NULL)) 451 { 452 /* Fail */ 453 Status = STATUS_NO_MEMORY; 454 goto Quit; 455 } 456 } 457 458 /* Wait until we un-pause the console */ 459 // Status = STATUS_PENDING; 460 } 461 else 462 { 463 /* We read all what we wanted. Set the number of bytes written. */ 464 WriteConsoleRequest->NumBytes = NrCharactersWritten * CharSize; 465 } 466 467 Quit: 468 ConSrvReleaseScreenBuffer(ScreenBuffer, FALSE); 469 return Status; 470 } 471 472 473 /* TEXT OUTPUT APIS ***********************************************************/ 474 475 NTSTATUS NTAPI 476 ConDrvReadConsoleOutput(IN PCONSOLE Console, 477 IN PTEXTMODE_SCREEN_BUFFER Buffer, 478 IN BOOLEAN Unicode, 479 OUT PCHAR_INFO CharInfo/*Buffer*/, 480 IN OUT PSMALL_RECT ReadRegion); 481 CSR_API(SrvReadConsoleOutput) 482 { 483 NTSTATUS Status; 484 PCONSOLE_READOUTPUT ReadOutputRequest = &((PCONSOLE_API_MESSAGE)ApiMessage)->Data.ReadOutputRequest; 485 PTEXTMODE_SCREEN_BUFFER Buffer; 486 487 ULONG NumCells; 488 PCHAR_INFO CharInfo; 489 490 DPRINT("SrvReadConsoleOutput\n"); 491 492 NumCells = (ReadOutputRequest->ReadRegion.Right - ReadOutputRequest->ReadRegion.Left + 1) * 493 (ReadOutputRequest->ReadRegion.Bottom - ReadOutputRequest->ReadRegion.Top + 1); 494 495 /* 496 * For optimization purposes, Windows (and hence ReactOS, too, for 497 * compatibility reasons) uses a static buffer if no more than one 498 * cell is read. Otherwise a new buffer is used. 499 * The client-side expects that we know this behaviour. 500 */ 501 if (NumCells <= 1) 502 { 503 /* 504 * Adjust the internal pointer, because its old value points to 505 * the static buffer in the original ApiMessage structure. 506 */ 507 // ReadOutputRequest->CharInfo = &ReadOutputRequest->StaticBuffer; 508 CharInfo = &ReadOutputRequest->StaticBuffer; 509 } 510 else 511 { 512 if (!CsrValidateMessageBuffer(ApiMessage, 513 (PVOID*)&ReadOutputRequest->CharInfo, 514 NumCells, 515 sizeof(CHAR_INFO))) 516 { 517 return STATUS_INVALID_PARAMETER; 518 } 519 520 CharInfo = ReadOutputRequest->CharInfo; 521 } 522 523 Status = ConSrvGetTextModeBuffer(ConsoleGetPerProcessData(CsrGetClientThread()->Process), 524 ReadOutputRequest->OutputHandle, 525 &Buffer, GENERIC_READ, TRUE); 526 if (!NT_SUCCESS(Status)) return Status; 527 528 Status = ConDrvReadConsoleOutput(Buffer->Header.Console, 529 Buffer, 530 ReadOutputRequest->Unicode, 531 CharInfo, 532 &ReadOutputRequest->ReadRegion); 533 534 ConSrvReleaseScreenBuffer(Buffer, TRUE); 535 return Status; 536 } 537 538 NTSTATUS NTAPI 539 ConDrvWriteConsoleOutput(IN PCONSOLE Console, 540 IN PTEXTMODE_SCREEN_BUFFER Buffer, 541 IN BOOLEAN Unicode, 542 IN PCHAR_INFO CharInfo/*Buffer*/, 543 IN OUT PSMALL_RECT WriteRegion); 544 CSR_API(SrvWriteConsoleOutput) 545 { 546 NTSTATUS Status; 547 PCONSOLE_WRITEOUTPUT WriteOutputRequest = &((PCONSOLE_API_MESSAGE)ApiMessage)->Data.WriteOutputRequest; 548 PTEXTMODE_SCREEN_BUFFER Buffer; 549 PCSR_PROCESS Process = CsrGetClientThread()->Process; 550 551 ULONG NumCells; 552 PCHAR_INFO CharInfo; 553 554 DPRINT("SrvWriteConsoleOutput\n"); 555 556 NumCells = (WriteOutputRequest->WriteRegion.Right - WriteOutputRequest->WriteRegion.Left + 1) * 557 (WriteOutputRequest->WriteRegion.Bottom - WriteOutputRequest->WriteRegion.Top + 1); 558 559 Status = ConSrvGetTextModeBuffer(ConsoleGetPerProcessData(Process), 560 WriteOutputRequest->OutputHandle, 561 &Buffer, GENERIC_WRITE, TRUE); 562 if (!NT_SUCCESS(Status)) return Status; 563 564 /* 565 * Validate the message buffer if we do not use a process' heap buffer 566 * (CsrAllocateCaptureBuffer succeeded because we haven't allocated 567 * a too large (>= 64 kB, size of the CSR heap) data buffer). 568 */ 569 if (!WriteOutputRequest->UseVirtualMemory) 570 { 571 /* 572 * For optimization purposes, Windows (and hence ReactOS, too, for 573 * compatibility reasons) uses a static buffer if no more than one 574 * cell is written. Otherwise a new buffer is used. 575 * The client-side expects that we know this behaviour. 576 */ 577 if (NumCells <= 1) 578 { 579 /* 580 * Adjust the internal pointer, because its old value points to 581 * the static buffer in the original ApiMessage structure. 582 */ 583 // WriteOutputRequest->CharInfo = &WriteOutputRequest->StaticBuffer; 584 CharInfo = &WriteOutputRequest->StaticBuffer; 585 } 586 else 587 { 588 if (!CsrValidateMessageBuffer(ApiMessage, 589 (PVOID*)&WriteOutputRequest->CharInfo, 590 NumCells, 591 sizeof(CHAR_INFO))) 592 { 593 Status = STATUS_INVALID_PARAMETER; 594 goto Quit; 595 } 596 597 CharInfo = WriteOutputRequest->CharInfo; 598 } 599 } 600 else 601 { 602 /* 603 * This was not the case: we use a heap buffer. Retrieve its contents. 604 */ 605 ULONG Size = NumCells * sizeof(CHAR_INFO); 606 607 CharInfo = ConsoleAllocHeap(HEAP_ZERO_MEMORY, Size); 608 if (CharInfo == NULL) 609 { 610 Status = STATUS_NO_MEMORY; 611 goto Quit; 612 } 613 614 Status = NtReadVirtualMemory(Process->ProcessHandle, 615 WriteOutputRequest->CharInfo, 616 CharInfo, 617 Size, 618 NULL); 619 if (!NT_SUCCESS(Status)) 620 { 621 ConsoleFreeHeap(CharInfo); 622 // Status = STATUS_NO_MEMORY; 623 goto Quit; 624 } 625 } 626 627 Status = ConDrvWriteConsoleOutput(Buffer->Header.Console, 628 Buffer, 629 WriteOutputRequest->Unicode, 630 CharInfo, 631 &WriteOutputRequest->WriteRegion); 632 633 /* Free the temporary buffer if we used the process' heap buffer */ 634 if (WriteOutputRequest->UseVirtualMemory && CharInfo) 635 ConsoleFreeHeap(CharInfo); 636 637 Quit: 638 ConSrvReleaseScreenBuffer(Buffer, TRUE); 639 return Status; 640 } 641 642 CSR_API(SrvWriteConsole) 643 { 644 NTSTATUS Status; 645 PCONSOLE_WRITECONSOLE WriteConsoleRequest = &((PCONSOLE_API_MESSAGE)ApiMessage)->Data.WriteConsoleRequest; 646 647 DPRINT("SrvWriteConsole\n"); 648 649 /* 650 * For optimization purposes, Windows (and hence ReactOS, too, for 651 * compatibility reasons) uses a static buffer if no more than eighty 652 * bytes are written. Otherwise a new buffer is used. 653 * The client-side expects that we know this behaviour. 654 */ 655 if (WriteConsoleRequest->UsingStaticBuffer && 656 WriteConsoleRequest->NumBytes <= sizeof(WriteConsoleRequest->StaticBuffer)) 657 { 658 /* 659 * Adjust the internal pointer, because its old value points to 660 * the static buffer in the original ApiMessage structure. 661 */ 662 // WriteConsoleRequest->Buffer = WriteConsoleRequest->StaticBuffer; 663 } 664 else 665 { 666 if (!CsrValidateMessageBuffer(ApiMessage, 667 (PVOID)&WriteConsoleRequest->Buffer, 668 WriteConsoleRequest->NumBytes, 669 sizeof(BYTE))) 670 { 671 return STATUS_INVALID_PARAMETER; 672 } 673 } 674 675 Status = DoWriteConsole(ApiMessage, CsrGetClientThread(), TRUE); 676 677 if (Status == STATUS_PENDING) *ReplyCode = CsrReplyPending; 678 679 return Status; 680 } 681 682 NTSTATUS NTAPI 683 ConDrvReadConsoleOutputString(IN PCONSOLE Console, 684 IN PTEXTMODE_SCREEN_BUFFER Buffer, 685 IN CODE_TYPE CodeType, 686 OUT PVOID StringBuffer, 687 IN ULONG NumCodesToRead, 688 IN PCOORD ReadCoord, 689 // OUT PCOORD EndCoord, 690 OUT PULONG NumCodesRead OPTIONAL); 691 CSR_API(SrvReadConsoleOutputString) 692 { 693 NTSTATUS Status; 694 PCONSOLE_READOUTPUTCODE ReadOutputCodeRequest = &((PCONSOLE_API_MESSAGE)ApiMessage)->Data.ReadOutputCodeRequest; 695 PTEXTMODE_SCREEN_BUFFER Buffer; 696 ULONG CodeSize; 697 698 PVOID pCode; 699 700 DPRINT("SrvReadConsoleOutputString\n"); 701 702 switch (ReadOutputCodeRequest->CodeType) 703 { 704 case CODE_ASCII: 705 CodeSize = RTL_FIELD_SIZE(CODE_ELEMENT, AsciiChar); 706 break; 707 708 case CODE_UNICODE: 709 CodeSize = RTL_FIELD_SIZE(CODE_ELEMENT, UnicodeChar); 710 break; 711 712 case CODE_ATTRIBUTE: 713 CodeSize = RTL_FIELD_SIZE(CODE_ELEMENT, Attribute); 714 break; 715 716 default: 717 return STATUS_INVALID_PARAMETER; 718 } 719 720 /* 721 * For optimization purposes, Windows (and hence ReactOS, too, for 722 * compatibility reasons) uses a static buffer if no more than eighty 723 * bytes are read. Otherwise a new buffer is used. 724 * The client-side expects that we know this behaviour. 725 */ 726 if (ReadOutputCodeRequest->NumCodes * CodeSize <= sizeof(ReadOutputCodeRequest->CodeStaticBuffer)) 727 { 728 /* 729 * Adjust the internal pointer, because its old value points to 730 * the static buffer in the original ApiMessage structure. 731 */ 732 // ReadOutputCodeRequest->pCode = ReadOutputCodeRequest->CodeStaticBuffer; 733 pCode = ReadOutputCodeRequest->CodeStaticBuffer; 734 } 735 else 736 { 737 if (!CsrValidateMessageBuffer(ApiMessage, 738 (PVOID*)&ReadOutputCodeRequest->pCode, 739 ReadOutputCodeRequest->NumCodes, 740 CodeSize)) 741 { 742 return STATUS_INVALID_PARAMETER; 743 } 744 745 pCode = ReadOutputCodeRequest->pCode; 746 } 747 748 Status = ConSrvGetTextModeBuffer(ConsoleGetPerProcessData(CsrGetClientThread()->Process), 749 ReadOutputCodeRequest->OutputHandle, 750 &Buffer, GENERIC_READ, TRUE); 751 if (!NT_SUCCESS(Status)) 752 { 753 ReadOutputCodeRequest->NumCodes = 0; 754 return Status; 755 } 756 757 Status = ConDrvReadConsoleOutputString(Buffer->Header.Console, 758 Buffer, 759 ReadOutputCodeRequest->CodeType, 760 pCode, 761 ReadOutputCodeRequest->NumCodes, 762 &ReadOutputCodeRequest->Coord, 763 // &ReadOutputCodeRequest->EndCoord, 764 &ReadOutputCodeRequest->NumCodes); 765 766 ConSrvReleaseScreenBuffer(Buffer, TRUE); 767 return Status; 768 } 769 770 NTSTATUS NTAPI 771 ConDrvWriteConsoleOutputString(IN PCONSOLE Console, 772 IN PTEXTMODE_SCREEN_BUFFER Buffer, 773 IN CODE_TYPE CodeType, 774 IN PVOID StringBuffer, 775 IN ULONG NumCodesToWrite, 776 IN PCOORD WriteCoord, 777 // OUT PCOORD EndCoord, 778 OUT PULONG NumCodesWritten OPTIONAL); 779 CSR_API(SrvWriteConsoleOutputString) 780 { 781 NTSTATUS Status; 782 PCONSOLE_WRITEOUTPUTCODE WriteOutputCodeRequest = &((PCONSOLE_API_MESSAGE)ApiMessage)->Data.WriteOutputCodeRequest; 783 PTEXTMODE_SCREEN_BUFFER Buffer; 784 ULONG CodeSize; 785 786 PVOID pCode; 787 788 DPRINT("SrvWriteConsoleOutputString\n"); 789 790 switch (WriteOutputCodeRequest->CodeType) 791 { 792 case CODE_ASCII: 793 CodeSize = RTL_FIELD_SIZE(CODE_ELEMENT, AsciiChar); 794 break; 795 796 case CODE_UNICODE: 797 CodeSize = RTL_FIELD_SIZE(CODE_ELEMENT, UnicodeChar); 798 break; 799 800 case CODE_ATTRIBUTE: 801 CodeSize = RTL_FIELD_SIZE(CODE_ELEMENT, Attribute); 802 break; 803 804 default: 805 return STATUS_INVALID_PARAMETER; 806 } 807 808 /* 809 * For optimization purposes, Windows (and hence ReactOS, too, for 810 * compatibility reasons) uses a static buffer if no more than eighty 811 * bytes are written. Otherwise a new buffer is used. 812 * The client-side expects that we know this behaviour. 813 */ 814 if (WriteOutputCodeRequest->NumCodes * CodeSize <= sizeof(WriteOutputCodeRequest->CodeStaticBuffer)) 815 { 816 /* 817 * Adjust the internal pointer, because its old value points to 818 * the static buffer in the original ApiMessage structure. 819 */ 820 // WriteOutputCodeRequest->pCode = WriteOutputCodeRequest->CodeStaticBuffer; 821 pCode = WriteOutputCodeRequest->CodeStaticBuffer; 822 } 823 else 824 { 825 if (!CsrValidateMessageBuffer(ApiMessage, 826 (PVOID*)&WriteOutputCodeRequest->pCode, 827 WriteOutputCodeRequest->NumCodes, 828 CodeSize)) 829 { 830 return STATUS_INVALID_PARAMETER; 831 } 832 833 pCode = WriteOutputCodeRequest->pCode; 834 } 835 836 Status = ConSrvGetTextModeBuffer(ConsoleGetPerProcessData(CsrGetClientThread()->Process), 837 WriteOutputCodeRequest->OutputHandle, 838 &Buffer, GENERIC_WRITE, TRUE); 839 if (!NT_SUCCESS(Status)) 840 { 841 WriteOutputCodeRequest->NumCodes = 0; 842 return Status; 843 } 844 845 Status = ConDrvWriteConsoleOutputString(Buffer->Header.Console, 846 Buffer, 847 WriteOutputCodeRequest->CodeType, 848 pCode, 849 WriteOutputCodeRequest->NumCodes, 850 &WriteOutputCodeRequest->Coord, 851 // &WriteOutputCodeRequest->EndCoord, 852 &WriteOutputCodeRequest->NumCodes); 853 854 ConSrvReleaseScreenBuffer(Buffer, TRUE); 855 return Status; 856 } 857 858 NTSTATUS NTAPI 859 ConDrvFillConsoleOutput(IN PCONSOLE Console, 860 IN PTEXTMODE_SCREEN_BUFFER Buffer, 861 IN CODE_TYPE CodeType, 862 IN CODE_ELEMENT Code, 863 IN ULONG NumCodesToWrite, 864 IN PCOORD WriteCoord, 865 OUT PULONG NumCodesWritten OPTIONAL); 866 CSR_API(SrvFillConsoleOutput) 867 { 868 NTSTATUS Status; 869 PCONSOLE_FILLOUTPUTCODE FillOutputRequest = &((PCONSOLE_API_MESSAGE)ApiMessage)->Data.FillOutputRequest; 870 PTEXTMODE_SCREEN_BUFFER Buffer; 871 CODE_TYPE CodeType = FillOutputRequest->CodeType; 872 873 DPRINT("SrvFillConsoleOutput\n"); 874 875 if ( (CodeType != CODE_ASCII ) && 876 (CodeType != CODE_UNICODE ) && 877 (CodeType != CODE_ATTRIBUTE) ) 878 { 879 return STATUS_INVALID_PARAMETER; 880 } 881 882 Status = ConSrvGetTextModeBuffer(ConsoleGetPerProcessData(CsrGetClientThread()->Process), 883 FillOutputRequest->OutputHandle, 884 &Buffer, GENERIC_WRITE, TRUE); 885 if (!NT_SUCCESS(Status)) 886 { 887 FillOutputRequest->NumCodes = 0; 888 return Status; 889 } 890 891 Status = ConDrvFillConsoleOutput(Buffer->Header.Console, 892 Buffer, 893 CodeType, 894 FillOutputRequest->Code, 895 FillOutputRequest->NumCodes, 896 &FillOutputRequest->WriteCoord, 897 &FillOutputRequest->NumCodes); 898 899 ConSrvReleaseScreenBuffer(Buffer, TRUE); 900 return Status; 901 } 902 903 NTSTATUS NTAPI 904 ConDrvGetConsoleScreenBufferInfo(IN PCONSOLE Console, 905 IN PTEXTMODE_SCREEN_BUFFER Buffer, 906 OUT PCOORD ScreenBufferSize, 907 OUT PCOORD CursorPosition, 908 OUT PCOORD ViewOrigin, 909 OUT PCOORD ViewSize, 910 OUT PCOORD MaximumViewSize, 911 OUT PWORD Attributes); 912 CSR_API(SrvGetConsoleScreenBufferInfo) 913 { 914 NTSTATUS Status; 915 PCONSOLE_GETSCREENBUFFERINFO ScreenBufferInfoRequest = &((PCONSOLE_API_MESSAGE)ApiMessage)->Data.ScreenBufferInfoRequest; 916 PTEXTMODE_SCREEN_BUFFER Buffer; 917 918 DPRINT("SrvGetConsoleScreenBufferInfo\n"); 919 920 Status = ConSrvGetTextModeBuffer(ConsoleGetPerProcessData(CsrGetClientThread()->Process), 921 ScreenBufferInfoRequest->OutputHandle, 922 &Buffer, GENERIC_READ, TRUE); 923 if (!NT_SUCCESS(Status)) return Status; 924 925 Status = ConDrvGetConsoleScreenBufferInfo(Buffer->Header.Console, 926 Buffer, 927 &ScreenBufferInfoRequest->ScreenBufferSize, 928 &ScreenBufferInfoRequest->CursorPosition, 929 &ScreenBufferInfoRequest->ViewOrigin, 930 &ScreenBufferInfoRequest->ViewSize, 931 &ScreenBufferInfoRequest->MaximumViewSize, 932 &ScreenBufferInfoRequest->Attributes); 933 934 ConSrvReleaseScreenBuffer(Buffer, TRUE); 935 return Status; 936 } 937 938 NTSTATUS NTAPI 939 ConDrvSetConsoleTextAttribute(IN PCONSOLE Console, 940 IN PTEXTMODE_SCREEN_BUFFER Buffer, 941 IN WORD Attributes); 942 CSR_API(SrvSetConsoleTextAttribute) 943 { 944 NTSTATUS Status; 945 PCONSOLE_SETTEXTATTRIB SetTextAttribRequest = &((PCONSOLE_API_MESSAGE)ApiMessage)->Data.SetTextAttribRequest; 946 PTEXTMODE_SCREEN_BUFFER Buffer; 947 948 DPRINT("SrvSetConsoleTextAttribute\n"); 949 950 Status = ConSrvGetTextModeBuffer(ConsoleGetPerProcessData(CsrGetClientThread()->Process), 951 SetTextAttribRequest->OutputHandle, 952 &Buffer, GENERIC_WRITE, TRUE); 953 if (!NT_SUCCESS(Status)) return Status; 954 955 Status = ConDrvSetConsoleTextAttribute(Buffer->Header.Console, 956 Buffer, 957 SetTextAttribRequest->Attributes); 958 959 ConSrvReleaseScreenBuffer(Buffer, TRUE); 960 return Status; 961 } 962 963 NTSTATUS NTAPI 964 ConDrvSetConsoleScreenBufferSize(IN PCONSOLE Console, 965 IN PTEXTMODE_SCREEN_BUFFER Buffer, 966 IN PCOORD Size); 967 CSR_API(SrvSetConsoleScreenBufferSize) 968 { 969 NTSTATUS Status; 970 PCONSOLE_SETSCREENBUFFERSIZE SetScreenBufferSizeRequest = &((PCONSOLE_API_MESSAGE)ApiMessage)->Data.SetScreenBufferSizeRequest; 971 PTEXTMODE_SCREEN_BUFFER Buffer; 972 973 DPRINT("SrvSetConsoleScreenBufferSize\n"); 974 975 Status = ConSrvGetTextModeBuffer(ConsoleGetPerProcessData(CsrGetClientThread()->Process), 976 SetScreenBufferSizeRequest->OutputHandle, 977 &Buffer, GENERIC_WRITE, TRUE); 978 if (!NT_SUCCESS(Status)) return Status; 979 980 Status = ConDrvSetConsoleScreenBufferSize(Buffer->Header.Console, 981 Buffer, 982 &SetScreenBufferSizeRequest->Size); 983 984 ConSrvReleaseScreenBuffer(Buffer, TRUE); 985 return Status; 986 } 987 988 NTSTATUS NTAPI 989 ConDrvScrollConsoleScreenBuffer(IN PCONSOLE Console, 990 IN PTEXTMODE_SCREEN_BUFFER Buffer, 991 IN BOOLEAN Unicode, 992 IN PSMALL_RECT ScrollRectangle, 993 IN BOOLEAN UseClipRectangle, 994 IN PSMALL_RECT ClipRectangle OPTIONAL, 995 IN PCOORD DestinationOrigin, 996 IN CHAR_INFO FillChar); 997 CSR_API(SrvScrollConsoleScreenBuffer) 998 { 999 NTSTATUS Status; 1000 PCONSOLE_SCROLLSCREENBUFFER ScrollScreenBufferRequest = &((PCONSOLE_API_MESSAGE)ApiMessage)->Data.ScrollScreenBufferRequest; 1001 PTEXTMODE_SCREEN_BUFFER Buffer; 1002 1003 DPRINT("SrvScrollConsoleScreenBuffer\n"); 1004 1005 Status = ConSrvGetTextModeBuffer(ConsoleGetPerProcessData(CsrGetClientThread()->Process), 1006 ScrollScreenBufferRequest->OutputHandle, 1007 &Buffer, GENERIC_WRITE, TRUE); 1008 if (!NT_SUCCESS(Status)) return Status; 1009 1010 Status = ConDrvScrollConsoleScreenBuffer(Buffer->Header.Console, 1011 Buffer, 1012 ScrollScreenBufferRequest->Unicode, 1013 &ScrollScreenBufferRequest->ScrollRectangle, 1014 ScrollScreenBufferRequest->UseClipRectangle, 1015 &ScrollScreenBufferRequest->ClipRectangle, 1016 &ScrollScreenBufferRequest->DestinationOrigin, 1017 ScrollScreenBufferRequest->Fill); 1018 1019 ConSrvReleaseScreenBuffer(Buffer, TRUE); 1020 return Status; 1021 } 1022 1023 NTSTATUS NTAPI 1024 ConDrvSetConsoleWindowInfo(IN PCONSOLE Console, 1025 IN PTEXTMODE_SCREEN_BUFFER Buffer, 1026 IN BOOLEAN Absolute, 1027 IN PSMALL_RECT WindowRect); 1028 CSR_API(SrvSetConsoleWindowInfo) 1029 { 1030 NTSTATUS Status; 1031 PCONSOLE_SETWINDOWINFO SetWindowInfoRequest = &((PCONSOLE_API_MESSAGE)ApiMessage)->Data.SetWindowInfoRequest; 1032 // PCONSOLE_SCREEN_BUFFER Buffer; 1033 PTEXTMODE_SCREEN_BUFFER Buffer; 1034 1035 DPRINT1("SrvSetConsoleWindowInfo(0x%08x, %d, {L%d, T%d, R%d, B%d}) called\n", 1036 SetWindowInfoRequest->OutputHandle, SetWindowInfoRequest->Absolute, 1037 SetWindowInfoRequest->WindowRect.Left , 1038 SetWindowInfoRequest->WindowRect.Top , 1039 SetWindowInfoRequest->WindowRect.Right, 1040 SetWindowInfoRequest->WindowRect.Bottom); 1041 1042 // ConSrvGetScreenBuffer 1043 Status = ConSrvGetTextModeBuffer(ConsoleGetPerProcessData(CsrGetClientThread()->Process), 1044 SetWindowInfoRequest->OutputHandle, 1045 &Buffer, GENERIC_READ, TRUE); 1046 if (!NT_SUCCESS(Status)) return Status; 1047 1048 Status = ConDrvSetConsoleWindowInfo(Buffer->Header.Console, 1049 Buffer, 1050 SetWindowInfoRequest->Absolute, 1051 &SetWindowInfoRequest->WindowRect); 1052 1053 ConSrvReleaseScreenBuffer(Buffer, TRUE); 1054 return Status; 1055 } 1056 1057 /* EOF */ 1058