Source code of Windows XP (NT5)
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 
 
 

611 lines
18 KiB

/*++
Copyright (c) 1995 Microsoft Corporation
Module Name:
job.c
Abstract:
A user mode app that allows creation and management of jobs.
Environment:
User mode only
Revision History:
03-26-96 : Created
--*/
//
// this module may be compiled at warning level 4 with the following
// warnings disabled:
//
#include <string.h>
#include <stdio.h>
#include <stdlib.h>
#include <tchar.h>
#include <assert.h>
#include <windows.h>
#include <devioctl.h>
#include "jobmgr.h"
typedef struct {
char Option;
JOBOBJECTINFOCLASS InfoClass;
char *Name;
DWORD (*Function)(HANDLE Job, JOBOBJECTINFOCLASS InfoClass);
} JOB_QUERY_OPTION, *PJOB_QUERY_OPTION;
#define MKOPTION(optChar, optClass) {optChar, optClass, #optClass, Dump##optClass}
DWORD DumpJobObjectBasicProcessIdList(HANDLE JobHandle,
JOBOBJECTINFOCLASS InfoClass);
DWORD DumpJobObjectBasicUIRestrictions(HANDLE JobHandle,
JOBOBJECTINFOCLASS InfoClass);
DWORD
DumpJobObjectBasicAndIoAccountingInformation(
HANDLE JobHandle,
JOBOBJECTINFOCLASS InfoClass
);
DWORD
DumpJobObjectExtendedLimitInformation(
HANDLE Job,
JOBOBJECTINFOCLASS InfoClass
);
DWORD
DumpJobObjectSecurityLimitInformation(
HANDLE JobHandle,
JOBOBJECTINFOCLASS InfoClass
);
JOB_QUERY_OPTION JobInfoClasses[] = {
MKOPTION('a', JobObjectBasicAndIoAccountingInformation),
MKOPTION('l', JobObjectExtendedLimitInformation),
MKOPTION('p', JobObjectBasicProcessIdList),
MKOPTION('s', JobObjectSecurityLimitInformation),
MKOPTION('u', JobObjectBasicUIRestrictions),
{'\0', 0, NULL}
};
DWORD
QueryJobCommand(
IN PCOMMAND CommandEntry,
IN int argc,
IN char* argv[]
)
{
TCHAR defaultOptions[] = {TEXT('p'), TEXT('\0')};
PTSTR options;
PTSTR jobName;
HANDLE job;
int i;
BOOLEAN matchAll = FALSE;
DWORD status;
if(argc == 0) {
return -1;
}
GetAllProcessInfo();
if((argc > 1) && (argv[0][0] == '-')) {
// a - accounting & io
// l - extended limit info
// p - process id list
// u - basic ui restrictions
// s - security limits
options = &(argv[0][1]);
argc -= 1;
argv += 1;
} else {
options = defaultOptions;
}
if(_tcschr(options, TEXT('*')) != NULL) {
if(_tcslen(options) != 1) {
printf("Cannot specify '*' with other flags\n");
return -1;
} else {
matchAll = TRUE;
options = defaultOptions;
}
}
jobName = argv[0];
_tprintf("Opening job object %s\n", jobName);
job = OpenJobObject(JOB_OBJECT_QUERY, FALSE, jobName);
if(job == NULL) {
return GetLastError();
}
for(i = 0; JobInfoClasses[i].Option != '\0'; i++) {
LPTSTR match;
if(!matchAll) {
match = _tcschr(options, JobInfoClasses[i].Option);
if(match == NULL) {
continue;
}
//
// Clear the option so we can report the invalid option flags at the
// end.
//
*match = ' ';
}
_tprintf("%s [%#x]:\n", JobInfoClasses[i].Name,
JobInfoClasses[i].InfoClass);
status = JobInfoClasses[i].Function(job,
JobInfoClasses[i].InfoClass);
_tprintf("\n");
if(status != ERROR_SUCCESS) {
DWORD length;
PVOID buffer;
_tprintf("Error %s querying info: ",
CommandArray[i].Name, status);
length = FormatMessage((FORMAT_MESSAGE_ALLOCATE_BUFFER |
FORMAT_MESSAGE_FROM_SYSTEM |
FORMAT_MESSAGE_IGNORE_INSERTS |
(FORMAT_MESSAGE_MAX_WIDTH_MASK & 0)),
NULL,
status,
MAKELANGID(LANG_NEUTRAL, SUBLANG_DEFAULT),
(LPTSTR) &buffer,
1,
NULL);
if(length != 0) {
_tprintf("%s", buffer);
LocalFree(buffer);
}
}
}
if(!matchAll) {
LPSTR header = "Option flag not understood:";
while(options[0] != TEXT('\0')) {
if(options[0] != TEXT(' ')) {
_tprintf("%s %c", header, options[0]);
header = "";
}
options += 1;
}
}
#if 0
for(;options[0] != '\0'; options += 1) {
int i;
for(i = 0; JobInfoClasses[i].Option != '\0'; i++) {
if((options[0] != TEXT('*')) &&
(JobInfoClasses[i].Option != tolower(options[0]))) {
continue;
}
_tprintf("%s [%#x]:\n", JobInfoClasses[i].Name,
JobInfoClasses[i].InfoClass);
status = JobInfoClasses[i].Function(job,
JobInfoClasses[i].InfoClass);
_tprintf("\n");
if(status != ERROR_SUCCESS) {
DWORD length;
PVOID buffer;
_tprintf("Error %s querying info: ",
CommandArray[i].Name, status);
length = FormatMessage((FORMAT_MESSAGE_ALLOCATE_BUFFER |
FORMAT_MESSAGE_FROM_SYSTEM |
FORMAT_MESSAGE_IGNORE_INSERTS |
(FORMAT_MESSAGE_MAX_WIDTH_MASK & 0)),
NULL,
status,
MAKELANGID(LANG_NEUTRAL, SUBLANG_DEFAULT),
(LPTSTR) &buffer,
1,
NULL);
if(length != 0) {
_tprintf("%s", buffer);
LocalFree(buffer);
}
}
break;
}
if(JobInfoClasses[i].Option == '\0') {
_tprintf("invalid option flag '%c'\n", options[0]);
}
}
#endif
CloseHandle(job);
return ERROR_SUCCESS;
}
DWORD
DumpJobObjectBasicProcessIdList(
HANDLE Job,
JOBOBJECTINFOCLASS InfoClass
)
{
JOBOBJECT_BASIC_PROCESS_ID_LIST buffer;
PJOBOBJECT_BASIC_PROCESS_ID_LIST idList = NULL;
ULONG bufferSize;
BOOL result;
DWORD status;
DWORD i;
result = QueryInformationJobObject(Job,
InfoClass,
&buffer,
sizeof(JOBOBJECT_BASIC_PROCESS_ID_LIST),
NULL);
status = GetLastError();
if((!result) && (status != ERROR_MORE_DATA)) {
return status;
}
do {
if(idList != NULL) {
buffer.NumberOfAssignedProcesses =
idList->NumberOfAssignedProcesses;
LocalFree(idList);
idList = NULL;
}
//
// Calculate the actual size of the list and allocate a buffer to hold it.
//
bufferSize = sizeof(JOBOBJECT_BASIC_PROCESS_ID_LIST);
bufferSize -= sizeof(ULONG_PTR);
bufferSize += sizeof(ULONG_PTR) * buffer.NumberOfAssignedProcesses;
assert(idList == NULL);
idList = LocalAlloc(LPTR, bufferSize);
if(idList == NULL) {
return ERROR_NOT_ENOUGH_MEMORY;
}
result = QueryInformationJobObject(Job,
InfoClass,
idList,
bufferSize,
NULL);
status = GetLastError();
if((!result) && (status != ERROR_MORE_DATA)) {
LocalFree(idList);
return status;
}
} while(idList->NumberOfAssignedProcesses >
idList->NumberOfProcessIdsInList);
assert(idList->NumberOfAssignedProcesses ==
idList->NumberOfProcessIdsInList);
//
// Dump the information.
//
_tprintf(" %d processes assigned to job:\n",
idList->NumberOfAssignedProcesses);
for(i = 0; i < idList->NumberOfAssignedProcesses; i++) {
_tprintf("%8d", idList->ProcessIdList[i]);
PrintProcessInfo(idList->ProcessIdList[i]);
_tprintf("\n");
}
LocalFree(idList);
return ERROR_SUCCESS;
}
DWORD
DumpJobObjectBasicUIRestrictions(
HANDLE Job,
JOBOBJECTINFOCLASS InfoClass
)
{
JOBOBJECT_BASIC_UI_RESTRICTIONS uiLimit;
static FLAG_NAME jobUiLimitFlags[] = {
FLAG_NAME(JOB_OBJECT_UILIMIT_HANDLES ), //0x00000001
FLAG_NAME(JOB_OBJECT_UILIMIT_READCLIPBOARD ), //0x00000002
FLAG_NAME(JOB_OBJECT_UILIMIT_WRITECLIPBOARD ), //0x00000004
FLAG_NAME(JOB_OBJECT_UILIMIT_SYSTEMPARAMETERS ), //0x00000008
FLAG_NAME(JOB_OBJECT_UILIMIT_DISPLAYSETTINGS ), //0x00000010
FLAG_NAME(JOB_OBJECT_UILIMIT_GLOBALATOMS ), //0x00000020
FLAG_NAME(JOB_OBJECT_UILIMIT_DESKTOP ), //0x00000040
FLAG_NAME(JOB_OBJECT_UILIMIT_EXITWINDOWS ), //0x00000080
{0,0}
};
BOOL result;
DWORD status;
DWORD i;
result = QueryInformationJobObject(Job,
InfoClass,
&uiLimit,
sizeof(JOBOBJECT_BASIC_UI_RESTRICTIONS),
NULL);
status = GetLastError();
if(!result) {
return status;
}
if(uiLimit.UIRestrictionsClass == JOB_OBJECT_UILIMIT_NONE) {
_tprintf(" Job has no UI restrictions\n");
return ERROR_SUCCESS;
}
DumpFlags(2,
"UI Restrictions",
uiLimit.UIRestrictionsClass,
jobUiLimitFlags);
return ERROR_SUCCESS;
}
DWORD
DumpJobObjectBasicAndIoAccountingInformation(
HANDLE Job,
JOBOBJECTINFOCLASS InfoClass
)
{
JOBOBJECT_BASIC_AND_IO_ACCOUNTING_INFORMATION info;
BOOL result;
DWORD status;
DWORD i;
result = QueryInformationJobObject(Job,
InfoClass,
&info,
sizeof(info),
NULL);
status = GetLastError();
if(!result) {
return status;
}
xprintf(2, "Basic Info\n");
xprintf(4, "TotalUserTime: %s\n", TicksToString(info.BasicInfo.TotalUserTime));
xprintf(4, "TotalKernelTime: %s\n", TicksToString(info.BasicInfo.TotalKernelTime));
xprintf(4, "ThisPeriodTotalUserTime: %s\n", TicksToString(info.BasicInfo.ThisPeriodTotalUserTime));
xprintf(4, "ThisPeriodTotalKernelTime: %s\n", TicksToString(info.BasicInfo.ThisPeriodTotalKernelTime));
xprintf(4, "TotalPageFaultCount: %d\n", info.BasicInfo.TotalPageFaultCount);
xprintf(4, "TotalProcesses: %d\n", info.BasicInfo.TotalProcesses);
xprintf(4, "ActiveProcesses: %d\n", info.BasicInfo.ActiveProcesses);
xprintf(4, "TotalTerminatedProcesses: %d\n", info.BasicInfo.TotalTerminatedProcesses);
xprintf(2, "I/O Info\n");
xprintf(4, "ReadOperationCount: %I64d\n", info.IoInfo.ReadOperationCount);
xprintf(4, "WriteOperationCount: %I64d\n", info.IoInfo.WriteOperationCount);
xprintf(4, "OtherOperationCount: %I64d\n", info.IoInfo.OtherOperationCount);
xprintf(4, "ReadTransferCount: %I64d\n", info.IoInfo.ReadTransferCount);
xprintf(4, "WriteTransferCount: %I64d\n", info.IoInfo.WriteTransferCount);
xprintf(4, "OtherTransferCount: %I64d\n", info.IoInfo.OtherTransferCount);
return ERROR_SUCCESS;
}
DWORD
DumpJobObjectExtendedLimitInformation(
HANDLE Job,
JOBOBJECTINFOCLASS InfoClass
)
{
JOBOBJECT_EXTENDED_LIMIT_INFORMATION info;
ULONG limits;
static FLAG_NAME basicJobLimitFlags[] = {
FLAG_NAME(JOB_OBJECT_LIMIT_WORKINGSET ), //0x00000001
FLAG_NAME(JOB_OBJECT_LIMIT_PROCESS_TIME ), //0x00000002
FLAG_NAME(JOB_OBJECT_LIMIT_JOB_TIME ), //0x00000004
FLAG_NAME(JOB_OBJECT_LIMIT_ACTIVE_PROCESS ), //0x00000008
FLAG_NAME(JOB_OBJECT_LIMIT_AFFINITY ), //0x00000010
FLAG_NAME(JOB_OBJECT_LIMIT_PRIORITY_CLASS ), //0x00000020
FLAG_NAME(JOB_OBJECT_LIMIT_PRESERVE_JOB_TIME ), //0x00000040
FLAG_NAME(JOB_OBJECT_LIMIT_SCHEDULING_CLASS ), //0x00000080
{0,0}
};
//
// Extended Limits
//
static FLAG_NAME extendedJobLimitFlags[] = {
FLAG_NAME(JOB_OBJECT_LIMIT_PROCESS_MEMORY ), //0x00000100
FLAG_NAME(JOB_OBJECT_LIMIT_JOB_MEMORY ), //0x00000200
FLAG_NAME(JOB_OBJECT_LIMIT_DIE_ON_UNHANDLED_EXCEPTION ), //0x00000400
FLAG_NAME(JOB_OBJECT_LIMIT_BREAKAWAY_OK ), //0x00000800
FLAG_NAME(JOB_OBJECT_LIMIT_SILENT_BREAKAWAY_OK ), //0x00001000
FLAG_NAME(JOB_OBJECT_LIMIT_KILL_ON_JOB_CLOSE ), //0x00002000
FLAG_NAME(JOB_OBJECT_LIMIT_RESERVED2 ), //0x00004000
FLAG_NAME(JOB_OBJECT_LIMIT_RESERVED3 ), //0x00008000
FLAG_NAME(JOB_OBJECT_LIMIT_RESERVED4 ), //0x00010000
FLAG_NAME(JOB_OBJECT_LIMIT_RESERVED5 ), //0x00020000
FLAG_NAME(JOB_OBJECT_LIMIT_RESERVED6 ), //0x00040000
{0,0}
};
BOOL result;
DWORD status;
DWORD i;
result = QueryInformationJobObject(Job,
InfoClass,
&info,
sizeof(info),
NULL);
status = GetLastError();
if(!result) {
return status;
}
limits = info.BasicLimitInformation.LimitFlags;
if(TEST_FLAG(limits, JOB_OBJECT_BASIC_LIMIT_VALID_FLAGS) == 0) {
xprintf(2, "No basic limits on job\n");
} else {
DumpFlags(2, "Basic Limit Flags", limits & JOB_OBJECT_BASIC_LIMIT_VALID_FLAGS, basicJobLimitFlags);
if(TEST_FLAG(limits, JOB_OBJECT_LIMIT_PROCESS_TIME)) {
xprintf(4, "PerProcessUserTimeLimit: %s\n", TicksToString(info.BasicLimitInformation.PerProcessUserTimeLimit));
}
if(TEST_FLAG(limits, JOB_OBJECT_LIMIT_JOB_TIME)) {
xprintf(4, "PerJobUserTimeLimit: %s\n", TicksToString(info.BasicLimitInformation.PerJobUserTimeLimit));
}
if(TEST_FLAG(limits, JOB_OBJECT_LIMIT_WORKINGSET)) {
xprintf(4, "MinimumWorkingSetSize: %I64d\n", (ULONGLONG) info.BasicLimitInformation.MinimumWorkingSetSize);
xprintf(4, "MaximumWorkingSetSize: %I64d\n", (ULONGLONG) info.BasicLimitInformation.MaximumWorkingSetSize);
}
if(TEST_FLAG(limits, JOB_OBJECT_LIMIT_ACTIVE_PROCESS)) {
xprintf(4, "ActiveProcessLimit: %d\n",info.BasicLimitInformation.ActiveProcessLimit);
}
if(TEST_FLAG(limits, JOB_OBJECT_LIMIT_AFFINITY)) {
xprintf(4, "Affinity: %#I64x\n", (ULONGLONG)info.BasicLimitInformation.Affinity);
}
if(TEST_FLAG(limits, JOB_OBJECT_LIMIT_PRIORITY_CLASS)) {
xprintf(4, "PriorityClass: %d\n",info.BasicLimitInformation.PriorityClass);
}
if(TEST_FLAG(limits, JOB_OBJECT_LIMIT_SCHEDULING_CLASS)) {
xprintf(4, "SchedulingClass: %d\n",info.BasicLimitInformation.SchedulingClass);
}
}
if(TEST_FLAG(limits, JOB_OBJECT_EXTENDED_LIMIT_VALID_FLAGS) == 0) {
xprintf(2, "No extended limits on job\n");
} else {
DumpFlags(2, "Extended Limit Flags", limits & JOB_OBJECT_EXTENDED_LIMIT_VALID_FLAGS & ~JOB_OBJECT_BASIC_LIMIT_VALID_FLAGS, extendedJobLimitFlags);
if(TEST_FLAG(limits, JOB_OBJECT_LIMIT_PROCESS_MEMORY)) {
xprintf(4, "ProcessMemoryLimit: %I64d\n", (ULONGLONG) info.ProcessMemoryLimit);
}
if(TEST_FLAG(limits, JOB_OBJECT_LIMIT_PROCESS_MEMORY)) {
xprintf(4, "JobMemoryLimit: %I64d\n", (ULONGLONG) info.JobMemoryLimit);
}
}
xprintf(2, "PeakProcessMemoryUsed: %I64d\n", (ULONGLONG) info.PeakProcessMemoryUsed);
xprintf(2, "PeakJobMemoryUsed: %I64d\n", (ULONGLONG) info.PeakJobMemoryUsed);
return ERROR_SUCCESS;
}
DWORD
DumpJobObjectSecurityLimitInformation(
HANDLE Job,
JOBOBJECTINFOCLASS InfoClass
)
{
JOBOBJECT_SECURITY_LIMIT_INFORMATION buffer;
PJOBOBJECT_SECURITY_LIMIT_INFORMATION info = NULL;
static FLAG_NAME jobSecurityLimitFlags[] = {
FLAG_NAME(JOB_OBJECT_SECURITY_NO_ADMIN ), //00000001
FLAG_NAME(JOB_OBJECT_SECURITY_RESTRICTED_TOKEN ), //00000002
FLAG_NAME(JOB_OBJECT_SECURITY_ONLY_TOKEN ), //00000004
FLAG_NAME(JOB_OBJECT_SECURITY_FILTER_TOKENS ), //00000008
{0, 0}
};
ULONG bufferSize;
BOOL result;
DWORD status;
DWORD i;
result = QueryInformationJobObject(Job,
InfoClass,
&buffer,
sizeof(buffer),
&bufferSize);
status = GetLastError();
if((!result) && (status != ERROR_MORE_DATA)) {
return status;
}
info = LocalAlloc(LPTR, bufferSize);
if(info == NULL) {
return GetLastError();
}
result = QueryInformationJobObject(Job, InfoClass, info, bufferSize, NULL);
if(!result) {
status = GetLastError();
LocalFree(info);
return status;
}
if(info->SecurityLimitFlags == 0) {
xprintf(2, "No security limitations on job\n");
} else {
DumpFlags(2, "SecurityLimitFlags", info->SecurityLimitFlags, jobSecurityLimitFlags);
}
LocalFree(info);
return ERROR_SUCCESS;
}