2005-01-06 13:58:04 +00:00
|
|
|
/* $Id$
|
2004-09-23 18:02:19 +00:00
|
|
|
*
|
|
|
|
* COPYRIGHT: See COPYING in the top level directory
|
|
|
|
* PROJECT: ReactOS system libraries
|
|
|
|
* FILE: lib/kernel32/process/job.c
|
|
|
|
* PURPOSE: Job functions
|
2004-09-23 18:31:51 +00:00
|
|
|
* PROGRAMMER: Thomas Weidenmueller <w3seek@reactos.com>
|
2004-09-23 18:02:19 +00:00
|
|
|
* UPDATE HISTORY:
|
|
|
|
* Created 9/23/2004
|
|
|
|
*/
|
|
|
|
|
|
|
|
/* INCLUDES ****************************************************************/
|
|
|
|
|
|
|
|
#include <k32.h>
|
|
|
|
|
|
|
|
#define NDEBUG
|
2007-09-02 19:42:22 +00:00
|
|
|
#include <debug.h>
|
2004-09-23 18:02:19 +00:00
|
|
|
|
|
|
|
/* FUNCTIONS ****************************************************************/
|
|
|
|
|
2004-09-23 18:31:51 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* @implemented
|
|
|
|
*/
|
|
|
|
HANDLE
|
2008-11-30 11:42:05 +00:00
|
|
|
WINAPI
|
2004-09-23 18:31:51 +00:00
|
|
|
CreateJobObjectA(LPSECURITY_ATTRIBUTES lpJobAttributes,
|
|
|
|
LPCSTR lpName)
|
|
|
|
{
|
2010-01-09 20:51:33 +00:00
|
|
|
HANDLE hJob;
|
|
|
|
ANSI_STRING AnsiName;
|
|
|
|
UNICODE_STRING UnicodeName;
|
2005-05-09 01:46:57 +00:00
|
|
|
|
2010-01-09 20:51:33 +00:00
|
|
|
if (lpName != NULL)
|
2004-09-23 18:31:51 +00:00
|
|
|
{
|
2010-01-09 20:51:33 +00:00
|
|
|
NTSTATUS Status;
|
|
|
|
|
|
|
|
RtlInitAnsiString(&AnsiName, lpName);
|
|
|
|
Status = RtlAnsiStringToUnicodeString(&UnicodeName, &AnsiName, TRUE);
|
|
|
|
if (!NT_SUCCESS(Status))
|
|
|
|
{
|
|
|
|
SetLastErrorByStatus(Status);
|
|
|
|
return FALSE;
|
|
|
|
}
|
2004-09-23 18:31:51 +00:00
|
|
|
}
|
2005-05-09 01:46:57 +00:00
|
|
|
|
2010-01-09 20:51:33 +00:00
|
|
|
hJob = CreateJobObjectW(lpJobAttributes,
|
|
|
|
((lpName != NULL) ? UnicodeName.Buffer : NULL));
|
2004-09-23 18:31:51 +00:00
|
|
|
|
2010-01-09 20:51:33 +00:00
|
|
|
if (lpName != NULL)
|
|
|
|
{
|
|
|
|
RtlFreeUnicodeString(&UnicodeName);
|
|
|
|
}
|
|
|
|
|
|
|
|
return hJob;
|
2004-09-23 18:31:51 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
* @implemented
|
|
|
|
*/
|
|
|
|
HANDLE
|
2008-11-30 11:42:05 +00:00
|
|
|
WINAPI
|
2004-09-23 18:31:51 +00:00
|
|
|
CreateJobObjectW(LPSECURITY_ATTRIBUTES lpJobAttributes,
|
|
|
|
LPCWSTR lpName)
|
|
|
|
{
|
2010-01-09 20:51:33 +00:00
|
|
|
UNICODE_STRING JobName;
|
|
|
|
OBJECT_ATTRIBUTES ObjectAttributes;
|
|
|
|
ULONG Attributes = 0;
|
|
|
|
PVOID SecurityDescriptor;
|
|
|
|
HANDLE hJob;
|
|
|
|
NTSTATUS Status;
|
2005-05-09 01:46:57 +00:00
|
|
|
|
2010-01-09 20:51:33 +00:00
|
|
|
if (lpName != NULL)
|
2004-09-23 18:31:51 +00:00
|
|
|
{
|
2010-01-09 20:51:33 +00:00
|
|
|
RtlInitUnicodeString(&JobName, lpName);
|
2004-09-23 18:31:51 +00:00
|
|
|
}
|
2005-05-09 01:46:57 +00:00
|
|
|
|
2010-01-09 20:51:33 +00:00
|
|
|
if (lpJobAttributes != NULL)
|
|
|
|
{
|
|
|
|
if (lpJobAttributes->bInheritHandle)
|
|
|
|
{
|
|
|
|
Attributes |= OBJ_INHERIT;
|
|
|
|
}
|
|
|
|
|
|
|
|
SecurityDescriptor = lpJobAttributes->lpSecurityDescriptor;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
SecurityDescriptor = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
InitializeObjectAttributes(&ObjectAttributes,
|
|
|
|
((lpName != NULL) ? &JobName : NULL),
|
|
|
|
Attributes,
|
|
|
|
NULL,
|
|
|
|
SecurityDescriptor);
|
|
|
|
|
|
|
|
Status = NtCreateJobObject(&hJob,
|
|
|
|
JOB_OBJECT_ALL_ACCESS,
|
|
|
|
&ObjectAttributes);
|
|
|
|
if (!NT_SUCCESS(Status))
|
|
|
|
{
|
|
|
|
SetLastErrorByStatus(Status);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
return hJob;
|
2004-09-23 18:31:51 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-09-23 18:46:10 +00:00
|
|
|
/*
|
|
|
|
* @implemented
|
|
|
|
*/
|
|
|
|
HANDLE
|
2008-11-30 11:42:05 +00:00
|
|
|
WINAPI
|
2004-09-23 18:46:10 +00:00
|
|
|
OpenJobObjectW(DWORD dwDesiredAccess,
|
|
|
|
BOOL bInheritHandle,
|
|
|
|
LPCWSTR lpName)
|
|
|
|
{
|
2010-01-09 20:51:33 +00:00
|
|
|
OBJECT_ATTRIBUTES ObjectAttributes;
|
|
|
|
UNICODE_STRING JobName;
|
|
|
|
HANDLE hJob;
|
|
|
|
NTSTATUS Status;
|
2004-09-23 18:46:10 +00:00
|
|
|
|
2010-01-09 20:51:33 +00:00
|
|
|
if (lpName == NULL)
|
|
|
|
{
|
|
|
|
SetLastError(ERROR_INVALID_PARAMETER);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
RtlInitUnicodeString(&JobName, lpName);
|
|
|
|
|
|
|
|
InitializeObjectAttributes(&ObjectAttributes,
|
|
|
|
&JobName,
|
|
|
|
(bInheritHandle ? OBJ_INHERIT : 0),
|
|
|
|
NULL,
|
|
|
|
NULL);
|
|
|
|
|
|
|
|
Status = NtOpenJobObject(&hJob,
|
|
|
|
dwDesiredAccess,
|
|
|
|
&ObjectAttributes);
|
|
|
|
if (!NT_SUCCESS(Status))
|
|
|
|
{
|
|
|
|
SetLastErrorByStatus(Status);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
return hJob;
|
2004-09-23 18:46:10 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
* @implemented
|
|
|
|
*/
|
|
|
|
HANDLE
|
2008-11-30 11:42:05 +00:00
|
|
|
WINAPI
|
2004-09-23 18:46:10 +00:00
|
|
|
OpenJobObjectA(DWORD dwDesiredAccess,
|
|
|
|
BOOL bInheritHandle,
|
|
|
|
LPCSTR lpName)
|
|
|
|
{
|
2010-01-09 20:51:33 +00:00
|
|
|
ANSI_STRING AnsiName;
|
|
|
|
UNICODE_STRING UnicodeName;
|
|
|
|
HANDLE hJob;
|
|
|
|
NTSTATUS Status;
|
2005-05-09 01:46:57 +00:00
|
|
|
|
2010-01-09 20:51:33 +00:00
|
|
|
if (lpName == NULL)
|
|
|
|
{
|
|
|
|
SetLastError(ERROR_INVALID_PARAMETER);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
RtlInitAnsiString(&AnsiName, lpName);
|
|
|
|
Status = RtlAnsiStringToUnicodeString(&UnicodeName, &AnsiName, TRUE);
|
|
|
|
if (!NT_SUCCESS(Status))
|
|
|
|
{
|
|
|
|
SetLastErrorByStatus(Status);
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
hJob = OpenJobObjectW(dwDesiredAccess,
|
|
|
|
bInheritHandle,
|
|
|
|
UnicodeName.Buffer);
|
2004-09-23 18:46:10 +00:00
|
|
|
|
2010-01-09 20:51:33 +00:00
|
|
|
RtlFreeUnicodeString(&UnicodeName);
|
|
|
|
return hJob;
|
2004-09-23 18:46:10 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-09-23 18:02:19 +00:00
|
|
|
/*
|
|
|
|
* @implemented
|
|
|
|
*/
|
|
|
|
BOOL
|
2008-11-30 11:42:05 +00:00
|
|
|
WINAPI
|
2004-09-23 18:02:19 +00:00
|
|
|
IsProcessInJob(HANDLE ProcessHandle,
|
|
|
|
HANDLE JobHandle,
|
|
|
|
PBOOL Result)
|
|
|
|
{
|
2010-01-09 20:51:33 +00:00
|
|
|
NTSTATUS Status;
|
2004-09-23 18:02:19 +00:00
|
|
|
|
2010-01-09 20:51:33 +00:00
|
|
|
Status = NtIsProcessInJob(ProcessHandle, JobHandle);
|
|
|
|
if (NT_SUCCESS(Status))
|
|
|
|
{
|
|
|
|
*Result = (Status == STATUS_PROCESS_IN_JOB);
|
|
|
|
return TRUE;
|
|
|
|
}
|
2005-05-09 01:46:57 +00:00
|
|
|
|
2010-01-09 20:51:33 +00:00
|
|
|
SetLastErrorByStatus(Status);
|
|
|
|
return FALSE;
|
2004-09-23 18:02:19 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
* @implemented
|
|
|
|
*/
|
|
|
|
BOOL
|
2008-11-30 11:42:05 +00:00
|
|
|
WINAPI
|
2004-09-23 18:02:19 +00:00
|
|
|
AssignProcessToJobObject(HANDLE hJob,
|
|
|
|
HANDLE hProcess)
|
|
|
|
{
|
2010-01-09 20:51:33 +00:00
|
|
|
NTSTATUS Status;
|
2004-09-23 18:02:19 +00:00
|
|
|
|
2010-01-09 20:51:33 +00:00
|
|
|
Status = NtAssignProcessToJobObject(hJob, hProcess);
|
|
|
|
if (!NT_SUCCESS(Status))
|
|
|
|
{
|
|
|
|
SetLastErrorByStatus(Status);
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
return TRUE;
|
2004-09-23 18:02:19 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
* @implemented
|
|
|
|
*/
|
|
|
|
BOOL
|
2008-11-30 11:42:05 +00:00
|
|
|
WINAPI
|
2004-09-23 18:02:19 +00:00
|
|
|
QueryInformationJobObject(HANDLE hJob,
|
|
|
|
JOBOBJECTINFOCLASS JobObjectInformationClass,
|
|
|
|
LPVOID lpJobObjectInformation,
|
|
|
|
DWORD cbJobObjectInformationLength,
|
|
|
|
LPDWORD lpReturnLength)
|
|
|
|
{
|
2010-01-09 20:51:33 +00:00
|
|
|
NTSTATUS Status;
|
2004-09-23 18:02:19 +00:00
|
|
|
|
2010-01-09 20:51:33 +00:00
|
|
|
Status = NtQueryInformationJobObject(hJob,
|
|
|
|
JobObjectInformationClass,
|
|
|
|
lpJobObjectInformation,
|
|
|
|
cbJobObjectInformationLength,
|
|
|
|
lpReturnLength);
|
|
|
|
if (NT_SUCCESS(Status))
|
2004-09-23 18:02:19 +00:00
|
|
|
{
|
2010-01-09 20:51:33 +00:00
|
|
|
PJOBOBJECT_BASIC_LIMIT_INFORMATION BasicInfo;
|
|
|
|
|
|
|
|
switch (JobObjectInformationClass)
|
|
|
|
{
|
|
|
|
case JobObjectBasicLimitInformation:
|
|
|
|
BasicInfo = (PJOBOBJECT_BASIC_LIMIT_INFORMATION)lpJobObjectInformation;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case JobObjectExtendedLimitInformation:
|
|
|
|
BasicInfo = &((PJOBOBJECT_EXTENDED_LIMIT_INFORMATION)lpJobObjectInformation)->BasicLimitInformation;
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
BasicInfo = NULL;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (BasicInfo != NULL)
|
|
|
|
{
|
|
|
|
/* we need to convert the process priority classes in the
|
|
|
|
JOBOBJECT_BASIC_LIMIT_INFORMATION structure the same way as
|
|
|
|
GetPriorityClass() converts it! */
|
|
|
|
switch (BasicInfo->PriorityClass)
|
|
|
|
{
|
|
|
|
case PROCESS_PRIORITY_CLASS_IDLE:
|
|
|
|
BasicInfo->PriorityClass = IDLE_PRIORITY_CLASS;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case PROCESS_PRIORITY_CLASS_BELOW_NORMAL:
|
|
|
|
BasicInfo->PriorityClass = BELOW_NORMAL_PRIORITY_CLASS;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case PROCESS_PRIORITY_CLASS_NORMAL:
|
|
|
|
BasicInfo->PriorityClass = NORMAL_PRIORITY_CLASS;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case PROCESS_PRIORITY_CLASS_ABOVE_NORMAL:
|
|
|
|
BasicInfo->PriorityClass = ABOVE_NORMAL_PRIORITY_CLASS;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case PROCESS_PRIORITY_CLASS_HIGH:
|
|
|
|
BasicInfo->PriorityClass = HIGH_PRIORITY_CLASS;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case PROCESS_PRIORITY_CLASS_REALTIME:
|
|
|
|
BasicInfo->PriorityClass = REALTIME_PRIORITY_CLASS;
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
BasicInfo->PriorityClass = NORMAL_PRIORITY_CLASS;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return TRUE;
|
2004-09-23 18:02:19 +00:00
|
|
|
}
|
|
|
|
|
2010-01-09 20:51:33 +00:00
|
|
|
SetLastErrorByStatus(Status);
|
|
|
|
return FALSE;
|
2004-09-23 18:02:19 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
* @implemented
|
|
|
|
*/
|
|
|
|
BOOL
|
2008-11-30 11:42:05 +00:00
|
|
|
WINAPI
|
2004-09-23 18:02:19 +00:00
|
|
|
SetInformationJobObject(HANDLE hJob,
|
|
|
|
JOBOBJECTINFOCLASS JobObjectInformationClass,
|
|
|
|
LPVOID lpJobObjectInformation,
|
|
|
|
DWORD cbJobObjectInformationLength)
|
|
|
|
{
|
2010-01-09 20:51:33 +00:00
|
|
|
JOBOBJECT_EXTENDED_LIMIT_INFORMATION ExtendedLimitInfo;
|
|
|
|
PJOBOBJECT_BASIC_LIMIT_INFORMATION BasicInfo;
|
|
|
|
PVOID ObjectInfo;
|
|
|
|
NTSTATUS Status;
|
|
|
|
|
|
|
|
switch (JobObjectInformationClass)
|
2004-09-23 18:02:19 +00:00
|
|
|
{
|
2010-01-09 20:51:33 +00:00
|
|
|
case JobObjectBasicLimitInformation:
|
|
|
|
if (cbJobObjectInformationLength != sizeof(JOBOBJECT_BASIC_LIMIT_INFORMATION))
|
|
|
|
{
|
|
|
|
SetLastError(ERROR_BAD_LENGTH);
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
ObjectInfo = &ExtendedLimitInfo.BasicLimitInformation;
|
|
|
|
BasicInfo = (PJOBOBJECT_BASIC_LIMIT_INFORMATION)ObjectInfo;
|
|
|
|
RtlCopyMemory(ObjectInfo, lpJobObjectInformation, cbJobObjectInformationLength);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case JobObjectExtendedLimitInformation:
|
|
|
|
if (cbJobObjectInformationLength != sizeof(JOBOBJECT_EXTENDED_LIMIT_INFORMATION))
|
|
|
|
{
|
|
|
|
SetLastError(ERROR_BAD_LENGTH);
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
ObjectInfo = &ExtendedLimitInfo;
|
|
|
|
BasicInfo = &ExtendedLimitInfo.BasicLimitInformation;
|
|
|
|
RtlCopyMemory(ObjectInfo, lpJobObjectInformation, cbJobObjectInformationLength);
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
ObjectInfo = lpJobObjectInformation;
|
|
|
|
BasicInfo = NULL;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (BasicInfo != NULL)
|
|
|
|
{
|
|
|
|
/* we need to convert the process priority classes in the
|
|
|
|
JOBOBJECT_BASIC_LIMIT_INFORMATION structure the same way as
|
|
|
|
SetPriorityClass() converts it! */
|
|
|
|
switch(BasicInfo->PriorityClass)
|
|
|
|
{
|
|
|
|
case IDLE_PRIORITY_CLASS:
|
|
|
|
BasicInfo->PriorityClass = PROCESS_PRIORITY_CLASS_IDLE;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case BELOW_NORMAL_PRIORITY_CLASS:
|
|
|
|
BasicInfo->PriorityClass = PROCESS_PRIORITY_CLASS_BELOW_NORMAL;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case NORMAL_PRIORITY_CLASS:
|
|
|
|
BasicInfo->PriorityClass = PROCESS_PRIORITY_CLASS_NORMAL;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case ABOVE_NORMAL_PRIORITY_CLASS:
|
|
|
|
BasicInfo->PriorityClass = PROCESS_PRIORITY_CLASS_ABOVE_NORMAL;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case HIGH_PRIORITY_CLASS:
|
|
|
|
BasicInfo->PriorityClass = PROCESS_PRIORITY_CLASS_HIGH;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case REALTIME_PRIORITY_CLASS:
|
|
|
|
BasicInfo->PriorityClass = PROCESS_PRIORITY_CLASS_REALTIME;
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
BasicInfo->PriorityClass = PROCESS_PRIORITY_CLASS_NORMAL;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Status = NtSetInformationJobObject(hJob,
|
|
|
|
JobObjectInformationClass,
|
|
|
|
ObjectInfo,
|
|
|
|
cbJobObjectInformationLength);
|
|
|
|
if (!NT_SUCCESS(Status))
|
|
|
|
{
|
|
|
|
SetLastErrorByStatus(Status);
|
|
|
|
return FALSE;
|
2004-09-23 18:02:19 +00:00
|
|
|
}
|
2005-05-09 01:46:57 +00:00
|
|
|
|
2010-01-09 20:51:33 +00:00
|
|
|
return TRUE;
|
2004-09-23 18:02:19 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
* @implemented
|
|
|
|
*/
|
|
|
|
BOOL
|
2008-11-30 11:42:05 +00:00
|
|
|
WINAPI
|
2004-09-23 18:02:19 +00:00
|
|
|
TerminateJobObject(HANDLE hJob,
|
|
|
|
UINT uExitCode)
|
|
|
|
{
|
2010-01-09 20:51:33 +00:00
|
|
|
NTSTATUS Status;
|
2005-05-09 01:46:57 +00:00
|
|
|
|
2010-01-09 20:51:33 +00:00
|
|
|
Status = NtTerminateJobObject(hJob, uExitCode);
|
|
|
|
if (!NT_SUCCESS(Status))
|
|
|
|
{
|
|
|
|
SetLastErrorByStatus(Status);
|
|
|
|
return FALSE;
|
|
|
|
}
|
2004-09-23 18:02:19 +00:00
|
|
|
|
2010-01-09 20:51:33 +00:00
|
|
|
return TRUE;
|
2004-09-23 18:02:19 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* EOF */
|