/* * PROJECT: ReactOS Task Manager * LICENSE: LGPL-2.1-or-later (https://spdx.org/licenses/LGPL-2.1-or-later) * PURPOSE: Process Termination. * COPYRIGHT: Copyright 1999-2001 Brian Palmer * Copyright 2005 Klemens Friedl * Copyright 2014 Ismael Ferreras Morezuelas */ #include "precomp.h" #define NTOS_MODE_USER #include void ProcessPage_OnEndProcess(void) { DWORD dwProcessId; HANDLE hProcess; WCHAR szTitle[256]; WCHAR strErrorText[260]; dwProcessId = GetSelectedProcessId(); if (dwProcessId == 0) return; hProcess = OpenProcess(PROCESS_TERMINATE | PROCESS_QUERY_INFORMATION, FALSE, dwProcessId); /* forbid killing system processes even if we have privileges -- sigh, windows kludge! */ if (hProcess && IsCriticalProcess(hProcess)) { LoadStringW(hInst, IDS_MSG_UNABLETERMINATEPRO, szTitle, 256); LoadStringW(hInst, IDS_MSG_CLOSESYSTEMPROCESS, strErrorText, 256); MessageBoxW(hMainWnd, strErrorText, szTitle, MB_OK|MB_ICONWARNING|MB_TOPMOST); CloseHandle(hProcess); return; } /* if this is a standard process just ask for confirmation before doing it */ LoadStringW(hInst, IDS_MSG_WARNINGTERMINATING, strErrorText, 256); LoadStringW(hInst, IDS_MSG_TASKMGRWARNING, szTitle, 256); if (!ConfirmMessageBox(hMainWnd, strErrorText, szTitle, MB_YESNO|MB_ICONWARNING|MB_TOPMOST)) { if (hProcess) CloseHandle(hProcess); return; } /* no such process or not enough privileges to open its token */ if (!hProcess) { GetLastErrorText(strErrorText, 260); LoadStringW(hInst, IDS_MSG_UNABLETERMINATEPRO, szTitle, 256); MessageBoxW(hMainWnd, strErrorText, szTitle, MB_OK|MB_ICONSTOP|MB_TOPMOST); return; } /* try to kill it, and notify the user if didn't work */ if (!TerminateProcess(hProcess, 1)) { GetLastErrorText(strErrorText, 260); LoadStringW(hInst, IDS_MSG_UNABLETERMINATEPRO, szTitle, 256); MessageBoxW(hMainWnd, strErrorText, szTitle, MB_OK|MB_ICONSTOP|MB_TOPMOST); } CloseHandle(hProcess); } BOOL IsCriticalProcess(HANDLE hProcess) { NTSTATUS status; ULONG BreakOnTermination; /* return early if the process handle does not exist */ if (!hProcess) return FALSE; /* the important system processes that we don't want to let the user kill come marked as critical, this simplifies the check greatly. a critical process brings the system down when is terminated: */ status = NtQueryInformationProcess(hProcess, ProcessBreakOnTermination, &BreakOnTermination, sizeof(ULONG), NULL); if (NT_SUCCESS(status) && BreakOnTermination) return TRUE; return FALSE; } BOOL ShutdownProcessTreeHelper(HANDLE hSnapshot, HANDLE hParentProcess, DWORD dwParentPID) { HANDLE hChildHandle; PROCESSENTRY32W ProcessEntry = {0}; ProcessEntry.dwSize = sizeof(ProcessEntry); if (Process32FirstW(hSnapshot, &ProcessEntry)) { do { if (ProcessEntry.th32ParentProcessID == dwParentPID) { hChildHandle = OpenProcess(PROCESS_TERMINATE | PROCESS_QUERY_INFORMATION, FALSE, ProcessEntry.th32ProcessID); if (!hChildHandle || IsCriticalProcess(hChildHandle)) { if (hChildHandle) { CloseHandle(hChildHandle); } continue; } if (!ShutdownProcessTreeHelper(hSnapshot, hChildHandle, ProcessEntry.th32ProcessID)) { CloseHandle(hChildHandle); return FALSE; } CloseHandle(hChildHandle); } } while (Process32NextW(hSnapshot, &ProcessEntry)); } return TerminateProcess(hParentProcess, 0); } BOOL ShutdownProcessTree(HANDLE hParentProcess, DWORD dwParentPID) { HANDLE hSnapshot = CreateToolhelp32Snapshot(TH32CS_SNAPPROCESS, 0); BOOL bResult; if (!hSnapshot) { return FALSE; } bResult = ShutdownProcessTreeHelper(hSnapshot, hParentProcess, dwParentPID); CloseHandle(hSnapshot); return bResult; } void ProcessPage_OnEndProcessTree(void) { DWORD dwProcessId; HANDLE hProcess; WCHAR szTitle[256]; WCHAR strErrorText[260]; dwProcessId = GetSelectedProcessId(); if (dwProcessId == 0) return; hProcess = OpenProcess(PROCESS_TERMINATE | PROCESS_QUERY_INFORMATION, FALSE, dwProcessId); /* forbid killing system processes even if we have privileges -- sigh, windows kludge! */ if (hProcess && IsCriticalProcess(hProcess)) { LoadStringW(hInst, IDS_MSG_UNABLETERMINATEPRO, szTitle, 256); LoadStringW(hInst, IDS_MSG_CLOSESYSTEMPROCESS, strErrorText, 256); MessageBoxW(hMainWnd, strErrorText, szTitle, MB_OK|MB_ICONWARNING|MB_TOPMOST); CloseHandle(hProcess); return; } LoadStringW(hInst, IDS_MSG_WARNINGTERMINATING, strErrorText, 256); LoadStringW(hInst, IDS_MSG_TASKMGRWARNING, szTitle, 256); if (!ConfirmMessageBox(hMainWnd, strErrorText, szTitle, MB_YESNO|MB_ICONWARNING)) { if (hProcess) CloseHandle(hProcess); return; } if (!hProcess) { GetLastErrorText(strErrorText, 260); LoadStringW(hInst, IDS_MSG_UNABLETERMINATEPRO, szTitle, 256); MessageBoxW(hMainWnd, strErrorText, szTitle, MB_OK|MB_ICONSTOP); return; } if (!ShutdownProcessTree(hProcess, dwProcessId)) { GetLastErrorText(strErrorText, 260); LoadStringW(hInst, IDS_MSG_UNABLETERMINATEPRO, szTitle, 256); MessageBoxW(hMainWnd, strErrorText, szTitle, MB_OK|MB_ICONSTOP); } CloseHandle(hProcess); }