simplify RemoveProcess

This commit is contained in:
nakst 2021-11-08 21:28:00 +00:00
parent 9f119b5104
commit 455a213349
3 changed files with 41 additions and 103 deletions

View File

@ -1665,24 +1665,9 @@ void MMBalanceThread() {
// Find a process to balance.
KMutexAcquire(&scheduler.allProcessesMutex);
Process *process = nullptr;
while (true) {
if (pmm.nextProcessToBalance) {
process = pmm.nextProcessToBalance;
} else {
process = scheduler.allProcesses.firstItem->thisItem;
}
pmm.nextProcessToBalance = process->allItem.nextItem ? process->allItem.nextItem->thisItem : nullptr;
if (process->handles) {
OpenHandleToObject(process, KERNEL_OBJECT_PROCESS, ES_FLAGS_DEFAULT);
break;
}
}
Process *process = pmm.nextProcessToBalance ?: scheduler.allProcesses.firstItem->thisItem;
pmm.nextProcessToBalance = process->allItem.nextItem ? process->allItem.nextItem->thisItem : nullptr;
OpenHandleToObject(process, KERNEL_OBJECT_PROCESS, ES_FLAGS_DEFAULT);
KMutexRelease(&scheduler.allProcessesMutex);
// For every memory region...

View File

@ -4,8 +4,6 @@
#define THREAD_PRIORITY_LOW (1)
#define THREAD_PRIORITY_COUNT (2)
void CloseHandleToProcess(void *_thread);
enum ThreadState : int8_t {
THREAD_ACTIVE, // An active thread. Not necessarily executing; `executing` determines if it executing.
THREAD_WAITING_MUTEX, // Waiting for a mutex to be released.
@ -136,7 +134,6 @@ struct Process {
bool preventNewThreads; // Set by TerminateProcess.
int exitStatus; // TODO Remove this.
KEvent killedEvent;
KAsyncTask removeAsyncTask;
// Executable state:
uint8_t executableState;
@ -194,14 +191,13 @@ struct Scheduler {
// - TerminateThread is called on each thread, leading to an eventual call to KillProcess.
// - This is optional because KillProcess is called if all threads get terminated naturally; in this case, preventNewThreads is never set.
// 2. KillProcess
// - Destroys the handle table and memory space, and sets killedEvent.
// - Removes the process from the lists, destroys the handle table and memory space, and sets killedEvent.
// - Sends a message to Desktop informing it the process was killed.
// - Since KillProcess is only called from KillThread, there is an associated closing of a process handle from the killed thread.
// 3. CloseHandleToObject KERNEL_OBJECT_PROCESS (CloseHandleToProcess)
// - If the last handle to the process has been closed, then RemoveProcess is called on an async task.
// 3. CloseHandleToObject KERNEL_OBJECT_PROCESS
// - If the last handle to the process has been closed, then RemoveProcess is called.
// 4. RemoveProcess
// - Removes the process from the lists, destroys its message queue, and closes its handle to its executable node.
// (These tasks are done here because processes that are created but never started will not reach KillProcess.)
// - Destroys the message queue, and closes the handle to the executable node.
// - The process and memory space structures are deallocated.
void TerminateProcess(Process *process, int status);
@ -499,42 +495,33 @@ Thread *Scheduler::SpawnThread(const char *cName, uintptr_t startAddress, uintpt
return nullptr;
}
void _RemoveProcess(KAsyncTask *task) {
Process *process = EsContainerOf(Process, removeAsyncTask, task);
scheduler.SetTemporaryAddressSpace(process->vmm);
scheduler.RemoveProcess(process);
}
void CloseHandleToProcess(void *_process) {
KSpinlockAcquire(&scheduler.lock);
Process *process = (Process *) _process;
uintptr_t previous = __sync_fetch_and_sub(&process->handles, 1);
if (!previous) KernelPanic("CloseHandleToProcess - All handles to process %x have been closed.\n", process);
bool removeProcess = previous == 1;
KernelLog(LOG_VERBOSE, "Scheduler", "close process handle", "Closed handle to process %d; %d handles remain.\n", process->id, process->handles);
if (removeProcess && process->executableStartRequest) {
// This must be done in the correct virtual address space!
KRegisterAsyncTask(&process->removeAsyncTask, _RemoveProcess);
}
KSpinlockRelease(&scheduler.lock);
if (removeProcess && !process->executableStartRequest) {
// The process was never started, so we can't make a RemoveProcess task, because it doesn't have an MMSpace yet.
scheduler.RemoveProcess(process);
}
ProcessorFakeTimerInterrupt(); // Process the asynchronous task.
}
void KillProcess(Process *process) {
// This function should only be called by KillThread, when the last thread in the process exits.
// There should be at least one remaining handle to the process here, owned by that thread.
// It will be closed at the end of the KillThread function.
if (!process->handles) {
KernelPanic("KillProcess - Process %x is on the allProcesses list but there are no handles to it.\n", process);
}
KernelLog(LOG_INFO, "Scheduler", "killing process", "Killing process (%d) %x...\n", process->id, process);
__sync_fetch_and_sub(&scheduler.activeProcessCount, 1);
// Remove the process from the list of processes.
KMutexAcquire(&scheduler.allProcessesMutex);
scheduler.allProcesses.Remove(&process->allItem);
if (pmm.nextProcessToBalance == process) {
// If the balance thread got interrupted while balancing this process,
// start at the beginning of the next process.
pmm.nextProcessToBalance = process->allItem.nextItem ? process->allItem.nextItem->thisItem : nullptr;
pmm.nextRegionToBalance = nullptr;
pmm.balanceResumePosition = 0;
}
KMutexRelease(&scheduler.allProcessesMutex);
KSpinlockAcquire(&scheduler.lock);
process->allThreadsTerminated = true;
@ -877,15 +864,14 @@ bool Process::StartWithNode(KNode *node) {
__sync_fetch_and_add(&scheduler.activeProcessCount, 1);
__sync_fetch_and_add(&scheduler.blockShutdownProcessCount, 1);
// Add the process to the list of all processes.
// Add the process to the list of all processes,
// and spawn the kernel thread to load the executable.
// This is synchronized under allProcessesMutex so that the process can't be terminated or paused
// until newProcessThread has been spawned.
KMutexAcquire(&scheduler.allProcessesMutex);
scheduler.allProcesses.InsertEnd(&allItem);
KMutexRelease(&scheduler.allProcessesMutex);
// Spawn the kernel thread to load the executable.
Thread *newProcessThread = scheduler.SpawnThread("NewProcess", (uintptr_t) NewProcess, 0, ES_FLAGS_DEFAULT, this);
KMutexRelease(&scheduler.allProcessesMutex);
if (!newProcessThread) {
CloseHandleToObject(this, KERNEL_OBJECT_PROCESS);
@ -997,58 +983,26 @@ void Scheduler::CreateProcessorThreads(CPULocalStorage *local) {
void Scheduler::RemoveProcess(Process *process) {
KernelLog(LOG_INFO, "Scheduler", "remove process", "Removing process %d.\n", process->id);
bool started = process->executableStartRequest;
if (started) {
// Make sure that the process cannot be opened.
KMutexAcquire(&allProcessesMutex);
allProcesses.Remove(&process->allItem);
if (pmm.nextProcessToBalance == process) {
// If the balance thread got interrupted while balancing this process,
// start at the beginning of the next process.
pmm.nextProcessToBalance = process->allItem.nextItem ? process->allItem.nextItem->thisItem : nullptr;
pmm.nextRegionToBalance = nullptr;
pmm.balanceResumePosition = 0;
}
KMutexRelease(&allProcessesMutex);
// At this point, no pointers to the process (should) remain (I think).
if (!process->allThreadsTerminated) {
KernelPanic("Scheduler::RemoveProcess - The process is being removed before all its threads have terminated?!\n");
}
if (process->executableNode) {
// Close the handle to the executable node.
CloseHandleToObject(process->executableNode, KERNEL_OBJECT_NODE, ES_FILE_READ);
process->executableNode = nullptr;
}
// Destroy the process's handle table, if it has already been destroyed.
// Destroy the process's handle table, if it hasn't already been destroyed.
// For most processes, the handle table is destroyed when the last thread terminates.
process->handleTable.Destroy();
// Free all the remaining messages in the message queue.
// This is done after closing all handles, since closing handles can generate messages.
process->messageQueue.messages.Free();
// Free the process.
MMSpaceCloseReference(process->vmm);
scheduler.processPool.Remove(process);
if (started) {
// If all processes (except the kernel process) have terminated, set the scheduler's killedEvent.
if (1 == __sync_fetch_and_sub(&scheduler.blockShutdownProcessCount, 1)) {
KEventSet(&scheduler.killedEvent);
}
if (1 == __sync_fetch_and_sub(&scheduler.blockShutdownProcessCount, 1)) {
KEventSet(&scheduler.killedEvent);
}
}
@ -1378,9 +1332,7 @@ Process *Scheduler::OpenProcess(uint64_t id) {
while (item) {
Process *process = item->thisItem;
if (process->id == id
&& process->handles /* if the process has no handles, it's about to be removed */
&& process->type != PROCESS_KERNEL /* the kernel process cannot be opened */) {
if (process->id == id && process->type != PROCESS_KERNEL /* the kernel process cannot be opened */) {
OpenHandleToObject(process, KERNEL_OBJECT_PROCESS, ES_FLAGS_DEFAULT);
result = item->thisItem;
break;

View File

@ -1274,6 +1274,7 @@ SYSCALL_IMPLEMENT(ES_SYSCALL_SYSTEM_TAKE_SNAPSHOT) {
snapshot->processes[index].isKernel = process->type == PROCESS_KERNEL;
snapshot->processes[index].nameBytes = EsCStringLength(process->cExecutableName);
EsMemoryCopy(snapshot->processes[index].name, process->cExecutableName, snapshot->processes[index].nameBytes);
EsAssert(process->handles);
item = item->nextItem, index++;
}