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
7 * <jimtabor@adsl-64-217-116-74.dsl.hstntx.swbell.net>
10 /* INCLUDES *******************************************************************/
18 /* GLOBALS ********************************************************************/
20 extern RTL_CRITICAL_SECTION ConsoleLock
;
21 extern BOOL ConsoleInitialized
;
22 extern BOOL WINAPI
IsDebuggerPresent(VOID
);
24 /* Console reserved "file" names */
25 static LPCWSTR BaseConFileName
= CONSOLE_FILE_NAME
;
26 static LPCWSTR BaseConInputFileName
= CONSOLE_INPUT_FILE_NAME
;
27 static LPCWSTR BaseConOutputFileName
= CONSOLE_OUTPUT_FILE_NAME
;
29 PHANDLER_ROUTINE InitialHandler
[1];
30 PHANDLER_ROUTINE
* CtrlHandlers
;
32 ULONG NrAllocatedHandlers
;
34 HANDLE InputWaitHandle
= INVALID_HANDLE_VALUE
;
36 #define INPUTEXENAME_BUFLEN 256
37 static WCHAR InputExeName
[INPUTEXENAME_BUFLEN
];
40 /* Default Console Control Handler ********************************************/
44 DefaultConsoleCtrlHandler(DWORD Event
)
46 DPRINT("Default handler called: %lx\n", Event
);
50 DPRINT("Ctrl-C Event\n");
53 case CTRL_BREAK_EVENT
:
54 DPRINT("Ctrl-Break Event\n");
57 case CTRL_CLOSE_EVENT
:
58 DPRINT("Ctrl Close Event\n");
61 case CTRL_LOGOFF_EVENT
:
62 DPRINT("Ctrl Logoff Event\n");
65 case CTRL_SHUTDOWN_EVENT
:
66 DPRINT("Ctrl Shutdown Event\n");
70 ExitProcess(CONTROL_C_EXIT
);
76 ConsoleControlDispatcher(IN LPVOID lpThreadParameter
)
79 DWORD CodeAndFlag
= PtrToUlong(lpThreadParameter
);
80 DWORD nCode
= CodeAndFlag
& MAXLONG
;
82 EXCEPTION_RECORD erException
;
84 DPRINT1("Console Dispatcher Active: %lx %lx\n", CodeAndFlag
, nCode
);
85 SetThreadPriority(GetCurrentThread(), THREAD_PRIORITY_HIGHEST
);
90 case CTRL_BREAK_EVENT
:
92 if (IsDebuggerPresent())
94 erException
.ExceptionCode
= (nCode
== CTRL_C_EVENT
?
95 DBG_CONTROL_C
: DBG_CONTROL_BREAK
);
96 erException
.ExceptionFlags
= 0;
97 erException
.ExceptionRecord
= NULL
;
98 erException
.ExceptionAddress
= DefaultConsoleCtrlHandler
;
99 erException
.NumberParameters
= 0;
103 RtlRaiseException(&erException
);
105 _SEH2_EXCEPT(EXCEPTION_EXECUTE_HANDLER
)
107 RtlEnterCriticalSection(&ConsoleLock
);
109 if ((nCode
!= CTRL_C_EVENT
) ||
110 (NtCurrentPeb()->ProcessParameters
->ConsoleFlags
!= 1))
112 for (i
= NrCtrlHandlers
; i
> 0; i
--)
114 if (CtrlHandlers
[i
- 1](nCode
)) break;
118 RtlLeaveCriticalSection(&ConsoleLock
);
127 case CTRL_CLOSE_EVENT
:
128 case CTRL_LOGOFF_EVENT
:
129 case CTRL_SHUTDOWN_EVENT
:
137 ExitProcess(CONTROL_C_EXIT
);
145 ASSERT(ConsoleInitialized
);
147 RtlEnterCriticalSection(&ConsoleLock
);
150 if ((nCode
!= CTRL_C_EVENT
) || (NtCurrentPeb()->ProcessParameters
->ConsoleFlags
!= 1))
152 for (i
= NrCtrlHandlers
; i
> 0; i
--)
155 (CodeAndFlag
& MINLONG
) &&
156 ((nCode
== CTRL_LOGOFF_EVENT
) || (nCode
== CTRL_SHUTDOWN_EVENT
)))
158 DPRINT("Skipping system/service apps\n");
162 if (CtrlHandlers
[i
- 1](nCode
))
166 case CTRL_CLOSE_EVENT
:
167 case CTRL_LOGOFF_EVENT
:
168 case CTRL_SHUTDOWN_EVENT
:
170 nExitCode
= CodeAndFlag
;
178 RtlLeaveCriticalSection(&ConsoleLock
);
180 ExitThread(nExitCode
);
181 return STATUS_SUCCESS
;
186 InitConsoleCtrlHandling(VOID
)
188 /* Initialize Console Ctrl Handler */
189 NrAllocatedHandlers
= NrCtrlHandlers
= 1;
190 CtrlHandlers
= InitialHandler
;
191 CtrlHandlers
[0] = DefaultConsoleCtrlHandler
;
195 /* FUNCTIONS ******************************************************************/
198 IntCheckForConsoleFileName(IN LPCWSTR pszName
,
199 IN DWORD dwDesiredAccess
)
201 LPCWSTR ConsoleName
= pszName
;
202 ULONG DeviceNameInfo
;
205 * Check whether we deal with a DOS device, and if so,
206 * strip the path till the file name.
207 * Therefore, things like \\.\CON or C:\some_path\CONIN$
208 * are transformed into CON or CONIN$, for example.
210 DeviceNameInfo
= RtlIsDosDeviceName_U(pszName
);
211 if (DeviceNameInfo
!= 0)
213 ConsoleName
= (LPCWSTR
)((ULONG_PTR
)ConsoleName
+ ((DeviceNameInfo
>> 16) & 0xFFFF));
216 /* Return a standard console "file" name according to what we passed in parameters */
217 if (_wcsicmp(ConsoleName
, BaseConInputFileName
) == 0)
219 return BaseConInputFileName
;
221 else if (_wcsicmp(ConsoleName
, BaseConOutputFileName
) == 0)
223 return BaseConOutputFileName
;
225 else if (_wcsicmp(ConsoleName
, BaseConFileName
) == 0)
227 if ((dwDesiredAccess
& (GENERIC_READ
| GENERIC_WRITE
)) == GENERIC_READ
)
229 return BaseConInputFileName
;
231 else if ((dwDesiredAccess
& (GENERIC_READ
| GENERIC_WRITE
)) == GENERIC_WRITE
)
233 return BaseConOutputFileName
;
237 /* If we are there, that means that either the file name or the desired access are wrong */
243 * @implemented (Undocumented)
244 * @note See http://undoc.airesoft.co.uk/kernel32.dll/ConsoleMenuControl.php
248 ConsoleMenuControl(HANDLE hConsoleOutput
,
252 CONSOLE_API_MESSAGE ApiMessage
;
253 PCONSOLE_MENUCONTROL MenuControlRequest
= &ApiMessage
.Data
.MenuControlRequest
;
255 MenuControlRequest
->OutputHandle
= hConsoleOutput
;
256 MenuControlRequest
->dwCmdIdLow
= dwCmdIdLow
;
257 MenuControlRequest
->dwCmdIdHigh
= dwCmdIdHigh
;
258 MenuControlRequest
->hMenu
= NULL
;
260 CsrClientCallServer((PCSR_API_MESSAGE
)&ApiMessage
,
262 CSR_CREATE_API_NUMBER(CONSRV_SERVERDLL_INDEX
, ConsolepMenuControl
),
263 sizeof(CONSOLE_MENUCONTROL
));
265 return MenuControlRequest
->hMenu
;
274 DuplicateConsoleHandle(HANDLE hConsole
,
275 DWORD dwDesiredAccess
,
280 CONSOLE_API_MESSAGE ApiMessage
;
281 PCONSOLE_DUPLICATEHANDLE DuplicateHandleRequest
= &ApiMessage
.Data
.DuplicateHandleRequest
;
283 if ( (dwOptions
& ~(DUPLICATE_CLOSE_SOURCE
| DUPLICATE_SAME_ACCESS
)) ||
284 (!(dwOptions
& DUPLICATE_SAME_ACCESS
) &&
285 (dwDesiredAccess
& ~(GENERIC_READ
| GENERIC_WRITE
))) )
287 SetLastError (ERROR_INVALID_PARAMETER
);
288 return INVALID_HANDLE_VALUE
;
291 DuplicateHandleRequest
->ConsoleHandle
= hConsole
;
292 DuplicateHandleRequest
->Access
= dwDesiredAccess
;
293 DuplicateHandleRequest
->Inheritable
= bInheritHandle
;
294 DuplicateHandleRequest
->Options
= dwOptions
;
296 Status
= CsrClientCallServer((PCSR_API_MESSAGE
)&ApiMessage
,
298 CSR_CREATE_API_NUMBER(CONSRV_SERVERDLL_INDEX
, ConsolepDuplicateHandle
),
299 sizeof(CONSOLE_DUPLICATEHANDLE
));
300 if (!NT_SUCCESS(Status
))
302 BaseSetLastNTError(Status
);
303 return INVALID_HANDLE_VALUE
;
306 return DuplicateHandleRequest
->ConsoleHandle
;
315 GetConsoleDisplayMode(LPDWORD lpModeFlags
)
318 CONSOLE_API_MESSAGE ApiMessage
;
319 PCONSOLE_GETDISPLAYMODE GetDisplayModeRequest
= &ApiMessage
.Data
.GetDisplayModeRequest
;
321 if (lpModeFlags
== NULL
)
323 SetLastError(ERROR_INVALID_PARAMETER
);
327 // GetDisplayModeRequest->OutputHandle = hConsoleOutput;
329 Status
= CsrClientCallServer((PCSR_API_MESSAGE
)&ApiMessage
,
331 CSR_CREATE_API_NUMBER(CONSRV_SERVERDLL_INDEX
, ConsolepGetDisplayMode
),
332 sizeof(CONSOLE_GETDISPLAYMODE
));
333 if (!NT_SUCCESS(Status
))
335 BaseSetLastNTError(Status
);
339 *lpModeFlags
= GetDisplayModeRequest
->DisplayMode
;
345 * @unimplemented (Undocumented)
349 GetConsoleFontInfo(DWORD Unknown0
,
354 DPRINT1("GetConsoleFontInfo(0x%x, 0x%x, 0x%x, 0x%x) UNIMPLEMENTED!\n", Unknown0
, Unknown1
, Unknown2
, Unknown3
);
355 SetLastError(ERROR_CALL_NOT_IMPLEMENTED
);
365 GetConsoleFontSize(HANDLE hConsoleOutput
,
368 COORD Empty
= {0, 0};
369 DPRINT1("GetConsoleFontSize(0x%x, 0x%x) UNIMPLEMENTED!\n", hConsoleOutput
, nFont
);
370 SetLastError(ERROR_CALL_NOT_IMPLEMENTED
);
376 * @implemented (Undocumented)
380 GetConsoleHardwareState(HANDLE hConsoleOutput
,
385 CONSOLE_API_MESSAGE ApiMessage
;
386 PCONSOLE_GETSETHWSTATE HardwareStateRequest
= &ApiMessage
.Data
.HardwareStateRequest
;
388 DPRINT1("GetConsoleHardwareState(%d, 0x%p) UNIMPLEMENTED!\n", Flags
, State
);
392 SetLastError(ERROR_INVALID_PARAMETER
);
396 HardwareStateRequest
->OutputHandle
= hConsoleOutput
;
398 Status
= CsrClientCallServer((PCSR_API_MESSAGE
)&ApiMessage
,
400 CSR_CREATE_API_NUMBER(CONSRV_SERVERDLL_INDEX
, ConsolepGetHardwareState
),
401 sizeof(CONSOLE_GETSETHWSTATE
));
402 if (!NT_SUCCESS(Status
))
404 BaseSetLastNTError(Status
);
408 *State
= HardwareStateRequest
->State
;
414 * @implemented (Undocumented)
418 GetConsoleInputWaitHandle(VOID
)
420 return InputWaitHandle
;
429 GetCurrentConsoleFont(HANDLE hConsoleOutput
,
431 PCONSOLE_FONT_INFO lpConsoleCurrentFont
)
433 DPRINT1("GetCurrentConsoleFont(0x%x, 0x%x, 0x%x) UNIMPLEMENTED!\n", hConsoleOutput
, bMaximumWindow
, lpConsoleCurrentFont
);
434 SetLastError(ERROR_CALL_NOT_IMPLEMENTED
);
440 * @unimplemented (Undocumented)
444 GetNumberOfConsoleFonts(VOID
)
446 DPRINT1("GetNumberOfConsoleFonts() UNIMPLEMENTED!\n");
447 SetLastError(ERROR_CALL_NOT_IMPLEMENTED
);
453 * @implemented (Undocumented)
454 * @note See http://blog.airesoft.co.uk/2012/10/things-ms-can-do-that-they-dont-tell-you-about-console-graphics/
458 InvalidateConsoleDIBits(IN HANDLE hConsoleOutput
,
459 IN PSMALL_RECT lpRect
)
462 CONSOLE_API_MESSAGE ApiMessage
;
463 PCONSOLE_INVALIDATEDIBITS InvalidateDIBitsRequest
= &ApiMessage
.Data
.InvalidateDIBitsRequest
;
467 SetLastError(ERROR_INVALID_PARAMETER
);
471 InvalidateDIBitsRequest
->OutputHandle
= hConsoleOutput
;
472 InvalidateDIBitsRequest
->Region
= *lpRect
;
474 Status
= CsrClientCallServer((PCSR_API_MESSAGE
)&ApiMessage
,
476 CSR_CREATE_API_NUMBER(CONSRV_SERVERDLL_INDEX
, ConsolepInvalidateBitMapRect
),
477 sizeof(CONSOLE_INVALIDATEDIBITS
));
478 if (!NT_SUCCESS(Status
))
480 BaseSetLastNTError(Status
);
489 * @unimplemented (Undocumented)
493 OpenConsoleW(LPCWSTR wsName
,
494 DWORD dwDesiredAccess
,
498 NTSTATUS Status
= STATUS_SUCCESS
;
499 CONSOLE_API_MESSAGE ApiMessage
;
500 PCONSOLE_OPENCONSOLE OpenConsoleRequest
= &ApiMessage
.Data
.OpenConsoleRequest
;
501 CONSOLE_HANDLE_TYPE HandleType
;
503 if (wsName
&& 0 == _wcsicmp(wsName
, BaseConInputFileName
))
505 HandleType
= HANDLE_INPUT
;
507 else if (wsName
&& 0 == _wcsicmp(wsName
, BaseConOutputFileName
))
509 HandleType
= HANDLE_OUTPUT
;
513 SetLastError(ERROR_INVALID_PARAMETER
);
514 return INVALID_HANDLE_VALUE
;
517 if ( (dwDesiredAccess
& ~(GENERIC_READ
| GENERIC_WRITE
)) ||
518 (dwShareMode
& ~(FILE_SHARE_READ
| FILE_SHARE_WRITE
)) )
520 SetLastError(ERROR_INVALID_PARAMETER
);
521 return INVALID_HANDLE_VALUE
;
524 OpenConsoleRequest
->HandleType
= HandleType
;
525 OpenConsoleRequest
->Access
= dwDesiredAccess
;
526 OpenConsoleRequest
->Inheritable
= bInheritHandle
;
527 OpenConsoleRequest
->ShareMode
= dwShareMode
;
529 Status
= CsrClientCallServer((PCSR_API_MESSAGE
)&ApiMessage
,
531 CSR_CREATE_API_NUMBER(CONSRV_SERVERDLL_INDEX
, ConsolepOpenConsole
),
532 sizeof(CONSOLE_OPENCONSOLE
));
533 if (!NT_SUCCESS(Status
))
535 BaseSetLastNTError(Status
);
536 return INVALID_HANDLE_VALUE
;
539 return OpenConsoleRequest
->ConsoleHandle
;
544 * @unimplemented (Undocumented)
548 SetConsoleCursor(DWORD Unknown0
,
551 DPRINT1("SetConsoleCursor(0x%x, 0x%x) UNIMPLEMENTED!\n", Unknown0
, Unknown1
);
552 SetLastError(ERROR_CALL_NOT_IMPLEMENTED
);
562 SetConsoleDisplayMode(HANDLE hConsoleOutput
,
564 PCOORD lpNewScreenBufferDimensions
)
567 CONSOLE_API_MESSAGE ApiMessage
;
568 PCONSOLE_SETDISPLAYMODE SetDisplayModeRequest
= &ApiMessage
.Data
.SetDisplayModeRequest
;
570 SetDisplayModeRequest
->OutputHandle
= hConsoleOutput
;
571 SetDisplayModeRequest
->DisplayMode
= dwFlags
;
572 SetDisplayModeRequest
->NewSBDim
.X
= 0;
573 SetDisplayModeRequest
->NewSBDim
.Y
= 0;
575 Status
= CsrClientCallServer((PCSR_API_MESSAGE
)&ApiMessage
,
577 CSR_CREATE_API_NUMBER(CONSRV_SERVERDLL_INDEX
, ConsolepSetDisplayMode
),
578 sizeof(CONSOLE_SETDISPLAYMODE
));
579 if (!NT_SUCCESS(Status
))
581 BaseSetLastNTError(Status
);
585 if (lpNewScreenBufferDimensions
)
586 *lpNewScreenBufferDimensions
= SetDisplayModeRequest
->NewSBDim
;
593 * @unimplemented (Undocumented)
597 SetConsoleFont(DWORD Unknown0
,
600 DPRINT1("SetConsoleFont(0x%x, 0x%x) UNIMPLEMENTED!\n", Unknown0
, Unknown1
);
601 SetLastError(ERROR_CALL_NOT_IMPLEMENTED
);
607 * @implemented (Undocumented)
611 SetConsoleHardwareState(HANDLE hConsoleOutput
,
616 CONSOLE_API_MESSAGE ApiMessage
;
617 PCONSOLE_GETSETHWSTATE HardwareStateRequest
= &ApiMessage
.Data
.HardwareStateRequest
;
619 DPRINT1("SetConsoleHardwareState(%d, %d) UNIMPLEMENTED!\n", Flags
, State
);
621 HardwareStateRequest
->OutputHandle
= hConsoleOutput
;
622 HardwareStateRequest
->State
= State
;
624 Status
= CsrClientCallServer((PCSR_API_MESSAGE
)&ApiMessage
,
626 CSR_CREATE_API_NUMBER(CONSRV_SERVERDLL_INDEX
, ConsolepSetHardwareState
),
627 sizeof(CONSOLE_GETSETHWSTATE
));
628 if (!NT_SUCCESS(Status
))
630 BaseSetLastNTError(Status
);
639 * @unimplemented (Undocumented)
643 SetConsoleKeyShortcuts(DWORD Unknown0
,
648 DPRINT1("SetConsoleKeyShortcuts(0x%x, 0x%x, 0x%x, 0x%x) UNIMPLEMENTED!\n", Unknown0
, Unknown1
, Unknown2
, Unknown3
);
649 SetLastError(ERROR_CALL_NOT_IMPLEMENTED
);
655 * @unimplemented (Undocumented)
659 SetConsoleMaximumWindowSize(DWORD Unknown0
,
662 DPRINT1("SetConsoleMaximumWindowSize(0x%x, 0x%x) UNIMPLEMENTED!\n", Unknown0
, Unknown1
);
663 SetLastError(ERROR_CALL_NOT_IMPLEMENTED
);
669 * @implemented (Undocumented)
673 SetConsoleMenuClose(BOOL bEnable
)
676 CONSOLE_API_MESSAGE ApiMessage
;
677 PCONSOLE_SETMENUCLOSE SetMenuCloseRequest
= &ApiMessage
.Data
.SetMenuCloseRequest
;
679 SetMenuCloseRequest
->Enable
= bEnable
;
681 Status
= CsrClientCallServer((PCSR_API_MESSAGE
)&ApiMessage
,
683 CSR_CREATE_API_NUMBER(CONSRV_SERVERDLL_INDEX
, ConsolepSetMenuClose
),
684 sizeof(CONSOLE_SETMENUCLOSE
));
685 if (!NT_SUCCESS(Status
))
687 BaseSetLastNTError(Status
);
696 * @unimplemented (Undocumented)
700 SetConsolePalette(DWORD Unknown0
,
704 DPRINT1("SetConsolePalette(0x%x, 0x%x, 0x%x) UNIMPLEMENTED!\n", Unknown0
, Unknown1
, Unknown2
);
705 SetLastError(ERROR_CALL_NOT_IMPLEMENTED
);
710 * @unimplemented (Undocumented)
714 ShowConsoleCursor(DWORD Unknown0
,
717 DPRINT1("ShowConsoleCursor(0x%x, 0x%x) UNIMPLEMENTED!\n", Unknown0
, Unknown1
);
718 SetLastError(ERROR_CALL_NOT_IMPLEMENTED
);
724 * FUNCTION: Checks whether the given handle is a valid console handle.
727 * Handle - Handle to be checked
730 * TRUE: Handle is a valid console handle
731 * FALSE: Handle is not a valid console handle.
733 * STATUS: Officially undocumented
739 VerifyConsoleIoHandle(HANDLE Handle
)
742 CONSOLE_API_MESSAGE ApiMessage
;
744 ApiMessage
.Data
.VerifyHandleRequest
.ConsoleHandle
= Handle
;
746 Status
= CsrClientCallServer((PCSR_API_MESSAGE
)&ApiMessage
,
748 CSR_CREATE_API_NUMBER(CONSRV_SERVERDLL_INDEX
, ConsolepVerifyIoHandle
),
749 sizeof(CONSOLE_VERIFYHANDLE
));
750 if (!NT_SUCCESS(Status
))
752 BaseSetLastNTError(Status
);
765 WriteConsoleInputVDMA(DWORD Unknown0
,
770 DPRINT1("WriteConsoleInputVDMA(0x%x, 0x%x, 0x%x, 0x%x) UNIMPLEMENTED!\n", Unknown0
, Unknown1
, Unknown2
, Unknown3
);
771 SetLastError(ERROR_CALL_NOT_IMPLEMENTED
);
781 WriteConsoleInputVDMW(DWORD Unknown0
,
786 DPRINT1("WriteConsoleInputVDMW(0x%x, 0x%x, 0x%x, 0x%x) UNIMPLEMENTED!\n", Unknown0
, Unknown1
, Unknown2
, Unknown3
);
787 SetLastError(ERROR_CALL_NOT_IMPLEMENTED
);
793 * @implemented (Undocumented)
797 CloseConsoleHandle(HANDLE Handle
)
800 CONSOLE_API_MESSAGE ApiMessage
;
802 ApiMessage
.Data
.CloseHandleRequest
.ConsoleHandle
= Handle
;
804 Status
= CsrClientCallServer((PCSR_API_MESSAGE
)&ApiMessage
,
806 CSR_CREATE_API_NUMBER(CONSRV_SERVERDLL_INDEX
, ConsolepCloseHandle
),
807 sizeof(CONSOLE_CLOSEHANDLE
));
808 if (!NT_SUCCESS(Status
))
810 BaseSetLastNTError(Status
);
823 GetStdHandle(DWORD nStdHandle
)
825 * FUNCTION: Get a handle for the standard input, standard output
826 * and a standard error device.
829 * nStdHandle - Specifies the device for which to return the handle.
831 * RETURNS: If the function succeeds, the return value is the handle
832 * of the specified device. Otherwise the value is INVALID_HANDLE_VALUE.
835 PRTL_USER_PROCESS_PARAMETERS Ppb
= NtCurrentPeb()->ProcessParameters
;
839 case STD_INPUT_HANDLE
:
840 return Ppb
->StandardInput
;
842 case STD_OUTPUT_HANDLE
:
843 return Ppb
->StandardOutput
;
845 case STD_ERROR_HANDLE
:
846 return Ppb
->StandardError
;
849 SetLastError(ERROR_INVALID_HANDLE
);
850 return INVALID_HANDLE_VALUE
;
859 SetStdHandle(DWORD nStdHandle
,
862 * FUNCTION: Set the handle for the standard input, standard output or
863 * the standard error device.
866 * nStdHandle - Specifies the handle to be set.
867 * hHandle - The handle to set.
869 * RETURNS: TRUE if the function succeeds, FALSE otherwise.
872 PRTL_USER_PROCESS_PARAMETERS Ppb
= NtCurrentPeb()->ProcessParameters
;
874 /* no need to check if hHandle == INVALID_HANDLE_VALUE */
878 case STD_INPUT_HANDLE
:
879 Ppb
->StandardInput
= hHandle
;
882 case STD_OUTPUT_HANDLE
:
883 Ppb
->StandardOutput
= hHandle
;
886 case STD_ERROR_HANDLE
:
887 Ppb
->StandardError
= hHandle
;
891 /* Windows for whatever reason sets the last error to ERROR_INVALID_HANDLE here */
892 SetLastError(ERROR_INVALID_HANDLE
);
897 /*--------------------------------------------------------------
907 PRTL_USER_PROCESS_PARAMETERS Parameters
= NtCurrentPeb()->ProcessParameters
;
908 CONSOLE_API_MESSAGE ApiMessage
;
909 PCONSOLE_ALLOCCONSOLE AllocConsoleRequest
= &ApiMessage
.Data
.AllocConsoleRequest
;
910 PCSR_CAPTURE_BUFFER CaptureBuffer
;
911 LPWSTR AppPath
= NULL
;
914 if (Parameters
->ConsoleHandle
)
916 DPRINT1("AllocConsole: Allocating a console to a process already having one\n");
917 SetLastError(ERROR_ACCESS_DENIED
);
921 CaptureBuffer
= CsrAllocateCaptureBuffer(1, sizeof(CONSOLE_START_INFO
));
922 if (CaptureBuffer
== NULL
)
924 DPRINT1("CsrAllocateCaptureBuffer failed!\n");
925 SetLastError(ERROR_NOT_ENOUGH_MEMORY
);
929 CsrAllocateMessagePointer(CaptureBuffer
,
930 sizeof(CONSOLE_START_INFO
),
931 (PVOID
*)&AllocConsoleRequest
->ConsoleStartInfo
);
933 /** Copied from BasepInitConsole **********************************************/
934 InitConsoleInfo(AllocConsoleRequest
->ConsoleStartInfo
);
936 AppPath
= AllocConsoleRequest
->ConsoleStartInfo
->AppPath
;
937 Length
= min(MAX_PATH
, Parameters
->ImagePathName
.Length
/ sizeof(WCHAR
));
938 wcsncpy(AppPath
, Parameters
->ImagePathName
.Buffer
, Length
);
939 AppPath
[Length
] = L
'\0';
940 /******************************************************************************/
942 AllocConsoleRequest
->Console
= NULL
;
943 AllocConsoleRequest
->CtrlDispatcher
= ConsoleControlDispatcher
;
944 AllocConsoleRequest
->PropDispatcher
= PropDialogHandler
;
946 Status
= CsrClientCallServer((PCSR_API_MESSAGE
)&ApiMessage
,
948 CSR_CREATE_API_NUMBER(CONSRV_SERVERDLL_INDEX
, ConsolepAlloc
),
949 sizeof(CONSOLE_ALLOCCONSOLE
));
951 CsrFreeCaptureBuffer(CaptureBuffer
);
953 if (!NT_SUCCESS(Status
))
955 BaseSetLastNTError(Status
);
959 Parameters
->ConsoleHandle
= AllocConsoleRequest
->Console
;
960 SetStdHandle(STD_INPUT_HANDLE
, AllocConsoleRequest
->InputHandle
);
961 SetStdHandle(STD_OUTPUT_HANDLE
, AllocConsoleRequest
->OutputHandle
);
962 SetStdHandle(STD_ERROR_HANDLE
, AllocConsoleRequest
->ErrorHandle
);
964 /* Initialize Console Ctrl Handler */
965 InitConsoleCtrlHandling();
967 InputWaitHandle
= AllocConsoleRequest
->InputWaitHandle
;
973 /*--------------------------------------------------------------
982 // AG: I'm not sure if this is correct (what happens to std handles?)
983 // but I just tried to reverse what AllocConsole() does...
986 CONSOLE_API_MESSAGE ApiMessage
;
988 Status
= CsrClientCallServer((PCSR_API_MESSAGE
)&ApiMessage
,
990 CSR_CREATE_API_NUMBER(CONSRV_SERVERDLL_INDEX
, ConsolepFree
),
991 sizeof(CONSOLE_FREECONSOLE
));
992 if (!NT_SUCCESS(Status
))
994 BaseSetLastNTError(Status
);
998 NtCurrentPeb()->ProcessParameters
->ConsoleHandle
= NULL
;
1000 CloseHandle(InputWaitHandle
);
1001 InputWaitHandle
= INVALID_HANDLE_VALUE
;
1007 /*--------------------------------------------------------------
1008 * GetConsoleScreenBufferInfo
1014 GetConsoleScreenBufferInfo(HANDLE hConsoleOutput
,
1015 PCONSOLE_SCREEN_BUFFER_INFO lpConsoleScreenBufferInfo
)
1018 CONSOLE_API_MESSAGE ApiMessage
;
1020 if (lpConsoleScreenBufferInfo
== NULL
)
1022 SetLastError(ERROR_INVALID_PARAMETER
);
1026 ApiMessage
.Data
.ScreenBufferInfoRequest
.OutputHandle
= hConsoleOutput
;
1028 Status
= CsrClientCallServer((PCSR_API_MESSAGE
)&ApiMessage
,
1030 CSR_CREATE_API_NUMBER(CONSRV_SERVERDLL_INDEX
, ConsolepGetScreenBufferInfo
),
1031 sizeof(CONSOLE_GETSCREENBUFFERINFO
));
1032 if (!NT_SUCCESS(Status
))
1034 BaseSetLastNTError(Status
);
1038 *lpConsoleScreenBufferInfo
= ApiMessage
.Data
.ScreenBufferInfoRequest
.Info
;
1044 /*--------------------------------------------------------------
1045 * SetConsoleCursorPosition
1051 SetConsoleCursorPosition(HANDLE hConsoleOutput
,
1052 COORD dwCursorPosition
)
1055 CONSOLE_API_MESSAGE ApiMessage
;
1057 ApiMessage
.Data
.SetCursorPositionRequest
.OutputHandle
= hConsoleOutput
;
1058 ApiMessage
.Data
.SetCursorPositionRequest
.Position
= dwCursorPosition
;
1060 Status
= CsrClientCallServer((PCSR_API_MESSAGE
)&ApiMessage
,
1062 CSR_CREATE_API_NUMBER(CONSRV_SERVERDLL_INDEX
, ConsolepSetCursorPosition
),
1063 sizeof(CONSOLE_SETCURSORPOSITION
));
1064 if (!NT_SUCCESS(Status
))
1066 BaseSetLastNTError(Status
);
1074 /*--------------------------------------------------------------
1081 GetConsoleMode(HANDLE hConsoleHandle
,
1085 CONSOLE_API_MESSAGE ApiMessage
;
1086 PCONSOLE_GETSETCONSOLEMODE ConsoleModeRequest
= &ApiMessage
.Data
.ConsoleModeRequest
;
1090 SetLastError(ERROR_INVALID_PARAMETER
);
1094 ConsoleModeRequest
->ConsoleHandle
= hConsoleHandle
;
1096 Status
= CsrClientCallServer((PCSR_API_MESSAGE
)&ApiMessage
,
1098 CSR_CREATE_API_NUMBER(CONSRV_SERVERDLL_INDEX
, ConsolepGetMode
),
1099 sizeof(CONSOLE_GETSETCONSOLEMODE
));
1100 if (!NT_SUCCESS(Status
))
1102 BaseSetLastNTError(Status
);
1106 *lpMode
= ConsoleModeRequest
->ConsoleMode
;
1112 /*--------------------------------------------------------------
1113 * GetNumberOfConsoleInputEvents
1119 GetNumberOfConsoleInputEvents(HANDLE hConsoleInput
,
1120 LPDWORD lpNumberOfEvents
)
1123 CONSOLE_API_MESSAGE ApiMessage
;
1124 PCONSOLE_GETNUMINPUTEVENTS GetNumInputEventsRequest
= &ApiMessage
.Data
.GetNumInputEventsRequest
;
1126 GetNumInputEventsRequest
->InputHandle
= hConsoleInput
;
1127 GetNumInputEventsRequest
->NumInputEvents
= 0;
1129 Status
= CsrClientCallServer((PCSR_API_MESSAGE
)&ApiMessage
,
1131 CSR_CREATE_API_NUMBER(CONSRV_SERVERDLL_INDEX
, ConsolepGetNumberOfInputEvents
),
1132 sizeof(CONSOLE_GETNUMINPUTEVENTS
));
1133 if (!NT_SUCCESS(Status
))
1135 BaseSetLastNTError(Status
);
1139 if (lpNumberOfEvents
== NULL
)
1141 SetLastError(ERROR_INVALID_ACCESS
);
1145 *lpNumberOfEvents
= GetNumInputEventsRequest
->NumInputEvents
;
1151 /*--------------------------------------------------------------
1152 * GetLargestConsoleWindowSize
1158 GetLargestConsoleWindowSize(HANDLE hConsoleOutput
)
1161 CONSOLE_API_MESSAGE ApiMessage
;
1162 PCONSOLE_GETLARGESTWINDOWSIZE GetLargestWindowSizeRequest
= &ApiMessage
.Data
.GetLargestWindowSizeRequest
;
1164 GetLargestWindowSizeRequest
->OutputHandle
= hConsoleOutput
;
1165 GetLargestWindowSizeRequest
->Size
.X
= 0;
1166 GetLargestWindowSizeRequest
->Size
.Y
= 0;
1168 Status
= CsrClientCallServer((PCSR_API_MESSAGE
)&ApiMessage
,
1170 CSR_CREATE_API_NUMBER(CONSRV_SERVERDLL_INDEX
, ConsolepGetLargestWindowSize
),
1171 sizeof(CONSOLE_GETLARGESTWINDOWSIZE
));
1172 if (!NT_SUCCESS(Status
))
1174 BaseSetLastNTError(Status
);
1177 DPRINT1("GetLargestConsoleWindowSize, X = %d, Y = %d\n", GetLargestWindowSizeRequest
->Size
.X
, GetLargestWindowSizeRequest
->Size
.Y
);
1178 return GetLargestWindowSizeRequest
->Size
;
1182 /*--------------------------------------------------------------
1183 * GetConsoleCursorInfo
1189 GetConsoleCursorInfo(HANDLE hConsoleOutput
,
1190 PCONSOLE_CURSOR_INFO lpConsoleCursorInfo
)
1193 CONSOLE_API_MESSAGE ApiMessage
;
1195 if (!lpConsoleCursorInfo
)
1197 if (!hConsoleOutput
)
1198 SetLastError(ERROR_INVALID_HANDLE
);
1200 SetLastError(ERROR_INVALID_ACCESS
);
1205 ApiMessage
.Data
.CursorInfoRequest
.OutputHandle
= hConsoleOutput
;
1207 Status
= CsrClientCallServer((PCSR_API_MESSAGE
)&ApiMessage
,
1209 CSR_CREATE_API_NUMBER(CONSRV_SERVERDLL_INDEX
, ConsolepGetCursorInfo
),
1210 sizeof(CONSOLE_GETSETCURSORINFO
));
1211 if (!NT_SUCCESS(Status
))
1213 BaseSetLastNTError(Status
);
1217 *lpConsoleCursorInfo
= ApiMessage
.Data
.CursorInfoRequest
.Info
;
1223 /*--------------------------------------------------------------
1224 * GetNumberOfConsoleMouseButtons
1230 GetNumberOfConsoleMouseButtons(LPDWORD lpNumberOfMouseButtons
)
1232 DPRINT1("GetNumberOfConsoleMouseButtons(0x%x) UNIMPLEMENTED!\n", lpNumberOfMouseButtons
);
1233 SetLastError(ERROR_CALL_NOT_IMPLEMENTED
);
1238 /*--------------------------------------------------------------
1245 SetConsoleMode(HANDLE hConsoleHandle
,
1249 CONSOLE_API_MESSAGE ApiMessage
;
1250 PCONSOLE_GETSETCONSOLEMODE ConsoleModeRequest
= &ApiMessage
.Data
.ConsoleModeRequest
;
1252 ConsoleModeRequest
->ConsoleHandle
= hConsoleHandle
;
1253 ConsoleModeRequest
->ConsoleMode
= dwMode
;
1255 Status
= CsrClientCallServer((PCSR_API_MESSAGE
)&ApiMessage
,
1257 CSR_CREATE_API_NUMBER(CONSRV_SERVERDLL_INDEX
, ConsolepSetMode
),
1258 sizeof(CONSOLE_GETSETCONSOLEMODE
));
1259 if (!NT_SUCCESS(Status
))
1261 BaseSetLastNTError(Status
);
1269 /*--------------------------------------------------------------
1270 * SetConsoleActiveScreenBuffer
1276 SetConsoleActiveScreenBuffer(HANDLE hConsoleOutput
)
1279 CONSOLE_API_MESSAGE ApiMessage
;
1281 ApiMessage
.Data
.SetScreenBufferRequest
.OutputHandle
= hConsoleOutput
;
1283 Status
= CsrClientCallServer((PCSR_API_MESSAGE
)&ApiMessage
,
1285 CSR_CREATE_API_NUMBER(CONSRV_SERVERDLL_INDEX
, ConsolepSetActiveScreenBuffer
),
1286 sizeof(CONSOLE_SETACTIVESCREENBUFFER
));
1287 if (!NT_SUCCESS(Status
))
1289 BaseSetLastNTError(Status
);
1297 /*--------------------------------------------------------------
1298 * FlushConsoleInputBuffer
1304 FlushConsoleInputBuffer(HANDLE hConsoleInput
)
1307 CONSOLE_API_MESSAGE ApiMessage
;
1309 ApiMessage
.Data
.FlushInputBufferRequest
.InputHandle
= hConsoleInput
;
1311 Status
= CsrClientCallServer((PCSR_API_MESSAGE
)&ApiMessage
,
1313 CSR_CREATE_API_NUMBER(CONSRV_SERVERDLL_INDEX
, ConsolepFlushInputBuffer
),
1314 sizeof(CONSOLE_FLUSHINPUTBUFFER
));
1315 if (!NT_SUCCESS(Status
))
1317 BaseSetLastNTError(Status
);
1325 /*--------------------------------------------------------------
1326 * SetConsoleScreenBufferSize
1332 SetConsoleScreenBufferSize(HANDLE hConsoleOutput
,
1336 CONSOLE_API_MESSAGE ApiMessage
;
1338 ApiMessage
.Data
.SetScreenBufferSizeRequest
.OutputHandle
= hConsoleOutput
;
1339 ApiMessage
.Data
.SetScreenBufferSizeRequest
.Size
= dwSize
;
1341 Status
= CsrClientCallServer((PCSR_API_MESSAGE
)&ApiMessage
,
1343 CSR_CREATE_API_NUMBER(CONSRV_SERVERDLL_INDEX
, ConsolepSetScreenBufferSize
),
1344 sizeof(CONSOLE_SETSCREENBUFFERSIZE
));
1345 if (!NT_SUCCESS(Status
))
1347 BaseSetLastNTError(Status
);
1355 /*--------------------------------------------------------------
1356 * SetConsoleCursorInfo
1362 SetConsoleCursorInfo(HANDLE hConsoleOutput
,
1363 CONST CONSOLE_CURSOR_INFO
*lpConsoleCursorInfo
)
1366 CONSOLE_API_MESSAGE ApiMessage
;
1368 ApiMessage
.Data
.CursorInfoRequest
.OutputHandle
= hConsoleOutput
;
1369 ApiMessage
.Data
.CursorInfoRequest
.Info
= *lpConsoleCursorInfo
;
1371 Status
= CsrClientCallServer((PCSR_API_MESSAGE
)&ApiMessage
,
1373 CSR_CREATE_API_NUMBER(CONSRV_SERVERDLL_INDEX
, ConsolepSetCursorInfo
),
1374 sizeof(CONSOLE_GETSETCURSORINFO
));
1375 if (!NT_SUCCESS(Status
))
1377 BaseSetLastNTError(Status
);
1387 IntScrollConsoleScreenBuffer(HANDLE hConsoleOutput
,
1388 const SMALL_RECT
*lpScrollRectangle
,
1389 const SMALL_RECT
*lpClipRectangle
,
1390 COORD dwDestinationOrigin
,
1391 const CHAR_INFO
*lpFill
,
1395 CONSOLE_API_MESSAGE ApiMessage
;
1396 PCONSOLE_SCROLLSCREENBUFFER ScrollScreenBufferRequest
= &ApiMessage
.Data
.ScrollScreenBufferRequest
;
1398 ScrollScreenBufferRequest
->OutputHandle
= hConsoleOutput
;
1399 ScrollScreenBufferRequest
->Unicode
= bUnicode
;
1400 ScrollScreenBufferRequest
->ScrollRectangle
= *lpScrollRectangle
;
1402 if (lpClipRectangle
!= NULL
)
1404 ScrollScreenBufferRequest
->UseClipRectangle
= TRUE
;
1405 ScrollScreenBufferRequest
->ClipRectangle
= *lpClipRectangle
;
1409 ScrollScreenBufferRequest
->UseClipRectangle
= FALSE
;
1412 ScrollScreenBufferRequest
->DestinationOrigin
= dwDestinationOrigin
;
1413 ScrollScreenBufferRequest
->Fill
= *lpFill
;
1415 Status
= CsrClientCallServer((PCSR_API_MESSAGE
)&ApiMessage
,
1417 CSR_CREATE_API_NUMBER(CONSRV_SERVERDLL_INDEX
, ConsolepScrollScreenBuffer
),
1418 sizeof(CONSOLE_SCROLLSCREENBUFFER
));
1420 if (!NT_SUCCESS(Status
))
1422 BaseSetLastNTError(Status
);
1430 /*--------------------------------------------------------------
1431 * ScrollConsoleScreenBufferA
1437 ScrollConsoleScreenBufferA(HANDLE hConsoleOutput
,
1438 CONST SMALL_RECT
*lpScrollRectangle
,
1439 CONST SMALL_RECT
*lpClipRectangle
,
1440 COORD dwDestinationOrigin
,
1441 CONST CHAR_INFO
*lpFill
)
1443 return IntScrollConsoleScreenBuffer(hConsoleOutput
,
1444 (PSMALL_RECT
)lpScrollRectangle
,
1445 (PSMALL_RECT
)lpClipRectangle
,
1446 dwDestinationOrigin
,
1452 /*--------------------------------------------------------------
1453 * ScrollConsoleScreenBufferW
1459 ScrollConsoleScreenBufferW(HANDLE hConsoleOutput
,
1460 CONST SMALL_RECT
*lpScrollRectangle
,
1461 CONST SMALL_RECT
*lpClipRectangle
,
1462 COORD dwDestinationOrigin
,
1463 CONST CHAR_INFO
*lpFill
)
1465 return IntScrollConsoleScreenBuffer(hConsoleOutput
,
1468 dwDestinationOrigin
,
1474 /*--------------------------------------------------------------
1475 * SetConsoleWindowInfo
1481 SetConsoleWindowInfo(HANDLE hConsoleOutput
,
1483 CONST SMALL_RECT
*lpConsoleWindow
)
1486 CONSOLE_API_MESSAGE ApiMessage
;
1487 PCONSOLE_SETWINDOWINFO SetWindowInfoRequest
= &ApiMessage
.Data
.SetWindowInfoRequest
;
1489 if (lpConsoleWindow
== NULL
)
1491 SetLastError(ERROR_INVALID_PARAMETER
);
1495 SetWindowInfoRequest
->OutputHandle
= hConsoleOutput
;
1496 SetWindowInfoRequest
->Absolute
= bAbsolute
;
1497 SetWindowInfoRequest
->WindowRect
= *lpConsoleWindow
;
1499 Status
= CsrClientCallServer((PCSR_API_MESSAGE
)&ApiMessage
,
1501 CSR_CREATE_API_NUMBER(CONSRV_SERVERDLL_INDEX
, ConsolepSetWindowInfo
),
1502 sizeof(CONSOLE_SETWINDOWINFO
));
1503 if (!NT_SUCCESS(Status
))
1505 BaseSetLastNTError(Status
);
1513 /*--------------------------------------------------------------
1514 * SetConsoleTextAttribute
1520 SetConsoleTextAttribute(HANDLE hConsoleOutput
,
1524 CONSOLE_API_MESSAGE ApiMessage
;
1526 ApiMessage
.Data
.SetTextAttribRequest
.OutputHandle
= hConsoleOutput
;
1527 ApiMessage
.Data
.SetTextAttribRequest
.Attrib
= wAttributes
;
1529 Status
= CsrClientCallServer((PCSR_API_MESSAGE
)&ApiMessage
,
1531 CSR_CREATE_API_NUMBER(CONSRV_SERVERDLL_INDEX
, ConsolepSetTextAttribute
),
1532 sizeof(CONSOLE_SETTEXTATTRIB
));
1533 if (!NT_SUCCESS(Status
))
1535 BaseSetLastNTError(Status
);
1545 AddConsoleCtrlHandler(PHANDLER_ROUTINE HandlerRoutine
)
1547 PHANDLER_ROUTINE
* NewCtrlHandlers
= NULL
;
1549 if (HandlerRoutine
== NULL
)
1551 NtCurrentPeb()->ProcessParameters
->ConsoleFlags
= TRUE
;
1555 if (NrCtrlHandlers
== NrAllocatedHandlers
)
1557 NewCtrlHandlers
= RtlAllocateHeap(RtlGetProcessHeap(),
1559 (NrCtrlHandlers
+ 4) * sizeof(PHANDLER_ROUTINE
));
1560 if (NewCtrlHandlers
== NULL
)
1562 SetLastError(ERROR_NOT_ENOUGH_MEMORY
);
1566 memmove(NewCtrlHandlers
, CtrlHandlers
, sizeof(PHANDLER_ROUTINE
) * NrCtrlHandlers
);
1568 if (NrAllocatedHandlers
> 1) RtlFreeHeap(RtlGetProcessHeap(), 0, CtrlHandlers
);
1570 CtrlHandlers
= NewCtrlHandlers
;
1571 NrAllocatedHandlers
+= 4;
1574 ASSERT(NrCtrlHandlers
< NrAllocatedHandlers
);
1576 CtrlHandlers
[NrCtrlHandlers
++] = HandlerRoutine
;
1583 RemoveConsoleCtrlHandler(PHANDLER_ROUTINE HandlerRoutine
)
1587 if (HandlerRoutine
== NULL
)
1589 NtCurrentPeb()->ProcessParameters
->ConsoleFlags
= FALSE
;
1593 for (i
= 0; i
< NrCtrlHandlers
; i
++)
1595 if (CtrlHandlers
[i
] == HandlerRoutine
)
1597 if (i
< (NrCtrlHandlers
- 1))
1599 memmove(&CtrlHandlers
[i
],
1601 (NrCtrlHandlers
- i
+ 1) * sizeof(PHANDLER_ROUTINE
));
1609 SetLastError(ERROR_INVALID_PARAMETER
);
1619 SetConsoleCtrlHandler(PHANDLER_ROUTINE HandlerRoutine
,
1624 RtlEnterCriticalSection(&BaseDllDirectoryLock
);
1627 Ret
= AddConsoleCtrlHandler(HandlerRoutine
);
1631 Ret
= RemoveConsoleCtrlHandler(HandlerRoutine
);
1634 RtlLeaveCriticalSection(&BaseDllDirectoryLock
);
1639 /*--------------------------------------------------------------
1640 * GenerateConsoleCtrlEvent
1646 GenerateConsoleCtrlEvent(DWORD dwCtrlEvent
,
1647 DWORD dwProcessGroupId
)
1650 CONSOLE_API_MESSAGE ApiMessage
;
1652 if (dwCtrlEvent
!= CTRL_C_EVENT
&& dwCtrlEvent
!= CTRL_BREAK_EVENT
)
1654 SetLastError(ERROR_INVALID_PARAMETER
);
1658 ApiMessage
.Data
.GenerateCtrlEventRequest
.Event
= dwCtrlEvent
;
1659 ApiMessage
.Data
.GenerateCtrlEventRequest
.ProcessGroup
= dwProcessGroupId
;
1661 Status
= CsrClientCallServer((PCSR_API_MESSAGE
)&ApiMessage
,
1663 CSR_CREATE_API_NUMBER(CONSRV_SERVERDLL_INDEX
, ConsolepGenerateCtrlEvent
),
1664 sizeof(CONSOLE_GENERATECTRLEVENT
));
1665 if (!NT_SUCCESS(Status
))
1667 BaseSetLastNTError(Status
);
1676 IntGetConsoleTitle(LPVOID lpConsoleTitle
, DWORD nSize
, BOOL bUnicode
)
1679 CONSOLE_API_MESSAGE ApiMessage
;
1680 PCONSOLE_GETSETCONSOLETITLE TitleRequest
= &ApiMessage
.Data
.TitleRequest
;
1681 PCSR_CAPTURE_BUFFER CaptureBuffer
;
1683 if (nSize
== 0) return 0;
1685 TitleRequest
->Length
= nSize
* (bUnicode
? 1 : sizeof(WCHAR
));
1686 CaptureBuffer
= CsrAllocateCaptureBuffer(1, TitleRequest
->Length
);
1687 if (CaptureBuffer
== NULL
)
1689 DPRINT1("CsrAllocateCaptureBuffer failed!\n");
1690 SetLastError(ERROR_NOT_ENOUGH_MEMORY
);
1694 CsrAllocateMessagePointer(CaptureBuffer
,
1695 TitleRequest
->Length
,
1696 (PVOID
*)&TitleRequest
->Title
);
1698 Status
= CsrClientCallServer((PCSR_API_MESSAGE
)&ApiMessage
,
1700 CSR_CREATE_API_NUMBER(CONSRV_SERVERDLL_INDEX
, ConsolepGetTitle
),
1701 sizeof(CONSOLE_GETSETCONSOLETITLE
));
1702 if (!NT_SUCCESS(Status
))
1704 CsrFreeCaptureBuffer(CaptureBuffer
);
1705 BaseSetLastNTError(Status
);
1711 if (nSize
>= sizeof(WCHAR
))
1712 wcscpy((LPWSTR
)lpConsoleTitle
, TitleRequest
->Title
);
1716 if (nSize
< TitleRequest
->Length
/ sizeof(WCHAR
) ||
1717 !WideCharToMultiByte(CP_ACP
, // ANSI code page
1718 0, // performance and mapping flags
1719 TitleRequest
->Title
, // address of wide-character string
1720 -1, // number of characters in string
1721 (LPSTR
)lpConsoleTitle
, // address of buffer for new string
1722 nSize
, // size of buffer
1726 /* Yes, if the buffer isn't big enough, it returns 0... Bad API */
1727 *(LPSTR
)lpConsoleTitle
= '\0';
1728 TitleRequest
->Length
= 0;
1731 CsrFreeCaptureBuffer(CaptureBuffer
);
1733 return TitleRequest
->Length
/ sizeof(WCHAR
);
1737 /*--------------------------------------------------------------
1744 GetConsoleTitleW(LPWSTR lpConsoleTitle
,
1747 return IntGetConsoleTitle(lpConsoleTitle
, nSize
, TRUE
);
1751 /*--------------------------------------------------------------
1758 GetConsoleTitleA(LPSTR lpConsoleTitle
,
1761 return IntGetConsoleTitle(lpConsoleTitle
, nSize
, FALSE
);
1765 /*--------------------------------------------------------------
1772 SetConsoleTitleW(LPCWSTR lpConsoleTitle
)
1775 CONSOLE_API_MESSAGE ApiMessage
;
1776 PCONSOLE_GETSETCONSOLETITLE TitleRequest
= &ApiMessage
.Data
.TitleRequest
;
1777 PCSR_CAPTURE_BUFFER CaptureBuffer
;
1779 TitleRequest
->Length
= wcslen(lpConsoleTitle
) * sizeof(WCHAR
);
1781 CaptureBuffer
= CsrAllocateCaptureBuffer(1, TitleRequest
->Length
);
1782 if (CaptureBuffer
== NULL
)
1784 DPRINT1("CsrAllocateCaptureBuffer failed!\n");
1785 SetLastError(ERROR_NOT_ENOUGH_MEMORY
);
1789 CsrCaptureMessageBuffer(CaptureBuffer
,
1790 (PVOID
)lpConsoleTitle
,
1791 TitleRequest
->Length
,
1792 (PVOID
*)&TitleRequest
->Title
);
1794 Status
= CsrClientCallServer((PCSR_API_MESSAGE
)&ApiMessage
,
1796 CSR_CREATE_API_NUMBER(CONSRV_SERVERDLL_INDEX
, ConsolepSetTitle
),
1797 sizeof(CONSOLE_GETSETCONSOLETITLE
));
1799 CsrFreeCaptureBuffer(CaptureBuffer
);
1801 if (!NT_SUCCESS(Status
))
1803 BaseSetLastNTError(Status
);
1811 /*--------------------------------------------------------------
1818 SetConsoleTitleA(LPCSTR lpConsoleTitle
)
1821 ULONG Length
= strlen(lpConsoleTitle
) + 1;
1822 LPWSTR WideTitle
= HeapAlloc(GetProcessHeap(), 0, Length
* sizeof(WCHAR
));
1826 SetLastError(ERROR_NOT_ENOUGH_MEMORY
);
1830 MultiByteToWideChar(CP_ACP
, 0, lpConsoleTitle
, -1, WideTitle
, Length
);
1832 Ret
= SetConsoleTitleW(WideTitle
);
1834 HeapFree(GetProcessHeap(), 0, WideTitle
);
1839 /*--------------------------------------------------------------
1840 * CreateConsoleScreenBuffer
1846 CreateConsoleScreenBuffer(DWORD dwDesiredAccess
,
1848 CONST SECURITY_ATTRIBUTES
*lpSecurityAttributes
,
1850 LPVOID lpScreenBufferData
)
1853 CONSOLE_API_MESSAGE ApiMessage
;
1854 PCONSOLE_CREATESCREENBUFFER CreateScreenBufferRequest
= &ApiMessage
.Data
.CreateScreenBufferRequest
;
1855 PCSR_CAPTURE_BUFFER CaptureBuffer
= NULL
;
1856 PCONSOLE_GRAPHICS_BUFFER_INFO GraphicsBufferInfo
= /*(PCONSOLE_GRAPHICS_BUFFER_INFO)*/lpScreenBufferData
;
1858 if ( (dwDesiredAccess
& ~(GENERIC_READ
| GENERIC_WRITE
)) ||
1859 (dwShareMode
& ~(FILE_SHARE_READ
| FILE_SHARE_WRITE
)) ||
1860 (dwFlags
!= CONSOLE_TEXTMODE_BUFFER
&& dwFlags
!= CONSOLE_GRAPHICS_BUFFER
) )
1862 SetLastError(ERROR_INVALID_PARAMETER
);
1863 return INVALID_HANDLE_VALUE
;
1866 CreateScreenBufferRequest
->ScreenBufferType
= dwFlags
;
1867 CreateScreenBufferRequest
->Access
= dwDesiredAccess
;
1868 CreateScreenBufferRequest
->ShareMode
= dwShareMode
;
1869 CreateScreenBufferRequest
->Inheritable
=
1870 (lpSecurityAttributes
? lpSecurityAttributes
->bInheritHandle
: FALSE
);
1872 if (dwFlags
== CONSOLE_GRAPHICS_BUFFER
)
1874 if (CreateScreenBufferRequest
->Inheritable
|| GraphicsBufferInfo
== NULL
)
1876 SetLastError(ERROR_INVALID_PARAMETER
);
1877 return INVALID_HANDLE_VALUE
;
1880 CreateScreenBufferRequest
->GraphicsBufferInfo
= *GraphicsBufferInfo
;
1882 CaptureBuffer
= CsrAllocateCaptureBuffer(1, GraphicsBufferInfo
->dwBitMapInfoLength
);
1883 if (CaptureBuffer
== NULL
)
1885 SetLastError(ERROR_NOT_ENOUGH_MEMORY
);
1889 CsrCaptureMessageBuffer(CaptureBuffer
,
1890 (PVOID
)GraphicsBufferInfo
->lpBitMapInfo
,
1891 GraphicsBufferInfo
->dwBitMapInfoLength
,
1892 (PVOID
*)&CreateScreenBufferRequest
->GraphicsBufferInfo
.lpBitMapInfo
);
1895 Status
= CsrClientCallServer((PCSR_API_MESSAGE
)&ApiMessage
,
1897 CSR_CREATE_API_NUMBER(CONSRV_SERVERDLL_INDEX
, ConsolepCreateScreenBuffer
),
1898 sizeof(CONSOLE_CREATESCREENBUFFER
));
1901 CsrFreeCaptureBuffer(CaptureBuffer
);
1903 if (!NT_SUCCESS(Status
))
1905 BaseSetLastNTError(Status
);
1906 return INVALID_HANDLE_VALUE
;
1909 if (dwFlags
== CONSOLE_GRAPHICS_BUFFER
&& GraphicsBufferInfo
)
1911 GraphicsBufferInfo
->hMutex
= CreateScreenBufferRequest
->GraphicsBufferInfo
.hMutex
;
1912 GraphicsBufferInfo
->lpBitMap
= CreateScreenBufferRequest
->GraphicsBufferInfo
.lpBitMap
;
1915 return CreateScreenBufferRequest
->OutputHandle
;
1919 /*--------------------------------------------------------------
1929 CONSOLE_API_MESSAGE ApiMessage
;
1931 /* Get the Input Code Page */
1932 ApiMessage
.Data
.ConsoleCPRequest
.InputCP
= TRUE
;
1934 Status
= CsrClientCallServer((PCSR_API_MESSAGE
)&ApiMessage
,
1936 CSR_CREATE_API_NUMBER(CONSRV_SERVERDLL_INDEX
, ConsolepGetCP
),
1937 sizeof(CONSOLE_GETSETINPUTOUTPUTCP
));
1938 if (!NT_SUCCESS(Status
))
1940 BaseSetLastNTError(Status
);
1944 return ApiMessage
.Data
.ConsoleCPRequest
.CodePage
;
1948 /*--------------------------------------------------------------
1955 SetConsoleCP(UINT wCodePageID
)
1958 CONSOLE_API_MESSAGE ApiMessage
;
1960 /* Set the Input Code Page */
1961 ApiMessage
.Data
.ConsoleCPRequest
.InputCP
= TRUE
;
1962 ApiMessage
.Data
.ConsoleCPRequest
.CodePage
= wCodePageID
;
1964 Status
= CsrClientCallServer((PCSR_API_MESSAGE
)&ApiMessage
,
1966 CSR_CREATE_API_NUMBER(CONSRV_SERVERDLL_INDEX
, ConsolepSetCP
),
1967 sizeof(CONSOLE_GETSETINPUTOUTPUTCP
));
1968 if (!NT_SUCCESS(Status
))
1970 BaseSetLastNTError(Status
);
1973 return NT_SUCCESS(Status
);
1977 /*--------------------------------------------------------------
1978 * GetConsoleOutputCP
1984 GetConsoleOutputCP(VOID
)
1987 CONSOLE_API_MESSAGE ApiMessage
;
1989 /* Get the Output Code Page */
1990 ApiMessage
.Data
.ConsoleCPRequest
.InputCP
= FALSE
;
1992 Status
= CsrClientCallServer((PCSR_API_MESSAGE
)&ApiMessage
,
1994 CSR_CREATE_API_NUMBER(CONSRV_SERVERDLL_INDEX
, ConsolepGetCP
),
1995 sizeof(CONSOLE_GETSETINPUTOUTPUTCP
));
1996 if (!NT_SUCCESS(Status
))
1998 BaseSetLastNTError (Status
);
2002 return ApiMessage
.Data
.ConsoleCPRequest
.CodePage
;
2006 /*--------------------------------------------------------------
2007 * SetConsoleOutputCP
2013 SetConsoleOutputCP(UINT wCodePageID
)
2016 CONSOLE_API_MESSAGE ApiMessage
;
2018 /* Set the Output Code Page */
2019 ApiMessage
.Data
.ConsoleCPRequest
.InputCP
= FALSE
;
2020 ApiMessage
.Data
.ConsoleCPRequest
.CodePage
= wCodePageID
;
2022 Status
= CsrClientCallServer((PCSR_API_MESSAGE
)&ApiMessage
,
2024 CSR_CREATE_API_NUMBER(CONSRV_SERVERDLL_INDEX
, ConsolepSetCP
),
2025 sizeof(CONSOLE_GETSETINPUTOUTPUTCP
));
2026 if (!NT_SUCCESS(Status
))
2028 BaseSetLastNTError(Status
);
2031 return NT_SUCCESS(Status
);
2035 /*--------------------------------------------------------------
2036 * GetConsoleProcessList
2042 GetConsoleProcessList(LPDWORD lpdwProcessList
,
2043 DWORD dwProcessCount
)
2046 CONSOLE_API_MESSAGE ApiMessage
;
2047 PCONSOLE_GETPROCESSLIST GetProcessListRequest
= &ApiMessage
.Data
.GetProcessListRequest
;
2048 PCSR_CAPTURE_BUFFER CaptureBuffer
;
2051 if (lpdwProcessList
== NULL
|| dwProcessCount
== 0)
2053 SetLastError(ERROR_INVALID_PARAMETER
);
2057 CaptureBuffer
= CsrAllocateCaptureBuffer(1, dwProcessCount
* sizeof(DWORD
));
2058 if (CaptureBuffer
== NULL
)
2060 DPRINT1("CsrAllocateCaptureBuffer failed!\n");
2061 SetLastError(ERROR_NOT_ENOUGH_MEMORY
);
2065 GetProcessListRequest
->nMaxIds
= dwProcessCount
;
2067 CsrAllocateMessagePointer(CaptureBuffer
,
2068 dwProcessCount
* sizeof(DWORD
),
2069 (PVOID
*)&GetProcessListRequest
->pProcessIds
);
2071 Status
= CsrClientCallServer((PCSR_API_MESSAGE
)&ApiMessage
,
2073 CSR_CREATE_API_NUMBER(CONSRV_SERVERDLL_INDEX
, ConsolepGetProcessList
),
2074 sizeof(CONSOLE_GETPROCESSLIST
));
2075 if (!NT_SUCCESS(Status
))
2077 BaseSetLastNTError (Status
);
2082 nProcesses
= GetProcessListRequest
->nProcessIdsTotal
;
2083 if (dwProcessCount
>= nProcesses
)
2085 memcpy(lpdwProcessList
, GetProcessListRequest
->pProcessIds
, nProcesses
* sizeof(DWORD
));
2089 CsrFreeCaptureBuffer(CaptureBuffer
);
2094 /*--------------------------------------------------------------
2095 * GetConsoleSelectionInfo
2101 GetConsoleSelectionInfo(PCONSOLE_SELECTION_INFO lpConsoleSelectionInfo
)
2104 CONSOLE_API_MESSAGE ApiMessage
;
2106 if (lpConsoleSelectionInfo
== NULL
)
2108 SetLastError(ERROR_INVALID_PARAMETER
);
2112 Status
= CsrClientCallServer((PCSR_API_MESSAGE
)&ApiMessage
,
2114 CSR_CREATE_API_NUMBER(CONSRV_SERVERDLL_INDEX
, ConsolepGetSelectionInfo
),
2115 sizeof(CONSOLE_GETSELECTIONINFO
));
2116 if (!NT_SUCCESS(Status
))
2118 BaseSetLastNTError(Status
);
2122 *lpConsoleSelectionInfo
= ApiMessage
.Data
.GetSelectionInfoRequest
.Info
;
2127 /*--------------------------------------------------------------
2132 * @note Strongly inspired by AllocConsole.
2136 AttachConsole(DWORD dwProcessId
)
2139 PRTL_USER_PROCESS_PARAMETERS Parameters
= NtCurrentPeb()->ProcessParameters
;
2140 CONSOLE_API_MESSAGE ApiMessage
;
2141 PCONSOLE_ATTACHCONSOLE AttachConsoleRequest
= &ApiMessage
.Data
.AttachConsoleRequest
;
2143 if (Parameters
->ConsoleHandle
)
2145 DPRINT1("AttachConsole: Attaching a console to a process already having one\n");
2146 SetLastError(ERROR_ACCESS_DENIED
);
2150 AttachConsoleRequest
->ProcessId
= dwProcessId
;
2151 AttachConsoleRequest
->CtrlDispatcher
= ConsoleControlDispatcher
;
2152 AttachConsoleRequest
->PropDispatcher
= PropDialogHandler
;
2154 Status
= CsrClientCallServer((PCSR_API_MESSAGE
)&ApiMessage
,
2156 CSR_CREATE_API_NUMBER(CONSRV_SERVERDLL_INDEX
, ConsolepAttach
),
2157 sizeof(CONSOLE_ATTACHCONSOLE
));
2158 if (!NT_SUCCESS(Status
))
2160 BaseSetLastNTError(Status
);
2164 Parameters
->ConsoleHandle
= AttachConsoleRequest
->Console
;
2165 SetStdHandle(STD_INPUT_HANDLE
, AttachConsoleRequest
->InputHandle
);
2166 SetStdHandle(STD_OUTPUT_HANDLE
, AttachConsoleRequest
->OutputHandle
);
2167 SetStdHandle(STD_ERROR_HANDLE
, AttachConsoleRequest
->ErrorHandle
);
2169 /* Initialize Console Ctrl Handler */
2170 InitConsoleCtrlHandling();
2172 InputWaitHandle
= AttachConsoleRequest
->InputWaitHandle
;
2178 /*--------------------------------------------------------------
2185 GetConsoleWindow(VOID
)
2188 CONSOLE_API_MESSAGE ApiMessage
;
2190 Status
= CsrClientCallServer((PCSR_API_MESSAGE
)&ApiMessage
,
2192 CSR_CREATE_API_NUMBER(CONSRV_SERVERDLL_INDEX
, ConsolepGetConsoleWindow
),
2193 sizeof(CONSOLE_GETWINDOW
));
2194 if (!NT_SUCCESS(Status
))
2196 BaseSetLastNTError(Status
);
2200 return ApiMessage
.Data
.GetWindowRequest
.WindowHandle
;
2204 /*--------------------------------------------------------------
2211 SetConsoleIcon(HICON hicon
)
2214 CONSOLE_API_MESSAGE ApiMessage
;
2216 ApiMessage
.Data
.SetIconRequest
.WindowIcon
= hicon
;
2218 Status
= CsrClientCallServer((PCSR_API_MESSAGE
)&ApiMessage
,
2220 CSR_CREATE_API_NUMBER(CONSRV_SERVERDLL_INDEX
, ConsolepSetIcon
),
2221 sizeof(CONSOLE_SETICON
));
2222 if (!NT_SUCCESS(Status
))
2224 BaseSetLastNTError(Status
);
2228 return NT_SUCCESS(Status
);
2232 /******************************************************************************
2233 * \name SetConsoleInputExeNameW
2234 * \brief Sets the console input file name from a unicode string.
2235 * \param lpInputExeName Pointer to a unicode string with the name.
2236 * \return TRUE if successful, FALSE if unsuccsedful.
2237 * \remarks If lpInputExeName is 0 or the string length is 0 or greater than 255,
2238 * the function fails and sets last error to ERROR_INVALID_PARAMETER.
2242 SetConsoleInputExeNameW(LPCWSTR lpInputExeName
)
2246 if ( !lpInputExeName
||
2247 (lenName
= lstrlenW(lpInputExeName
)) == 0 ||
2248 lenName
> INPUTEXENAME_BUFLEN
- 1 )
2250 /* Fail if string is empty or too long */
2251 SetLastError(ERROR_INVALID_PARAMETER
);
2255 RtlEnterCriticalSection(&ConsoleLock
);
2258 RtlCopyMemory(InputExeName
, lpInputExeName
, lenName
* sizeof(WCHAR
));
2259 InputExeName
[lenName
] = L
'\0';
2263 RtlLeaveCriticalSection(&ConsoleLock
);
2271 /******************************************************************************
2272 * \name SetConsoleInputExeNameA
2273 * \brief Sets the console input file name from an ansi string.
2274 * \param lpInputExeName Pointer to an ansi string with the name.
2275 * \return TRUE if successful, FALSE if unsuccsedful.
2276 * \remarks If lpInputExeName is 0 or the string length is 0 or greater than 255,
2277 * the function fails and sets last error to ERROR_INVALID_PARAMETER.
2281 SetConsoleInputExeNameA(LPCSTR lpInputExeName
)
2283 WCHAR Buffer
[INPUTEXENAME_BUFLEN
];
2284 ANSI_STRING InputExeNameA
;
2285 UNICODE_STRING InputExeNameU
;
2288 RtlInitAnsiString(&InputExeNameA
, lpInputExeName
);
2290 if ( InputExeNameA
.Length
== 0 ||
2291 InputExeNameA
.Length
> INPUTEXENAME_BUFLEN
- 1 )
2293 /* Fail if string is empty or too long */
2294 SetLastError(ERROR_INVALID_PARAMETER
);
2298 InputExeNameU
.Buffer
= Buffer
;
2299 InputExeNameU
.MaximumLength
= sizeof(Buffer
);
2300 InputExeNameU
.Length
= 0;
2302 Status
= RtlAnsiStringToUnicodeString(&InputExeNameU
, &InputExeNameA
, FALSE
);
2303 if (!NT_SUCCESS(Status
))
2305 BaseSetLastNTError(Status
);
2309 return SetConsoleInputExeNameW(InputExeNameU
.Buffer
);
2313 /******************************************************************************
2314 * \name GetConsoleInputExeNameW
2315 * \brief Retrieves the console input file name as unicode string.
2316 * \param nBufferLength Length of the buffer in WCHARs.
2317 * Specify 0 to receive the needed buffer length.
2318 * \param lpBuffer Pointer to a buffer that receives the string.
2319 * \return Needed buffer size if \p nBufferLength is 0.
2320 * Otherwise 1 if successful, 2 if buffer is too small.
2321 * \remarks Sets last error value to ERROR_BUFFER_OVERFLOW if the buffer
2322 * is not big enough.
2326 GetConsoleInputExeNameW(DWORD nBufferLength
, LPWSTR lpBuffer
)
2328 ULONG lenName
= lstrlenW(InputExeName
);
2330 if (nBufferLength
== 0)
2332 /* Buffer size is requested, return it */
2336 if (lenName
+ 1 > nBufferLength
)
2338 /* Buffer is not large enough! */
2339 SetLastError(ERROR_BUFFER_OVERFLOW
);
2343 RtlEnterCriticalSection(&ConsoleLock
);
2346 RtlCopyMemory(lpBuffer
, InputExeName
, lenName
* sizeof(WCHAR
));
2347 lpBuffer
[lenName
] = '\0';
2351 RtlLeaveCriticalSection(&ConsoleLock
);
2355 /* Success, return 1 */
2360 /******************************************************************************
2361 * \name GetConsoleInputExeNameA
2362 * \brief Retrieves the console input file name as ansi string.
2363 * \param nBufferLength Length of the buffer in CHARs.
2364 * \param lpBuffer Pointer to a buffer that receives the string.
2365 * \return 1 if successful, 2 if buffer is too small.
2366 * \remarks Sets last error value to ERROR_BUFFER_OVERFLOW if the buffer
2367 * is not big enough. The buffer receives as much characters as fit.
2371 GetConsoleInputExeNameA(DWORD nBufferLength
, LPSTR lpBuffer
)
2373 WCHAR Buffer
[INPUTEXENAME_BUFLEN
];
2375 UNICODE_STRING BufferU
;
2376 ANSI_STRING BufferA
;
2378 /* Get the unicode name */
2379 Ret
= GetConsoleInputExeNameW(sizeof(Buffer
) / sizeof(Buffer
[0]), Buffer
);
2381 /* Initialize strings for conversion */
2382 RtlInitUnicodeString(&BufferU
, Buffer
);
2384 BufferA
.MaximumLength
= (USHORT
)nBufferLength
;
2385 BufferA
.Buffer
= lpBuffer
;
2387 /* Convert unicode name to ansi, copying as much chars as fit */
2388 RtlUnicodeStringToAnsiString(&BufferA
, &BufferU
, FALSE
);
2390 /* Error handling */
2391 if (nBufferLength
<= BufferU
.Length
/ sizeof(WCHAR
))
2393 SetLastError(ERROR_BUFFER_OVERFLOW
);
2402 GetConsoleCharType(HANDLE hConsole
, COORD Coord
, PDWORD Type
)
2410 GetConsoleCursorMode(HANDLE hConsole
, PBOOL pUnknown1
, PBOOL pUnknown2
)
2418 GetConsoleNlsMode(HANDLE hConsole
, LPDWORD lpMode
)
2426 SetConsoleCursorMode(HANDLE hConsole
, BOOL Unknown1
, BOOL Unknown2
)
2434 SetConsoleLocalEUDC(DWORD Unknown1
, DWORD Unknown2
, DWORD Unknown3
, DWORD Unknown4
)
2442 SetConsoleNlsMode(HANDLE hConsole
, DWORD dwMode
)
2450 RegisterConsoleIME(HWND hWnd
, LPDWORD ThreadId
)
2458 RegisterConsoleOS2(BOOL bUnknown
)
2466 SetConsoleOS2OemFormat(BOOL bUnknown
)
2474 UnregisterConsoleIME(VOID
)
2484 BOOL WINAPI
GetConsoleKeyboardLayoutNameA(LPSTR name
)
2493 BOOL WINAPI
GetConsoleKeyboardLayoutNameW(LPWSTR name
)
2504 SetLastConsoleEventActive(VOID
)