introduce allProcessesMutex

This commit is contained in:
nakst 2021-11-08 20:51:57 +00:00
parent fb14dd87e0
commit 9f119b5104
3 changed files with 32 additions and 48 deletions

View File

@ -1664,7 +1664,7 @@ void MMBalanceThread() {
// Find a process to balance. // Find a process to balance.
KSpinlockAcquire(&scheduler.lock); KMutexAcquire(&scheduler.allProcessesMutex);
Process *process = nullptr; Process *process = nullptr;
@ -1683,7 +1683,7 @@ void MMBalanceThread() {
} }
} }
KSpinlockRelease(&scheduler.lock); KMutexRelease(&scheduler.allProcessesMutex);
// For every memory region... // For every memory region...

View File

@ -235,6 +235,7 @@ struct Scheduler {
KSpinlock lock; // The general lock. TODO Break this up! KSpinlock lock; // The general lock. TODO Break this up!
KMutex allThreadsMutex; // For accessing the allThreads list. KMutex allThreadsMutex; // For accessing the allThreads list.
KMutex allProcessesMutex; // For accessing the allProcesses list.
KSpinlock activeTimersSpinlock; // For accessing the activeTimers lists. KSpinlock activeTimersSpinlock; // For accessing the activeTimers lists.
KEvent killedEvent; // Set during shutdown when all processes have been terminated. KEvent killedEvent; // Set during shutdown when all processes have been terminated.
@ -878,9 +879,9 @@ bool Process::StartWithNode(KNode *node) {
// Add the process to the list of all processes. // Add the process to the list of all processes.
KSpinlockAcquire(&scheduler.lock); KMutexAcquire(&scheduler.allProcessesMutex);
scheduler.allProcesses.InsertEnd(&allItem); scheduler.allProcesses.InsertEnd(&allItem);
KSpinlockRelease(&scheduler.lock); KMutexRelease(&scheduler.allProcessesMutex);
// Spawn the kernel thread to load the executable. // Spawn the kernel thread to load the executable.
@ -1001,7 +1002,7 @@ void Scheduler::RemoveProcess(Process *process) {
if (started) { if (started) {
// Make sure that the process cannot be opened. // Make sure that the process cannot be opened.
KSpinlockAcquire(&lock); KMutexAcquire(&allProcessesMutex);
allProcesses.Remove(&process->allItem); allProcesses.Remove(&process->allItem);
@ -1014,7 +1015,7 @@ void Scheduler::RemoveProcess(Process *process) {
pmm.balanceResumePosition = 0; pmm.balanceResumePosition = 0;
} }
KSpinlockRelease(&lock); KMutexRelease(&allProcessesMutex);
// At this point, no pointers to the process (should) remain (I think). // At this point, no pointers to the process (should) remain (I think).
@ -1352,7 +1353,7 @@ void Scheduler::Shutdown() {
KernelLog(LOG_INFO, "Scheduler", "killing all processes", "Scheduler::Destroy - Killing all processes....\n"); KernelLog(LOG_INFO, "Scheduler", "killing all processes", "Scheduler::Destroy - Killing all processes....\n");
while (true) { while (true) {
KSpinlockAcquire(&lock); KMutexAcquire(&allProcessesMutex);
Process *process = allProcesses.firstItem->thisItem; Process *process = allProcesses.firstItem->thisItem;
while (process && (process->preventNewThreads || process == kernelProcess)) { while (process && (process->preventNewThreads || process == kernelProcess)) {
@ -1360,7 +1361,7 @@ void Scheduler::Shutdown() {
process = item ? item->thisItem : nullptr; process = item ? item->thisItem : nullptr;
} }
KSpinlockRelease(&lock); KMutexRelease(&allProcessesMutex);
if (!process) break; if (!process) break;
TerminateProcess(process, -1); TerminateProcess(process, -1);
@ -1370,9 +1371,9 @@ void Scheduler::Shutdown() {
} }
Process *Scheduler::OpenProcess(uint64_t id) { Process *Scheduler::OpenProcess(uint64_t id) {
KSpinlockAcquire(&scheduler.lock); KMutexAcquire(&allProcessesMutex);
LinkedItem<Process> *item = scheduler.allProcesses.firstItem; LinkedItem<Process> *item = scheduler.allProcesses.firstItem;
Process *result = nullptr;
while (item) { while (item) {
Process *process = item->thisItem; Process *process = item->thisItem;
@ -1381,15 +1382,15 @@ Process *Scheduler::OpenProcess(uint64_t id) {
&& process->handles /* if the process has no handles, it's about to be removed */ && process->handles /* if the process has no handles, it's about to be removed */
&& process->type != PROCESS_KERNEL /* the kernel process cannot be opened */) { && process->type != PROCESS_KERNEL /* the kernel process cannot be opened */) {
OpenHandleToObject(process, KERNEL_OBJECT_PROCESS, ES_FLAGS_DEFAULT); OpenHandleToObject(process, KERNEL_OBJECT_PROCESS, ES_FLAGS_DEFAULT);
result = item->thisItem;
break; break;
} }
item = item->nextItem; item = item->nextItem;
} }
KSpinlockRelease(&scheduler.lock); KMutexRelease(&allProcessesMutex);
return result;
return item ? item->thisItem : nullptr;
} }
bool KThreadCreate(const char *cName, void (*startAddress)(uintptr_t), uintptr_t argument) { bool KThreadCreate(const char *cName, void (*startAddress)(uintptr_t), uintptr_t argument) {

View File

@ -1249,54 +1249,37 @@ SYSCALL_IMPLEMENT(ES_SYSCALL_SYSTEM_TAKE_SNAPSHOT) {
switch (type) { switch (type) {
case ES_SYSTEM_SNAPSHOT_PROCESSES: { case ES_SYSTEM_SNAPSHOT_PROCESSES: {
KSpinlockAcquire(&scheduler.lock); KMutexAcquire(&scheduler.allProcessesMutex);
size_t count = scheduler.allProcesses.count + 8;
bufferSize = sizeof(EsSnapshotProcesses) + sizeof(EsSnapshotProcessesItem) * count; bufferSize = sizeof(EsSnapshotProcesses) + sizeof(EsSnapshotProcessesItem) * scheduler.allProcesses.count;
KSpinlockRelease(&scheduler.lock);
buffer = EsHeapAllocate(bufferSize, true, K_FIXED); buffer = EsHeapAllocate(bufferSize, true, K_FIXED);
if (!buffer) { if (!buffer) {
KMutexRelease(&scheduler.allProcessesMutex);
SYSCALL_RETURN(ES_ERROR_INSUFFICIENT_RESOURCES, false); SYSCALL_RETURN(ES_ERROR_INSUFFICIENT_RESOURCES, false);
} }
EsMemoryZero(buffer, bufferSize);
KSpinlockAcquire(&scheduler.lock);
if (scheduler.allProcesses.count < count) {
count = scheduler.allProcesses.count;
}
EsSnapshotProcesses *snapshot = (EsSnapshotProcesses *) buffer; EsSnapshotProcesses *snapshot = (EsSnapshotProcesses *) buffer;
LinkedItem<Process> *item = scheduler.allProcesses.firstItem; LinkedItem<Process> *item = scheduler.allProcesses.firstItem;
uintptr_t index = 0; uintptr_t index = 0;
while (item && index < count) { while (item) {
Process *process = item->thisItem; Process *process = item->thisItem;
snapshot->processes[index].pid = process->id;
{ snapshot->processes[index].memoryUsage = process->vmm->commit * K_PAGE_SIZE;
snapshot->processes[index].pid = process->id; snapshot->processes[index].cpuTimeSlices = process->cpuTimeSlices;
snapshot->processes[index].memoryUsage = process->vmm->commit * K_PAGE_SIZE; snapshot->processes[index].idleTimeSlices = process->idleTimeSlices;
snapshot->processes[index].cpuTimeSlices = process->cpuTimeSlices; snapshot->processes[index].handleCount = process->handleTable.handleCount;
snapshot->processes[index].idleTimeSlices = process->idleTimeSlices; snapshot->processes[index].threadCount = process->threads.count;
snapshot->processes[index].handleCount = process->handleTable.handleCount; snapshot->processes[index].isKernel = process->type == PROCESS_KERNEL;
snapshot->processes[index].threadCount = process->threads.count; snapshot->processes[index].nameBytes = EsCStringLength(process->cExecutableName);
snapshot->processes[index].isKernel = process->type == PROCESS_KERNEL; EsMemoryCopy(snapshot->processes[index].name, process->cExecutableName, snapshot->processes[index].nameBytes);
item = item->nextItem, index++;
snapshot->processes[index].nameBytes = EsCStringLength(process->cExecutableName);
EsMemoryCopy(snapshot->processes[index].name, process->cExecutableName, snapshot->processes[index].nameBytes);
index++;
}
item = item->nextItem;
} }
snapshot->count = index; EsAssert(index == scheduler.allProcesses.count);
bufferSize = sizeof(EsSnapshotProcesses) + sizeof(EsSnapshotProcessesItem) * index; snapshot->count = scheduler.allProcesses.count;
KSpinlockRelease(&scheduler.lock); KMutexRelease(&scheduler.allProcessesMutex);
} break; } break;
default: { default: {