1 /* 2 * COPYRIGHT: See COPYING in the top level directory 3 * PROJECT: ReactOS User API Server DLL 4 * FILE: win32ss/user/winsrv/usersrv/shutdown.c 5 * PURPOSE: Logout/shutdown 6 * PROGRAMMERS: 7 */ 8 9 /* INCLUDES *******************************************************************/ 10 11 #include "usersrv.h" 12 13 #include <commctrl.h> 14 #include <psapi.h> 15 16 #include "resource.h" 17 18 #define NDEBUG 19 #include <debug.h> 20 21 /* GLOBALS ********************************************************************/ 22 23 // Those flags (that are used for CsrProcess->ShutdownFlags) are named 24 // in accordance to the only public one: SHUTDOWN_NORETRY used for the 25 // SetProcessShutdownParameters API. 26 #if !defined(SHUTDOWN_SYSTEMCONTEXT) && !defined(SHUTDOWN_OTHERCONTEXT) 27 #define SHUTDOWN_SYSTEMCONTEXT CsrShutdownSystem 28 #define SHUTDOWN_OTHERCONTEXT CsrShutdownOther 29 #endif 30 31 // The DPRINTs that need to really be removed as soon as everything works. 32 #define MY_DPRINT DPRINT1 33 #define MY_DPRINT2 DPRINT 34 35 typedef struct tagNOTIFY_CONTEXT 36 { 37 UINT Msg; 38 WPARAM wParam; 39 LPARAM lParam; 40 // HDESK Desktop; 41 // HDESK OldDesktop; 42 DWORD StartTime; 43 DWORD QueryResult; 44 HWND Dlg; 45 DWORD EndNowResult; 46 BOOL ShowUI; 47 HANDLE UIThread; 48 HWND WndClient; 49 PSHUTDOWN_SETTINGS ShutdownSettings; 50 } NOTIFY_CONTEXT, *PNOTIFY_CONTEXT; 51 52 #define QUERY_RESULT_ABORT 0 53 #define QUERY_RESULT_CONTINUE 1 54 #define QUERY_RESULT_TIMEOUT 2 55 #define QUERY_RESULT_ERROR 3 56 #define QUERY_RESULT_FORCE 4 57 58 typedef void (WINAPI *INITCOMMONCONTROLS_PROC)(void); 59 60 typedef struct tagMESSAGE_CONTEXT 61 { 62 HWND Wnd; 63 UINT Msg; 64 WPARAM wParam; 65 LPARAM lParam; 66 DWORD Timeout; 67 } MESSAGE_CONTEXT, *PMESSAGE_CONTEXT; 68 69 70 /* FUNCTIONS ******************************************************************/ 71 72 static HMODULE hComCtl32Lib = NULL; 73 74 static VOID 75 CallInitCommonControls(VOID) 76 { 77 static BOOL Initialized = FALSE; 78 INITCOMMONCONTROLS_PROC InitProc; 79 80 if (Initialized) return; 81 82 hComCtl32Lib = LoadLibraryW(L"COMCTL32.DLL"); 83 if (hComCtl32Lib == NULL) return; 84 85 InitProc = (INITCOMMONCONTROLS_PROC)GetProcAddress(hComCtl32Lib, "InitCommonControls"); 86 if (InitProc == NULL) return; 87 88 (*InitProc)(); 89 90 Initialized = TRUE; 91 } 92 93 static VOID FASTCALL 94 UpdateProgressBar(HWND ProgressBar, PNOTIFY_CONTEXT NotifyContext) 95 { 96 DWORD Passed; 97 98 Passed = GetTickCount() - NotifyContext->StartTime; 99 Passed -= NotifyContext->ShutdownSettings->HungAppTimeout; 100 if (NotifyContext->ShutdownSettings->WaitToKillAppTimeout < Passed) 101 { 102 Passed = NotifyContext->ShutdownSettings->WaitToKillAppTimeout; 103 } 104 SendMessageW(ProgressBar, PBM_SETPOS, Passed / 2, 0); 105 } 106 107 static INT_PTR CALLBACK 108 EndNowDlgProc(HWND Dlg, UINT Msg, WPARAM wParam, LPARAM lParam) 109 { 110 INT_PTR Result; 111 PNOTIFY_CONTEXT NotifyContext; 112 HWND ProgressBar; 113 DWORD TitleLength; 114 int Len; 115 LPWSTR Title; 116 117 switch(Msg) 118 { 119 case WM_INITDIALOG: 120 NotifyContext = (PNOTIFY_CONTEXT)lParam; 121 NotifyContext->EndNowResult = QUERY_RESULT_ABORT; 122 SetWindowLongPtrW(Dlg, DWLP_USER, (LONG_PTR)lParam); 123 TitleLength = SendMessageW(NotifyContext->WndClient, WM_GETTEXTLENGTH, 124 0, 0) + 125 GetWindowTextLengthW(Dlg); 126 Title = HeapAlloc(UserServerHeap, 0, (TitleLength + 1) * sizeof(WCHAR)); 127 if (Title) 128 { 129 Len = GetWindowTextW(Dlg, Title, TitleLength + 1); 130 SendMessageW(NotifyContext->WndClient, WM_GETTEXT, 131 TitleLength + 1 - Len, (LPARAM)(Title + Len)); 132 SetWindowTextW(Dlg, Title); 133 HeapFree(UserServerHeap, 0, Title); 134 } 135 ProgressBar = GetDlgItem(Dlg, IDC_PROGRESS); 136 SendMessageW(ProgressBar, PBM_SETRANGE32, 0, 137 NotifyContext->ShutdownSettings->WaitToKillAppTimeout / 2); 138 UpdateProgressBar(ProgressBar, NotifyContext); 139 SetTimer(Dlg, 0, 200, NULL); 140 Result = FALSE; 141 break; 142 143 case WM_TIMER: 144 NotifyContext = (PNOTIFY_CONTEXT)GetWindowLongPtrW(Dlg, DWLP_USER); 145 ProgressBar = GetDlgItem(Dlg, IDC_PROGRESS); 146 UpdateProgressBar(ProgressBar, NotifyContext); 147 Result = TRUE; 148 break; 149 150 case WM_COMMAND: 151 if (BN_CLICKED == HIWORD(wParam) && IDC_END_NOW == LOWORD(wParam)) 152 { 153 NotifyContext = (PNOTIFY_CONTEXT)GetWindowLongPtrW(Dlg, DWLP_USER); 154 NotifyContext->EndNowResult = QUERY_RESULT_FORCE; 155 MY_DPRINT("Closing progress dlg by hand\n"); 156 SendMessageW(Dlg, WM_CLOSE, 0, 0); 157 Result = TRUE; 158 } 159 else 160 { 161 Result = FALSE; 162 } 163 break; 164 165 case WM_CLOSE: 166 MY_DPRINT("WM_CLOSE\n"); 167 DestroyWindow(Dlg); 168 Result = TRUE; 169 break; 170 171 case WM_DESTROY: 172 MY_DPRINT("WM_DESTROY\n"); 173 NotifyContext = (PNOTIFY_CONTEXT)GetWindowLongPtrW(Dlg, DWLP_USER); 174 NotifyContext->Dlg = NULL; 175 KillTimer(Dlg, 0); 176 PostQuitMessage(NotifyContext->EndNowResult); 177 Result = TRUE; 178 break; 179 180 default: 181 Result = FALSE; 182 break; 183 } 184 185 return Result; 186 } 187 188 static DWORD WINAPI 189 EndNowThreadProc(LPVOID Parameter) 190 { 191 PNOTIFY_CONTEXT NotifyContext = (PNOTIFY_CONTEXT)Parameter; 192 MSG Msg; 193 194 #if 0 195 SetThreadDesktop(NotifyContext->Desktop); 196 SwitchDesktop(NotifyContext->Desktop); 197 #else 198 /* For now show the end task dialog in the active desktop */ 199 NtUserSetInformationThread(NtCurrentThread(), 200 UserThreadUseActiveDesktop, 201 NULL, 202 0); 203 #endif 204 205 CallInitCommonControls(); 206 NotifyContext->Dlg = CreateDialogParam(UserServerDllInstance, 207 MAKEINTRESOURCE(IDD_END_NOW), NULL, 208 EndNowDlgProc, (LPARAM)NotifyContext); 209 if (NotifyContext->Dlg == NULL) 210 return 0; 211 212 ShowWindow(NotifyContext->Dlg, SW_SHOWNORMAL); 213 214 while (GetMessageW(&Msg, NULL, 0, 0)) 215 { 216 if (!IsDialogMessage(NotifyContext->Dlg, &Msg)) 217 { 218 TranslateMessage(&Msg); 219 DispatchMessageW(&Msg); 220 } 221 } 222 223 return Msg.wParam; 224 } 225 226 static DWORD WINAPI 227 SendClientShutdown(LPVOID Parameter) 228 { 229 PMESSAGE_CONTEXT Context = (PMESSAGE_CONTEXT)Parameter; 230 DWORD_PTR Result; 231 232 /* If the shutdown is aborted, just notify the process, there is no need to wait */ 233 if ((Context->wParam & (MCS_QUERYENDSESSION | MCS_ENDSESSION)) == 0) 234 { 235 DPRINT("Called WM_CLIENTSHUTDOWN with wParam == 0 ...\n"); 236 SendNotifyMessageW(Context->Wnd, WM_CLIENTSHUTDOWN, 237 Context->wParam, Context->lParam); 238 return QUERY_RESULT_CONTINUE; 239 } 240 241 if (SendMessageTimeoutW(Context->Wnd, WM_CLIENTSHUTDOWN, 242 Context->wParam, Context->lParam, 243 SMTO_NORMAL, Context->Timeout, &Result)) 244 { 245 DWORD Ret; 246 247 if (Context->wParam & MCS_QUERYENDSESSION) 248 { 249 /* WM_QUERYENDSESSION case */ 250 switch (Result) 251 { 252 case MCSR_DONOTSHUTDOWN: 253 Ret = QUERY_RESULT_ABORT; 254 break; 255 256 case MCSR_GOODFORSHUTDOWN: 257 case MCSR_SHUTDOWNFINISHED: 258 default: 259 Ret = QUERY_RESULT_CONTINUE; 260 } 261 } 262 else 263 { 264 /* WM_ENDSESSION case */ 265 Ret = QUERY_RESULT_CONTINUE; 266 } 267 268 DPRINT("SendClientShutdown -- Return == %s\n", 269 Ret == QUERY_RESULT_CONTINUE ? "Continue" : "Abort"); 270 return Ret; 271 } 272 273 DPRINT1("SendClientShutdown -- Error == %s\n", 274 GetLastError() == 0 ? "Timeout" : "error"); 275 276 return (GetLastError() == 0 ? QUERY_RESULT_TIMEOUT : QUERY_RESULT_ERROR); 277 } 278 279 static BOOL 280 NotifyTopLevelWindow(HWND Wnd, PNOTIFY_CONTEXT NotifyContext) 281 { 282 MESSAGE_CONTEXT MessageContext; 283 DWORD Now, Passed; 284 DWORD Timeout, WaitStatus; 285 HANDLE MessageThread; 286 HANDLE Threads[2]; 287 288 SetForegroundWindow(Wnd); 289 290 Now = GetTickCount(); 291 if (NotifyContext->StartTime == 0) 292 NotifyContext->StartTime = Now; 293 294 /* 295 * Note: Passed is computed correctly even when GetTickCount() 296 * wraps due to unsigned arithmetic. 297 */ 298 Passed = Now - NotifyContext->StartTime; 299 MessageContext.Wnd = Wnd; 300 MessageContext.Msg = NotifyContext->Msg; 301 MessageContext.wParam = NotifyContext->wParam; 302 MessageContext.lParam = NotifyContext->lParam; 303 MessageContext.Timeout = NotifyContext->ShutdownSettings->HungAppTimeout; 304 if (!NotifyContext->ShutdownSettings->AutoEndTasks) 305 { 306 MessageContext.Timeout += NotifyContext->ShutdownSettings->WaitToKillAppTimeout; 307 } 308 if (Passed < MessageContext.Timeout) 309 { 310 MessageContext.Timeout -= Passed; 311 MessageThread = CreateThread(NULL, 0, SendClientShutdown, 312 (LPVOID)&MessageContext, 0, NULL); 313 if (MessageThread == NULL) 314 { 315 NotifyContext->QueryResult = QUERY_RESULT_ERROR; 316 return FALSE; 317 } 318 Timeout = NotifyContext->ShutdownSettings->HungAppTimeout; 319 if (Passed < Timeout) 320 { 321 Timeout -= Passed; 322 WaitStatus = WaitForSingleObjectEx(MessageThread, Timeout, FALSE); 323 } 324 else 325 { 326 WaitStatus = WAIT_TIMEOUT; 327 } 328 if (WAIT_TIMEOUT == WaitStatus) 329 { 330 NotifyContext->WndClient = Wnd; 331 if (NotifyContext->UIThread == NULL && NotifyContext->ShowUI) 332 { 333 NotifyContext->UIThread = CreateThread(NULL, 0, 334 EndNowThreadProc, 335 (LPVOID)NotifyContext, 336 0, NULL); 337 } 338 Threads[0] = MessageThread; 339 Threads[1] = NotifyContext->UIThread; 340 WaitStatus = WaitForMultipleObjectsEx(NotifyContext->UIThread == NULL ? 341 1 : 2, 342 Threads, FALSE, INFINITE, 343 FALSE); 344 if (WaitStatus == WAIT_OBJECT_0) 345 { 346 if (!GetExitCodeThread(MessageThread, &NotifyContext->QueryResult)) 347 { 348 NotifyContext->QueryResult = QUERY_RESULT_ERROR; 349 } 350 } 351 else if (WaitStatus == WAIT_OBJECT_0 + 1) 352 { 353 if (!GetExitCodeThread(NotifyContext->UIThread, 354 &NotifyContext->QueryResult)) 355 { 356 NotifyContext->QueryResult = QUERY_RESULT_ERROR; 357 } 358 } 359 else 360 { 361 NotifyContext->QueryResult = QUERY_RESULT_ERROR; 362 } 363 if (WaitStatus != WAIT_OBJECT_0) 364 { 365 TerminateThread(MessageThread, QUERY_RESULT_TIMEOUT); 366 } 367 } 368 else if (WaitStatus == WAIT_OBJECT_0) 369 { 370 if (!GetExitCodeThread(MessageThread, 371 &NotifyContext->QueryResult)) 372 { 373 NotifyContext->QueryResult = QUERY_RESULT_ERROR; 374 } 375 } 376 else 377 { 378 NotifyContext->QueryResult = QUERY_RESULT_ERROR; 379 } 380 CloseHandle(MessageThread); 381 } 382 else 383 { 384 NotifyContext->QueryResult = QUERY_RESULT_TIMEOUT; 385 } 386 387 DPRINT("NotifyContext->QueryResult == %d\n", NotifyContext->QueryResult); 388 return (NotifyContext->QueryResult == QUERY_RESULT_CONTINUE); 389 } 390 391 static BOOLEAN 392 IsConsoleMode(VOID) 393 { 394 return (BOOLEAN)NtUserCallNoParam(NOPARAM_ROUTINE_ISCONSOLEMODE); 395 } 396 397 /************************************************/ 398 399 400 static BOOL 401 ThreadShutdownNotify(IN PCSR_THREAD CsrThread, 402 IN ULONG Flags, 403 IN ULONG Flags2, 404 IN PNOTIFY_CONTEXT Context) 405 { 406 HWND TopWnd = NULL; 407 408 EnumThreadWindows(HandleToUlong(CsrThread->ClientId.UniqueThread), 409 FindTopLevelWnd, (LPARAM)&TopWnd); 410 if (TopWnd) 411 { 412 HWND hWndOwner; 413 414 /*** FOR TESTING PURPOSES ONLY!! ***/ 415 HWND tmpWnd; 416 tmpWnd = TopWnd; 417 /***********************************/ 418 419 while ((hWndOwner = GetWindow(TopWnd, GW_OWNER)) != NULL) 420 { 421 MY_DPRINT("GetWindow(TopWnd, GW_OWNER) not returned NULL...\n"); 422 TopWnd = hWndOwner; 423 } 424 if (TopWnd != tmpWnd) MY_DPRINT("(TopWnd = %x) != (tmpWnd = %x)\n", TopWnd, tmpWnd); 425 } 426 else 427 { 428 return FALSE; 429 } 430 431 Context->wParam = Flags2; 432 Context->lParam = (0 != (Flags & EWX_CALLER_WINLOGON_LOGOFF) ? 433 ENDSESSION_LOGOFF : 0); 434 435 Context->StartTime = 0; 436 Context->UIThread = NULL; 437 Context->ShowUI = !IsConsoleMode() && (Flags2 & (MCS_QUERYENDSESSION | MCS_ENDSESSION)); 438 Context->Dlg = NULL; 439 440 #if 0 // Obviously, switching desktops like that from within WINSRV doesn't work... 441 { 442 BOOL Success; 443 Context->OldDesktop = GetThreadDesktop(GetCurrentThreadId()); 444 // Context->Desktop = GetThreadDesktop(HandleToUlong(CsrThread->ClientId.UniqueThread)); 445 Context->Desktop = GetThreadDesktop(GetWindowThreadProcessId(TopWnd, NULL)); 446 MY_DPRINT("Last error = %d\n", GetLastError()); 447 MY_DPRINT("Before switching to desktop 0x%x\n", Context->Desktop); 448 Success = SwitchDesktop(Context->Desktop); 449 MY_DPRINT("After switching to desktop (Success = %s ; last error = %d); going to notify top-level...\n", 450 Success ? "TRUE" : "FALSE", GetLastError()); 451 } 452 #endif 453 454 NotifyTopLevelWindow(TopWnd, Context); 455 456 /******************************************************************************/ 457 #if 1 458 if (Context->UIThread) 459 { 460 MY_DPRINT("Context->UIThread != NULL\n"); 461 if (Context->Dlg) 462 { 463 MY_DPRINT("Sending WM_CLOSE because Dlg is != NULL\n"); 464 SendMessageW(Context->Dlg, WM_CLOSE, 0, 0); 465 } 466 else 467 { 468 MY_DPRINT("Terminating UIThread thread with QUERY_RESULT_ERROR\n"); 469 TerminateThread(Context->UIThread, QUERY_RESULT_ERROR); 470 } 471 CloseHandle(Context->UIThread); 472 /**/Context->UIThread = NULL;/**/ 473 /**/Context->Dlg = NULL;/**/ 474 } 475 #endif 476 /******************************************************************************/ 477 478 #if 0 479 MY_DPRINT("Switch back to old desktop 0x%x\n", Context->OldDesktop); 480 SwitchDesktop(Context->OldDesktop); 481 MY_DPRINT("Switched back ok\n"); 482 #endif 483 484 return TRUE; 485 } 486 487 static ULONG 488 NotifyUserProcessForShutdown(PCSR_PROCESS CsrProcess, 489 PSHUTDOWN_SETTINGS ShutdownSettings, 490 UINT Flags) 491 { 492 DWORD QueryResult = QUERY_RESULT_CONTINUE; 493 PCSR_PROCESS Process; 494 PCSR_THREAD Thread; 495 PLIST_ENTRY NextEntry; 496 NOTIFY_CONTEXT Context; 497 BOOL FoundWindows = FALSE; 498 499 /* In case we make a forced shutdown, just kill the process */ 500 if (Flags & EWX_FORCE) 501 return CsrShutdownCsrProcess; 502 503 Context.ShutdownSettings = ShutdownSettings; 504 Context.QueryResult = QUERY_RESULT_CONTINUE; // We continue shutdown by default. 505 506 /* Lock the process */ 507 CsrLockProcessByClientId(CsrProcess->ClientId.UniqueProcess, &Process); 508 509 /* Send first the QUERYENDSESSION messages to all the threads of the process */ 510 MY_DPRINT2("Sending the QUERYENDSESSION messages...\n"); 511 512 NextEntry = CsrProcess->ThreadList.Flink; 513 while (NextEntry != &CsrProcess->ThreadList) 514 { 515 /* Get the current thread entry */ 516 Thread = CONTAINING_RECORD(NextEntry, CSR_THREAD, Link); 517 518 /* Move to the next entry */ 519 NextEntry = NextEntry->Flink; 520 521 /* If the thread is being terminated, just skip it */ 522 if (Thread->Flags & CsrThreadTerminated) continue; 523 524 /* Reference the thread and temporarily unlock the process */ 525 CsrReferenceThread(Thread); 526 CsrUnlockProcess(Process); 527 528 Context.QueryResult = QUERY_RESULT_CONTINUE; 529 if (ThreadShutdownNotify(Thread, Flags, MCS_QUERYENDSESSION, &Context)) 530 { 531 FoundWindows = TRUE; 532 } 533 534 /* Lock the process again and dereference the thread */ 535 CsrLockProcessByClientId(CsrProcess->ClientId.UniqueProcess, &Process); 536 CsrDereferenceThread(Thread); 537 538 // FIXME: Analyze Context.QueryResult !! 539 /**/if (Context.QueryResult == QUERY_RESULT_ABORT) goto Quit;/**/ 540 } 541 542 if (!FoundWindows) 543 { 544 /* We looped all threads but no top level window was found so we didn't send any message */ 545 /* Let the console server run the generic process shutdown handler */ 546 CsrUnlockProcess(Process); 547 return CsrShutdownNonCsrProcess; 548 } 549 550 QueryResult = Context.QueryResult; 551 MY_DPRINT2("QueryResult = %s\n", 552 QueryResult == QUERY_RESULT_ABORT ? "Abort" : "Continue"); 553 554 /* Now send the ENDSESSION messages to the threads */ 555 MY_DPRINT2("Now sending the ENDSESSION messages...\n"); 556 557 NextEntry = CsrProcess->ThreadList.Flink; 558 while (NextEntry != &CsrProcess->ThreadList) 559 { 560 /* Get the current thread entry */ 561 Thread = CONTAINING_RECORD(NextEntry, CSR_THREAD, Link); 562 563 /* Move to the next entry */ 564 NextEntry = NextEntry->Flink; 565 566 /* If the thread is being terminated, just skip it */ 567 if (Thread->Flags & CsrThreadTerminated) continue; 568 569 /* Reference the thread and temporarily unlock the process */ 570 CsrReferenceThread(Thread); 571 CsrUnlockProcess(Process); 572 573 Context.QueryResult = QUERY_RESULT_CONTINUE; 574 ThreadShutdownNotify(Thread, Flags, 575 (QUERY_RESULT_ABORT != QueryResult) ? MCS_ENDSESSION : 0, 576 &Context); 577 578 /* Lock the process again and dereference the thread */ 579 CsrLockProcessByClientId(CsrProcess->ClientId.UniqueProcess, &Process); 580 CsrDereferenceThread(Thread); 581 } 582 583 Quit: 584 /* Unlock the process */ 585 CsrUnlockProcess(Process); 586 587 #if 0 588 if (Context.UIThread) 589 { 590 if (Context.Dlg) 591 { 592 SendMessageW(Context.Dlg, WM_CLOSE, 0, 0); 593 } 594 else 595 { 596 TerminateThread(Context.UIThread, QUERY_RESULT_ERROR); 597 } 598 CloseHandle(Context.UIThread); 599 } 600 #endif 601 602 /* Kill the process unless we abort shutdown */ 603 if (QueryResult == QUERY_RESULT_ABORT) 604 return CsrShutdownCancelled; 605 606 return CsrShutdownCsrProcess; 607 } 608 609 static NTSTATUS FASTCALL 610 UserExitReactOS(PCSR_THREAD CsrThread, UINT Flags) 611 { 612 NTSTATUS Status; 613 LUID CallerLuid; 614 615 DWORD ProcessId = HandleToUlong(CsrThread->ClientId.UniqueProcess); 616 DWORD ThreadId = HandleToUlong(CsrThread->ClientId.UniqueThread); 617 618 DPRINT1("SrvExitWindowsEx(ClientId: %lx.%lx, Flags: 0x%x)\n", 619 ProcessId, ThreadId, Flags); 620 621 /* 622 * Check for flags validity 623 */ 624 625 if (Flags & EWX_CALLER_WINLOGON) 626 { 627 /* Only Winlogon can call this */ 628 if (ProcessId != LogonProcessId) 629 { 630 DPRINT1("SrvExitWindowsEx call not from Winlogon\n"); 631 return STATUS_ACCESS_DENIED; 632 } 633 } 634 635 /* Implicitely add the shutdown flag when we poweroff or reboot */ 636 if (Flags & (EWX_POWEROFF | EWX_REBOOT)) 637 Flags |= EWX_SHUTDOWN; 638 639 /* 640 * Impersonate and retrieve the caller's LUID so that 641 * we can only shutdown processes in its context. 642 */ 643 if (!CsrImpersonateClient(NULL)) 644 return STATUS_BAD_IMPERSONATION_LEVEL; 645 646 Status = CsrGetProcessLuid(NULL, &CallerLuid); 647 if (!NT_SUCCESS(Status)) 648 { 649 DPRINT1("Unable to get caller LUID, Status = 0x%08x\n", Status); 650 goto Quit; 651 } 652 653 DPRINT("Caller LUID is: %lx.%lx\n", CallerLuid.HighPart, CallerLuid.LowPart); 654 655 /* Shutdown loop */ 656 while (TRUE) 657 { 658 /* Notify Win32k and potentially Winlogon of the shutdown */ 659 Status = NtUserSetInformationThread(CsrThread->ThreadHandle, 660 UserThreadInitiateShutdown, 661 &Flags, sizeof(Flags)); 662 DPRINT("Win32k says: %lx\n", Status); 663 switch (Status) 664 { 665 /* We cannot wait here, the caller should start a new thread */ 666 case STATUS_CANT_WAIT: 667 DPRINT1("NtUserSetInformationThread returned STATUS_CANT_WAIT\n"); 668 goto Quit; 669 670 /* Shutdown is in progress */ 671 case STATUS_PENDING: 672 DPRINT1("NtUserSetInformationThread returned STATUS_PENDING\n"); 673 goto Quit; 674 675 /* Abort */ 676 case STATUS_RETRY: 677 { 678 DPRINT1("NtUserSetInformationThread returned STATUS_RETRY\n"); 679 UNIMPLEMENTED; 680 continue; 681 } 682 683 default: 684 { 685 if (!NT_SUCCESS(Status)) 686 { 687 // FIXME: Use some UserSetLastNTError or SetLastNtError 688 // that we have defined for user32 or win32k usage only... 689 SetLastError(RtlNtStatusToDosError(Status)); 690 goto Quit; 691 } 692 } 693 } 694 695 /* All good */ 696 break; 697 } 698 699 /* 700 * OK we can continue. Now magic happens: 701 * 702 * Terminate all Win32 processes, stop if we find one kicking 703 * and screaming it doesn't want to die. 704 * 705 * This function calls the ShutdownProcessCallback callback of 706 * each CSR server for each Win32 process. 707 */ 708 Status = CsrShutdownProcesses(&CallerLuid, Flags); 709 if (!NT_SUCCESS(Status)) 710 { 711 DPRINT1("Failed to shutdown processes, Status = 0x%08x\n", Status); 712 } 713 714 // FIXME: If Status == STATUS_CANCELLED, call RecordShutdownReason 715 716 /* Tell Win32k and potentially Winlogon that we're done */ 717 NtUserSetInformationThread(CsrThread->ThreadHandle, 718 UserThreadEndShutdown, 719 &Status, sizeof(Status)); 720 721 DPRINT("SrvExitWindowsEx returned 0x%08x\n", Status); 722 723 Quit: 724 /* We are done */ 725 CsrRevertToSelf(); 726 return Status; 727 } 728 729 730 ULONG 731 NTAPI 732 UserClientShutdown(IN PCSR_PROCESS CsrProcess, 733 IN ULONG Flags, 734 IN BOOLEAN FirstPhase) 735 { 736 ULONG result; 737 738 DPRINT("UserClientShutdown(0x%p, 0x%x, %s) - [0x%x, 0x%x], ShutdownFlags: %lu\n", 739 CsrProcess, Flags, FirstPhase ? "FirstPhase" : "LastPhase", 740 CsrProcess->ClientId.UniqueProcess, CsrProcess->ClientId.UniqueThread, 741 CsrProcess->ShutdownFlags); 742 743 /* 744 * Check for process validity 745 */ 746 747 /* Do not kill system processes when a user is logging off */ 748 if ((Flags & EWX_SHUTDOWN) == EWX_LOGOFF && 749 (CsrProcess->ShutdownFlags & (SHUTDOWN_OTHERCONTEXT | SHUTDOWN_SYSTEMCONTEXT))) 750 { 751 DPRINT("Do not kill a system process in a logoff request!\n"); 752 return CsrShutdownNonCsrProcess; 753 } 754 755 /* Do not kill Winlogon */ 756 if (CsrProcess->ClientId.UniqueProcess == UlongToHandle(LogonProcessId)) 757 { 758 DPRINT("Not killing Winlogon; CsrProcess->ShutdownFlags = %lu\n", 759 CsrProcess->ShutdownFlags); 760 761 /* Returning CsrShutdownCsrProcess means that we handled this process by doing nothing */ 762 /* This will mark winlogon as processed so consrv won't be notified again for it */ 763 CsrDereferenceProcess(CsrProcess); 764 return CsrShutdownCsrProcess; 765 } 766 767 /* Notify the process for shutdown if needed */ 768 result = NotifyUserProcessForShutdown(CsrProcess, &ShutdownSettings, Flags); 769 if (result == CsrShutdownCancelled || result == CsrShutdownNonCsrProcess) 770 { 771 if (result == CsrShutdownCancelled) 772 DPRINT1("Process 0x%x aborted shutdown\n", CsrProcess->ClientId.UniqueProcess); 773 return result; 774 } 775 776 /* Terminate this process */ 777 #if DBG 778 { 779 WCHAR buffer[MAX_PATH]; 780 if (!GetProcessImageFileNameW(CsrProcess->ProcessHandle, buffer, MAX_PATH)) 781 { 782 DPRINT1("Terminating process %x\n", CsrProcess->ClientId.UniqueProcess); 783 } 784 else 785 { 786 DPRINT1("Terminating process %x (%S)\n", CsrProcess->ClientId.UniqueProcess, buffer); 787 } 788 } 789 #endif 790 NtTerminateProcess(CsrProcess->ProcessHandle, 0); 791 792 WaitForSingleObject(CsrProcess->ProcessHandle, ShutdownSettings.ProcessTerminateTimeout); 793 794 /* We are done */ 795 CsrDereferenceProcess(CsrProcess); 796 return CsrShutdownCsrProcess; 797 } 798 799 800 /* PUBLIC SERVER APIS *********************************************************/ 801 802 CSR_API(SrvExitWindowsEx) 803 { 804 NTSTATUS Status; 805 PUSER_EXIT_REACTOS ExitReactOSRequest = &((PUSER_API_MESSAGE)ApiMessage)->Data.ExitReactOSRequest; 806 807 Status = NtUserSetInformationThread(NtCurrentThread(), 808 UserThreadUseActiveDesktop, 809 NULL, 810 0); 811 if (!NT_SUCCESS(Status)) 812 { 813 DPRINT1("Failed to set thread desktop!\n"); 814 return Status; 815 } 816 817 Status = UserExitReactOS(CsrGetClientThread(), ExitReactOSRequest->Flags); 818 ExitReactOSRequest->Success = NT_SUCCESS(Status); 819 ExitReactOSRequest->LastError = GetLastError(); 820 821 NtUserSetInformationThread(NtCurrentThread(), UserThreadRestoreDesktop, NULL, 0); 822 823 return Status; 824 } 825 826 CSR_API(SrvEndTask) 827 { 828 PUSER_END_TASK EndTaskRequest = &((PUSER_API_MESSAGE)ApiMessage)->Data.EndTaskRequest; 829 NTSTATUS Status; 830 831 // FIXME: This is HACK-plemented!! 832 DPRINT1("SrvEndTask is HACKPLEMENTED!!\n"); 833 834 Status = NtUserSetInformationThread(NtCurrentThread(), 835 UserThreadUseActiveDesktop, 836 NULL, 837 0); 838 if (!NT_SUCCESS(Status)) 839 { 840 DPRINT1("Failed to set thread desktop!\n"); 841 return Status; 842 } 843 844 SendMessageW(EndTaskRequest->WndHandle, WM_CLOSE, 0, 0); 845 // PostMessageW(EndTaskRequest->WndHandle, WM_CLOSE, 0, 0); 846 847 if (IsWindow(EndTaskRequest->WndHandle)) 848 { 849 if (EndTaskRequest->Force) 850 { 851 EndTaskRequest->Success = DestroyWindow(EndTaskRequest->WndHandle); 852 EndTaskRequest->LastError = GetLastError(); 853 } 854 else 855 { 856 EndTaskRequest->Success = FALSE; 857 } 858 } 859 else 860 { 861 EndTaskRequest->Success = TRUE; 862 EndTaskRequest->LastError = ERROR_SUCCESS; 863 } 864 865 NtUserSetInformationThread(NtCurrentThread(), UserThreadRestoreDesktop, NULL, 0); 866 867 return STATUS_SUCCESS; 868 } 869 870 CSR_API(SrvRecordShutdownReason) 871 { 872 DPRINT1("%s not yet implemented\n", __FUNCTION__); 873 return STATUS_NOT_IMPLEMENTED; 874 } 875 876 /* EOF */ 877