/*++ Copyright (c) 1989 Microsoft Corporation Module Name: smss.c Abstract: Author: Mark Lucovsky (markl) 04-Oct-1989 Revision History: --*/ #include "smsrvp.h" #if defined(REMOTE_BOOT) char SmpFormatKeyword[] = "NETBOOTFORMAT"; char SmpDisconnectedKeyword[] = "NETBOOTDISCONNECTED"; char SmpNetbootKeyword[] = "NETBOOT"; char SmpHalKeyword[] = "NETBOOTHAL"; BOOLEAN SmpAutoFormat = FALSE; BOOLEAN SmpNetboot = FALSE; BOOLEAN SmpNetbootDisconnected = FALSE; char SmpHalName[MAX_HAL_NAME_LENGTH + 1] = ""; #endif // defined(REMOTE_BOOT) void SmpTerminate( ULONG_PTR Parameters[] ); EXCEPTION_DISPOSITION SmpUnhandledExceptionFilter( struct _EXCEPTION_POINTERS *ExceptionInfo, ULONG_PTR Parameters[] ); void __cdecl main( int argc, char *argv[], char *envp[], ULONG DebugParameter OPTIONAL ) { NTSTATUS Status; KPRIORITY SetBasePriority; UNICODE_STRING InitialCommand, DebugInitialCommand, UnicodeParameter; HANDLE ProcessHandles[ 2 ]; ULONG_PTR Parameters[ 4 ]; PROCESS_BASIC_INFORMATION ProcessInfo; ULONG MuSessionId = 0; // First instance (console) has MuSessionId = 0 RtlSetProcessIsCritical(TRUE, NULL, TRUE); RtlSetThreadIsCritical(TRUE, NULL, TRUE); SetBasePriority = FOREGROUND_BASE_PRIORITY+2; Status = NtSetInformationProcess( NtCurrentProcess(), ProcessBasePriority, (PVOID) &SetBasePriority, sizeof( SetBasePriority ) ); ASSERT(NT_SUCCESS(Status)); if (ARGUMENT_PRESENT( (PVOID)(ULONG_PTR) DebugParameter )) { SmpDebug = DebugParameter; } try { Parameters[ 0 ] = (ULONG_PTR)&UnicodeParameter; Parameters[ 1 ] = 0; Parameters[ 2 ] = 0; Parameters[ 3 ] = 0; Status = SmpInit( &InitialCommand, &ProcessHandles[ 0 ] ); if (!NT_SUCCESS( Status )) { KdPrint(( "SMSS: SmpInit return failure - Status == %x\n", Status )); RtlInitUnicodeString( &UnicodeParameter, L"Session Manager Initialization" ); Parameters[ 1 ] = (ULONG)Status; } else { SYSTEM_FLAGS_INFORMATION FlagInfo; Status = NtQuerySystemInformation( SystemFlagsInformation, &FlagInfo, sizeof( FlagInfo ), NULL); ASSERT(NT_SUCCESS(Status)); if (FlagInfo.Flags & (FLG_DEBUG_INITIAL_COMMAND | FLG_DEBUG_INITIAL_COMMAND_EX) ) { DebugInitialCommand.MaximumLength = InitialCommand.Length + 64; DebugInitialCommand.Length = 0; DebugInitialCommand.Buffer = RtlAllocateHeap( RtlProcessHeap(), MAKE_TAG( INIT_TAG ), DebugInitialCommand.MaximumLength ); if (DebugInitialCommand.Buffer == NULL) { Status = STATUS_NO_MEMORY; KdPrint(( "SMSS: SmpInit heap allocation failed\n")); } else { if (FlagInfo.Flags & FLG_ENABLE_CSRDEBUG) { RtlAppendUnicodeToString( &DebugInitialCommand, L"ntsd -isd -p -1 -d " ); } else { RtlAppendUnicodeToString( &DebugInitialCommand, L"ntsd -isd -d " ); } if (FlagInfo.Flags & FLG_DEBUG_INITIAL_COMMAND_EX ) { RtlAppendUnicodeToString( &DebugInitialCommand, L"-g -x " ); } RtlAppendUnicodeStringToString( &DebugInitialCommand, &InitialCommand ); InitialCommand = DebugInitialCommand; } } if (NT_SUCCESS( Status )) { Status = SmpExecuteInitialCommand( 0L, &InitialCommand, &ProcessHandles[ 1 ], NULL ); } if (NT_SUCCESS( Status )) { // // Detach the session manager from the session space as soon as // we have executed the initial command (winlogon). // PVOID State; Status = SmpAcquirePrivilege( SE_LOAD_DRIVER_PRIVILEGE, &State ); if (NT_SUCCESS( Status )) { // // If we are attached to a session space, leave it // so we can create a new one // if( (AttachedSessionId != (-1)) ) { Status = NtSetSystemInformation( SystemSessionDetach, (PVOID)&AttachedSessionId, sizeof(MuSessionId) ); ASSERT(NT_SUCCESS(Status)); AttachedSessionId = (-1); } SmpReleasePrivilege( State ); } } if (NT_SUCCESS( Status )) { Status = NtWaitForMultipleObjects( 2, ProcessHandles, WaitAny, FALSE, NULL ); } if (Status == STATUS_WAIT_0) { RtlInitUnicodeString( &UnicodeParameter, L"Windows SubSystem" ); Status = NtQueryInformationProcess( ProcessHandles[ 0 ], ProcessBasicInformation, &ProcessInfo, sizeof( ProcessInfo ), NULL ); KdPrint(( "SMSS: Windows subsystem terminated when it wasn't supposed to.\n" )); } else { RtlInitUnicodeString( &UnicodeParameter, L"Windows Logon Process" ); if (Status == STATUS_WAIT_1) { Status = NtQueryInformationProcess( ProcessHandles[ 1 ], ProcessBasicInformation, &ProcessInfo, sizeof( ProcessInfo ), NULL ); } else { ProcessInfo.ExitStatus = Status; Status = STATUS_SUCCESS; } KdPrint(( "SMSS: Initial command '%wZ' terminated when it wasn't supposed to.\n", &InitialCommand )); } if (NT_SUCCESS( Status )) { Parameters[ 1 ] = (ULONG)ProcessInfo.ExitStatus; } else { Parameters[ 1 ] = (ULONG)STATUS_UNSUCCESSFUL; } } } except( SmpUnhandledExceptionFilter( GetExceptionInformation(), Parameters ) ) { /* not reached */ } SmpTerminate(Parameters); /* not reached */ } void SmpTerminate( ULONG_PTR Parameters[] ) { NTSTATUS Status; ULONG Response; BOOLEAN WasEnabled; // // We are hosed, so raise a fatal system error to shutdown the system. // (Basically a user mode KeBugCheck). // Status = RtlAdjustPrivilege( SE_SHUTDOWN_PRIVILEGE, (BOOLEAN)TRUE, TRUE, &WasEnabled ); if (Status == STATUS_NO_TOKEN) { // // No thread token, use the process token. // Status = RtlAdjustPrivilege( SE_SHUTDOWN_PRIVILEGE, (BOOLEAN)TRUE, FALSE, &WasEnabled ); } Status = NtRaiseHardError( STATUS_SYSTEM_PROCESS_TERMINATED, 4, 1, Parameters, OptionShutdownSystem, &Response ); // // If this returns, give up. // NtTerminateProcess( NtCurrentProcess(), Status ); } EXCEPTION_DISPOSITION SmpUnhandledExceptionFilter( struct _EXCEPTION_POINTERS *ExceptionInfo, ULONG_PTR Parameters[] ) { UNICODE_STRING ExUnicodeParameter; #if DBG DbgPrint( "SMSS: Unhandled exception - Status == %x IP == %x\n", ExceptionInfo->ExceptionRecord->ExceptionCode, ExceptionInfo->ExceptionRecord->ExceptionAddress ); DbgPrint( " Memory Address: %x Read/Write: %x\n", ExceptionInfo->ExceptionRecord->ExceptionInformation[ 0 ], ExceptionInfo->ExceptionRecord->ExceptionInformation[ 1 ] ); DbgBreakPoint(); #endif RtlInitUnicodeString( &ExUnicodeParameter, L"Unhandled Exception in Session Manager" ); Parameters[ 0 ] = (ULONG_PTR)&ExUnicodeParameter; Parameters[ 1 ] = (ULONG_PTR)ExceptionInfo->ExceptionRecord->ExceptionCode; Parameters[ 2 ] = (ULONG_PTR)ExceptionInfo->ExceptionRecord->ExceptionAddress; Parameters[ 3 ] = (ULONG_PTR)ExceptionInfo->ContextRecord; // // SmpTerminate will raise a hard error with the exception info still valid. // SmpTerminate(Parameters); // not reached return EXCEPTION_EXECUTE_HANDLER; }