2004-03-23 02:19:54 +01:00
|
|
|
/*
|
|
|
|
* ReactOS Task Manager
|
|
|
|
*
|
|
|
|
* affinity.c
|
|
|
|
*
|
|
|
|
* Copyright (C) 1999 - 2001 Brian Palmer <brianp@reactos.org>
|
2008-08-20 10:32:24 +02:00
|
|
|
* Copyright (C) 2008 Vladimir Pankratov
|
2019-03-26 15:48:41 +01:00
|
|
|
* Copyright (C) 2019 Isira Seneviratne
|
2004-03-23 02:19:54 +01:00
|
|
|
*
|
|
|
|
* This library is free software; you can redistribute it and/or
|
|
|
|
* modify it under the terms of the GNU Lesser General Public
|
|
|
|
* License as published by the Free Software Foundation; either
|
|
|
|
* version 2.1 of the License, or (at your option) any later version.
|
|
|
|
*
|
|
|
|
* This library is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
|
|
* Lesser General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU Lesser General Public
|
|
|
|
* License along with this library; if not, write to the Free Software
|
2006-05-18 14:49:52 +02:00
|
|
|
* Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
|
2004-03-23 02:19:54 +01:00
|
|
|
*/
|
2012-01-23 12:04:05 +01:00
|
|
|
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
|
2004-03-23 02:19:54 +01:00
|
|
|
#include <windows.h>
|
|
|
|
#include <commctrl.h>
|
|
|
|
#include <winnt.h>
|
2012-01-23 12:04:05 +01:00
|
|
|
|
2004-03-23 02:19:54 +01:00
|
|
|
#include "taskmgr.h"
|
|
|
|
#include "perfdata.h"
|
|
|
|
|
|
|
|
HANDLE hProcessAffinityHandle;
|
|
|
|
|
2009-06-24 14:39:19 +02:00
|
|
|
WCHAR wszUnable2Access[255];
|
2008-08-20 10:32:24 +02:00
|
|
|
|
2006-03-21 18:32:40 +01:00
|
|
|
static INT_PTR CALLBACK
|
2005-09-14 12:06:09 +02:00
|
|
|
AffinityDialogWndProc(HWND hDlg, UINT message, WPARAM wParam, LPARAM lParam)
|
2004-03-23 02:19:54 +01:00
|
|
|
{
|
2006-06-06 07:21:44 +02:00
|
|
|
DWORD_PTR dwProcessAffinityMask = 0;
|
|
|
|
DWORD_PTR dwSystemAffinityMask = 0;
|
2008-08-20 10:32:24 +02:00
|
|
|
WCHAR wstrErrorText[256];
|
2019-03-26 15:48:41 +01:00
|
|
|
int i;
|
2004-03-23 02:19:54 +01:00
|
|
|
|
|
|
|
switch (message) {
|
|
|
|
case WM_INITDIALOG:
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Get the current affinity mask for the process and
|
|
|
|
* the number of CPUs present in the system
|
|
|
|
*/
|
|
|
|
if (!GetProcessAffinityMask(hProcessAffinityHandle, &dwProcessAffinityMask, &dwSystemAffinityMask)) {
|
2018-07-24 00:01:16 +02:00
|
|
|
GetLastErrorText(wstrErrorText, ARRAY_SIZE(wstrErrorText));
|
2004-03-23 02:19:54 +01:00
|
|
|
EndDialog(hDlg, 0);
|
2018-07-24 00:01:16 +02:00
|
|
|
LoadStringW(hInst, IDS_AFFINITY_UNABLE2ACCESS, wszUnable2Access, ARRAY_SIZE(wszUnable2Access));
|
2008-08-20 10:32:24 +02:00
|
|
|
MessageBoxW(hMainWnd, wstrErrorText, wszUnable2Access, MB_OK|MB_ICONSTOP);
|
2004-03-23 02:19:54 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Enable a checkbox for each processor present in the system
|
|
|
|
*/
|
2019-03-26 15:48:41 +01:00
|
|
|
for (i = 0; i < 32; i++)
|
|
|
|
if (dwSystemAffinityMask & (1 << i))
|
|
|
|
EnableWindow(GetDlgItem(hDlg, IDC_CPU0 + i), TRUE);
|
2004-03-23 02:19:54 +01:00
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Check each checkbox that the current process
|
|
|
|
* has affinity with
|
|
|
|
*/
|
2019-03-26 15:48:41 +01:00
|
|
|
for (i = 0; i < 32; i++)
|
|
|
|
if (dwProcessAffinityMask & (1 << i))
|
|
|
|
SendMessageW(GetDlgItem(hDlg, IDC_CPU0 + i), BM_SETCHECK, BST_CHECKED, 0);
|
2004-03-23 02:19:54 +01:00
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
|
|
|
|
case WM_COMMAND:
|
|
|
|
|
|
|
|
/*
|
|
|
|
* If the user has cancelled the dialog box
|
|
|
|
* then just close it
|
|
|
|
*/
|
|
|
|
if (LOWORD(wParam) == IDCANCEL) {
|
|
|
|
EndDialog(hDlg, LOWORD(wParam));
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* The user has clicked OK -- so now we have
|
|
|
|
* to adjust the process affinity mask
|
|
|
|
*/
|
|
|
|
if (LOWORD(wParam) == IDOK) {
|
|
|
|
/*
|
|
|
|
* First we have to create a mask out of each
|
|
|
|
* checkbox that the user checked.
|
|
|
|
*/
|
2019-03-26 15:48:41 +01:00
|
|
|
for (i = 0; i < 32; i++)
|
|
|
|
if (SendMessageW(GetDlgItem(hDlg, IDC_CPU0 + i), BM_GETCHECK, 0, 0))
|
|
|
|
dwProcessAffinityMask |= (1 << i);
|
2004-03-23 02:19:54 +01:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Make sure they are giving the process affinity
|
|
|
|
* with at least one processor. I'd hate to see a
|
|
|
|
* process that is not in a wait state get deprived
|
2014-02-07 22:42:49 +01:00
|
|
|
* of its cpu time.
|
2004-03-23 02:19:54 +01:00
|
|
|
*/
|
|
|
|
if (!dwProcessAffinityMask) {
|
2009-06-24 14:39:19 +02:00
|
|
|
WCHAR wszErrorMsg[255];
|
|
|
|
WCHAR wszErrorTitle[255];
|
2018-07-24 00:01:16 +02:00
|
|
|
LoadStringW(hInst, IDS_AFFINITY_ERROR_MESSAGE, wszErrorMsg, ARRAY_SIZE(wszErrorMsg));
|
|
|
|
LoadStringW(hInst, IDS_AFFINITY_ERROR_TITLE, wszErrorTitle, ARRAY_SIZE(wszErrorTitle));
|
2008-08-20 10:32:24 +02:00
|
|
|
MessageBoxW(hDlg, wszErrorMsg, wszErrorTitle, MB_OK|MB_ICONSTOP);
|
2004-03-23 02:19:54 +01:00
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Try to set the process affinity
|
|
|
|
*/
|
|
|
|
if (!SetProcessAffinityMask(hProcessAffinityHandle, dwProcessAffinityMask)) {
|
2018-07-24 00:01:16 +02:00
|
|
|
GetLastErrorText(wstrErrorText, ARRAY_SIZE(wstrErrorText));
|
2004-03-23 02:19:54 +01:00
|
|
|
EndDialog(hDlg, LOWORD(wParam));
|
2018-07-24 00:01:16 +02:00
|
|
|
LoadStringW(hInst, IDS_AFFINITY_UNABLE2ACCESS, wszUnable2Access, ARRAY_SIZE(wszUnable2Access));
|
2008-08-20 10:32:24 +02:00
|
|
|
MessageBoxW(hMainWnd, wstrErrorText, wszUnable2Access, MB_OK|MB_ICONSTOP);
|
2004-03-23 02:19:54 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
EndDialog(hDlg, LOWORD(wParam));
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
2006-03-21 18:32:40 +01:00
|
|
|
|
|
|
|
void ProcessPage_OnSetAffinity(void)
|
|
|
|
{
|
2008-08-20 10:32:24 +02:00
|
|
|
LV_ITEMW lvitem;
|
2011-05-17 12:15:03 +02:00
|
|
|
ULONG Index, Count;
|
2006-03-21 18:32:40 +01:00
|
|
|
DWORD dwProcessId;
|
2008-08-20 10:32:24 +02:00
|
|
|
WCHAR wstrErrorText[256];
|
2006-03-21 18:32:40 +01:00
|
|
|
|
2011-05-17 12:15:03 +02:00
|
|
|
Count = SendMessageW(hProcessPageListCtrl, LVM_GETITEMCOUNT, 0, 0);
|
|
|
|
for (Index=0; Index<Count; Index++) {
|
2008-08-20 10:32:24 +02:00
|
|
|
memset(&lvitem, 0, sizeof(LV_ITEMW));
|
2006-03-21 18:32:40 +01:00
|
|
|
lvitem.mask = LVIF_STATE;
|
|
|
|
lvitem.stateMask = LVIS_SELECTED;
|
|
|
|
lvitem.iItem = Index;
|
2008-08-20 10:32:24 +02:00
|
|
|
SendMessageW(hProcessPageListCtrl, LVM_GETITEMW, 0, (LPARAM) &lvitem);
|
2006-03-21 18:32:40 +01:00
|
|
|
if (lvitem.state & LVIS_SELECTED)
|
|
|
|
break;
|
|
|
|
}
|
2011-05-17 12:15:03 +02:00
|
|
|
|
|
|
|
Count = SendMessageW(hProcessPageListCtrl, LVM_GETSELECTEDCOUNT, 0, 0);
|
2006-03-21 18:32:40 +01:00
|
|
|
dwProcessId = PerfDataGetProcessId(Index);
|
2011-05-17 12:15:03 +02:00
|
|
|
if ((Count != 1) || (dwProcessId == 0))
|
2006-03-21 18:32:40 +01:00
|
|
|
return;
|
|
|
|
hProcessAffinityHandle = OpenProcess(PROCESS_QUERY_INFORMATION|PROCESS_SET_INFORMATION, FALSE, dwProcessId);
|
|
|
|
if (!hProcessAffinityHandle) {
|
2018-07-24 00:01:16 +02:00
|
|
|
GetLastErrorText(wstrErrorText, ARRAY_SIZE(wstrErrorText));
|
|
|
|
LoadStringW(hInst, IDS_AFFINITY_UNABLE2ACCESS, wszUnable2Access, ARRAY_SIZE(wszUnable2Access));
|
2008-08-20 10:32:24 +02:00
|
|
|
MessageBoxW(hMainWnd, wstrErrorText, wszUnable2Access, MB_OK|MB_ICONSTOP);
|
2006-03-21 18:32:40 +01:00
|
|
|
return;
|
|
|
|
}
|
2008-08-20 10:32:24 +02:00
|
|
|
DialogBoxW(hInst, MAKEINTRESOURCEW(IDD_AFFINITY_DIALOG), hMainWnd, AffinityDialogWndProc);
|
2006-03-21 18:32:40 +01:00
|
|
|
if (hProcessAffinityHandle) {
|
|
|
|
CloseHandle(hProcessAffinityHandle);
|
|
|
|
hProcessAffinityHandle = NULL;
|
|
|
|
}
|
|
|
|
}
|