mirror of https://gitlab.com/nakst/essence
document TerminateThread and TerminateProcess
This commit is contained in:
parent
95e6cc1780
commit
f7894f74a6
|
@ -106,7 +106,6 @@ void RunACPICAEvent(void *e) {
|
||||||
ACPICAEvent *event = (ACPICAEvent *) e;
|
ACPICAEvent *event = (ACPICAEvent *) e;
|
||||||
event->function(event->context);
|
event->function(event->context);
|
||||||
EsHeapFree(event, 0, K_FIXED);
|
EsHeapFree(event, 0, K_FIXED);
|
||||||
scheduler.TerminateThread(GetCurrentThread());
|
|
||||||
}
|
}
|
||||||
|
|
||||||
ES_EXTERN_C ACPI_STATUS AcpiOsExecute(ACPI_EXECUTE_TYPE type, ACPI_OSD_EXEC_CALLBACK function, void *context) {
|
ES_EXTERN_C ACPI_STATUS AcpiOsExecute(ACPI_EXECUTE_TYPE type, ACPI_OSD_EXEC_CALLBACK function, void *context) {
|
||||||
|
|
|
@ -250,7 +250,17 @@ void CloseHandleToObject(void *object, KernelObjectType type, uint32_t flags) {
|
||||||
} break;
|
} break;
|
||||||
|
|
||||||
case KERNEL_OBJECT_THREAD: {
|
case KERNEL_OBJECT_THREAD: {
|
||||||
CloseHandleToThread(object);
|
KSpinlockAcquire(&scheduler.lock);
|
||||||
|
Thread *thread = (Thread *) object;
|
||||||
|
if (!thread->handles) KernelPanic("CloseHandleToThread - All handles to the thread have been closed.\n");
|
||||||
|
thread->handles--;
|
||||||
|
bool removeThread = thread->handles == 0;
|
||||||
|
// EsPrint("Thread %d has %d handles\n", thread->id, thread->handles);
|
||||||
|
KSpinlockRelease(&scheduler.lock);
|
||||||
|
|
||||||
|
if (removeThread) {
|
||||||
|
scheduler.RemoveThread(thread);
|
||||||
|
}
|
||||||
} break;
|
} break;
|
||||||
|
|
||||||
case KERNEL_OBJECT_NODE: {
|
case KERNEL_OBJECT_NODE: {
|
||||||
|
|
|
@ -1,18 +1,15 @@
|
||||||
#ifndef IMPLEMENTATION
|
#ifndef IMPLEMENTATION
|
||||||
|
|
||||||
#define PREEMPT_AFTER_MUTEX_RELEASE
|
|
||||||
|
|
||||||
#define THREAD_PRIORITY_NORMAL (0) // Lower value = higher priority.
|
#define THREAD_PRIORITY_NORMAL (0) // Lower value = higher priority.
|
||||||
#define THREAD_PRIORITY_LOW (1)
|
#define THREAD_PRIORITY_LOW (1)
|
||||||
#define THREAD_PRIORITY_COUNT (2)
|
#define THREAD_PRIORITY_COUNT (2)
|
||||||
|
|
||||||
void CloseHandleToThread(void *_thread);
|
|
||||||
void CloseHandleToProcess(void *_thread);
|
void CloseHandleToProcess(void *_thread);
|
||||||
void KillThread(EsGeneric _thread);
|
void KillThread(EsGeneric _thread);
|
||||||
|
|
||||||
void RegisterAsyncTask(KAsyncTaskCallback callback, EsGeneric argument, struct Process *targetProcess,
|
void RegisterAsyncTask(KAsyncTaskCallback callback, EsGeneric argument, struct Process *targetProcess,
|
||||||
bool needed /*If false, the task may not be registered if there are many queued tasks.*/,
|
bool needed /* if false, the task may not be registered if there are many queued tasks */,
|
||||||
bool unlocked = false /*Set to true if you haven't acquired the scheduler's lock.*/);
|
bool unlocked = false /* set to true if you haven't acquired the scheduler's lock */);
|
||||||
|
|
||||||
enum ThreadState : int8_t {
|
enum ThreadState : int8_t {
|
||||||
THREAD_ACTIVE, // An active thread. Not necessarily executing; `executing` determines if it executing.
|
THREAD_ACTIVE, // An active thread. Not necessarily executing; `executing` determines if it executing.
|
||||||
|
@ -20,7 +17,6 @@ enum ThreadState : int8_t {
|
||||||
THREAD_WAITING_EVENT, // Waiting for a event to be notified.
|
THREAD_WAITING_EVENT, // Waiting for a event to be notified.
|
||||||
THREAD_WAITING_WRITER_LOCK, // Waiting for a writer lock to be notified.
|
THREAD_WAITING_WRITER_LOCK, // Waiting for a writer lock to be notified.
|
||||||
THREAD_TERMINATED, // The thread has been terminated. It will be deallocated when all handles are closed.
|
THREAD_TERMINATED, // The thread has been terminated. It will be deallocated when all handles are closed.
|
||||||
// I believe this is called a "zombie thread" in UNIX terminology.
|
|
||||||
};
|
};
|
||||||
|
|
||||||
enum ThreadType : int8_t {
|
enum ThreadType : int8_t {
|
||||||
|
@ -143,7 +139,7 @@ struct Process {
|
||||||
|
|
||||||
// Termination:
|
// Termination:
|
||||||
bool allThreadsTerminated;
|
bool allThreadsTerminated;
|
||||||
bool terminating;
|
bool terminating; // This never gets set if TerminateProcess is not called, and instead the process is killed because all its threads exit naturally.
|
||||||
int exitStatus; // TODO Remove this.
|
int exitStatus; // TODO Remove this.
|
||||||
KEvent killedEvent;
|
KEvent killedEvent;
|
||||||
|
|
||||||
|
@ -174,14 +170,46 @@ struct Scheduler {
|
||||||
#define SPAWN_THREAD_PAUSED (8)
|
#define SPAWN_THREAD_PAUSED (8)
|
||||||
Thread *SpawnThread(const char *cName, uintptr_t startAddress, uintptr_t argument1 = 0,
|
Thread *SpawnThread(const char *cName, uintptr_t startAddress, uintptr_t argument1 = 0,
|
||||||
uint32_t flags = ES_FLAGS_DEFAULT, Process *process = nullptr, uintptr_t argument2 = 0);
|
uint32_t flags = ES_FLAGS_DEFAULT, Process *process = nullptr, uintptr_t argument2 = 0);
|
||||||
void TerminateThread(Thread *thread, bool lockAlreadyAcquired = false);
|
|
||||||
void PauseThread(Thread *thread, bool resume /*true to resume, false to pause*/, bool lockAlreadyAcquired = false);
|
void PauseThread(Thread *thread, bool resume /*true to resume, false to pause*/, bool lockAlreadyAcquired = false);
|
||||||
|
|
||||||
Process *SpawnProcess(ProcessType processType = PROCESS_NORMAL);
|
Process *SpawnProcess(ProcessType processType = PROCESS_NORMAL);
|
||||||
void TerminateProcess(Process *process, int status);
|
|
||||||
void PauseProcess(Process *process, bool resume);
|
void PauseProcess(Process *process, bool resume);
|
||||||
void CrashProcess(Process *process, EsCrashReason *reason);
|
void CrashProcess(Process *process, EsCrashReason *reason);
|
||||||
|
|
||||||
|
// How thread termination works:
|
||||||
|
// 1. TerminateThread
|
||||||
|
// - terminating is set to true.
|
||||||
|
// - If the thread is executing, then on the next context switch, KillThread is called on a async task.
|
||||||
|
// - If it is not executing, KillThread is called on a async task.
|
||||||
|
// - Note, terminatableState must be set to THREAD_TERMINATABLE.
|
||||||
|
// - When a thread terminates itself, its terminatableState is automatically set to THREAD_TERMINATABLE.
|
||||||
|
// 2. KillThread
|
||||||
|
// - Removes the thread from the lists, frees the stacks, and sets killedEvent.
|
||||||
|
// - The thread's handles to itself and its process are closed.
|
||||||
|
// - If this is the last thread in the process, KillProcess is called.
|
||||||
|
// 3. CloseHandleToObject KERNEL_OBJECT_THREAD
|
||||||
|
// - If the last handle to the thread has been closed, then RemoveThread is called.
|
||||||
|
// 4. RemoveThread
|
||||||
|
// - The thread structure is deallocated.
|
||||||
|
void TerminateThread(Thread *thread, bool lockAlreadyAcquired = false);
|
||||||
|
|
||||||
|
// How process termination works:
|
||||||
|
// 1. TerminateProcess (optional)
|
||||||
|
// - terminating is set to true (to prevent creation of new threads).
|
||||||
|
// - 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, terminating is never set.
|
||||||
|
// 2. KillProcess
|
||||||
|
// - 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.
|
||||||
|
// 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.)
|
||||||
|
// - The process and memory space structures are deallocated.
|
||||||
|
void TerminateProcess(Process *process, int status);
|
||||||
|
|
||||||
Process *OpenProcess(uint64_t id);
|
Process *OpenProcess(uint64_t id);
|
||||||
|
|
||||||
void WaitMutex(KMutex *mutex);
|
void WaitMutex(KMutex *mutex);
|
||||||
|
@ -195,7 +223,7 @@ struct Scheduler {
|
||||||
|
|
||||||
void RemoveProcess(Process *process); // Do not call. Use TerminateProcess/CloseHandleToObject.
|
void RemoveProcess(Process *process); // Do not call. Use TerminateProcess/CloseHandleToObject.
|
||||||
void RemoveThread(Thread *thread); // Do not call. Use TerminateThread/CloseHandleToObject.
|
void RemoveThread(Thread *thread); // Do not call. Use TerminateThread/CloseHandleToObject.
|
||||||
void AddActiveThread(Thread *thread, bool start /*Put it at the start*/); // Add an active thread into the queue.
|
void AddActiveThread(Thread *thread, bool start /* put it at the start of the active list */); // Add an active thread into the queue.
|
||||||
void InsertNewThread(Thread *thread, bool addToActiveList, Process *owner); // Used during thread creation.
|
void InsertNewThread(Thread *thread, bool addToActiveList, Process *owner); // Used during thread creation.
|
||||||
void MaybeUpdateActiveList(Thread *thread); // After changing the priority of a thread, call this to move it to the correct active thread queue if needed.
|
void MaybeUpdateActiveList(Thread *thread); // After changing the priority of a thread, call this to move it to the correct active thread queue if needed.
|
||||||
|
|
||||||
|
@ -957,8 +985,6 @@ void Scheduler::RemoveThread(Thread *thread) {
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
// TODO Deallocate user and kernel stacks.
|
|
||||||
|
|
||||||
scheduler.threadPool.Remove(thread);
|
scheduler.threadPool.Remove(thread);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1108,34 +1134,8 @@ void CloseHandleToProcess(void *_process) {
|
||||||
ProcessorFakeTimerInterrupt(); // Process the asynchronous task.
|
ProcessorFakeTimerInterrupt(); // Process the asynchronous task.
|
||||||
}
|
}
|
||||||
|
|
||||||
void CloseHandleToThread(void *_thread) {
|
void KillProcess(Process *process) {
|
||||||
KSpinlockAcquire(&scheduler.lock);
|
KernelLog(LOG_INFO, "Scheduler", "killing process", "Killing process (%d) %x...\n", process->id, process);
|
||||||
Thread *thread = (Thread *) _thread;
|
|
||||||
if (!thread->handles) KernelPanic("CloseHandleToThread - All handles to the thread have been closed.\n");
|
|
||||||
thread->handles--;
|
|
||||||
bool removeThread = thread->handles == 0;
|
|
||||||
// EsPrint("Thread %d has %d handles\n", thread->id, thread->handles);
|
|
||||||
KSpinlockRelease(&scheduler.lock);
|
|
||||||
|
|
||||||
if (removeThread) {
|
|
||||||
scheduler.RemoveThread(thread);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void KillThread(EsGeneric _thread) {
|
|
||||||
Thread *thread = (Thread *) _thread.p;
|
|
||||||
|
|
||||||
KSpinlockAcquire(&scheduler.lock);
|
|
||||||
scheduler.allThreads.Remove(&thread->allItem);
|
|
||||||
thread->process->threads.Remove(&thread->processItem);
|
|
||||||
|
|
||||||
KernelLog(LOG_INFO, "Scheduler", "killing thread",
|
|
||||||
"Killing thread (ID %d, %d remain in process %d) %x...\n", thread->id, thread->process->threads.count, thread->process->id, _thread);
|
|
||||||
|
|
||||||
if (thread->process->threads.count == 0) {
|
|
||||||
Process *process = thread->process;
|
|
||||||
KernelLog(LOG_INFO, "Scheduler", "killing process",
|
|
||||||
"Killing process (%d) %x...\n", process->id, process);
|
|
||||||
|
|
||||||
process->allThreadsTerminated = true;
|
process->allThreadsTerminated = true;
|
||||||
scheduler.activeProcessCount--;
|
scheduler.activeProcessCount--;
|
||||||
|
@ -1177,6 +1177,20 @@ void KillThread(EsGeneric _thread) {
|
||||||
m.message.crash.pid = process->id;
|
m.message.crash.pid = process->id;
|
||||||
desktopProcess->messageQueue.SendMessage(&m);
|
desktopProcess->messageQueue.SendMessage(&m);
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void KillThread(EsGeneric _thread) {
|
||||||
|
Thread *thread = (Thread *) _thread.p;
|
||||||
|
|
||||||
|
KSpinlockAcquire(&scheduler.lock);
|
||||||
|
scheduler.allThreads.Remove(&thread->allItem);
|
||||||
|
thread->process->threads.Remove(&thread->processItem);
|
||||||
|
|
||||||
|
KernelLog(LOG_INFO, "Scheduler", "killing thread",
|
||||||
|
"Killing thread (ID %d, %d remain in process %d) %x...\n", thread->id, thread->process->threads.count, thread->process->id, _thread);
|
||||||
|
|
||||||
|
if (thread->process->threads.count == 0) {
|
||||||
|
KillProcess(thread->process); // Releases the scheduler's lock.
|
||||||
} else {
|
} else {
|
||||||
KSpinlockRelease(&scheduler.lock);
|
KSpinlockRelease(&scheduler.lock);
|
||||||
}
|
}
|
||||||
|
@ -1186,10 +1200,9 @@ void KillThread(EsGeneric _thread) {
|
||||||
|
|
||||||
KEventSet(&thread->killedEvent);
|
KEventSet(&thread->killedEvent);
|
||||||
|
|
||||||
// Close the handle that this thread owns of its owner process.
|
// Close the handle that this thread owns of its owner process, and the handle it owns of itself.
|
||||||
CloseHandleToObject(thread->process, KERNEL_OBJECT_PROCESS);
|
CloseHandleToObject(thread->process, KERNEL_OBJECT_PROCESS);
|
||||||
|
CloseHandleToObject(thread, KERNEL_OBJECT_THREAD);
|
||||||
CloseHandleToThread(thread);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
Thread *Scheduler::PickThread(CPULocalStorage *local) {
|
Thread *Scheduler::PickThread(CPULocalStorage *local) {
|
||||||
|
|
|
@ -244,9 +244,7 @@ void KMutexRelease(KMutex *mutex) {
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#ifdef PREEMPT_AFTER_MUTEX_RELEASE
|
|
||||||
if (preempt) ProcessorFakeTimerInterrupt();
|
if (preempt) ProcessorFakeTimerInterrupt();
|
||||||
#endif
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void KMutexAssertLocked(KMutex *mutex) {
|
void KMutexAssertLocked(KMutex *mutex) {
|
||||||
|
|
Loading…
Reference in New Issue