1 /* 2 * COPYRIGHT: See COPYING in the top level directory 3 * PROJECT: ReactOS system libraries 4 * FILE: dll/win32/kernel32/client/console/console.c 5 * PURPOSE: Win32 server console functions 6 * PROGRAMMERS: James Tabor <jimtabor@adsl-64-217-116-74.dsl.hstntx.swbell.net> 7 * Hermes Belusca-Maito (hermes.belusca@sfr.fr) 8 */ 9 10 /* INCLUDES *******************************************************************/ 11 12 #include <k32.h> 13 14 #define NDEBUG 15 #include <debug.h> 16 17 18 /* GLOBALS ********************************************************************/ 19 20 extern RTL_CRITICAL_SECTION ConsoleLock; 21 extern BOOLEAN ConsoleInitialized; 22 23 /* Console reserved "file" names */ 24 static LPCWSTR BaseConFileName = CONSOLE_FILE_NAME; 25 static LPCWSTR BaseConInputFileName = CONSOLE_INPUT_FILE_NAME; 26 static LPCWSTR BaseConOutputFileName = CONSOLE_OUTPUT_FILE_NAME; 27 28 /* Console Control handling */ 29 static PHANDLER_ROUTINE InitialHandler[1]; 30 static PHANDLER_ROUTINE* CtrlHandlers; 31 static ULONG NrCtrlHandlers; 32 static ULONG NrAllocatedHandlers; 33 static BOOLEAN LastCloseNotify = FALSE; 34 35 extern BOOL WINAPI IsDebuggerPresent(VOID); 36 37 /* Console Input facilities */ 38 HANDLE InputWaitHandle = INVALID_HANDLE_VALUE; 39 40 #define EXENAME_LENGTH (255 + 1) 41 static RTL_CRITICAL_SECTION ExeNameLock; 42 static BOOLEAN ExeNameInitialized; 43 static WCHAR ExeNameBuffer[EXENAME_LENGTH]; // NULL-terminated 44 static USHORT ExeNameLength; // Count in number of characters without NULL 45 static WCHAR StartDirBuffer[MAX_PATH + 1]; // NULL-terminated 46 static USHORT StartDirLength; // Count in number of characters without NULL 47 48 49 /* Default Console Control Handler ********************************************/ 50 51 static BOOL 52 WINAPI 53 DefaultConsoleCtrlHandler(DWORD Event) 54 { 55 DPRINT("Default handler called: %lx\n", Event); 56 switch(Event) 57 { 58 case CTRL_C_EVENT: 59 DPRINT("Ctrl-C Event\n"); 60 break; 61 62 case CTRL_BREAK_EVENT: 63 DPRINT("Ctrl-Break Event\n"); 64 break; 65 66 case CTRL_CLOSE_EVENT: 67 DPRINT("Ctrl Close Event\n"); 68 break; 69 70 case CTRL_LAST_CLOSE_EVENT: 71 DPRINT("Ctrl Last Close Event\n"); 72 break; 73 74 case CTRL_LOGOFF_EVENT: 75 DPRINT("Ctrl Logoff Event\n"); 76 break; 77 78 case CTRL_SHUTDOWN_EVENT: 79 DPRINT("Ctrl Shutdown Event\n"); 80 break; 81 } 82 83 ExitProcess(CONTROL_C_EXIT); 84 return TRUE; 85 } 86 87 DWORD 88 WINAPI 89 ConsoleControlDispatcher(IN LPVOID lpThreadParameter) 90 { 91 DWORD nExitCode = 0; 92 DWORD CodeAndFlag = PtrToUlong(lpThreadParameter); 93 DWORD nCode = CodeAndFlag & MAXLONG; 94 UINT i; 95 EXCEPTION_RECORD erException; 96 97 DPRINT1("Console Dispatcher Active: %lx %lx\n", CodeAndFlag, nCode); 98 SetThreadPriority(GetCurrentThread(), THREAD_PRIORITY_HIGHEST); 99 100 switch(nCode) 101 { 102 case CTRL_C_EVENT: 103 case CTRL_BREAK_EVENT: 104 { 105 if (IsDebuggerPresent()) 106 { 107 erException.ExceptionCode = (nCode == CTRL_C_EVENT ? 108 DBG_CONTROL_C : DBG_CONTROL_BREAK); 109 erException.ExceptionFlags = 0; 110 erException.ExceptionRecord = NULL; 111 erException.ExceptionAddress = DefaultConsoleCtrlHandler; 112 erException.NumberParameters = 0; 113 114 _SEH2_TRY 115 { 116 RtlRaiseException(&erException); 117 } 118 _SEH2_EXCEPT(EXCEPTION_EXECUTE_HANDLER) 119 { 120 RtlEnterCriticalSection(&ConsoleLock); 121 122 if ((nCode != CTRL_C_EVENT) || 123 (NtCurrentPeb()->ProcessParameters->ConsoleFlags != 1)) 124 { 125 for (i = NrCtrlHandlers; i > 0; i--) 126 { 127 if (CtrlHandlers[i - 1](nCode)) break; 128 } 129 } 130 131 RtlLeaveCriticalSection(&ConsoleLock); 132 } 133 _SEH2_END; 134 135 ExitThread(0); 136 } 137 break; 138 } 139 140 case CTRL_CLOSE_EVENT: 141 case CTRL_LOGOFF_EVENT: 142 case CTRL_SHUTDOWN_EVENT: 143 break; 144 145 case CTRL_LAST_CLOSE_EVENT: 146 /* 147 * In case the console app hasn't register for last close notification, 148 * just kill this console handler thread. We don't want that such apps 149 * get killed for unexpected reasons. On the contrary apps that registered 150 * can be killed because they expect to be. 151 */ 152 if (!LastCloseNotify) ExitThread(0); 153 break; 154 155 case 4: 156 ExitProcess(CONTROL_C_EXIT); 157 break; 158 159 default: 160 ASSERT(FALSE); 161 break; 162 } 163 164 ASSERT(ConsoleInitialized); 165 166 RtlEnterCriticalSection(&ConsoleLock); 167 168 nExitCode = 0; 169 if ((nCode != CTRL_C_EVENT) || (NtCurrentPeb()->ProcessParameters->ConsoleFlags != 1)) 170 { 171 for (i = NrCtrlHandlers; i > 0; i--) 172 { 173 if ((i == 1) && 174 (CodeAndFlag & MINLONG) && 175 ((nCode == CTRL_LOGOFF_EVENT) || (nCode == CTRL_SHUTDOWN_EVENT))) 176 { 177 DPRINT("Skipping system/service apps\n"); 178 break; 179 } 180 181 if (CtrlHandlers[i - 1](nCode)) 182 { 183 switch(nCode) 184 { 185 case CTRL_CLOSE_EVENT: 186 case CTRL_LAST_CLOSE_EVENT: 187 case CTRL_LOGOFF_EVENT: 188 case CTRL_SHUTDOWN_EVENT: 189 nExitCode = CodeAndFlag; 190 break; 191 } 192 break; 193 } 194 } 195 } 196 197 RtlLeaveCriticalSection(&ConsoleLock); 198 199 ExitThread(nExitCode); 200 return STATUS_SUCCESS; 201 } 202 203 VOID 204 InitializeCtrlHandling(VOID) 205 { 206 /* Initialize Console Ctrl Handler */ 207 NrAllocatedHandlers = NrCtrlHandlers = 1; 208 CtrlHandlers = InitialHandler; 209 CtrlHandlers[0] = DefaultConsoleCtrlHandler; 210 } 211 212 213 /* Input EXE Name Support *****************************************************/ 214 215 VOID 216 InitExeName(VOID) 217 { 218 NTSTATUS Status; 219 PPEB Peb = NtCurrentPeb(); 220 PCURDIR CurrentDirectory = &Peb->ProcessParameters->CurrentDirectory; 221 PLDR_DATA_TABLE_ENTRY ImageEntry; 222 223 if (ExeNameInitialized) return; 224 225 /* Initialize the EXE name lock */ 226 Status = RtlInitializeCriticalSection(&ExeNameLock); 227 if (!NT_SUCCESS(Status)) return; 228 ExeNameInitialized = TRUE; 229 230 ImageEntry = CONTAINING_RECORD(Peb->Ldr->InLoadOrderModuleList.Flink, 231 LDR_DATA_TABLE_ENTRY, 232 InLoadOrderLinks); 233 234 /* Retrieve the EXE name, NULL-terminate it... */ 235 ExeNameLength = min(sizeof(ExeNameBuffer)/sizeof(ExeNameBuffer[0]), 236 ImageEntry->BaseDllName.Length / sizeof(WCHAR)); 237 RtlCopyMemory(ExeNameBuffer, 238 ImageEntry->BaseDllName.Buffer, 239 ImageEntry->BaseDllName.Length); 240 ExeNameBuffer[ExeNameLength] = UNICODE_NULL; 241 242 /* ... and retrieve the current directory path and NULL-terminate it. */ 243 StartDirLength = min(sizeof(StartDirBuffer)/sizeof(StartDirBuffer[0]), 244 CurrentDirectory->DosPath.Length / sizeof(WCHAR)); 245 RtlCopyMemory(StartDirBuffer, 246 CurrentDirectory->DosPath.Buffer, 247 CurrentDirectory->DosPath.Length); 248 StartDirBuffer[StartDirLength] = UNICODE_NULL; 249 } 250 251 /* 252 * NOTE: 253 * The "LPDWORD Length" parameters point on input to the maximum size of 254 * the buffers that can hold data (if != 0), and on output they hold the 255 * real size of the data. If "Length" are == 0 on input, then on output 256 * they receive the full size of the data. 257 * The "LPWSTR* String" parameters have a double meaning: 258 * - when "CaptureStrings" is TRUE, data is copied to the buffers pointed 259 * by the pointers (*String). 260 * - when "CaptureStrings" is FALSE, "*String" are set to the addresses of 261 * the source data. 262 */ 263 VOID 264 SetUpAppName(IN BOOLEAN CaptureStrings, 265 IN OUT LPDWORD CurDirLength, 266 IN OUT LPWSTR* CurDir, 267 IN OUT LPDWORD AppNameLength, 268 IN OUT LPWSTR* AppName) 269 { 270 DWORD Length; 271 272 /* Retrieve the needed buffer size */ 273 Length = (StartDirLength + 1) * sizeof(WCHAR); 274 if (*CurDirLength > 0) Length = min(Length, *CurDirLength); 275 *CurDirLength = Length; 276 277 /* Capture the data if needed, or, return a pointer to it */ 278 if (CaptureStrings) 279 { 280 /* 281 * Length is always >= sizeof(WCHAR). Copy everything but the 282 * possible trailing NULL character, and then NULL-terminate. 283 */ 284 Length -= sizeof(WCHAR); 285 RtlCopyMemory(*CurDir, StartDirBuffer, Length); 286 (*CurDir)[Length / sizeof(WCHAR)] = UNICODE_NULL; 287 } 288 else 289 { 290 *CurDir = StartDirBuffer; 291 } 292 293 /* Retrieve the needed buffer size */ 294 Length = (ExeNameLength + 1) * sizeof(WCHAR); 295 if (*AppNameLength > 0) Length = min(Length, *AppNameLength); 296 *AppNameLength = Length; 297 298 /* Capture the data if needed, or, return a pointer to it */ 299 if (CaptureStrings) 300 { 301 /* 302 * Length is always >= sizeof(WCHAR). Copy everything but the 303 * possible trailing NULL character, and then NULL-terminate. 304 */ 305 Length -= sizeof(WCHAR); 306 RtlCopyMemory(*AppName, ExeNameBuffer, Length); 307 (*AppName)[Length / sizeof(WCHAR)] = UNICODE_NULL; 308 } 309 else 310 { 311 *AppName = ExeNameBuffer; 312 } 313 } 314 315 USHORT 316 GetCurrentExeName(OUT PWCHAR ExeName, 317 IN USHORT BufferSize) 318 { 319 USHORT ExeLength; 320 321 if (ExeNameInitialized) 322 { 323 RtlEnterCriticalSection(&ExeNameLock); 324 325 if (BufferSize > ExeNameLength * sizeof(WCHAR)) 326 BufferSize = ExeNameLength * sizeof(WCHAR); 327 328 RtlCopyMemory(ExeName, ExeNameBuffer, BufferSize); 329 330 RtlLeaveCriticalSection(&ExeNameLock); 331 ExeLength = BufferSize; 332 } 333 else 334 { 335 *ExeName = UNICODE_NULL; 336 ExeLength = 0; 337 } 338 339 return ExeLength; 340 } 341 342 /* FUNCTIONS ******************************************************************/ 343 344 LPCWSTR 345 IntCheckForConsoleFileName(IN LPCWSTR pszName, 346 IN DWORD dwDesiredAccess) 347 { 348 LPCWSTR ConsoleName = pszName; 349 ULONG DeviceNameInfo; 350 351 /* 352 * Check whether we deal with a DOS device, and if so, 353 * strip the path till the file name. 354 * Therefore, things like \\.\CON or C:\some_path\CONIN$ 355 * are transformed into CON or CONIN$, for example. 356 */ 357 DeviceNameInfo = RtlIsDosDeviceName_U(pszName); 358 if (DeviceNameInfo != 0) 359 { 360 ConsoleName = (LPCWSTR)((ULONG_PTR)ConsoleName + ((DeviceNameInfo >> 16) & 0xFFFF)); 361 } 362 363 /* Return a standard console "file" name according to what we passed in parameters */ 364 if (_wcsicmp(ConsoleName, BaseConInputFileName) == 0) 365 { 366 return BaseConInputFileName; 367 } 368 else if (_wcsicmp(ConsoleName, BaseConOutputFileName) == 0) 369 { 370 return BaseConOutputFileName; 371 } 372 else if (_wcsicmp(ConsoleName, BaseConFileName) == 0) 373 { 374 if ((dwDesiredAccess & (GENERIC_READ | GENERIC_WRITE)) == GENERIC_READ) 375 { 376 return BaseConInputFileName; 377 } 378 else if ((dwDesiredAccess & (GENERIC_READ | GENERIC_WRITE)) == GENERIC_WRITE) 379 { 380 return BaseConOutputFileName; 381 } 382 } 383 384 /* If we are there, that means that either the file name or the desired access are wrong */ 385 return NULL; 386 } 387 388 389 /* 390 * @implemented (Undocumented) 391 * @note See http://undoc.airesoft.co.uk/kernel32.dll/ConsoleMenuControl.php 392 */ 393 HMENU 394 WINAPI 395 DECLSPEC_HOTPATCH 396 ConsoleMenuControl(HANDLE hConsoleOutput, 397 DWORD dwCmdIdLow, 398 DWORD dwCmdIdHigh) 399 { 400 CONSOLE_API_MESSAGE ApiMessage; 401 PCONSOLE_MENUCONTROL MenuControlRequest = &ApiMessage.Data.MenuControlRequest; 402 403 MenuControlRequest->ConsoleHandle = NtCurrentPeb()->ProcessParameters->ConsoleHandle; 404 MenuControlRequest->OutputHandle = hConsoleOutput; 405 MenuControlRequest->CmdIdLow = dwCmdIdLow; 406 MenuControlRequest->CmdIdHigh = dwCmdIdHigh; 407 MenuControlRequest->MenuHandle = NULL; 408 409 CsrClientCallServer((PCSR_API_MESSAGE)&ApiMessage, 410 NULL, 411 CSR_CREATE_API_NUMBER(CONSRV_SERVERDLL_INDEX, ConsolepMenuControl), 412 sizeof(*MenuControlRequest)); 413 414 return MenuControlRequest->MenuHandle; 415 } 416 417 418 /* 419 * @implemented 420 */ 421 HANDLE 422 WINAPI 423 DECLSPEC_HOTPATCH 424 DuplicateConsoleHandle(HANDLE hConsole, 425 DWORD dwDesiredAccess, 426 BOOL bInheritHandle, 427 DWORD dwOptions) 428 { 429 CONSOLE_API_MESSAGE ApiMessage; 430 PCONSOLE_DUPLICATEHANDLE DuplicateHandleRequest = &ApiMessage.Data.DuplicateHandleRequest; 431 432 if ( (dwOptions & ~(DUPLICATE_CLOSE_SOURCE | DUPLICATE_SAME_ACCESS)) || 433 (!(dwOptions & DUPLICATE_SAME_ACCESS) && 434 (dwDesiredAccess & ~(GENERIC_READ | GENERIC_WRITE))) ) 435 { 436 SetLastError(ERROR_INVALID_PARAMETER); 437 return INVALID_HANDLE_VALUE; 438 } 439 440 DuplicateHandleRequest->ConsoleHandle = NtCurrentPeb()->ProcessParameters->ConsoleHandle; 441 DuplicateHandleRequest->SourceHandle = hConsole; 442 DuplicateHandleRequest->DesiredAccess = dwDesiredAccess; 443 DuplicateHandleRequest->InheritHandle = bInheritHandle; 444 DuplicateHandleRequest->Options = dwOptions; 445 446 CsrClientCallServer((PCSR_API_MESSAGE)&ApiMessage, 447 NULL, 448 CSR_CREATE_API_NUMBER(CONSRV_SERVERDLL_INDEX, ConsolepDuplicateHandle), 449 sizeof(*DuplicateHandleRequest)); 450 if (!NT_SUCCESS(ApiMessage.Status)) 451 { 452 BaseSetLastNTError(ApiMessage.Status); 453 return INVALID_HANDLE_VALUE; 454 } 455 456 return DuplicateHandleRequest->TargetHandle; 457 } 458 459 460 /* 461 * @implemented 462 */ 463 BOOL 464 WINAPI 465 GetConsoleHandleInformation(IN HANDLE hHandle, 466 OUT LPDWORD lpdwFlags) 467 { 468 CONSOLE_API_MESSAGE ApiMessage; 469 PCONSOLE_GETHANDLEINFO GetHandleInfoRequest = &ApiMessage.Data.GetHandleInfoRequest; 470 471 GetHandleInfoRequest->ConsoleHandle = NtCurrentPeb()->ProcessParameters->ConsoleHandle; 472 GetHandleInfoRequest->Handle = hHandle; 473 474 CsrClientCallServer((PCSR_API_MESSAGE)&ApiMessage, 475 NULL, 476 CSR_CREATE_API_NUMBER(CONSRV_SERVERDLL_INDEX, ConsolepGetHandleInformation), 477 sizeof(*GetHandleInfoRequest)); 478 if (!NT_SUCCESS(ApiMessage.Status)) 479 { 480 BaseSetLastNTError(ApiMessage.Status); 481 return FALSE; 482 } 483 484 *lpdwFlags = GetHandleInfoRequest->Flags; 485 486 return TRUE; 487 } 488 489 490 /* 491 * @implemented 492 */ 493 BOOL 494 WINAPI 495 SetConsoleHandleInformation(IN HANDLE hHandle, 496 IN DWORD dwMask, 497 IN DWORD dwFlags) 498 { 499 CONSOLE_API_MESSAGE ApiMessage; 500 PCONSOLE_SETHANDLEINFO SetHandleInfoRequest = &ApiMessage.Data.SetHandleInfoRequest; 501 502 SetHandleInfoRequest->ConsoleHandle = NtCurrentPeb()->ProcessParameters->ConsoleHandle; 503 SetHandleInfoRequest->Handle = hHandle; 504 SetHandleInfoRequest->Mask = dwMask; 505 SetHandleInfoRequest->Flags = dwFlags; 506 507 CsrClientCallServer((PCSR_API_MESSAGE)&ApiMessage, 508 NULL, 509 CSR_CREATE_API_NUMBER(CONSRV_SERVERDLL_INDEX, ConsolepSetHandleInformation), 510 sizeof(*SetHandleInfoRequest)); 511 if (!NT_SUCCESS(ApiMessage.Status)) 512 { 513 BaseSetLastNTError(ApiMessage.Status); 514 return FALSE; 515 } 516 517 return TRUE; 518 } 519 520 521 /* 522 * @implemented 523 */ 524 BOOL 525 WINAPI 526 GetConsoleDisplayMode(LPDWORD lpModeFlags) 527 { 528 CONSOLE_API_MESSAGE ApiMessage; 529 PCONSOLE_GETDISPLAYMODE GetDisplayModeRequest = &ApiMessage.Data.GetDisplayModeRequest; 530 531 if (lpModeFlags == NULL) 532 { 533 SetLastError(ERROR_INVALID_PARAMETER); 534 return FALSE; 535 } 536 537 GetDisplayModeRequest->ConsoleHandle = NtCurrentPeb()->ProcessParameters->ConsoleHandle; 538 539 CsrClientCallServer((PCSR_API_MESSAGE)&ApiMessage, 540 NULL, 541 CSR_CREATE_API_NUMBER(CONSRV_SERVERDLL_INDEX, ConsolepGetDisplayMode), 542 sizeof(*GetDisplayModeRequest)); 543 if (!NT_SUCCESS(ApiMessage.Status)) 544 { 545 BaseSetLastNTError(ApiMessage.Status); 546 return FALSE; 547 } 548 549 *lpModeFlags = GetDisplayModeRequest->DisplayMode; // ModeFlags 550 551 return TRUE; 552 } 553 554 555 /* 556 * @implemented (Undocumented) 557 * @note See http://cboard.cprogramming.com/windows-programming/102187-console-font-size.html 558 */ 559 DWORD 560 WINAPI 561 GetConsoleFontInfo(IN HANDLE hConsoleOutput, 562 IN BOOL bMaximumWindow, 563 IN DWORD nFontCount, 564 OUT PCONSOLE_FONT_INFO lpConsoleFontInfo) 565 { 566 CONSOLE_API_MESSAGE ApiMessage; 567 PCONSOLE_GETFONTINFO GetFontInfoRequest = &ApiMessage.Data.GetFontInfoRequest; 568 PCSR_CAPTURE_BUFFER CaptureBuffer; 569 570 GetFontInfoRequest->ConsoleHandle = NtCurrentPeb()->ProcessParameters->ConsoleHandle; 571 GetFontInfoRequest->OutputHandle = hConsoleOutput; 572 GetFontInfoRequest->MaximumWindow = bMaximumWindow; 573 GetFontInfoRequest->NumFonts = nFontCount; 574 575 CaptureBuffer = CsrAllocateCaptureBuffer(1, nFontCount * sizeof(CONSOLE_FONT_INFO)); 576 if (CaptureBuffer == NULL) 577 { 578 DPRINT1("CsrAllocateCaptureBuffer failed!\n"); 579 SetLastError(ERROR_NOT_ENOUGH_MEMORY); 580 return 0; 581 } 582 583 CsrAllocateMessagePointer(CaptureBuffer, 584 nFontCount * sizeof(CONSOLE_FONT_INFO), 585 (PVOID*)&GetFontInfoRequest->FontInfo); 586 587 CsrClientCallServer((PCSR_API_MESSAGE)&ApiMessage, 588 CaptureBuffer, 589 CSR_CREATE_API_NUMBER(CONSRV_SERVERDLL_INDEX, ConsolepGetFontInfo), 590 sizeof(*GetFontInfoRequest)); 591 if (!NT_SUCCESS(ApiMessage.Status)) 592 { 593 BaseSetLastNTError(ApiMessage.Status); 594 } 595 else 596 { 597 RtlCopyMemory(lpConsoleFontInfo, 598 GetFontInfoRequest->FontInfo, 599 GetFontInfoRequest->NumFonts * sizeof(CONSOLE_FONT_INFO)); 600 } 601 602 CsrFreeCaptureBuffer(CaptureBuffer); 603 return GetFontInfoRequest->NumFonts; 604 } 605 606 607 /* 608 * @implemented 609 */ 610 COORD 611 WINAPI 612 DECLSPEC_HOTPATCH 613 GetConsoleFontSize(IN HANDLE hConsoleOutput, 614 IN DWORD nFont) 615 { 616 CONSOLE_API_MESSAGE ApiMessage; 617 PCONSOLE_GETFONTSIZE GetFontSizeRequest = &ApiMessage.Data.GetFontSizeRequest; 618 COORD Empty = {0, 0}; 619 620 GetFontSizeRequest->ConsoleHandle = NtCurrentPeb()->ProcessParameters->ConsoleHandle; 621 GetFontSizeRequest->OutputHandle = hConsoleOutput; 622 GetFontSizeRequest->FontIndex = nFont; 623 624 CsrClientCallServer((PCSR_API_MESSAGE)&ApiMessage, 625 NULL, 626 CSR_CREATE_API_NUMBER(CONSRV_SERVERDLL_INDEX, ConsolepGetFontSize), 627 sizeof(*GetFontSizeRequest)); 628 if (!NT_SUCCESS(ApiMessage.Status)) 629 { 630 BaseSetLastNTError(ApiMessage.Status); 631 return Empty; 632 } 633 634 return GetFontSizeRequest->FontSize; 635 } 636 637 638 /* 639 * @implemented (Undocumented) 640 */ 641 BOOL 642 WINAPI 643 GetConsoleHardwareState(HANDLE hConsoleOutput, 644 PDWORD Flags, 645 PDWORD State) 646 { 647 CONSOLE_API_MESSAGE ApiMessage; 648 PCONSOLE_GETSETHWSTATE HardwareStateRequest = &ApiMessage.Data.HardwareStateRequest; 649 650 DPRINT1("GetConsoleHardwareState(%lu, 0x%p) UNIMPLEMENTED!\n", Flags, State); 651 652 if (Flags == NULL || State == NULL) 653 { 654 SetLastError(ERROR_INVALID_PARAMETER); 655 return FALSE; 656 } 657 658 HardwareStateRequest->ConsoleHandle = NtCurrentPeb()->ProcessParameters->ConsoleHandle; 659 HardwareStateRequest->OutputHandle = hConsoleOutput; 660 661 CsrClientCallServer((PCSR_API_MESSAGE)&ApiMessage, 662 NULL, 663 CSR_CREATE_API_NUMBER(CONSRV_SERVERDLL_INDEX, ConsolepGetHardwareState), 664 sizeof(*HardwareStateRequest)); 665 if (!NT_SUCCESS(ApiMessage.Status)) 666 { 667 BaseSetLastNTError(ApiMessage.Status); 668 return FALSE; 669 } 670 671 *Flags = HardwareStateRequest->Flags; 672 *State = HardwareStateRequest->State; 673 674 return TRUE; 675 } 676 677 678 /* 679 * @implemented (Undocumented) 680 */ 681 HANDLE 682 WINAPI 683 GetConsoleInputWaitHandle(VOID) 684 { 685 return InputWaitHandle; 686 } 687 688 689 /* 690 * @implemented 691 */ 692 BOOL 693 WINAPI 694 GetCurrentConsoleFont(IN HANDLE hConsoleOutput, 695 IN BOOL bMaximumWindow, 696 OUT PCONSOLE_FONT_INFO lpConsoleCurrentFont) 697 { 698 CONSOLE_API_MESSAGE ApiMessage; 699 PCONSOLE_GETCURRENTFONT GetCurrentFontRequest = &ApiMessage.Data.GetCurrentFontRequest; 700 701 GetCurrentFontRequest->ConsoleHandle = NtCurrentPeb()->ProcessParameters->ConsoleHandle; 702 GetCurrentFontRequest->OutputHandle = hConsoleOutput; 703 GetCurrentFontRequest->MaximumWindow = bMaximumWindow; 704 705 CsrClientCallServer((PCSR_API_MESSAGE)&ApiMessage, 706 NULL, 707 CSR_CREATE_API_NUMBER(CONSRV_SERVERDLL_INDEX, ConsolepGetCurrentFont), 708 sizeof(*GetCurrentFontRequest)); 709 if (!NT_SUCCESS(ApiMessage.Status)) 710 { 711 BaseSetLastNTError(ApiMessage.Status); 712 return FALSE; 713 } 714 715 lpConsoleCurrentFont->dwFontSize = GetCurrentFontRequest->FontSize; 716 lpConsoleCurrentFont->nFont = GetCurrentFontRequest->FontIndex; 717 718 return TRUE; 719 } 720 721 722 /* 723 * @implemented (Undocumented) 724 * @note See http://cboard.cprogramming.com/windows-programming/102187-console-font-size.html 725 */ 726 DWORD 727 WINAPI 728 DECLSPEC_HOTPATCH 729 GetNumberOfConsoleFonts(VOID) 730 { 731 CONSOLE_API_MESSAGE ApiMessage; 732 PCONSOLE_GETNUMFONTS GetNumFontsRequest = &ApiMessage.Data.GetNumFontsRequest; 733 734 GetNumFontsRequest->ConsoleHandle = NtCurrentPeb()->ProcessParameters->ConsoleHandle; 735 736 CsrClientCallServer((PCSR_API_MESSAGE)&ApiMessage, 737 NULL, 738 CSR_CREATE_API_NUMBER(CONSRV_SERVERDLL_INDEX, ConsolepGetNumberOfFonts), 739 sizeof(*GetNumFontsRequest)); 740 if (!NT_SUCCESS(ApiMessage.Status)) 741 { 742 BaseSetLastNTError(ApiMessage.Status); 743 return 0; 744 } 745 746 return GetNumFontsRequest->NumFonts; 747 } 748 749 750 /* 751 * @implemented (Undocumented) 752 * @note See http://blog.airesoft.co.uk/2012/10/things-ms-can-do-that-they-dont-tell-you-about-console-graphics/ 753 */ 754 BOOL 755 WINAPI 756 InvalidateConsoleDIBits(IN HANDLE hConsoleOutput, 757 IN PSMALL_RECT lpRect) 758 { 759 CONSOLE_API_MESSAGE ApiMessage; 760 PCONSOLE_INVALIDATEDIBITS InvalidateDIBitsRequest = &ApiMessage.Data.InvalidateDIBitsRequest; 761 762 if (lpRect == NULL) 763 { 764 SetLastError(ERROR_INVALID_PARAMETER); 765 return FALSE; 766 } 767 768 InvalidateDIBitsRequest->ConsoleHandle = NtCurrentPeb()->ProcessParameters->ConsoleHandle; 769 InvalidateDIBitsRequest->OutputHandle = hConsoleOutput; 770 InvalidateDIBitsRequest->Region = *lpRect; 771 772 CsrClientCallServer((PCSR_API_MESSAGE)&ApiMessage, 773 NULL, 774 CSR_CREATE_API_NUMBER(CONSRV_SERVERDLL_INDEX, ConsolepInvalidateBitMapRect), 775 sizeof(*InvalidateDIBitsRequest)); 776 if (!NT_SUCCESS(ApiMessage.Status)) 777 { 778 BaseSetLastNTError(ApiMessage.Status); 779 return FALSE; 780 } 781 782 return TRUE; 783 } 784 785 786 /* 787 * @implemented (Undocumented) 788 */ 789 HANDLE 790 WINAPI 791 OpenConsoleW(LPCWSTR wsName, 792 DWORD dwDesiredAccess, 793 BOOL bInheritHandle, 794 DWORD dwShareMode) 795 { 796 CONSOLE_API_MESSAGE ApiMessage; 797 PCONSOLE_OPENCONSOLE OpenConsoleRequest = &ApiMessage.Data.OpenConsoleRequest; 798 CONSOLE_HANDLE_TYPE HandleType; 799 800 if (wsName && (_wcsicmp(wsName, BaseConInputFileName) == 0)) 801 { 802 HandleType = HANDLE_INPUT; 803 } 804 else if (wsName && (_wcsicmp(wsName, BaseConOutputFileName) == 0)) 805 { 806 HandleType = HANDLE_OUTPUT; 807 } 808 else 809 { 810 SetLastError(ERROR_INVALID_PARAMETER); 811 return INVALID_HANDLE_VALUE; 812 } 813 814 if ( (dwDesiredAccess & ~(GENERIC_READ | GENERIC_WRITE)) || 815 (dwShareMode & ~(FILE_SHARE_READ | FILE_SHARE_WRITE)) ) 816 { 817 SetLastError(ERROR_INVALID_PARAMETER); 818 return INVALID_HANDLE_VALUE; 819 } 820 821 OpenConsoleRequest->ConsoleHandle = NtCurrentPeb()->ProcessParameters->ConsoleHandle; 822 OpenConsoleRequest->HandleType = HandleType; 823 OpenConsoleRequest->DesiredAccess = dwDesiredAccess; 824 OpenConsoleRequest->InheritHandle = bInheritHandle; 825 OpenConsoleRequest->ShareMode = dwShareMode; 826 827 CsrClientCallServer((PCSR_API_MESSAGE)&ApiMessage, 828 NULL, 829 CSR_CREATE_API_NUMBER(CONSRV_SERVERDLL_INDEX, ConsolepOpenConsole), 830 sizeof(*OpenConsoleRequest)); 831 if (!NT_SUCCESS(ApiMessage.Status)) 832 { 833 BaseSetLastNTError(ApiMessage.Status); 834 return INVALID_HANDLE_VALUE; 835 } 836 837 return OpenConsoleRequest->Handle; 838 } 839 840 841 /* 842 * @implemented (Undocumented) 843 * @note See http://undoc.airesoft.co.uk/kernel32.dll/SetConsoleCursor.php 844 */ 845 BOOL 846 WINAPI 847 DECLSPEC_HOTPATCH 848 SetConsoleCursor(HANDLE hConsoleOutput, 849 HCURSOR hCursor) 850 { 851 CONSOLE_API_MESSAGE ApiMessage; 852 PCONSOLE_SETCURSOR SetCursorRequest = &ApiMessage.Data.SetCursorRequest; 853 854 SetCursorRequest->ConsoleHandle = NtCurrentPeb()->ProcessParameters->ConsoleHandle; 855 SetCursorRequest->OutputHandle = hConsoleOutput; 856 SetCursorRequest->CursorHandle = hCursor; 857 858 CsrClientCallServer((PCSR_API_MESSAGE)&ApiMessage, 859 NULL, 860 CSR_CREATE_API_NUMBER(CONSRV_SERVERDLL_INDEX, ConsolepSetCursor), 861 sizeof(*SetCursorRequest)); 862 if (!NT_SUCCESS(ApiMessage.Status)) 863 { 864 BaseSetLastNTError(ApiMessage.Status); 865 return FALSE; 866 } 867 868 return TRUE; 869 } 870 871 872 /* 873 * @implemented 874 */ 875 BOOL 876 WINAPI 877 SetConsoleDisplayMode(HANDLE hConsoleOutput, 878 DWORD dwFlags, // dwModeFlags 879 PCOORD lpNewScreenBufferDimensions) 880 { 881 CONSOLE_API_MESSAGE ApiMessage; 882 PCONSOLE_SETDISPLAYMODE SetDisplayModeRequest = &ApiMessage.Data.SetDisplayModeRequest; 883 884 SetDisplayModeRequest->ConsoleHandle = NtCurrentPeb()->ProcessParameters->ConsoleHandle; 885 SetDisplayModeRequest->OutputHandle = hConsoleOutput; 886 SetDisplayModeRequest->DisplayMode = dwFlags; // ModeFlags ; dwModeFlags 887 SetDisplayModeRequest->NewSBDim.X = 0; 888 SetDisplayModeRequest->NewSBDim.Y = 0; 889 /* SetDisplayModeRequest->EventHandle; */ 890 891 CsrClientCallServer((PCSR_API_MESSAGE)&ApiMessage, 892 NULL, 893 CSR_CREATE_API_NUMBER(CONSRV_SERVERDLL_INDEX, ConsolepSetDisplayMode), 894 sizeof(*SetDisplayModeRequest)); 895 if (!NT_SUCCESS(ApiMessage.Status)) 896 { 897 BaseSetLastNTError(ApiMessage.Status); 898 return FALSE; 899 } 900 901 if (lpNewScreenBufferDimensions) 902 *lpNewScreenBufferDimensions = SetDisplayModeRequest->NewSBDim; 903 904 return TRUE; 905 } 906 907 908 /* 909 * @implemented (Undocumented) 910 * @note See http://cboard.cprogramming.com/windows-programming/102187-console-font-size.html 911 */ 912 BOOL 913 WINAPI 914 DECLSPEC_HOTPATCH 915 SetConsoleFont(IN HANDLE hConsoleOutput, 916 IN DWORD nFont) 917 { 918 CONSOLE_API_MESSAGE ApiMessage; 919 PCONSOLE_SETFONT SetFontRequest = &ApiMessage.Data.SetFontRequest; 920 921 SetFontRequest->ConsoleHandle = NtCurrentPeb()->ProcessParameters->ConsoleHandle; 922 SetFontRequest->OutputHandle = hConsoleOutput; 923 SetFontRequest->FontIndex = nFont; 924 925 CsrClientCallServer((PCSR_API_MESSAGE)&ApiMessage, 926 NULL, 927 CSR_CREATE_API_NUMBER(CONSRV_SERVERDLL_INDEX, ConsolepSetFont), 928 sizeof(*SetFontRequest)); 929 if (!NT_SUCCESS(ApiMessage.Status)) 930 { 931 BaseSetLastNTError(ApiMessage.Status); 932 return FALSE; 933 } 934 935 return TRUE; 936 } 937 938 939 /* 940 * @implemented (Undocumented) 941 */ 942 BOOL 943 WINAPI 944 SetConsoleHardwareState(HANDLE hConsoleOutput, 945 DWORD Flags, 946 DWORD State) 947 { 948 CONSOLE_API_MESSAGE ApiMessage; 949 PCONSOLE_GETSETHWSTATE HardwareStateRequest = &ApiMessage.Data.HardwareStateRequest; 950 951 DPRINT1("SetConsoleHardwareState(%lu, %lu) UNIMPLEMENTED!\n", Flags, State); 952 953 HardwareStateRequest->ConsoleHandle = NtCurrentPeb()->ProcessParameters->ConsoleHandle; 954 HardwareStateRequest->OutputHandle = hConsoleOutput; 955 HardwareStateRequest->Flags = Flags; 956 HardwareStateRequest->State = State; 957 958 CsrClientCallServer((PCSR_API_MESSAGE)&ApiMessage, 959 NULL, 960 CSR_CREATE_API_NUMBER(CONSRV_SERVERDLL_INDEX, ConsolepSetHardwareState), 961 sizeof(*HardwareStateRequest)); 962 if (!NT_SUCCESS(ApiMessage.Status)) 963 { 964 BaseSetLastNTError(ApiMessage.Status); 965 return FALSE; 966 } 967 968 return TRUE; 969 } 970 971 972 /* 973 * @unimplemented (Undocumented) 974 */ 975 BOOL 976 WINAPI 977 DECLSPEC_HOTPATCH 978 SetConsoleKeyShortcuts(DWORD Unknown0, 979 DWORD Unknown1, 980 DWORD Unknown2, 981 DWORD Unknown3) 982 { 983 DPRINT1("SetConsoleKeyShortcuts(0x%x, 0x%x, 0x%x, 0x%x) UNIMPLEMENTED!\n", Unknown0, Unknown1, Unknown2, Unknown3); 984 SetLastError(ERROR_CALL_NOT_IMPLEMENTED); 985 return FALSE; 986 } 987 988 989 /* 990 * @implemented (Undocumented) 991 * @note See http://undoc.airesoft.co.uk/kernel32.dll/SetConsoleMaximumWindowSize.php 992 * Does nothing, returns TRUE only. Checked on Windows Server 2003. 993 */ 994 BOOL 995 WINAPI 996 SetConsoleMaximumWindowSize(HANDLE hConsoleOutput, 997 COORD dwMaximumSize) 998 { 999 DPRINT1("SetConsoleMaximumWindowSize(0x%p, {%d, %d}) does nothing\n", 1000 hConsoleOutput, dwMaximumSize.X, dwMaximumSize.Y); 1001 return TRUE; 1002 } 1003 1004 1005 /* 1006 * @implemented (Undocumented) 1007 */ 1008 BOOL 1009 WINAPI 1010 DECLSPEC_HOTPATCH 1011 SetConsoleMenuClose(BOOL bEnable) 1012 { 1013 CONSOLE_API_MESSAGE ApiMessage; 1014 PCONSOLE_SETMENUCLOSE SetMenuCloseRequest = &ApiMessage.Data.SetMenuCloseRequest; 1015 1016 SetMenuCloseRequest->ConsoleHandle = NtCurrentPeb()->ProcessParameters->ConsoleHandle; 1017 SetMenuCloseRequest->Enable = bEnable; 1018 1019 CsrClientCallServer((PCSR_API_MESSAGE)&ApiMessage, 1020 NULL, 1021 CSR_CREATE_API_NUMBER(CONSRV_SERVERDLL_INDEX, ConsolepSetMenuClose), 1022 sizeof(*SetMenuCloseRequest)); 1023 if (!NT_SUCCESS(ApiMessage.Status)) 1024 { 1025 BaseSetLastNTError(ApiMessage.Status); 1026 return FALSE; 1027 } 1028 1029 return TRUE; 1030 } 1031 1032 1033 /* 1034 * @implemented (Undocumented) 1035 * @note See http://comments.gmane.org/gmane.comp.lang.harbour.devel/27844 1036 * Usage example: https://github.com/harbour/core/commit/d79a1b7b812cbde6ddf718ebfd6939a24f633e52 1037 */ 1038 BOOL 1039 WINAPI 1040 DECLSPEC_HOTPATCH 1041 SetConsolePalette(HANDLE hConsoleOutput, 1042 HPALETTE hPalette, 1043 UINT dwUsage) 1044 { 1045 CONSOLE_API_MESSAGE ApiMessage; 1046 PCONSOLE_SETPALETTE SetPaletteRequest = &ApiMessage.Data.SetPaletteRequest; 1047 1048 SetPaletteRequest->ConsoleHandle = NtCurrentPeb()->ProcessParameters->ConsoleHandle; 1049 SetPaletteRequest->OutputHandle = hConsoleOutput; 1050 SetPaletteRequest->PaletteHandle = hPalette; 1051 SetPaletteRequest->Usage = dwUsage; 1052 1053 CsrClientCallServer((PCSR_API_MESSAGE)&ApiMessage, 1054 NULL, 1055 CSR_CREATE_API_NUMBER(CONSRV_SERVERDLL_INDEX, ConsolepSetPalette), 1056 sizeof(*SetPaletteRequest)); 1057 if (!NT_SUCCESS(ApiMessage.Status)) 1058 { 1059 BaseSetLastNTError(ApiMessage.Status); 1060 return FALSE; 1061 } 1062 1063 return TRUE; 1064 } 1065 1066 /* 1067 * @implemented (Undocumented) 1068 * @note See http://undoc.airesoft.co.uk/kernel32.dll/ShowConsoleCursor.php 1069 */ 1070 INT 1071 WINAPI 1072 DECLSPEC_HOTPATCH 1073 ShowConsoleCursor(HANDLE hConsoleOutput, 1074 BOOL bShow) 1075 { 1076 CONSOLE_API_MESSAGE ApiMessage; 1077 PCONSOLE_SHOWCURSOR ShowCursorRequest = &ApiMessage.Data.ShowCursorRequest; 1078 1079 ShowCursorRequest->ConsoleHandle = NtCurrentPeb()->ProcessParameters->ConsoleHandle; 1080 ShowCursorRequest->OutputHandle = hConsoleOutput; 1081 ShowCursorRequest->Show = bShow; 1082 ShowCursorRequest->RefCount = 0; 1083 1084 CsrClientCallServer((PCSR_API_MESSAGE)&ApiMessage, 1085 NULL, 1086 CSR_CREATE_API_NUMBER(CONSRV_SERVERDLL_INDEX, ConsolepShowCursor), 1087 sizeof(*ShowCursorRequest)); 1088 1089 return ShowCursorRequest->RefCount; 1090 } 1091 1092 1093 /* 1094 * FUNCTION: Checks whether the given handle is a valid console handle. 1095 * 1096 * ARGUMENTS: 1097 * hIoHandle - Handle to be checked. 1098 * 1099 * RETURNS: 1100 * TRUE : Handle is a valid console handle. 1101 * FALSE: Handle is not a valid console handle. 1102 * 1103 * STATUS: Officially undocumented 1104 * 1105 * @implemented 1106 */ 1107 BOOL 1108 WINAPI 1109 DECLSPEC_HOTPATCH 1110 VerifyConsoleIoHandle(HANDLE hIoHandle) 1111 { 1112 CONSOLE_API_MESSAGE ApiMessage; 1113 PCONSOLE_VERIFYHANDLE VerifyHandleRequest = &ApiMessage.Data.VerifyHandleRequest; 1114 1115 VerifyHandleRequest->ConsoleHandle = NtCurrentPeb()->ProcessParameters->ConsoleHandle; 1116 VerifyHandleRequest->Handle = hIoHandle; 1117 VerifyHandleRequest->IsValid = FALSE; 1118 1119 /* If the process is not attached to a console, return invalid handle */ 1120 if (VerifyHandleRequest->ConsoleHandle == NULL) return FALSE; 1121 1122 CsrClientCallServer((PCSR_API_MESSAGE)&ApiMessage, 1123 NULL, 1124 CSR_CREATE_API_NUMBER(CONSRV_SERVERDLL_INDEX, ConsolepVerifyIoHandle), 1125 sizeof(*VerifyHandleRequest)); 1126 if (!NT_SUCCESS(ApiMessage.Status)) 1127 { 1128 BaseSetLastNTError(ApiMessage.Status); 1129 return FALSE; 1130 } 1131 1132 return VerifyHandleRequest->IsValid; 1133 } 1134 1135 1136 /* 1137 * @implemented (Undocumented) 1138 */ 1139 BOOL 1140 WINAPI 1141 DECLSPEC_HOTPATCH 1142 CloseConsoleHandle(HANDLE hHandle) 1143 { 1144 CONSOLE_API_MESSAGE ApiMessage; 1145 PCONSOLE_CLOSEHANDLE CloseHandleRequest = &ApiMessage.Data.CloseHandleRequest; 1146 1147 CloseHandleRequest->ConsoleHandle = NtCurrentPeb()->ProcessParameters->ConsoleHandle; 1148 CloseHandleRequest->Handle = hHandle; 1149 1150 CsrClientCallServer((PCSR_API_MESSAGE)&ApiMessage, 1151 NULL, 1152 CSR_CREATE_API_NUMBER(CONSRV_SERVERDLL_INDEX, ConsolepCloseHandle), 1153 sizeof(*CloseHandleRequest)); 1154 if (!NT_SUCCESS(ApiMessage.Status)) 1155 { 1156 BaseSetLastNTError(ApiMessage.Status); 1157 return FALSE; 1158 } 1159 1160 return TRUE; 1161 } 1162 1163 1164 /* 1165 * @implemented 1166 */ 1167 HANDLE 1168 WINAPI 1169 DECLSPEC_HOTPATCH 1170 GetStdHandle(DWORD nStdHandle) 1171 /* 1172 * FUNCTION: Get a handle for the standard input, standard output 1173 * and a standard error device. 1174 * 1175 * ARGUMENTS: 1176 * nStdHandle - Specifies the device for which to return the handle. 1177 * 1178 * RETURNS: If the function succeeds, the return value is the handle 1179 * of the specified device. Otherwise the value is INVALID_HANDLE_VALUE. 1180 */ 1181 { 1182 PRTL_USER_PROCESS_PARAMETERS Ppb = NtCurrentPeb()->ProcessParameters; 1183 HANDLE Handle = INVALID_HANDLE_VALUE; 1184 1185 switch (nStdHandle) 1186 { 1187 case STD_INPUT_HANDLE: 1188 Handle = Ppb->StandardInput; 1189 break; 1190 1191 case STD_OUTPUT_HANDLE: 1192 Handle = Ppb->StandardOutput; 1193 break; 1194 1195 case STD_ERROR_HANDLE: 1196 Handle = Ppb->StandardError; 1197 break; 1198 } 1199 1200 /* If the returned handle is invalid, set last error */ 1201 if (Handle == INVALID_HANDLE_VALUE) SetLastError(ERROR_INVALID_HANDLE); 1202 1203 return Handle; 1204 } 1205 1206 1207 /* 1208 * @implemented 1209 */ 1210 BOOL 1211 WINAPI 1212 DECLSPEC_HOTPATCH 1213 SetStdHandle(DWORD nStdHandle, 1214 HANDLE hHandle) 1215 /* 1216 * FUNCTION: Set the handle for the standard input, standard output or 1217 * the standard error device. 1218 * 1219 * ARGUMENTS: 1220 * nStdHandle - Specifies the handle to be set. 1221 * hHandle - The handle to set. 1222 * 1223 * RETURNS: TRUE if the function succeeds, FALSE otherwise. 1224 */ 1225 { 1226 PRTL_USER_PROCESS_PARAMETERS Ppb = NtCurrentPeb()->ProcessParameters; 1227 1228 /* No need to check if hHandle == INVALID_HANDLE_VALUE */ 1229 1230 switch (nStdHandle) 1231 { 1232 case STD_INPUT_HANDLE: 1233 Ppb->StandardInput = hHandle; 1234 return TRUE; 1235 1236 case STD_OUTPUT_HANDLE: 1237 Ppb->StandardOutput = hHandle; 1238 return TRUE; 1239 1240 case STD_ERROR_HANDLE: 1241 Ppb->StandardError = hHandle; 1242 return TRUE; 1243 } 1244 1245 /* nStdHandle was invalid, bail out */ 1246 SetLastError(ERROR_INVALID_HANDLE); 1247 return FALSE; 1248 } 1249 1250 1251 /* 1252 * @implemented 1253 */ 1254 static BOOL 1255 IntAllocConsole(LPWSTR Title, 1256 DWORD TitleLength, 1257 LPWSTR Desktop, 1258 DWORD DesktopLength, 1259 LPWSTR CurDir, 1260 DWORD CurDirLength, 1261 LPWSTR AppName, 1262 DWORD AppNameLength, 1263 LPTHREAD_START_ROUTINE CtrlRoutine, 1264 LPTHREAD_START_ROUTINE PropRoutine, 1265 PCONSOLE_START_INFO ConsoleStartInfo) 1266 { 1267 BOOL Success = TRUE; 1268 NTSTATUS Status; 1269 1270 CONSOLE_API_MESSAGE ApiMessage; 1271 PCONSOLE_ALLOCCONSOLE AllocConsoleRequest = &ApiMessage.Data.AllocConsoleRequest; 1272 PCSR_CAPTURE_BUFFER CaptureBuffer; 1273 1274 AllocConsoleRequest->CtrlRoutine = CtrlRoutine; 1275 AllocConsoleRequest->PropRoutine = PropRoutine; 1276 1277 CaptureBuffer = CsrAllocateCaptureBuffer(5, TitleLength + 1278 DesktopLength + 1279 CurDirLength + 1280 AppNameLength + 1281 sizeof(CONSOLE_START_INFO)); 1282 if (CaptureBuffer == NULL) 1283 { 1284 SetLastError(ERROR_NOT_ENOUGH_MEMORY); 1285 Success = FALSE; 1286 goto Quit; 1287 } 1288 1289 CsrCaptureMessageBuffer(CaptureBuffer, 1290 ConsoleStartInfo, 1291 sizeof(CONSOLE_START_INFO), 1292 (PVOID*)&AllocConsoleRequest->ConsoleStartInfo); 1293 1294 AllocConsoleRequest->TitleLength = TitleLength; 1295 CsrCaptureMessageBuffer(CaptureBuffer, 1296 Title, 1297 TitleLength, 1298 (PVOID*)&AllocConsoleRequest->ConsoleTitle); 1299 1300 AllocConsoleRequest->DesktopLength = DesktopLength; 1301 CsrCaptureMessageBuffer(CaptureBuffer, 1302 Desktop, 1303 DesktopLength, 1304 (PVOID*)&AllocConsoleRequest->Desktop); 1305 1306 AllocConsoleRequest->CurDirLength = CurDirLength; 1307 CsrCaptureMessageBuffer(CaptureBuffer, 1308 CurDir, 1309 CurDirLength, 1310 (PVOID*)&AllocConsoleRequest->CurDir); 1311 1312 AllocConsoleRequest->AppNameLength = AppNameLength; 1313 CsrCaptureMessageBuffer(CaptureBuffer, 1314 AppName, 1315 AppNameLength, 1316 (PVOID*)&AllocConsoleRequest->AppName); 1317 1318 CsrClientCallServer((PCSR_API_MESSAGE)&ApiMessage, 1319 CaptureBuffer, 1320 CSR_CREATE_API_NUMBER(CONSRV_SERVERDLL_INDEX, ConsolepAlloc), 1321 sizeof(*AllocConsoleRequest)); 1322 if (!NT_SUCCESS(ApiMessage.Status)) 1323 { 1324 BaseSetLastNTError(ApiMessage.Status); 1325 Success = FALSE; 1326 goto Quit; 1327 } 1328 1329 // Is AllocConsoleRequest->ConsoleStartInfo->InitEvents aligned on handle boundary ???? 1330 Status = NtWaitForMultipleObjects(MAX_INIT_EVENTS, 1331 AllocConsoleRequest->ConsoleStartInfo->InitEvents, 1332 WaitAny, FALSE, NULL); 1333 if (!NT_SUCCESS(Status)) 1334 { 1335 BaseSetLastNTError(Status); 1336 Success = FALSE; 1337 goto Quit; 1338 } 1339 1340 NtClose(AllocConsoleRequest->ConsoleStartInfo->InitEvents[INIT_SUCCESS]); 1341 NtClose(AllocConsoleRequest->ConsoleStartInfo->InitEvents[INIT_FAILURE]); 1342 if (Status != INIT_SUCCESS) 1343 { 1344 NtCurrentPeb()->ProcessParameters->ConsoleHandle = NULL; 1345 Success = FALSE; 1346 } 1347 else 1348 { 1349 RtlCopyMemory(ConsoleStartInfo, 1350 AllocConsoleRequest->ConsoleStartInfo, 1351 sizeof(CONSOLE_START_INFO)); 1352 Success = TRUE; 1353 } 1354 1355 Quit: 1356 if (CaptureBuffer) CsrFreeCaptureBuffer(CaptureBuffer); 1357 return Success; 1358 } 1359 1360 BOOL 1361 WINAPI 1362 DECLSPEC_HOTPATCH 1363 AllocConsole(VOID) 1364 { 1365 BOOL Success; 1366 CONSOLE_START_INFO ConsoleStartInfo; 1367 1368 PWCHAR ConsoleTitle; 1369 PWCHAR Desktop; 1370 PWCHAR AppName; 1371 PWCHAR CurDir; 1372 1373 ULONG TitleLength = (MAX_PATH + 1) * sizeof(WCHAR); 1374 ULONG DesktopLength = (MAX_PATH + 1) * sizeof(WCHAR); 1375 ULONG AppNameLength = 128 * sizeof(WCHAR); 1376 ULONG CurDirLength = (MAX_PATH + 1) * sizeof(WCHAR); 1377 1378 LCID lcid; 1379 1380 RtlEnterCriticalSection(&ConsoleLock); 1381 1382 if (NtCurrentPeb()->ProcessParameters->ConsoleHandle) 1383 { 1384 DPRINT1("AllocConsole: Allocating a console to a process already having one\n"); 1385 SetLastError(ERROR_ACCESS_DENIED); 1386 Success = FALSE; 1387 goto Quit; 1388 } 1389 1390 /* Set up the console properties */ 1391 SetUpConsoleInfo(FALSE, 1392 &TitleLength, 1393 &ConsoleTitle, 1394 &DesktopLength, 1395 &Desktop, 1396 &ConsoleStartInfo); 1397 DPRINT("ConsoleTitle = '%S' - Desktop = '%S'\n", 1398 ConsoleTitle, Desktop); 1399 1400 /* Initialize the Input EXE name */ 1401 InitExeName(); 1402 SetUpAppName(FALSE, 1403 &CurDirLength, 1404 &CurDir, 1405 &AppNameLength, 1406 &AppName); 1407 DPRINT("CurDir = '%S' - AppName = '%S'\n", 1408 CurDir, AppName); 1409 1410 Success = IntAllocConsole(ConsoleTitle, 1411 TitleLength, 1412 Desktop, 1413 DesktopLength, 1414 CurDir, 1415 CurDirLength, 1416 AppName, 1417 AppNameLength, 1418 ConsoleControlDispatcher, 1419 PropDialogHandler, 1420 &ConsoleStartInfo); 1421 if (Success) 1422 { 1423 /* Set up the handles */ 1424 SetUpHandles(&ConsoleStartInfo); 1425 InputWaitHandle = ConsoleStartInfo.InputWaitHandle; 1426 1427 /* Initialize Console Ctrl Handling */ 1428 InitializeCtrlHandling(); 1429 1430 /* Sets the current console locale for this thread */ 1431 SetTEBLangID(lcid); 1432 } 1433 1434 Quit: 1435 RtlLeaveCriticalSection(&ConsoleLock); 1436 return Success; 1437 } 1438 1439 1440 /* 1441 * @implemented 1442 */ 1443 BOOL 1444 WINAPI 1445 DECLSPEC_HOTPATCH 1446 FreeConsole(VOID) 1447 { 1448 BOOL Success = TRUE; 1449 CONSOLE_API_MESSAGE ApiMessage; 1450 PCONSOLE_FREECONSOLE FreeConsoleRequest = &ApiMessage.Data.FreeConsoleRequest; 1451 HANDLE ConsoleHandle = NtCurrentPeb()->ProcessParameters->ConsoleHandle; 1452 1453 RtlEnterCriticalSection(&ConsoleLock); 1454 1455 /* We must have a non-trivial handle to close */ 1456 if (ConsoleHandle == NULL) // IsConsoleHandle(ConsoleHandle) 1457 { 1458 SetLastError(ERROR_INVALID_PARAMETER); 1459 Success = FALSE; 1460 goto Quit; 1461 } 1462 1463 /* Set up the data to send to the Console Server */ 1464 FreeConsoleRequest->ConsoleHandle = ConsoleHandle; 1465 1466 /* Call the server */ 1467 CsrClientCallServer((PCSR_API_MESSAGE)&ApiMessage, 1468 NULL, 1469 CSR_CREATE_API_NUMBER(CONSRV_SERVERDLL_INDEX, ConsolepFree), 1470 sizeof(*FreeConsoleRequest)); 1471 1472 /* Check for success */ 1473 if (!NT_SUCCESS(ApiMessage.Status)) 1474 { 1475 BaseSetLastNTError(ApiMessage.Status); 1476 Success = FALSE; 1477 goto Quit; 1478 } 1479 1480 /* Reset the console handle */ 1481 NtCurrentPeb()->ProcessParameters->ConsoleHandle = NULL; 1482 1483 /* Close the associated input handle */ 1484 CloseHandle(InputWaitHandle); 1485 InputWaitHandle = INVALID_HANDLE_VALUE; 1486 1487 Quit: 1488 RtlLeaveCriticalSection(&ConsoleLock); 1489 return Success; 1490 } 1491 1492 1493 /* 1494 * @implemented 1495 */ 1496 BOOL 1497 WINAPI 1498 GetConsoleScreenBufferInfo(HANDLE hConsoleOutput, 1499 PCONSOLE_SCREEN_BUFFER_INFO lpConsoleScreenBufferInfo) 1500 { 1501 CONSOLE_API_MESSAGE ApiMessage; 1502 PCONSOLE_GETSCREENBUFFERINFO ScreenBufferInfoRequest = &ApiMessage.Data.ScreenBufferInfoRequest; 1503 1504 if (lpConsoleScreenBufferInfo == NULL) 1505 { 1506 SetLastError(ERROR_INVALID_PARAMETER); 1507 return FALSE; 1508 } 1509 1510 ScreenBufferInfoRequest->ConsoleHandle = NtCurrentPeb()->ProcessParameters->ConsoleHandle; 1511 ScreenBufferInfoRequest->OutputHandle = hConsoleOutput; 1512 1513 CsrClientCallServer((PCSR_API_MESSAGE)&ApiMessage, 1514 NULL, 1515 CSR_CREATE_API_NUMBER(CONSRV_SERVERDLL_INDEX, ConsolepGetScreenBufferInfo), 1516 sizeof(*ScreenBufferInfoRequest)); 1517 if (!NT_SUCCESS(ApiMessage.Status)) 1518 { 1519 BaseSetLastNTError(ApiMessage.Status); 1520 return FALSE; 1521 } 1522 1523 lpConsoleScreenBufferInfo->dwSize = ScreenBufferInfoRequest->ScreenBufferSize; 1524 lpConsoleScreenBufferInfo->dwCursorPosition = ScreenBufferInfoRequest->CursorPosition; 1525 lpConsoleScreenBufferInfo->wAttributes = ScreenBufferInfoRequest->Attributes; 1526 lpConsoleScreenBufferInfo->srWindow.Left = ScreenBufferInfoRequest->ViewOrigin.X; 1527 lpConsoleScreenBufferInfo->srWindow.Top = ScreenBufferInfoRequest->ViewOrigin.Y; 1528 lpConsoleScreenBufferInfo->srWindow.Right = ScreenBufferInfoRequest->ViewOrigin.X + ScreenBufferInfoRequest->ViewSize.X - 1; 1529 lpConsoleScreenBufferInfo->srWindow.Bottom = ScreenBufferInfoRequest->ViewOrigin.Y + ScreenBufferInfoRequest->ViewSize.Y - 1; 1530 lpConsoleScreenBufferInfo->dwMaximumWindowSize = ScreenBufferInfoRequest->MaximumViewSize; 1531 1532 return TRUE; 1533 } 1534 1535 1536 /* 1537 * @implemented 1538 */ 1539 BOOL 1540 WINAPI 1541 DECLSPEC_HOTPATCH 1542 SetConsoleCursorPosition(HANDLE hConsoleOutput, 1543 COORD dwCursorPosition) 1544 { 1545 CONSOLE_API_MESSAGE ApiMessage; 1546 PCONSOLE_SETCURSORPOSITION SetCursorPositionRequest = &ApiMessage.Data.SetCursorPositionRequest; 1547 1548 SetCursorPositionRequest->ConsoleHandle = NtCurrentPeb()->ProcessParameters->ConsoleHandle; 1549 SetCursorPositionRequest->OutputHandle = hConsoleOutput; 1550 SetCursorPositionRequest->Position = dwCursorPosition; 1551 1552 CsrClientCallServer((PCSR_API_MESSAGE)&ApiMessage, 1553 NULL, 1554 CSR_CREATE_API_NUMBER(CONSRV_SERVERDLL_INDEX, ConsolepSetCursorPosition), 1555 sizeof(*SetCursorPositionRequest)); 1556 if (!NT_SUCCESS(ApiMessage.Status)) 1557 { 1558 BaseSetLastNTError(ApiMessage.Status); 1559 return FALSE; 1560 } 1561 1562 return TRUE; 1563 } 1564 1565 1566 /* 1567 * @implemented 1568 */ 1569 BOOL 1570 WINAPI 1571 GetConsoleMode(HANDLE hConsoleHandle, 1572 LPDWORD lpMode) 1573 { 1574 CONSOLE_API_MESSAGE ApiMessage; 1575 PCONSOLE_GETSETCONSOLEMODE ConsoleModeRequest = &ApiMessage.Data.ConsoleModeRequest; 1576 1577 if (lpMode == NULL) 1578 { 1579 SetLastError(ERROR_INVALID_PARAMETER); 1580 return FALSE; 1581 } 1582 1583 ConsoleModeRequest->ConsoleHandle = NtCurrentPeb()->ProcessParameters->ConsoleHandle; 1584 ConsoleModeRequest->Handle = hConsoleHandle; 1585 1586 CsrClientCallServer((PCSR_API_MESSAGE)&ApiMessage, 1587 NULL, 1588 CSR_CREATE_API_NUMBER(CONSRV_SERVERDLL_INDEX, ConsolepGetMode), 1589 sizeof(*ConsoleModeRequest)); 1590 if (!NT_SUCCESS(ApiMessage.Status)) 1591 { 1592 BaseSetLastNTError(ApiMessage.Status); 1593 return FALSE; 1594 } 1595 1596 *lpMode = ConsoleModeRequest->Mode; 1597 1598 return TRUE; 1599 } 1600 1601 1602 /* 1603 * @implemented 1604 */ 1605 BOOL 1606 WINAPI 1607 DECLSPEC_HOTPATCH 1608 SetConsoleMode(HANDLE hConsoleHandle, 1609 DWORD dwMode) 1610 { 1611 CONSOLE_API_MESSAGE ApiMessage; 1612 PCONSOLE_GETSETCONSOLEMODE ConsoleModeRequest = &ApiMessage.Data.ConsoleModeRequest; 1613 1614 ConsoleModeRequest->ConsoleHandle = NtCurrentPeb()->ProcessParameters->ConsoleHandle; 1615 ConsoleModeRequest->Handle = hConsoleHandle; 1616 ConsoleModeRequest->Mode = dwMode; 1617 1618 CsrClientCallServer((PCSR_API_MESSAGE)&ApiMessage, 1619 NULL, 1620 CSR_CREATE_API_NUMBER(CONSRV_SERVERDLL_INDEX, ConsolepSetMode), 1621 sizeof(*ConsoleModeRequest)); 1622 if (!NT_SUCCESS(ApiMessage.Status)) 1623 { 1624 BaseSetLastNTError(ApiMessage.Status); 1625 return FALSE; 1626 } 1627 1628 return TRUE; 1629 } 1630 1631 1632 /* 1633 * @implemented 1634 */ 1635 BOOL 1636 WINAPI 1637 GetNumberOfConsoleInputEvents(HANDLE hConsoleInput, 1638 LPDWORD lpNumberOfEvents) 1639 { 1640 CONSOLE_API_MESSAGE ApiMessage; 1641 PCONSOLE_GETNUMINPUTEVENTS GetNumInputEventsRequest = &ApiMessage.Data.GetNumInputEventsRequest; 1642 1643 GetNumInputEventsRequest->ConsoleHandle = NtCurrentPeb()->ProcessParameters->ConsoleHandle; 1644 GetNumInputEventsRequest->InputHandle = hConsoleInput; 1645 GetNumInputEventsRequest->NumberOfEvents = 0; 1646 1647 CsrClientCallServer((PCSR_API_MESSAGE)&ApiMessage, 1648 NULL, 1649 CSR_CREATE_API_NUMBER(CONSRV_SERVERDLL_INDEX, ConsolepGetNumberOfInputEvents), 1650 sizeof(*GetNumInputEventsRequest)); 1651 if (!NT_SUCCESS(ApiMessage.Status)) 1652 { 1653 BaseSetLastNTError(ApiMessage.Status); 1654 return FALSE; 1655 } 1656 1657 if (lpNumberOfEvents == NULL) 1658 { 1659 SetLastError(ERROR_INVALID_ACCESS); 1660 return FALSE; 1661 } 1662 1663 *lpNumberOfEvents = GetNumInputEventsRequest->NumberOfEvents; 1664 1665 return TRUE; 1666 } 1667 1668 1669 /* 1670 * @implemented 1671 */ 1672 COORD 1673 WINAPI 1674 DECLSPEC_HOTPATCH 1675 GetLargestConsoleWindowSize(HANDLE hConsoleOutput) 1676 { 1677 CONSOLE_API_MESSAGE ApiMessage; 1678 PCONSOLE_GETLARGESTWINDOWSIZE GetLargestWindowSizeRequest = &ApiMessage.Data.GetLargestWindowSizeRequest; 1679 1680 GetLargestWindowSizeRequest->ConsoleHandle = NtCurrentPeb()->ProcessParameters->ConsoleHandle; 1681 GetLargestWindowSizeRequest->OutputHandle = hConsoleOutput; 1682 GetLargestWindowSizeRequest->Size.X = 0; 1683 GetLargestWindowSizeRequest->Size.Y = 0; 1684 1685 CsrClientCallServer((PCSR_API_MESSAGE)&ApiMessage, 1686 NULL, 1687 CSR_CREATE_API_NUMBER(CONSRV_SERVERDLL_INDEX, ConsolepGetLargestWindowSize), 1688 sizeof(*GetLargestWindowSizeRequest)); 1689 if (!NT_SUCCESS(ApiMessage.Status)) 1690 { 1691 BaseSetLastNTError(ApiMessage.Status); 1692 } 1693 1694 DPRINT("GetLargestConsoleWindowSize, X = %d, Y = %d\n", GetLargestWindowSizeRequest->Size.X, GetLargestWindowSizeRequest->Size.Y); 1695 return GetLargestWindowSizeRequest->Size; 1696 } 1697 1698 1699 /* 1700 * @implemented 1701 */ 1702 BOOL 1703 WINAPI 1704 GetConsoleCursorInfo(HANDLE hConsoleOutput, 1705 PCONSOLE_CURSOR_INFO lpConsoleCursorInfo) 1706 { 1707 CONSOLE_API_MESSAGE ApiMessage; 1708 PCONSOLE_GETSETCURSORINFO CursorInfoRequest = &ApiMessage.Data.CursorInfoRequest; 1709 1710 if (!lpConsoleCursorInfo) 1711 { 1712 if (!hConsoleOutput) 1713 SetLastError(ERROR_INVALID_HANDLE); 1714 else 1715 SetLastError(ERROR_INVALID_ACCESS); 1716 1717 return FALSE; 1718 } 1719 1720 CursorInfoRequest->ConsoleHandle = NtCurrentPeb()->ProcessParameters->ConsoleHandle; 1721 CursorInfoRequest->OutputHandle = hConsoleOutput; 1722 1723 CsrClientCallServer((PCSR_API_MESSAGE)&ApiMessage, 1724 NULL, 1725 CSR_CREATE_API_NUMBER(CONSRV_SERVERDLL_INDEX, ConsolepGetCursorInfo), 1726 sizeof(*CursorInfoRequest)); 1727 if (!NT_SUCCESS(ApiMessage.Status)) 1728 { 1729 BaseSetLastNTError(ApiMessage.Status); 1730 return FALSE; 1731 } 1732 1733 *lpConsoleCursorInfo = CursorInfoRequest->Info; 1734 1735 return TRUE; 1736 } 1737 1738 1739 /* 1740 * @implemented 1741 */ 1742 BOOL 1743 WINAPI 1744 SetConsoleCursorInfo(HANDLE hConsoleOutput, 1745 CONST CONSOLE_CURSOR_INFO *lpConsoleCursorInfo) 1746 { 1747 CONSOLE_API_MESSAGE ApiMessage; 1748 PCONSOLE_GETSETCURSORINFO CursorInfoRequest = &ApiMessage.Data.CursorInfoRequest; 1749 1750 CursorInfoRequest->ConsoleHandle = NtCurrentPeb()->ProcessParameters->ConsoleHandle; 1751 CursorInfoRequest->OutputHandle = hConsoleOutput; 1752 CursorInfoRequest->Info = *lpConsoleCursorInfo; 1753 1754 CsrClientCallServer((PCSR_API_MESSAGE)&ApiMessage, 1755 NULL, 1756 CSR_CREATE_API_NUMBER(CONSRV_SERVERDLL_INDEX, ConsolepSetCursorInfo), 1757 sizeof(*CursorInfoRequest)); 1758 if (!NT_SUCCESS(ApiMessage.Status)) 1759 { 1760 BaseSetLastNTError(ApiMessage.Status); 1761 return FALSE; 1762 } 1763 1764 return TRUE; 1765 } 1766 1767 1768 /* 1769 * @implemented 1770 */ 1771 BOOL 1772 WINAPI 1773 GetNumberOfConsoleMouseButtons(LPDWORD lpNumberOfMouseButtons) 1774 { 1775 CONSOLE_API_MESSAGE ApiMessage; 1776 PCONSOLE_GETMOUSEINFO GetMouseInfoRequest = &ApiMessage.Data.GetMouseInfoRequest; 1777 1778 GetMouseInfoRequest->ConsoleHandle = NtCurrentPeb()->ProcessParameters->ConsoleHandle; 1779 1780 CsrClientCallServer((PCSR_API_MESSAGE)&ApiMessage, 1781 NULL, 1782 CSR_CREATE_API_NUMBER(CONSRV_SERVERDLL_INDEX, ConsolepGetMouseInfo), 1783 sizeof(*GetMouseInfoRequest)); 1784 if (!NT_SUCCESS(ApiMessage.Status)) 1785 { 1786 BaseSetLastNTError(ApiMessage.Status); 1787 return FALSE; 1788 } 1789 1790 *lpNumberOfMouseButtons = GetMouseInfoRequest->NumButtons; 1791 return TRUE; 1792 } 1793 1794 1795 /* 1796 * @implemented 1797 */ 1798 BOOL 1799 WINAPI 1800 DECLSPEC_HOTPATCH 1801 SetConsoleActiveScreenBuffer(HANDLE hConsoleOutput) 1802 { 1803 CONSOLE_API_MESSAGE ApiMessage; 1804 PCONSOLE_SETACTIVESCREENBUFFER SetScreenBufferRequest = &ApiMessage.Data.SetScreenBufferRequest; 1805 1806 SetScreenBufferRequest->ConsoleHandle = NtCurrentPeb()->ProcessParameters->ConsoleHandle; 1807 SetScreenBufferRequest->OutputHandle = hConsoleOutput; 1808 1809 CsrClientCallServer((PCSR_API_MESSAGE)&ApiMessage, 1810 NULL, 1811 CSR_CREATE_API_NUMBER(CONSRV_SERVERDLL_INDEX, ConsolepSetActiveScreenBuffer), 1812 sizeof(*SetScreenBufferRequest)); 1813 if (!NT_SUCCESS(ApiMessage.Status)) 1814 { 1815 BaseSetLastNTError(ApiMessage.Status); 1816 return FALSE; 1817 } 1818 1819 return TRUE; 1820 } 1821 1822 1823 /* 1824 * @implemented 1825 */ 1826 BOOL 1827 WINAPI 1828 DECLSPEC_HOTPATCH 1829 FlushConsoleInputBuffer(HANDLE hConsoleInput) 1830 { 1831 CONSOLE_API_MESSAGE ApiMessage; 1832 PCONSOLE_FLUSHINPUTBUFFER FlushInputBufferRequest = &ApiMessage.Data.FlushInputBufferRequest; 1833 1834 FlushInputBufferRequest->ConsoleHandle = NtCurrentPeb()->ProcessParameters->ConsoleHandle; 1835 FlushInputBufferRequest->InputHandle = hConsoleInput; 1836 1837 CsrClientCallServer((PCSR_API_MESSAGE)&ApiMessage, 1838 NULL, 1839 CSR_CREATE_API_NUMBER(CONSRV_SERVERDLL_INDEX, ConsolepFlushInputBuffer), 1840 sizeof(*FlushInputBufferRequest)); 1841 if (!NT_SUCCESS(ApiMessage.Status)) 1842 { 1843 BaseSetLastNTError(ApiMessage.Status); 1844 return FALSE; 1845 } 1846 1847 return TRUE; 1848 } 1849 1850 1851 /* 1852 * @implemented 1853 */ 1854 BOOL 1855 WINAPI 1856 DECLSPEC_HOTPATCH 1857 SetConsoleScreenBufferSize(HANDLE hConsoleOutput, 1858 COORD dwSize) 1859 { 1860 CONSOLE_API_MESSAGE ApiMessage; 1861 PCONSOLE_SETSCREENBUFFERSIZE SetScreenBufferSizeRequest = &ApiMessage.Data.SetScreenBufferSizeRequest; 1862 1863 SetScreenBufferSizeRequest->ConsoleHandle = NtCurrentPeb()->ProcessParameters->ConsoleHandle; 1864 SetScreenBufferSizeRequest->OutputHandle = hConsoleOutput; 1865 SetScreenBufferSizeRequest->Size = dwSize; 1866 1867 CsrClientCallServer((PCSR_API_MESSAGE)&ApiMessage, 1868 NULL, 1869 CSR_CREATE_API_NUMBER(CONSRV_SERVERDLL_INDEX, ConsolepSetScreenBufferSize), 1870 sizeof(*SetScreenBufferSizeRequest)); 1871 if (!NT_SUCCESS(ApiMessage.Status)) 1872 { 1873 BaseSetLastNTError(ApiMessage.Status); 1874 return FALSE; 1875 } 1876 1877 return TRUE; 1878 } 1879 1880 1881 static 1882 BOOL 1883 IntScrollConsoleScreenBuffer(HANDLE hConsoleOutput, 1884 CONST SMALL_RECT* lpScrollRectangle, 1885 CONST SMALL_RECT* lpClipRectangle, 1886 COORD dwDestinationOrigin, 1887 CONST CHAR_INFO* lpFill, 1888 BOOL bUnicode) 1889 { 1890 CONSOLE_API_MESSAGE ApiMessage; 1891 PCONSOLE_SCROLLSCREENBUFFER ScrollScreenBufferRequest = &ApiMessage.Data.ScrollScreenBufferRequest; 1892 1893 ScrollScreenBufferRequest->ConsoleHandle = NtCurrentPeb()->ProcessParameters->ConsoleHandle; 1894 ScrollScreenBufferRequest->OutputHandle = hConsoleOutput; 1895 ScrollScreenBufferRequest->ScrollRectangle = *lpScrollRectangle; 1896 1897 if (lpClipRectangle != NULL) 1898 { 1899 ScrollScreenBufferRequest->UseClipRectangle = TRUE; 1900 ScrollScreenBufferRequest->ClipRectangle = *lpClipRectangle; 1901 } 1902 else 1903 { 1904 ScrollScreenBufferRequest->UseClipRectangle = FALSE; 1905 } 1906 1907 ScrollScreenBufferRequest->DestinationOrigin = dwDestinationOrigin; 1908 ScrollScreenBufferRequest->Fill = *lpFill; 1909 ScrollScreenBufferRequest->Unicode = bUnicode; 1910 1911 CsrClientCallServer((PCSR_API_MESSAGE)&ApiMessage, 1912 NULL, 1913 CSR_CREATE_API_NUMBER(CONSRV_SERVERDLL_INDEX, ConsolepScrollScreenBuffer), 1914 sizeof(*ScrollScreenBufferRequest)); 1915 if (!NT_SUCCESS(ApiMessage.Status)) 1916 { 1917 BaseSetLastNTError(ApiMessage.Status); 1918 return FALSE; 1919 } 1920 1921 return TRUE; 1922 } 1923 1924 1925 /* 1926 * @implemented 1927 */ 1928 BOOL 1929 WINAPI 1930 DECLSPEC_HOTPATCH 1931 ScrollConsoleScreenBufferA(HANDLE hConsoleOutput, 1932 CONST SMALL_RECT* lpScrollRectangle, 1933 CONST SMALL_RECT* lpClipRectangle, 1934 COORD dwDestinationOrigin, 1935 CONST CHAR_INFO* lpFill) 1936 { 1937 return IntScrollConsoleScreenBuffer(hConsoleOutput, 1938 lpScrollRectangle, 1939 lpClipRectangle, 1940 dwDestinationOrigin, 1941 lpFill, 1942 FALSE); 1943 } 1944 1945 1946 /* 1947 * @implemented 1948 */ 1949 BOOL 1950 WINAPI 1951 DECLSPEC_HOTPATCH 1952 ScrollConsoleScreenBufferW(HANDLE hConsoleOutput, 1953 CONST SMALL_RECT *lpScrollRectangle, 1954 CONST SMALL_RECT *lpClipRectangle, 1955 COORD dwDestinationOrigin, 1956 CONST CHAR_INFO *lpFill) 1957 { 1958 return IntScrollConsoleScreenBuffer(hConsoleOutput, 1959 lpScrollRectangle, 1960 lpClipRectangle, 1961 dwDestinationOrigin, 1962 lpFill, 1963 TRUE); 1964 } 1965 1966 1967 /* 1968 * @implemented 1969 */ 1970 BOOL 1971 WINAPI 1972 SetConsoleWindowInfo(HANDLE hConsoleOutput, 1973 BOOL bAbsolute, 1974 CONST SMALL_RECT *lpConsoleWindow) 1975 { 1976 CONSOLE_API_MESSAGE ApiMessage; 1977 PCONSOLE_SETWINDOWINFO SetWindowInfoRequest = &ApiMessage.Data.SetWindowInfoRequest; 1978 1979 if (lpConsoleWindow == NULL) 1980 { 1981 SetLastError(ERROR_INVALID_PARAMETER); 1982 return FALSE; 1983 } 1984 1985 SetWindowInfoRequest->ConsoleHandle = NtCurrentPeb()->ProcessParameters->ConsoleHandle; 1986 SetWindowInfoRequest->OutputHandle = hConsoleOutput; 1987 SetWindowInfoRequest->Absolute = bAbsolute; 1988 SetWindowInfoRequest->WindowRect = *lpConsoleWindow; 1989 1990 CsrClientCallServer((PCSR_API_MESSAGE)&ApiMessage, 1991 NULL, 1992 CSR_CREATE_API_NUMBER(CONSRV_SERVERDLL_INDEX, ConsolepSetWindowInfo), 1993 sizeof(*SetWindowInfoRequest)); 1994 if (!NT_SUCCESS(ApiMessage.Status)) 1995 { 1996 BaseSetLastNTError(ApiMessage.Status); 1997 return FALSE; 1998 } 1999 2000 return TRUE; 2001 } 2002 2003 2004 /* 2005 * @implemented 2006 */ 2007 BOOL 2008 WINAPI 2009 DECLSPEC_HOTPATCH 2010 SetConsoleTextAttribute(HANDLE hConsoleOutput, 2011 WORD wAttributes) 2012 { 2013 CONSOLE_API_MESSAGE ApiMessage; 2014 PCONSOLE_SETTEXTATTRIB SetTextAttribRequest = &ApiMessage.Data.SetTextAttribRequest; 2015 2016 SetTextAttribRequest->ConsoleHandle = NtCurrentPeb()->ProcessParameters->ConsoleHandle; 2017 SetTextAttribRequest->OutputHandle = hConsoleOutput; 2018 SetTextAttribRequest->Attributes = wAttributes; 2019 2020 CsrClientCallServer((PCSR_API_MESSAGE)&ApiMessage, 2021 NULL, 2022 CSR_CREATE_API_NUMBER(CONSRV_SERVERDLL_INDEX, ConsolepSetTextAttribute), 2023 sizeof(*SetTextAttribRequest)); 2024 if (!NT_SUCCESS(ApiMessage.Status)) 2025 { 2026 BaseSetLastNTError(ApiMessage.Status); 2027 return FALSE; 2028 } 2029 2030 return TRUE; 2031 } 2032 2033 2034 static 2035 BOOL 2036 AddConsoleCtrlHandler(PHANDLER_ROUTINE HandlerRoutine) 2037 { 2038 PHANDLER_ROUTINE* NewCtrlHandlers = NULL; 2039 2040 if (HandlerRoutine == NULL) 2041 { 2042 NtCurrentPeb()->ProcessParameters->ConsoleFlags = TRUE; 2043 return TRUE; 2044 } 2045 2046 if (NrCtrlHandlers == NrAllocatedHandlers) 2047 { 2048 NewCtrlHandlers = RtlAllocateHeap(RtlGetProcessHeap(), 2049 0, 2050 (NrCtrlHandlers + 4) * sizeof(PHANDLER_ROUTINE)); 2051 if (NewCtrlHandlers == NULL) 2052 { 2053 SetLastError(ERROR_NOT_ENOUGH_MEMORY); 2054 return FALSE; 2055 } 2056 2057 memmove(NewCtrlHandlers, CtrlHandlers, sizeof(PHANDLER_ROUTINE) * NrCtrlHandlers); 2058 2059 if (NrAllocatedHandlers > 1) RtlFreeHeap(RtlGetProcessHeap(), 0, CtrlHandlers); 2060 2061 CtrlHandlers = NewCtrlHandlers; 2062 NrAllocatedHandlers += 4; 2063 } 2064 2065 ASSERT(NrCtrlHandlers < NrAllocatedHandlers); 2066 2067 CtrlHandlers[NrCtrlHandlers++] = HandlerRoutine; 2068 return TRUE; 2069 } 2070 2071 2072 static 2073 BOOL 2074 RemoveConsoleCtrlHandler(PHANDLER_ROUTINE HandlerRoutine) 2075 { 2076 ULONG i; 2077 2078 if (HandlerRoutine == NULL) 2079 { 2080 NtCurrentPeb()->ProcessParameters->ConsoleFlags = FALSE; 2081 return TRUE; 2082 } 2083 2084 for (i = 0; i < NrCtrlHandlers; i++) 2085 { 2086 if (CtrlHandlers[i] == HandlerRoutine) 2087 { 2088 if (i < (NrCtrlHandlers - 1)) 2089 { 2090 memmove(&CtrlHandlers[i], 2091 &CtrlHandlers[i+1], 2092 (NrCtrlHandlers - i + 1) * sizeof(PHANDLER_ROUTINE)); 2093 } 2094 2095 NrCtrlHandlers--; 2096 return TRUE; 2097 } 2098 } 2099 2100 SetLastError(ERROR_INVALID_PARAMETER); 2101 return FALSE; 2102 } 2103 2104 2105 /* 2106 * @implemented 2107 */ 2108 BOOL 2109 WINAPI 2110 DECLSPEC_HOTPATCH 2111 SetConsoleCtrlHandler(PHANDLER_ROUTINE HandlerRoutine, 2112 BOOL Add) 2113 { 2114 BOOL Ret; 2115 2116 RtlEnterCriticalSection(&ConsoleLock); 2117 2118 if (Add) 2119 Ret = AddConsoleCtrlHandler(HandlerRoutine); 2120 else 2121 Ret = RemoveConsoleCtrlHandler(HandlerRoutine); 2122 2123 RtlLeaveCriticalSection(&ConsoleLock); 2124 return Ret; 2125 } 2126 2127 2128 /* 2129 * @implemented 2130 */ 2131 BOOL 2132 WINAPI 2133 DECLSPEC_HOTPATCH 2134 GenerateConsoleCtrlEvent(DWORD dwCtrlEvent, 2135 DWORD dwProcessGroupId) 2136 { 2137 CONSOLE_API_MESSAGE ApiMessage; 2138 PCONSOLE_GENERATECTRLEVENT GenerateCtrlEventRequest = &ApiMessage.Data.GenerateCtrlEventRequest; 2139 2140 if (dwCtrlEvent != CTRL_C_EVENT && dwCtrlEvent != CTRL_BREAK_EVENT) 2141 { 2142 SetLastError(ERROR_INVALID_PARAMETER); 2143 return FALSE; 2144 } 2145 2146 GenerateCtrlEventRequest->ConsoleHandle = NtCurrentPeb()->ProcessParameters->ConsoleHandle; 2147 GenerateCtrlEventRequest->CtrlEvent = dwCtrlEvent; 2148 GenerateCtrlEventRequest->ProcessGroupId = dwProcessGroupId; 2149 2150 CsrClientCallServer((PCSR_API_MESSAGE)&ApiMessage, 2151 NULL, 2152 CSR_CREATE_API_NUMBER(CONSRV_SERVERDLL_INDEX, ConsolepGenerateCtrlEvent), 2153 sizeof(*GenerateCtrlEventRequest)); 2154 if (!NT_SUCCESS(ApiMessage.Status)) 2155 { 2156 BaseSetLastNTError(ApiMessage.Status); 2157 return FALSE; 2158 } 2159 2160 return TRUE; 2161 } 2162 2163 2164 static DWORD 2165 IntGetConsoleTitle(LPVOID lpConsoleTitle, DWORD dwNumChars, BOOLEAN bUnicode) 2166 { 2167 CONSOLE_API_MESSAGE ApiMessage; 2168 PCONSOLE_GETSETCONSOLETITLE TitleRequest = &ApiMessage.Data.TitleRequest; 2169 PCSR_CAPTURE_BUFFER CaptureBuffer; 2170 2171 if (dwNumChars == 0) return 0; 2172 2173 TitleRequest->ConsoleHandle = NtCurrentPeb()->ProcessParameters->ConsoleHandle; 2174 TitleRequest->Length = dwNumChars * (bUnicode ? sizeof(WCHAR) : sizeof(CHAR)); 2175 TitleRequest->Unicode = bUnicode; 2176 2177 CaptureBuffer = CsrAllocateCaptureBuffer(1, TitleRequest->Length); 2178 if (CaptureBuffer == NULL) 2179 { 2180 DPRINT1("CsrAllocateCaptureBuffer failed!\n"); 2181 SetLastError(ERROR_NOT_ENOUGH_MEMORY); 2182 return 0; 2183 } 2184 2185 CsrAllocateMessagePointer(CaptureBuffer, 2186 TitleRequest->Length, 2187 (PVOID*)&TitleRequest->Title); 2188 2189 CsrClientCallServer((PCSR_API_MESSAGE)&ApiMessage, 2190 CaptureBuffer, 2191 CSR_CREATE_API_NUMBER(CONSRV_SERVERDLL_INDEX, ConsolepGetTitle), 2192 sizeof(*TitleRequest)); 2193 if (!NT_SUCCESS(ApiMessage.Status)) 2194 { 2195 CsrFreeCaptureBuffer(CaptureBuffer); 2196 BaseSetLastNTError(ApiMessage.Status); 2197 return 0; 2198 } 2199 2200 dwNumChars = TitleRequest->Length / (bUnicode ? sizeof(WCHAR) : sizeof(CHAR)); 2201 2202 if (dwNumChars > 0) 2203 { 2204 RtlCopyMemory(lpConsoleTitle, TitleRequest->Title, TitleRequest->Length); 2205 2206 if (bUnicode) 2207 ((LPWSTR)lpConsoleTitle)[dwNumChars] = UNICODE_NULL; 2208 else 2209 ((LPSTR)lpConsoleTitle)[dwNumChars] = ANSI_NULL; 2210 } 2211 2212 CsrFreeCaptureBuffer(CaptureBuffer); 2213 2214 return dwNumChars; 2215 } 2216 2217 2218 /* 2219 * @implemented 2220 */ 2221 DWORD 2222 WINAPI 2223 DECLSPEC_HOTPATCH 2224 GetConsoleTitleW(LPWSTR lpConsoleTitle, 2225 DWORD nSize) 2226 { 2227 return IntGetConsoleTitle(lpConsoleTitle, nSize, TRUE); 2228 } 2229 2230 2231 /* 2232 * @implemented 2233 */ 2234 DWORD 2235 WINAPI 2236 DECLSPEC_HOTPATCH 2237 GetConsoleTitleA(LPSTR lpConsoleTitle, 2238 DWORD nSize) 2239 { 2240 return IntGetConsoleTitle(lpConsoleTitle, nSize, FALSE); 2241 } 2242 2243 2244 static BOOL 2245 IntSetConsoleTitle(CONST VOID *lpConsoleTitle, BOOLEAN bUnicode) 2246 { 2247 CONSOLE_API_MESSAGE ApiMessage; 2248 PCONSOLE_GETSETCONSOLETITLE TitleRequest = &ApiMessage.Data.TitleRequest; 2249 PCSR_CAPTURE_BUFFER CaptureBuffer; 2250 2251 ULONG NumChars = (ULONG)(lpConsoleTitle ? (bUnicode ? wcslen(lpConsoleTitle) : strlen(lpConsoleTitle)) : 0); 2252 2253 TitleRequest->ConsoleHandle = NtCurrentPeb()->ProcessParameters->ConsoleHandle; 2254 TitleRequest->Length = NumChars * (bUnicode ? sizeof(WCHAR) : sizeof(CHAR)); 2255 TitleRequest->Unicode = bUnicode; 2256 2257 CaptureBuffer = CsrAllocateCaptureBuffer(1, TitleRequest->Length); 2258 if (CaptureBuffer == NULL) 2259 { 2260 DPRINT1("CsrAllocateCaptureBuffer failed!\n"); 2261 SetLastError(ERROR_NOT_ENOUGH_MEMORY); 2262 return FALSE; 2263 } 2264 2265 CsrCaptureMessageBuffer(CaptureBuffer, 2266 (PVOID)lpConsoleTitle, 2267 TitleRequest->Length, 2268 (PVOID*)&TitleRequest->Title); 2269 2270 CsrClientCallServer((PCSR_API_MESSAGE)&ApiMessage, 2271 CaptureBuffer, 2272 CSR_CREATE_API_NUMBER(CONSRV_SERVERDLL_INDEX, ConsolepSetTitle), 2273 sizeof(*TitleRequest)); 2274 2275 CsrFreeCaptureBuffer(CaptureBuffer); 2276 2277 if (!NT_SUCCESS(ApiMessage.Status)) 2278 { 2279 BaseSetLastNTError(ApiMessage.Status); 2280 return FALSE; 2281 } 2282 2283 return TRUE; 2284 } 2285 2286 /* 2287 * @implemented 2288 */ 2289 BOOL 2290 WINAPI 2291 DECLSPEC_HOTPATCH 2292 SetConsoleTitleW(LPCWSTR lpConsoleTitle) 2293 { 2294 return IntSetConsoleTitle(lpConsoleTitle, TRUE); 2295 } 2296 2297 2298 /* 2299 * @implemented 2300 */ 2301 BOOL 2302 WINAPI 2303 DECLSPEC_HOTPATCH 2304 SetConsoleTitleA(LPCSTR lpConsoleTitle) 2305 { 2306 return IntSetConsoleTitle(lpConsoleTitle, FALSE); 2307 } 2308 2309 2310 /* 2311 * @implemented 2312 */ 2313 HANDLE 2314 WINAPI 2315 CreateConsoleScreenBuffer(DWORD dwDesiredAccess, 2316 DWORD dwShareMode, 2317 CONST SECURITY_ATTRIBUTES *lpSecurityAttributes, 2318 DWORD dwFlags, 2319 LPVOID lpScreenBufferData) 2320 { 2321 CONSOLE_API_MESSAGE ApiMessage; 2322 PCONSOLE_CREATESCREENBUFFER CreateScreenBufferRequest = &ApiMessage.Data.CreateScreenBufferRequest; 2323 PCSR_CAPTURE_BUFFER CaptureBuffer = NULL; 2324 PCONSOLE_GRAPHICS_BUFFER_INFO GraphicsBufferInfo = lpScreenBufferData; 2325 2326 if ( (dwDesiredAccess & ~(GENERIC_READ | GENERIC_WRITE)) || 2327 (dwShareMode & ~(FILE_SHARE_READ | FILE_SHARE_WRITE)) || 2328 (dwFlags != CONSOLE_TEXTMODE_BUFFER && dwFlags != CONSOLE_GRAPHICS_BUFFER) ) 2329 { 2330 SetLastError(ERROR_INVALID_PARAMETER); 2331 return INVALID_HANDLE_VALUE; 2332 } 2333 2334 CreateScreenBufferRequest->ConsoleHandle = NtCurrentPeb()->ProcessParameters->ConsoleHandle; 2335 CreateScreenBufferRequest->DesiredAccess = dwDesiredAccess; 2336 CreateScreenBufferRequest->InheritHandle = 2337 (lpSecurityAttributes ? lpSecurityAttributes->bInheritHandle : FALSE); 2338 CreateScreenBufferRequest->ShareMode = dwShareMode; 2339 CreateScreenBufferRequest->ScreenBufferType = dwFlags; 2340 2341 if (dwFlags == CONSOLE_GRAPHICS_BUFFER) 2342 { 2343 if (CreateScreenBufferRequest->InheritHandle || GraphicsBufferInfo == NULL) 2344 { 2345 SetLastError(ERROR_INVALID_PARAMETER); 2346 return INVALID_HANDLE_VALUE; 2347 } 2348 2349 CreateScreenBufferRequest->GraphicsBufferInfo = *GraphicsBufferInfo; 2350 2351 CaptureBuffer = CsrAllocateCaptureBuffer(1, GraphicsBufferInfo->dwBitMapInfoLength); 2352 if (CaptureBuffer == NULL) 2353 { 2354 SetLastError(ERROR_NOT_ENOUGH_MEMORY); 2355 return INVALID_HANDLE_VALUE; 2356 } 2357 2358 CsrCaptureMessageBuffer(CaptureBuffer, 2359 (PVOID)GraphicsBufferInfo->lpBitMapInfo, 2360 GraphicsBufferInfo->dwBitMapInfoLength, 2361 (PVOID*)&CreateScreenBufferRequest->GraphicsBufferInfo.lpBitMapInfo); 2362 } 2363 2364 CsrClientCallServer((PCSR_API_MESSAGE)&ApiMessage, 2365 CaptureBuffer, 2366 CSR_CREATE_API_NUMBER(CONSRV_SERVERDLL_INDEX, ConsolepCreateScreenBuffer), 2367 sizeof(*CreateScreenBufferRequest)); 2368 2369 if (CaptureBuffer) CsrFreeCaptureBuffer(CaptureBuffer); 2370 2371 if (!NT_SUCCESS(ApiMessage.Status)) 2372 { 2373 BaseSetLastNTError(ApiMessage.Status); 2374 return INVALID_HANDLE_VALUE; 2375 } 2376 2377 if (dwFlags == CONSOLE_GRAPHICS_BUFFER && GraphicsBufferInfo) 2378 { 2379 GraphicsBufferInfo->hMutex = CreateScreenBufferRequest->hMutex ; // CreateScreenBufferRequest->GraphicsBufferInfo.hMutex ; 2380 GraphicsBufferInfo->lpBitMap = CreateScreenBufferRequest->lpBitMap; // CreateScreenBufferRequest->GraphicsBufferInfo.lpBitMap; 2381 } 2382 2383 return CreateScreenBufferRequest->OutputHandle; 2384 } 2385 2386 2387 /* 2388 * @implemented 2389 */ 2390 UINT 2391 WINAPI 2392 DECLSPEC_HOTPATCH 2393 GetConsoleCP(VOID) 2394 { 2395 CONSOLE_API_MESSAGE ApiMessage; 2396 PCONSOLE_GETINPUTOUTPUTCP GetConsoleCPRequest = &ApiMessage.Data.GetConsoleCPRequest; 2397 2398 /* Get the Input Code Page */ 2399 GetConsoleCPRequest->ConsoleHandle = NtCurrentPeb()->ProcessParameters->ConsoleHandle; 2400 GetConsoleCPRequest->OutputCP = FALSE; 2401 2402 CsrClientCallServer((PCSR_API_MESSAGE)&ApiMessage, 2403 NULL, 2404 CSR_CREATE_API_NUMBER(CONSRV_SERVERDLL_INDEX, ConsolepGetCP), 2405 sizeof(*GetConsoleCPRequest)); 2406 if (!NT_SUCCESS(ApiMessage.Status)) 2407 { 2408 BaseSetLastNTError(ApiMessage.Status); 2409 return 0; 2410 } 2411 2412 return GetConsoleCPRequest->CodePage; 2413 } 2414 2415 2416 /* 2417 * @implemented 2418 */ 2419 BOOL 2420 WINAPI 2421 DECLSPEC_HOTPATCH 2422 SetConsoleCP(UINT wCodePageID) 2423 { 2424 CONSOLE_API_MESSAGE ApiMessage; 2425 PCONSOLE_SETINPUTOUTPUTCP SetConsoleCPRequest = &ApiMessage.Data.SetConsoleCPRequest; 2426 2427 /* Set the Input Code Page */ 2428 SetConsoleCPRequest->ConsoleHandle = NtCurrentPeb()->ProcessParameters->ConsoleHandle; 2429 SetConsoleCPRequest->CodePage = wCodePageID; 2430 SetConsoleCPRequest->OutputCP = FALSE; 2431 /* SetConsoleCPRequest->EventHandle; */ 2432 2433 CsrClientCallServer((PCSR_API_MESSAGE)&ApiMessage, 2434 NULL, 2435 CSR_CREATE_API_NUMBER(CONSRV_SERVERDLL_INDEX, ConsolepSetCP), 2436 sizeof(*SetConsoleCPRequest)); 2437 if (!NT_SUCCESS(ApiMessage.Status)) 2438 { 2439 BaseSetLastNTError(ApiMessage.Status); 2440 return FALSE; 2441 } 2442 2443 return TRUE; 2444 } 2445 2446 2447 /* 2448 * @implemented 2449 */ 2450 UINT 2451 WINAPI 2452 DECLSPEC_HOTPATCH 2453 GetConsoleOutputCP(VOID) 2454 { 2455 CONSOLE_API_MESSAGE ApiMessage; 2456 PCONSOLE_GETINPUTOUTPUTCP GetConsoleCPRequest = &ApiMessage.Data.GetConsoleCPRequest; 2457 2458 /* Get the Output Code Page */ 2459 GetConsoleCPRequest->ConsoleHandle = NtCurrentPeb()->ProcessParameters->ConsoleHandle; 2460 GetConsoleCPRequest->OutputCP = TRUE; 2461 2462 CsrClientCallServer((PCSR_API_MESSAGE)&ApiMessage, 2463 NULL, 2464 CSR_CREATE_API_NUMBER(CONSRV_SERVERDLL_INDEX, ConsolepGetCP), 2465 sizeof(*GetConsoleCPRequest)); 2466 if (!NT_SUCCESS(ApiMessage.Status)) 2467 { 2468 BaseSetLastNTError(ApiMessage.Status); 2469 return 0; 2470 } 2471 2472 return GetConsoleCPRequest->CodePage; 2473 } 2474 2475 2476 /* 2477 * @implemented 2478 */ 2479 BOOL 2480 WINAPI 2481 DECLSPEC_HOTPATCH 2482 SetConsoleOutputCP(UINT wCodePageID) 2483 { 2484 CONSOLE_API_MESSAGE ApiMessage; 2485 PCONSOLE_SETINPUTOUTPUTCP SetConsoleCPRequest = &ApiMessage.Data.SetConsoleCPRequest; 2486 2487 /* Set the Output Code Page */ 2488 SetConsoleCPRequest->ConsoleHandle = NtCurrentPeb()->ProcessParameters->ConsoleHandle; 2489 SetConsoleCPRequest->CodePage = wCodePageID; 2490 SetConsoleCPRequest->OutputCP = TRUE; 2491 /* SetConsoleCPRequest->EventHandle; */ 2492 2493 CsrClientCallServer((PCSR_API_MESSAGE)&ApiMessage, 2494 NULL, 2495 CSR_CREATE_API_NUMBER(CONSRV_SERVERDLL_INDEX, ConsolepSetCP), 2496 sizeof(*SetConsoleCPRequest)); 2497 if (!NT_SUCCESS(ApiMessage.Status)) 2498 { 2499 BaseSetLastNTError(ApiMessage.Status); 2500 return FALSE; 2501 } 2502 2503 return TRUE; 2504 } 2505 2506 2507 /* 2508 * @implemented 2509 */ 2510 DWORD 2511 WINAPI 2512 GetConsoleProcessList(LPDWORD lpdwProcessList, 2513 DWORD dwProcessCount) 2514 { 2515 CONSOLE_API_MESSAGE ApiMessage; 2516 PCONSOLE_GETPROCESSLIST GetProcessListRequest = &ApiMessage.Data.GetProcessListRequest; 2517 PCSR_CAPTURE_BUFFER CaptureBuffer; 2518 ULONG nProcesses = 0; 2519 2520 if (lpdwProcessList == NULL || dwProcessCount == 0) 2521 { 2522 SetLastError(ERROR_INVALID_PARAMETER); 2523 return 0; 2524 } 2525 2526 CaptureBuffer = CsrAllocateCaptureBuffer(1, dwProcessCount * sizeof(DWORD)); 2527 if (CaptureBuffer == NULL) 2528 { 2529 DPRINT1("CsrAllocateCaptureBuffer failed!\n"); 2530 SetLastError(ERROR_NOT_ENOUGH_MEMORY); 2531 return 0; 2532 } 2533 2534 GetProcessListRequest->ConsoleHandle = NtCurrentPeb()->ProcessParameters->ConsoleHandle; 2535 GetProcessListRequest->ProcessCount = dwProcessCount; 2536 2537 CsrAllocateMessagePointer(CaptureBuffer, 2538 dwProcessCount * sizeof(DWORD), 2539 (PVOID*)&GetProcessListRequest->ProcessIdsList); 2540 2541 CsrClientCallServer((PCSR_API_MESSAGE)&ApiMessage, 2542 CaptureBuffer, 2543 CSR_CREATE_API_NUMBER(CONSRV_SERVERDLL_INDEX, ConsolepGetProcessList), 2544 sizeof(*GetProcessListRequest)); 2545 if (!NT_SUCCESS(ApiMessage.Status)) 2546 { 2547 BaseSetLastNTError(ApiMessage.Status); 2548 } 2549 else 2550 { 2551 nProcesses = GetProcessListRequest->ProcessCount; 2552 if (dwProcessCount >= nProcesses) 2553 { 2554 RtlCopyMemory(lpdwProcessList, GetProcessListRequest->ProcessIdsList, nProcesses * sizeof(DWORD)); 2555 } 2556 } 2557 2558 CsrFreeCaptureBuffer(CaptureBuffer); 2559 return nProcesses; 2560 } 2561 2562 2563 /* 2564 * @implemented 2565 */ 2566 BOOL 2567 WINAPI 2568 GetConsoleSelectionInfo(PCONSOLE_SELECTION_INFO lpConsoleSelectionInfo) 2569 { 2570 CONSOLE_API_MESSAGE ApiMessage; 2571 PCONSOLE_GETSELECTIONINFO GetSelectionInfoRequest = &ApiMessage.Data.GetSelectionInfoRequest; 2572 2573 if (lpConsoleSelectionInfo == NULL) 2574 { 2575 SetLastError(ERROR_INVALID_PARAMETER); 2576 return FALSE; 2577 } 2578 2579 GetSelectionInfoRequest->ConsoleHandle = NtCurrentPeb()->ProcessParameters->ConsoleHandle; 2580 2581 CsrClientCallServer((PCSR_API_MESSAGE)&ApiMessage, 2582 NULL, 2583 CSR_CREATE_API_NUMBER(CONSRV_SERVERDLL_INDEX, ConsolepGetSelectionInfo), 2584 sizeof(*GetSelectionInfoRequest)); 2585 if (!NT_SUCCESS(ApiMessage.Status)) 2586 { 2587 BaseSetLastNTError(ApiMessage.Status); 2588 return FALSE; 2589 } 2590 2591 *lpConsoleSelectionInfo = GetSelectionInfoRequest->Info; 2592 2593 return TRUE; 2594 } 2595 2596 2597 /* 2598 * @implemented 2599 * @note Strongly inspired by AllocConsole. 2600 */ 2601 static BOOL 2602 IntAttachConsole(DWORD ProcessId, 2603 LPTHREAD_START_ROUTINE CtrlRoutine, 2604 LPTHREAD_START_ROUTINE PropRoutine, 2605 PCONSOLE_START_INFO ConsoleStartInfo) 2606 { 2607 BOOL Success = TRUE; 2608 NTSTATUS Status; 2609 2610 CONSOLE_API_MESSAGE ApiMessage; 2611 PCONSOLE_ATTACHCONSOLE AttachConsoleRequest = &ApiMessage.Data.AttachConsoleRequest; 2612 PCSR_CAPTURE_BUFFER CaptureBuffer; 2613 2614 AttachConsoleRequest->ProcessId = ProcessId; 2615 AttachConsoleRequest->CtrlRoutine = CtrlRoutine; 2616 AttachConsoleRequest->PropRoutine = PropRoutine; 2617 2618 CaptureBuffer = CsrAllocateCaptureBuffer(1, sizeof(CONSOLE_START_INFO)); 2619 if (CaptureBuffer == NULL) 2620 { 2621 SetLastError(ERROR_NOT_ENOUGH_MEMORY); 2622 Success = FALSE; 2623 goto Quit; 2624 } 2625 2626 CsrCaptureMessageBuffer(CaptureBuffer, 2627 ConsoleStartInfo, 2628 sizeof(CONSOLE_START_INFO), 2629 (PVOID*)&AttachConsoleRequest->ConsoleStartInfo); 2630 2631 CsrClientCallServer((PCSR_API_MESSAGE)&ApiMessage, 2632 CaptureBuffer, 2633 CSR_CREATE_API_NUMBER(CONSRV_SERVERDLL_INDEX, ConsolepAttach), 2634 sizeof(*AttachConsoleRequest)); 2635 if (!NT_SUCCESS(ApiMessage.Status)) 2636 { 2637 BaseSetLastNTError(ApiMessage.Status); 2638 Success = FALSE; 2639 goto Quit; 2640 } 2641 2642 // Is AttachConsoleRequest->ConsoleStartInfo->InitEvents aligned on handle boundary ???? 2643 Status = NtWaitForMultipleObjects(MAX_INIT_EVENTS, 2644 AttachConsoleRequest->ConsoleStartInfo->InitEvents, 2645 WaitAny, FALSE, NULL); 2646 if (!NT_SUCCESS(Status)) 2647 { 2648 BaseSetLastNTError(Status); 2649 Success = FALSE; 2650 goto Quit; 2651 } 2652 2653 NtClose(AttachConsoleRequest->ConsoleStartInfo->InitEvents[INIT_SUCCESS]); 2654 NtClose(AttachConsoleRequest->ConsoleStartInfo->InitEvents[INIT_FAILURE]); 2655 if (Status != INIT_SUCCESS) 2656 { 2657 NtCurrentPeb()->ProcessParameters->ConsoleHandle = NULL; 2658 Success = FALSE; 2659 } 2660 else 2661 { 2662 RtlCopyMemory(ConsoleStartInfo, 2663 AttachConsoleRequest->ConsoleStartInfo, 2664 sizeof(CONSOLE_START_INFO)); 2665 Success = TRUE; 2666 } 2667 2668 Quit: 2669 if (CaptureBuffer) CsrFreeCaptureBuffer(CaptureBuffer); 2670 return Success; 2671 } 2672 2673 BOOL 2674 WINAPI 2675 AttachConsole(DWORD dwProcessId) 2676 { 2677 BOOL Success; 2678 CONSOLE_START_INFO ConsoleStartInfo; 2679 2680 DWORD dummy; 2681 LCID lcid; 2682 2683 RtlEnterCriticalSection(&ConsoleLock); 2684 2685 if (NtCurrentPeb()->ProcessParameters->ConsoleHandle) 2686 { 2687 DPRINT1("AttachConsole: Attaching a console to a process already having one\n"); 2688 SetLastError(ERROR_ACCESS_DENIED); 2689 Success = FALSE; 2690 goto Quit; 2691 } 2692 2693 /* Set up the console properties */ 2694 SetUpConsoleInfo(FALSE, 2695 &dummy, 2696 NULL, 2697 &dummy, 2698 NULL, 2699 &ConsoleStartInfo); 2700 2701 Success = IntAttachConsole(dwProcessId, 2702 ConsoleControlDispatcher, 2703 PropDialogHandler, 2704 &ConsoleStartInfo); 2705 if (Success) 2706 { 2707 /* Set up the handles */ 2708 SetUpHandles(&ConsoleStartInfo); 2709 InputWaitHandle = ConsoleStartInfo.InputWaitHandle; 2710 2711 /* Initialize Console Ctrl Handling */ 2712 InitializeCtrlHandling(); 2713 2714 /* Sets the current console locale for this thread */ 2715 SetTEBLangID(lcid); 2716 } 2717 2718 Quit: 2719 RtlLeaveCriticalSection(&ConsoleLock); 2720 return Success; 2721 } 2722 2723 2724 /* 2725 * @implemented 2726 */ 2727 HWND 2728 WINAPI 2729 DECLSPEC_HOTPATCH 2730 GetConsoleWindow(VOID) 2731 { 2732 CONSOLE_API_MESSAGE ApiMessage; 2733 PCONSOLE_GETWINDOW GetWindowRequest = &ApiMessage.Data.GetWindowRequest; 2734 2735 GetWindowRequest->ConsoleHandle = NtCurrentPeb()->ProcessParameters->ConsoleHandle; 2736 2737 CsrClientCallServer((PCSR_API_MESSAGE)&ApiMessage, 2738 NULL, 2739 CSR_CREATE_API_NUMBER(CONSRV_SERVERDLL_INDEX, ConsolepGetConsoleWindow), 2740 sizeof(*GetWindowRequest)); 2741 if (!NT_SUCCESS(ApiMessage.Status)) 2742 { 2743 BaseSetLastNTError(ApiMessage.Status); 2744 return (HWND)NULL; 2745 } 2746 2747 return GetWindowRequest->WindowHandle; 2748 } 2749 2750 2751 /* 2752 * @implemented 2753 */ 2754 BOOL 2755 WINAPI 2756 DECLSPEC_HOTPATCH 2757 SetConsoleIcon(HICON hIcon) 2758 { 2759 CONSOLE_API_MESSAGE ApiMessage; 2760 PCONSOLE_SETICON SetIconRequest = &ApiMessage.Data.SetIconRequest; 2761 2762 SetIconRequest->ConsoleHandle = NtCurrentPeb()->ProcessParameters->ConsoleHandle; 2763 SetIconRequest->IconHandle = hIcon; 2764 2765 CsrClientCallServer((PCSR_API_MESSAGE)&ApiMessage, 2766 NULL, 2767 CSR_CREATE_API_NUMBER(CONSRV_SERVERDLL_INDEX, ConsolepSetIcon), 2768 sizeof(*SetIconRequest)); 2769 if (!NT_SUCCESS(ApiMessage.Status)) 2770 { 2771 BaseSetLastNTError(ApiMessage.Status); 2772 return FALSE; 2773 } 2774 2775 return TRUE; 2776 } 2777 2778 2779 /****************************************************************************** 2780 * \name SetConsoleInputExeNameW 2781 * \brief Sets the console input file name from a unicode string. 2782 * \param lpExeName Pointer to a unicode string with the name. 2783 * \return TRUE if successful, FALSE if unsuccessful. 2784 * \remarks If lpExeName is 0 or the string length is 0 or greater than 255, 2785 * the function fails and sets last error to ERROR_INVALID_PARAMETER. 2786 */ 2787 BOOL 2788 WINAPI 2789 DECLSPEC_HOTPATCH 2790 SetConsoleInputExeNameW(IN LPCWSTR lpExeName) 2791 { 2792 DWORD ExeLength; 2793 2794 ExeLength = lstrlenW(lpExeName); 2795 if ((ExeLength == 0) || (ExeLength >= EXENAME_LENGTH)) 2796 { 2797 /* Fail if string is empty or too long */ 2798 SetLastError(ERROR_INVALID_PARAMETER); 2799 return FALSE; 2800 } 2801 2802 RtlEnterCriticalSection(&ExeNameLock); 2803 _SEH2_TRY 2804 { 2805 /* Set the input EXE name, not NULL terminated */ 2806 RtlCopyMemory(ExeNameBuffer, lpExeName, ExeLength * sizeof(WCHAR)); 2807 ExeNameLength = (USHORT)ExeLength; 2808 } 2809 _SEH2_FINALLY 2810 { 2811 RtlLeaveCriticalSection(&ExeNameLock); 2812 } 2813 _SEH2_END; 2814 2815 return TRUE; 2816 } 2817 2818 2819 /****************************************************************************** 2820 * \name SetConsoleInputExeNameA 2821 * \brief Sets the console input file name from an ansi string. 2822 * \param lpExeName Pointer to an ansi string with the name. 2823 * \return TRUE if successful, FALSE if unsuccessful. 2824 * \remarks If lpExeName is 0 or the string length is 0 or greater than 255, 2825 * the function fails and sets last error to ERROR_INVALID_PARAMETER. 2826 */ 2827 BOOL 2828 WINAPI 2829 DECLSPEC_HOTPATCH 2830 SetConsoleInputExeNameA(IN LPCSTR lpExeName) 2831 { 2832 NTSTATUS Status; 2833 #ifdef USE_TEB_STATIC_USTR 2834 PUNICODE_STRING ExeNameU; 2835 #else 2836 UNICODE_STRING ExeNameU; 2837 #endif 2838 ANSI_STRING ExeNameA; 2839 #ifndef USE_TEB_STATIC_USTR 2840 WCHAR Buffer[EXENAME_LENGTH]; 2841 #endif 2842 2843 #ifdef USE_TEB_STATIC_USTR 2844 /* 2845 * Use the TEB static UNICODE string for storage. It is already 2846 * initialized at process creation time by the Memory Manager. 2847 */ 2848 ExeNameU = &NtCurrentTeb()->StaticUnicodeString; 2849 #endif 2850 2851 /* Initialize string for conversion */ 2852 RtlInitAnsiString(&ExeNameA, lpExeName); 2853 2854 #if 1 2855 if ((ExeNameA.Length == 0) || (ExeNameA.Length >= EXENAME_LENGTH)) 2856 { 2857 /* Fail if string is empty or too long */ 2858 SetLastError(ERROR_INVALID_PARAMETER); 2859 return FALSE; 2860 } 2861 #endif 2862 #ifndef USE_TEB_STATIC_USTR 2863 ExeNameU.Length = 0; 2864 ExeNameU.MaximumLength = (USHORT)sizeof(Buffer); 2865 ExeNameU.Buffer = Buffer; 2866 #endif 2867 2868 #ifdef USE_TEB_STATIC_USTR 2869 Status = RtlAnsiStringToUnicodeString(ExeNameU, &ExeNameA, FALSE); 2870 #else 2871 Status = RtlAnsiStringToUnicodeString(&ExeNameU, &ExeNameA, FALSE); 2872 #endif 2873 if (!NT_SUCCESS(Status)) 2874 { 2875 /* Fail if string is empty or too long */ 2876 if (Status == STATUS_BUFFER_OVERFLOW) 2877 SetLastError(ERROR_FILENAME_EXCED_RANGE); 2878 else 2879 SetLastError(ERROR_INVALID_PARAMETER); 2880 2881 return FALSE; 2882 } 2883 2884 #ifdef USE_TEB_STATIC_USTR 2885 return SetConsoleInputExeNameW(ExeNameU->Buffer); 2886 #else 2887 return SetConsoleInputExeNameW(ExeNameU.Buffer); 2888 #endif 2889 } 2890 2891 2892 /****************************************************************************** 2893 * \name GetConsoleInputExeNameW 2894 * \brief Retrieves the console input file name as unicode string. 2895 * \param nBufferLength Length of the buffer in WCHARs. 2896 * Specify 0 to receive the needed buffer length. 2897 * \param lpBuffer Pointer to a buffer that receives the string. 2898 * \return Needed buffer size if \p nBufferLength is 0. 2899 * Otherwise 1 if successful, 2 if buffer is too small. 2900 * \remarks Sets last error value to ERROR_BUFFER_OVERFLOW if the buffer 2901 * is not big enough. 2902 */ 2903 DWORD 2904 WINAPI 2905 DECLSPEC_HOTPATCH 2906 GetConsoleInputExeNameW(IN DWORD nBufferLength, 2907 OUT LPWSTR lpExeName) 2908 { 2909 if (nBufferLength <= ExeNameLength) 2910 { 2911 /* Buffer is not large enough! Return the correct size. */ 2912 SetLastError(ERROR_BUFFER_OVERFLOW); 2913 return ExeNameLength + 1; 2914 } 2915 2916 RtlEnterCriticalSection(&ExeNameLock); 2917 _SEH2_TRY 2918 { 2919 /* Copy the input EXE name and NULL-terminate it */ 2920 RtlCopyMemory(lpExeName, ExeNameBuffer, ExeNameLength * sizeof(WCHAR)); 2921 lpExeName[ExeNameLength] = UNICODE_NULL; 2922 } 2923 _SEH2_FINALLY 2924 { 2925 RtlLeaveCriticalSection(&ExeNameLock); 2926 } 2927 _SEH2_END; 2928 2929 return TRUE; 2930 } 2931 2932 2933 /****************************************************************************** 2934 * \name GetConsoleInputExeNameA 2935 * \brief Retrieves the console input file name as ansi string. 2936 * \param nBufferLength Length of the buffer in CHARs. 2937 * \param lpBuffer Pointer to a buffer that receives the string. 2938 * \return 1 if successful, 2 if buffer is too small. 2939 * \remarks Sets last error value to ERROR_BUFFER_OVERFLOW if the buffer 2940 * is not big enough. The buffer receives as much characters as fit. 2941 */ 2942 DWORD 2943 WINAPI 2944 DECLSPEC_HOTPATCH 2945 GetConsoleInputExeNameA(IN DWORD nBufferLength, 2946 OUT LPSTR lpExeName) 2947 { 2948 NTSTATUS Status; 2949 DWORD ExeLength; 2950 UNICODE_STRING BufferU; 2951 ANSI_STRING BufferA; 2952 WCHAR Buffer[EXENAME_LENGTH]; 2953 2954 /* Get the UNICODE name */ 2955 ExeLength = GetConsoleInputExeNameW(EXENAME_LENGTH, Buffer); 2956 2957 if ((ExeLength == 0) || (ExeLength >= EXENAME_LENGTH)) 2958 return ExeLength; 2959 2960 /* Initialize the strings for conversion */ 2961 RtlInitUnicodeString(&BufferU, Buffer); 2962 BufferA.Length = 0; 2963 BufferA.MaximumLength = (USHORT)nBufferLength; 2964 BufferA.Buffer = lpExeName; 2965 2966 /* Convert UNICODE name to ANSI, copying as much chars as it can fit */ 2967 Status = RtlUnicodeStringToAnsiString(&BufferA, &BufferU, FALSE); 2968 if (!NT_SUCCESS(Status)) 2969 { 2970 if (Status == STATUS_BUFFER_OVERFLOW) 2971 { 2972 SetLastError(ERROR_BUFFER_OVERFLOW); 2973 return ExeLength + 1; 2974 } 2975 SetLastError(ERROR_INVALID_PARAMETER); 2976 } 2977 2978 return ExeLength; 2979 } 2980 2981 BOOL 2982 WINAPI 2983 GetConsoleCharType(HANDLE hConsole, COORD Coord, PDWORD Type) 2984 { 2985 STUB; 2986 return FALSE; 2987 } 2988 2989 BOOL 2990 WINAPI 2991 DECLSPEC_HOTPATCH 2992 GetConsoleCursorMode(HANDLE hConsole, PBOOL pUnknown1, PBOOL pUnknown2) 2993 { 2994 STUB; 2995 return FALSE; 2996 } 2997 2998 BOOL 2999 WINAPI 3000 DECLSPEC_HOTPATCH 3001 SetConsoleCursorMode(HANDLE hConsole, BOOL Unknown1, BOOL Unknown2) 3002 { 3003 STUB; 3004 return FALSE; 3005 } 3006 3007 BOOL 3008 WINAPI 3009 DECLSPEC_HOTPATCH 3010 GetConsoleNlsMode(HANDLE hConsole, LPDWORD lpMode) 3011 { 3012 STUB; 3013 return FALSE; 3014 } 3015 3016 BOOL 3017 WINAPI 3018 DECLSPEC_HOTPATCH 3019 SetConsoleNlsMode(HANDLE hConsole, DWORD dwMode) 3020 { 3021 STUB; 3022 return FALSE; 3023 } 3024 3025 BOOL 3026 WINAPI 3027 DECLSPEC_HOTPATCH 3028 SetConsoleLocalEUDC(DWORD Unknown1, DWORD Unknown2, DWORD Unknown3, DWORD Unknown4) 3029 { 3030 STUB; 3031 return FALSE; 3032 } 3033 3034 BOOL 3035 WINAPI 3036 DECLSPEC_HOTPATCH 3037 RegisterConsoleIME(HWND hWnd, LPDWORD ThreadId) 3038 { 3039 STUB; 3040 return FALSE; 3041 } 3042 3043 BOOL 3044 WINAPI 3045 DECLSPEC_HOTPATCH 3046 RegisterConsoleOS2(BOOL bUnknown) 3047 { 3048 STUB; 3049 return FALSE; 3050 } 3051 3052 BOOL 3053 WINAPI 3054 DECLSPEC_HOTPATCH 3055 SetConsoleOS2OemFormat(BOOL bUnknown) 3056 { 3057 STUB; 3058 return FALSE; 3059 } 3060 3061 BOOL 3062 WINAPI 3063 DECLSPEC_HOTPATCH 3064 UnregisterConsoleIME(VOID) 3065 { 3066 STUB; 3067 return FALSE; 3068 } 3069 3070 3071 static 3072 BOOL 3073 IntGetConsoleKeyboardLayoutName(OUT PVOID pszLayoutName, 3074 IN BOOL bAnsi) 3075 { 3076 CONSOLE_API_MESSAGE ApiMessage; 3077 PCONSOLE_GETKBDLAYOUTNAME GetKbdLayoutNameRequest = &ApiMessage.Data.GetKbdLayoutNameRequest; 3078 3079 /* Set up the data to send to the Console Server */ 3080 GetKbdLayoutNameRequest->ConsoleHandle = NtCurrentPeb()->ProcessParameters->ConsoleHandle; 3081 GetKbdLayoutNameRequest->Ansi = bAnsi; 3082 3083 /* Call the server */ 3084 CsrClientCallServer((PCSR_API_MESSAGE)&ApiMessage, 3085 NULL, 3086 CSR_CREATE_API_NUMBER(CONSRV_SERVERDLL_INDEX, ConsolepGetKeyboardLayoutName), 3087 sizeof(*GetKbdLayoutNameRequest)); 3088 3089 /* Check for success */ 3090 if (!NT_SUCCESS(ApiMessage.Status)) 3091 { 3092 BaseSetLastNTError(ApiMessage.Status); 3093 return FALSE; 3094 } 3095 3096 /* Retrieve the results */ 3097 _SEH2_TRY 3098 { 3099 /* Copy only KL_NAMELENGTH == 9 characters, ANSI or UNICODE */ 3100 if (bAnsi) 3101 strncpy(pszLayoutName, (PCHAR)GetKbdLayoutNameRequest->LayoutBuffer, KL_NAMELENGTH); 3102 else 3103 wcsncpy(pszLayoutName, (PWCHAR)GetKbdLayoutNameRequest->LayoutBuffer, KL_NAMELENGTH); 3104 } 3105 _SEH2_EXCEPT(EXCEPTION_EXECUTE_HANDLER) 3106 { 3107 SetLastError(ERROR_INVALID_ACCESS); 3108 _SEH2_YIELD(return FALSE); 3109 } 3110 _SEH2_END; 3111 3112 return TRUE; 3113 } 3114 3115 /* 3116 * @implemented (undocumented) 3117 */ 3118 BOOL 3119 WINAPI 3120 DECLSPEC_HOTPATCH 3121 GetConsoleKeyboardLayoutNameA(OUT LPSTR pszLayoutName) 3122 { 3123 return IntGetConsoleKeyboardLayoutName(pszLayoutName, TRUE); 3124 } 3125 3126 /* 3127 * @implemented (undocumented) 3128 */ 3129 BOOL 3130 WINAPI 3131 DECLSPEC_HOTPATCH 3132 GetConsoleKeyboardLayoutNameW(OUT LPWSTR pszLayoutName) 3133 { 3134 return IntGetConsoleKeyboardLayoutName(pszLayoutName, FALSE); 3135 } 3136 3137 /* 3138 * @implemented 3139 */ 3140 DWORD 3141 WINAPI 3142 SetLastConsoleEventActive(VOID) 3143 { 3144 CONSOLE_API_MESSAGE ApiMessage; 3145 PCONSOLE_NOTIFYLASTCLOSE NotifyLastCloseRequest = &ApiMessage.Data.NotifyLastCloseRequest; 3146 3147 /* Set the flag used by the console control dispatcher */ 3148 LastCloseNotify = TRUE; 3149 3150 /* Set up the input arguments */ 3151 NotifyLastCloseRequest->ConsoleHandle = NtCurrentPeb()->ProcessParameters->ConsoleHandle; 3152 3153 /* Call CSRSS; just return the NTSTATUS cast to DWORD */ 3154 return CsrClientCallServer((PCSR_API_MESSAGE)&ApiMessage, 3155 NULL, 3156 CSR_CREATE_API_NUMBER(CONSRV_SERVERDLL_INDEX, ConsolepNotifyLastClose), 3157 sizeof(*NotifyLastCloseRequest)); 3158 } 3159 3160 /* EOF */ 3161