/*On each core, tasks are linked in a circular doubly-linked list queue with the Daemon task as the head. On Core0, the queue order represents the front-to-back window stack order with the window mgr as the wallpaper. The scheduler is round-robin. It checks if a task is ready and runs it or skips it. Swapping tasks just involves storing and restoring regs (no disk I/O for virtual memory and no addr map changes). It is always fully identity-mapped on all cores. Tasks can be switched in half a microsecond. The scheduler checks if a task is waiting for a certain time or waiting on a message and skips if not ready. A task runs until it voluntarily yields ctrl with a call to $LK,"Yield",A="MN:Yield"$(). Tasks waiting on I/O often loop, checking for a status and $LK,"Yield",A="MN:Yield"$ing. This does not really degrade performance, but pegs the CPU Load. The scheduler checks for a few keys: kill a task. reboots. Next task. breaks execution of a program. Each core has its own circular task queue. For AP processors, they have a "Daemon" task which stays in a loop waiting for jobs or requests to spawn tasks. See $LK,"CoreAPDaemonTask",A="MN:CoreAPDaemonTask"$(). $HL,1$*/ U0 TaskFocusNext() { CTask *task,*_task=sys_focus_task; sys_focus_task=NULL; if (!_task) _task=zenith_task; task=_task->next_task; do { if (!Bt(&task->win_inhibit,WIf_SELF_FOCUS)) { sys_focus_task=task; CallExtNum(EXT_WIN_TO_TOP,task,TRUE); return; } task=task->next_task; } while (task!=_task); } asm { TASK_CONTEXT_SAVE:: //OUT: RSI=FS PUSH RSI PUSHFD XOR RSI,RSI MOV RSI,FS:CTask.addr[RSI] POP U64 CTask.rflags[RSI] POP U64 CTask.rsi[RSI] MOV U64 CTask.rax[RSI],RAX /*Divert the stack to the Task memory and push onto it and divert it back. It's a little faster. */ MOV RAX,RSP LEA RSP,U64 CTask.r15+8[RSI] PUSH R15 PUSH R14 PUSH R13 PUSH R12 PUSH R11 PUSH R10 PUSH R9 PUSH R8 PUSH RDI PUSH RBP PUSH RBX PUSH RDX PUSH RCX MOV RSP,RAX MOV RAX,U64 CTask.fpu_mmx[RSI] FXSAVE U64 [RAX] MOV RDX,U64 CTask.bpt_list[RSI] @@05: TEST RDX,RDX JZ @@10 MOV RDI,U64 CBpt.addr[RDX] MOV AL,U8 CBpt.val[RDX] MOV U8 [RDI],AL MOV RDX,U64 CBpt.next[RDX] JMP @@05 @@10: RET //************************************ _TASK_CONTEXT_RESTORE:: XOR RAX,RAX INC U64 GS:CCPU.swap_counter[RAX] MOV RSI,FS:CTask.addr[RAX] BT U32 CTask.rflags[RSI],RFLAGf_INT JNC @@05 BTS U32 GS:CCPU.cpu_flags[RAX],CPUf_RAN_A_TASK @@05: BT U64 CTask.task_flags[RSI],TASKf_DISABLE_BPTS JC @@15 MOV RDX,U64 CTask.bpt_list[RSI] @@10: TEST RDX,RDX JZ @@15 MOV RDI,U64 CBpt.addr[RDX] MOV U8 [RDI],OC_BPT MOV RDX,U64 CBpt.next[RDX] JMP @@10 @@15: INC U64 CTask.swap_counter[RSI] MOV RAX,U64 CTask.fpu_mmx[RSI] FXRSTOR U64 [RAX] MOV RAX,RSP LEA RSP,U64 CTask.rcx[RSI] POP RCX POP RDX POP RBX POP RBP POP RDI POP R8 POP R9 POP R10 POP R11 POP R12 POP R13 POP R14 POP R15 MOV RSP,RAX MOV RAX,U64 CTask.rax[RSI] PUSH CGDT.ds PUSH U64 CTask.rsp[RSI] PUSH U64 CTask.rflags[RSI] PUSH CGDT.cs64 PUSH U64 CTask.rip[RSI] MOV RSI,U64 CTask.rsi[RSI] IRET //************************************ END_RSI_TASK: MOV RAX,RSI CALL SET_FS_BASE _TASK_END_NOW:: CALL &TaskEnd MOV RSI,RAX CALL SET_FS_BASE JMP I8 RESTORE_RSI_TASK _YIELD:: PUSHFD TEST U8 [SYS_SEMAS+SEMA_SINGLE_USER*DEFAULT_CACHE_LINE_WIDTH],1 JZ @@05 POPFD //If single user, don't change task. RET @@05: CLI CALL TASK_CONTEXT_SAVE MOV EBX,U32 _RET MOV U64 CTask.rip[RSI],RBX POP U64 CTask.rflags[RSI] MOV U64 CTask.rsp[RSI],RSP MOV RSI,U64 CTask.next_task[RSI] RESTORE_RSI_TASK: TEST U64 [SYS_CTRL_ALT_FLAGS],1<last_task; last->next_task=pred->last_task=task; task->last_task=last; task->next_task=pred; POPFD } U0 TaskQueueRemove(CTask *task) {//Remove a task from the scheduler running task queue. //Use $LK,"Suspend",A="MN:Suspend"$(). CTask *next,*last; PUSHFD CLI next=task->next_task; last=task->last_task; last->next_task=next; next->last_task=last; POPFD } U0 TaskQueueInsChild(CTask *task) { CTask *last,*pred; PUSHFD CLI pred=task->parent_task->last_child_task; last=pred->last_sibling_task; last->next_sibling_task=pred->last_sibling_task=task; task->last_sibling_task=last; task->next_sibling_task=pred; POPFD }