scheduler: Try to keep thread on the same logical CPU
Some SMT implementations (e.g. recent AMD microarchitectures) have separate L1d cache for each SMT thread (which AMD decides to call "cores"). This means that we shouldn't move threads to another logical processor too often even if it belongs to the same core. We aren't very strict about this as it would complicate load balancing, but we try to reduce unnecessary migrations.
This commit is contained in:
parent
ad6b9a1df8
commit
c08ed2db65
@ -175,9 +175,8 @@ enqueue(Thread* thread, bool newOne)
|
|||||||
thread);
|
thread);
|
||||||
|
|
||||||
int32 heapPriority = CPUPriorityHeap::GetKey(targetCPU);
|
int32 heapPriority = CPUPriorityHeap::GetKey(targetCPU);
|
||||||
if (threadPriority > atomic_get(&targetCPU->fPriority)
|
if (threadPriority > heapPriority
|
||||||
&& (threadPriority > heapPriority
|
|| (threadPriority == heapPriority && rescheduleNeeded)) {
|
||||||
|| (threadPriority == heapPriority && rescheduleNeeded))) {
|
|
||||||
|
|
||||||
if (targetCPU->fCPUNumber == smp_get_current_cpu())
|
if (targetCPU->fCPUNumber == smp_get_current_cpu())
|
||||||
gCPU[targetCPU->fCPUNumber].invoke_scheduler = true;
|
gCPU[targetCPU->fCPUNumber].invoke_scheduler = true;
|
||||||
@ -237,7 +236,6 @@ scheduler_set_thread_priority(Thread *thread, int32 priority)
|
|||||||
CPUEntry* cpu = &gCPUEntries[thread->cpu->cpu_num];
|
CPUEntry* cpu = &gCPUEntries[thread->cpu->cpu_num];
|
||||||
|
|
||||||
SpinLocker coreLocker(threadData->GetCore()->fCPULock);
|
SpinLocker coreLocker(threadData->GetCore()->fCPULock);
|
||||||
cpu->fPriority = priority;
|
|
||||||
cpu->UpdatePriority(priority);
|
cpu->UpdatePriority(priority);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -475,7 +473,9 @@ reschedule(int32 nextState)
|
|||||||
}
|
}
|
||||||
|
|
||||||
Thread* nextThread = nextThreadData->GetThread();
|
Thread* nextThread = nextThreadData->GetThread();
|
||||||
atomic_set(&cpu->fPriority, nextThreadData->GetEffectivePriority());
|
SpinLocker cpuLocker(core->fCPULock);
|
||||||
|
cpu->UpdatePriority(nextThreadData->GetEffectivePriority());
|
||||||
|
cpuLocker.Unlock();
|
||||||
|
|
||||||
if (nextThread != oldThread) {
|
if (nextThread != oldThread) {
|
||||||
if (enqueueOldThread) {
|
if (enqueueOldThread) {
|
||||||
@ -650,7 +650,6 @@ scheduler_set_cpu_enabled(int32 cpuID, bool enabled)
|
|||||||
if (enabled)
|
if (enabled)
|
||||||
core->fCPUCount++;
|
core->fCPUCount++;
|
||||||
else {
|
else {
|
||||||
cpu->fPriority = B_IDLE_PRIORITY;
|
|
||||||
cpu->UpdatePriority(B_IDLE_PRIORITY);
|
cpu->UpdatePriority(B_IDLE_PRIORITY);
|
||||||
core->fCPUCount--;
|
core->fCPUCount--;
|
||||||
}
|
}
|
||||||
|
@ -42,7 +42,6 @@ ThreadRunQueue::Dump() const
|
|||||||
|
|
||||||
CPUEntry::CPUEntry()
|
CPUEntry::CPUEntry()
|
||||||
:
|
:
|
||||||
fPriority(B_IDLE_PRIORITY),
|
|
||||||
fLoad(0),
|
fLoad(0),
|
||||||
fMeasureActiveTime(0),
|
fMeasureActiveTime(0),
|
||||||
fMeasureTime(0)
|
fMeasureTime(0)
|
||||||
|
@ -54,7 +54,6 @@ struct CPUEntry : public MinMaxHeapLinkImpl<CPUEntry, int32> {
|
|||||||
|
|
||||||
rw_spinlock fSchedulerModeLock;
|
rw_spinlock fSchedulerModeLock;
|
||||||
|
|
||||||
int32 fPriority;
|
|
||||||
ThreadRunQueue fRunQueue;
|
ThreadRunQueue fRunQueue;
|
||||||
|
|
||||||
int32 fLoad;
|
int32 fLoad;
|
||||||
|
@ -121,6 +121,45 @@ ThreadData::ComputeQuantum()
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
inline CoreEntry*
|
||||||
|
ThreadData::_ChooseCore() const
|
||||||
|
{
|
||||||
|
ASSERT(!gSingleCore);
|
||||||
|
return gCurrentMode->choose_core(this);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
inline CPUEntry*
|
||||||
|
ThreadData::_ChooseCPU(CoreEntry* core, bool& rescheduleNeeded) const
|
||||||
|
{
|
||||||
|
int32 threadPriority = GetEffectivePriority();
|
||||||
|
|
||||||
|
if (fThread->previous_cpu != NULL) {
|
||||||
|
CPUEntry* previousCPU = &gCPUEntries[fThread->previous_cpu->cpu_num];
|
||||||
|
if (previousCPU->fCore == core) {
|
||||||
|
SpinLocker cpuLocker(core->fCPULock);
|
||||||
|
if (CPUPriorityHeap::GetKey(previousCPU) < threadPriority) {
|
||||||
|
previousCPU->UpdatePriority(threadPriority);
|
||||||
|
rescheduleNeeded = true;
|
||||||
|
return previousCPU;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
SpinLocker cpuLocker(core->fCPULock);
|
||||||
|
CPUEntry* cpu = core->fCPUHeap.PeekMinimum();
|
||||||
|
ASSERT(cpu != NULL);
|
||||||
|
|
||||||
|
if (CPUPriorityHeap::GetKey(cpu) < threadPriority) {
|
||||||
|
cpu->UpdatePriority(threadPriority);
|
||||||
|
rescheduleNeeded = true;
|
||||||
|
} else
|
||||||
|
rescheduleNeeded = false;
|
||||||
|
|
||||||
|
return cpu;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
inline bigtime_t
|
inline bigtime_t
|
||||||
ThreadData::_GetBaseQuantum() const
|
ThreadData::_GetBaseQuantum() const
|
||||||
{
|
{
|
||||||
|
@ -335,32 +335,6 @@ ThreadData::_GetMinimalPriority() const
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
inline CoreEntry*
|
|
||||||
ThreadData::_ChooseCore() const
|
|
||||||
{
|
|
||||||
ASSERT(!gSingleCore);
|
|
||||||
return gCurrentMode->choose_core(this);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
inline CPUEntry*
|
|
||||||
ThreadData::_ChooseCPU(CoreEntry* core, bool& rescheduleNeeded) const
|
|
||||||
{
|
|
||||||
SpinLocker cpuLocker(core->fCPULock);
|
|
||||||
CPUEntry* cpu = core->fCPUHeap.PeekMinimum();
|
|
||||||
ASSERT(cpu != NULL);
|
|
||||||
|
|
||||||
int32 threadPriority = GetEffectivePriority();
|
|
||||||
if (CPUPriorityHeap::GetKey(cpu) < threadPriority) {
|
|
||||||
cpu->UpdatePriority(threadPriority);
|
|
||||||
rescheduleNeeded = true;
|
|
||||||
} else
|
|
||||||
rescheduleNeeded = false;
|
|
||||||
|
|
||||||
return cpu;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
} // namespace Scheduler
|
} // namespace Scheduler
|
||||||
|
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user