- Thread Support cleanup 1: Separate kthread.c into thrdobj.c (Exported and globally accesible functions dealing with the KTHREAD object) and thrdschd.c (Fully internal thread scheduling routines for Ke* only).

- Also fix KeSetAffinityThread to return the old affinity, instead of NTSTATUS and move NtYieldExecution from Ps to Ke, and move NtDelayExecution to wait.c
- No code changes.

svn path=/trunk/; revision=24024
This commit is contained in:
Alex Ionescu 2006-09-10 15:23:20 +00:00
parent f9b0ee1a03
commit aafcd436d5
8 changed files with 1672 additions and 1630 deletions

View file

@ -73,7 +73,7 @@ KeSetEventBoostPriority(
IN PKTHREAD *Thread OPTIONAL
);
NTSTATUS
KAFFINITY
NTAPI
KeSetAffinityThread(
PKTHREAD Thread,

View file

@ -126,7 +126,7 @@ extern LIST_ENTRY KiStackInSwapListHead;
extern KEVENT KiSwapEvent;
extern PKPRCB KiProcessorBlock[];
extern ULONG KiMask32Array[MAXIMUM_PRIORITY];
extern ULONG IdleProcessorMask;
extern ULONG KiIdleSummary;
extern VOID KiTrap8(VOID);
extern VOID KiTrap2(VOID);
@ -242,7 +242,8 @@ KAFFINITY
NTAPI
KiSetAffinityThread(
IN PKTHREAD Thread,
IN KAFFINITY Affinity
IN KAFFINITY Affinity,
IN PBOOLEAN Released // hack
);
PKTHREAD

View file

@ -337,7 +337,7 @@ KeInsertQueueDpc(IN PKDPC Dpc,
if (((Dpc->Importance == HighImportance) ||
(DpcData->DpcQueueDepth >=
Prcb->MaximumDpcQueueDepth)) &&
(!(AFFINITY_MASK(Cpu) & IdleProcessorMask) ||
(!(AFFINITY_MASK(Cpu) & KiIdleSummary) ||
(Prcb->Sleeping)))
{
/* Set interrupt requested */

File diff suppressed because it is too large Load diff

View file

@ -0,0 +1,430 @@
/*
* PROJECT: ReactOS Kernel
* LICENSE: GPL - See COPYING in the top level directory
* FILE: ntoskrnl/ke/thrdschd.c
* PURPOSE: Kernel Thread Scheduler (Affinity, Priority, Scheduling)
* PROGRAMMERS: Alex Ionescu (alex.ionescu@reactos.org)
*/
/* INCLUDES ******************************************************************/
#include <ntoskrnl.h>
#define NDEBUG
#include <debug.h>
/* GLOBALS *******************************************************************/
LIST_ENTRY PriorityListHead[MAXIMUM_PRIORITY];
ULONG PriorityListMask = 0;
ULONG KiIdleSummary;
ULONG KiIdleSMTSummary;
/* FUNCTIONS *****************************************************************/
static
VOID
KiRequestReschedule(CCHAR Processor)
{
PKPCR Pcr;
Pcr = (PKPCR)(KPCR_BASE + Processor * PAGE_SIZE);
Pcr->Prcb->QuantumEnd = TRUE;
KiIpiSendRequest(1 << Processor, IPI_DPC);
}
static
VOID
KiInsertIntoThreadList(KPRIORITY Priority,
PKTHREAD Thread)
{
ASSERT(Ready == Thread->State);
ASSERT(Thread->Priority == Priority);
if (Priority >= MAXIMUM_PRIORITY || Priority < LOW_PRIORITY) {
DPRINT1("Invalid thread priority (%d)\n", Priority);
KEBUGCHECK(0);
}
InsertTailList(&PriorityListHead[Priority], &Thread->WaitListEntry);
PriorityListMask |= (1 << Priority);
}
static
VOID
KiRemoveFromThreadList(PKTHREAD Thread)
{
ASSERT(Ready == Thread->State);
RemoveEntryList(&Thread->WaitListEntry);
if (IsListEmpty(&PriorityListHead[(ULONG)Thread->Priority])) {
PriorityListMask &= ~(1 << Thread->Priority);
}
}
static
PKTHREAD
KiScanThreadList(KPRIORITY Priority,
KAFFINITY Affinity)
{
PKTHREAD current;
ULONG Mask;
Mask = (1 << Priority);
if (PriorityListMask & Mask) {
LIST_FOR_EACH(current, &PriorityListHead[Priority], KTHREAD, WaitListEntry) {
if (current->State != Ready) {
DPRINT1("%p/%d\n", current, current->State);
}
ASSERT(current->State == Ready);
if (current->Affinity & Affinity) {
KiRemoveFromThreadList(current);
return(current);
}
}
}
return(NULL);
}
BOOLEAN
STDCALL
KiDispatchThreadNoLock(ULONG NewThreadStatus)
{
KPRIORITY CurrentPriority;
PKTHREAD Candidate;
ULONG Affinity;
PKTHREAD CurrentThread = KeGetCurrentThread();
BOOLEAN ApcState;
DPRINT("KiDispatchThreadNoLock() %d/%d/%d/%d\n", KeGetCurrentProcessorNumber(),
CurrentThread, NewThreadStatus, CurrentThread->State);
CurrentThread->State = (UCHAR)NewThreadStatus;
if (NewThreadStatus == Ready) {
KiInsertIntoThreadList(CurrentThread->Priority,
CurrentThread);
}
Affinity = 1 << KeGetCurrentProcessorNumber();
for (CurrentPriority = HIGH_PRIORITY; CurrentPriority >= LOW_PRIORITY; CurrentPriority--) {
Candidate = KiScanThreadList(CurrentPriority, Affinity);
if (Candidate == CurrentThread) {
Candidate->State = Running;
KeReleaseDispatcherDatabaseLockFromDpcLevel();
return FALSE;
}
if (Candidate != NULL) {
PKTHREAD OldThread;
PKTHREAD IdleThread;
DPRINT("Scheduling %x(%d)\n",Candidate, CurrentPriority);
Candidate->State = Running;
OldThread = CurrentThread;
CurrentThread = Candidate;
IdleThread = KeGetCurrentPrcb()->IdleThread;
if (OldThread == IdleThread) {
KiIdleSummary &= ~Affinity;
} else if (CurrentThread == IdleThread) {
KiIdleSummary |= Affinity;
}
MmUpdatePageDir((PEPROCESS)PsGetCurrentProcess(),((PETHREAD)CurrentThread)->ThreadsProcess, sizeof(EPROCESS));
/* Special note for Filip: This will release the Dispatcher DB Lock ;-) -- Alex */
DPRINT("You are : %x, swapping to: %x.\n", OldThread, CurrentThread);
ApcState = KiSwapContext(OldThread, CurrentThread);
DPRINT("You are : %x, swapped from: %x\n", OldThread, CurrentThread);
return ApcState;
}
}
DPRINT1("CRITICAL: No threads are ready (CPU%d)\n", KeGetCurrentProcessorNumber());
KEBUGCHECK(0);
return FALSE;
}
NTSTATUS
FASTCALL
KiSwapThread(IN PKTHREAD CurrentThread,
IN PKPRCB Prcb)
{
BOOLEAN ApcState;
/* Find a new thread to run */
DPRINT("Dispatching Thread as blocked\n");
ApcState = KiDispatchThreadNoLock(Waiting);
/* Check if we need to deliver APCs */
if (ApcState)
{
/* Lower to APC_LEVEL */
KeLowerIrql(APC_LEVEL);
/* Deliver APCs */
KiDeliverApc(KernelMode, NULL, NULL);
ASSERT(CurrentThread->WaitIrql == 0);
}
/* Lower IRQL back to what it was */
KfLowerIrql(CurrentThread->WaitIrql);
/* Return the wait status */
return CurrentThread->WaitStatus;
}
VOID
STDCALL
KiDispatchThread(ULONG NewThreadStatus)
{
KIRQL OldIrql;
if (KeGetCurrentPrcb()->IdleThread == NULL) {
return;
}
OldIrql = KeAcquireDispatcherDatabaseLock();
KiDispatchThreadNoLock(NewThreadStatus);
KeLowerIrql(OldIrql);
}
VOID
NTAPI
KiReadyThread(IN PKTHREAD Thread)
{
/* Makes a thread ready */
Thread->State = Ready;
KiInsertIntoThreadList(Thread->Priority, Thread);
}
VOID
STDCALL
KiAdjustQuantumThread(IN PKTHREAD Thread)
{
KPRIORITY Priority;
/* Don't adjust for RT threads */
if ((Thread->Priority < LOW_REALTIME_PRIORITY) &&
Thread->BasePriority < LOW_REALTIME_PRIORITY - 2)
{
/* Decrease Quantum by one and see if we've ran out */
if (--Thread->Quantum <= 0)
{
/* Return quantum */
Thread->Quantum = Thread->QuantumReset;
/* Calculate new Priority */
Priority = Thread->Priority - (Thread->PriorityDecrement + 1);
/* Normalize it if we've gone too low */
if (Priority < Thread->BasePriority) Priority = Thread->BasePriority;
/* Reset the priority decrement, we've done it */
Thread->PriorityDecrement = 0;
/* Set the new priority, if needed */
if (Priority != Thread->Priority)
{
/*
* FIXME: This should be a call to KiSetPriorityThread but
* due to the current ""scheduler"" in ROS, it can't be done
* cleanly since it actualyl dispatches threads instead.
*/
Thread->Priority = Priority;
}
else
{
/* FIXME: Priority hasn't changed, find a new thread */
}
}
}
/* Nothing to do... */
return;
}
VOID
STDCALL
KiSetPriorityThread(PKTHREAD Thread,
KPRIORITY Priority,
PBOOLEAN Released)
{
KPRIORITY OldPriority = Thread->Priority;
ULONG Mask;
int i;
PKPCR Pcr;
DPRINT("Changing prio to : %lx\n", Priority);
/* Check if priority changed */
if (OldPriority != Priority)
{
/* Set it */
Thread->Priority = Priority;
/* Choose action based on thread's state */
if (Thread->State == Ready)
{
/* Remove it from the current queue */
KiRemoveFromThreadList(Thread);
/* Re-insert it at its current priority */
KiInsertIntoThreadList(Priority, Thread);
/* Check if the old priority was lower */
if (KeGetCurrentThread()->Priority < Priority)
{
/* Dispatch it immediately */
KiDispatchThreadNoLock(Ready);
*Released = TRUE;
return;
}
}
else if (Thread->State == Running)
{
/* Check if the new priority is lower */
if (Priority < OldPriority)
{
/* Check for threads with a higher priority */
Mask = ~((1 << (Priority + 1)) - 1);
if (PriorityListMask & Mask)
{
/* Found a thread, is it us? */
if (Thread == KeGetCurrentThread())
{
/* Dispatch us */
KiDispatchThreadNoLock(Ready);
*Released = TRUE;
return;
}
else
{
/* Loop every CPU */
for (i = 0; i < KeNumberProcessors; i++)
{
/* Get the PCR for this CPU */
Pcr = (PKPCR)(KPCR_BASE + i * PAGE_SIZE);
/* Reschedule if the new one is already on a CPU */
if (Pcr->Prcb->CurrentThread == Thread)
{
KeReleaseDispatcherDatabaseLockFromDpcLevel();
KiRequestReschedule(i);
*Released = TRUE;
return;
}
}
}
}
}
}
}
/* Return to caller */
return;
}
KAFFINITY
NTAPI
KiSetAffinityThread(IN PKTHREAD Thread,
IN KAFFINITY Affinity,
PBOOLEAN Released)
{
KAFFINITY OldAffinity;
ULONG ProcessorMask;
ULONG i;
PKPCR Pcr;
/* Make sure that the affinity is valid */
if (((Affinity & Thread->ApcState.Process->Affinity) != (Affinity)) ||
(!Affinity))
{
/* Bugcheck the system */
KeBugCheck(INVALID_AFFINITY_SET);
}
/* Get the old affinity */
OldAffinity = Thread->UserAffinity;
Thread->UserAffinity = Affinity;
if (Thread->SystemAffinityActive == FALSE) {
Thread->Affinity = Affinity;
if (Thread->State == Running) {
ProcessorMask = 1 << KeGetCurrentProcessorNumber();
if (Thread == KeGetCurrentThread()) {
if (!(Affinity & ProcessorMask)) {
KiDispatchThreadNoLock(Ready);
*Released = TRUE;
return OldAffinity;
}
} else {
for (i = 0; i < KeNumberProcessors; i++) {
Pcr = (PKPCR)(KPCR_BASE + i * PAGE_SIZE);
if (Pcr->Prcb->CurrentThread == Thread) {
if (!(Affinity & ProcessorMask)) {
KeReleaseDispatcherDatabaseLockFromDpcLevel();
KiRequestReschedule(i);
*Released = TRUE;
return OldAffinity;
}
break;
}
}
ASSERT (i < KeNumberProcessors);
}
}
}
*Released = FALSE;
return OldAffinity;
}
/*
* @implemented
*/
NTSTATUS
NTAPI
NtYieldExecution(VOID)
{
//
// TODO (nothing too hard, just want to test out other code)
//
//DPRINT1("NO YIELD PERFORMED! If you see this, contact Alex\n");
//return STATUS_NO_YIELD_PERFORMED;
KiDispatchThread(Ready);
return STATUS_SUCCESS;
}

View file

@ -861,4 +861,41 @@ DontWait:
return WaitStatus;
}
NTSTATUS
NTAPI
NtDelayExecution(IN BOOLEAN Alertable,
IN PLARGE_INTEGER DelayInterval)
{
KPROCESSOR_MODE PreviousMode = ExGetPreviousMode();
LARGE_INTEGER SafeInterval;
NTSTATUS Status = STATUS_SUCCESS;
/* Check if parameters are valid */
if(PreviousMode != KernelMode)
{
_SEH_TRY
{
/* make a copy on the kernel stack and let DelayInterval point to it so
we don't need to wrap KeDelayExecutionThread in SEH! */
SafeInterval = ProbeForReadLargeInteger(DelayInterval);
DelayInterval = &SafeInterval;
}
_SEH_HANDLE
{
Status = _SEH_GetExceptionCode();
}
_SEH_END;
if (!NT_SUCCESS(Status)) return Status;
}
/* Call the Kernel Function */
Status = KeDelayExecutionThread(PreviousMode,
Alertable,
DelayInterval);
/* Return Status */
return Status;
}
/* EOF */

View file

@ -57,13 +57,14 @@
<file>ipi.c</file>
<file>kqueue.c</file>
<file>krnlinit.c</file>
<file>kthread.c</file>
<file>mutex.c</file>
<file>process.c</file>
<file>profile.c</file>
<file>queue.c</file>
<file>sem.c</file>
<file>spinlock.c</file>
<file>thrdschd.c</file>
<file>thrdobj.c</file>
<file>timer.c</file>
<file>usercall.c</file>
<file>wait.c</file>

View file

@ -1077,15 +1077,4 @@ NtOpenThread(OUT PHANDLE ThreadHandle,
return Status;
}
/*
* @implemented
*/
NTSTATUS
NTAPI
NtYieldExecution(VOID)
{
KiDispatchThread(Ready);
return STATUS_SUCCESS;
}
/* EOF */