;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; RDOS operating system ; Copyright (C) 1988-2000, Leif Ekblad ; ; This program is free software; you can redistribute it and/or modify ; it under the terms of the GNU General Public License as published by ; the Free Software Foundation; either version 2 of the License, or ; (at your option) any later version. The only exception to this rule ; is for commercial usage in embedded systems. For information on ; usage in commercial embedded systems, contact embedded@rdos.net ; ; This program is distributed in the hope that it will be useful, ; but WITHOUT ANY WARRANTY; without even the implied warranty of ; MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the ; GNU General Public License for more details. ; ; You should have received a copy of the GNU General Public License ; along with this program; if not, write to the Free Software ; Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA ; ; The author of this program may be contacted at leif@rdos.net ; ; TASK.ASM ; Scheduling and thread handling module ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; INCLUDE ..\driver.def INCLUDE port.def INCLUDE protseg.def INCLUDE ..\user.def INCLUDE ..\os.def INCLUDE ..\os.inc INCLUDE system.def INCLUDE system.inc INCLUDE ..\user.inc INCLUDE ..\pcdev\apic.inc INCLUDE proc.inc INCLUDE ..\handle.inc INCLUDE ..\apicheck.inc include ..\wait.inc include gate.def MSR_SYSENTER_CS = 174h MSR_SYSENTER_ESP = 175h MSR_SYSENTER_EIP = 176h MAX_CORES = 64 SLEEP_TYPE_WAIT = 1 SLEEP_TYPE_SIGNAL = 2 SLEEP_TYPE_FUTEX = 3 SLEEP_TYPE_SECTION = 4 SLEEP_TYPE_DEBUG = 5 SLEEP_TYPE_SUSPEND = 6 section_handle_seg STRUC us_base handle_header <> us_value DW ? us_list DW ? us_owner DW ? us_count DW ? us_lock DW ? section_handle_seg ENDS futex_handle_seg STRUC fh_base handle_header <> fh_list DW ? fh_lock DW ? futex_handle_seg ENDS proc_handle_seg STRUC ph_base handle_header <> ph_lib_sel DW ? ph_proc_sel DW ? proc_handle_seg ENDS proc_end_wait_header STRUC pew_obj wait_obj_header <> pew_proc_sel DW ? proc_end_wait_header ENDS process_callback_seg STRUC cm_mode DW ? cm_stack DD ? cm_process DW ? cm_cs DW ? cm_eip DD ? cm_flags DW ? cm_eax DD ? cm_ebx DD ? cm_ecx DD ? cm_edx DD ? cm_esi DD ? cm_edi DD ? cm_ebp DD ? process_callback_seg ENDS cr_seg EQU 28 cr_offs EQU 24 cr_prio EQU 22 cr_stack EQU 18 cr_mode EQU 16 cr_name EQU 10 cr_cs EQU 8 cr_eip EQU 4 cr_ebp EQU 0 cr_flags EQU -2 cr_ds EQU -4 cr_es EQU -6 cr_fs EQU -8 cr_gs EQU -10 cr_eax EQU -14 cr_ebx EQU -18 cr_ecx EQU -22 cr_edx EQU -26 cr_esi EQU -30 cr_edi EQU -34 data SEGMENT byte public 'DATA' term_thread_list DW ? term_proc_list DW ? list_lock DW ? sys_lsb_tics_base DD ? sys_msb_tics_base DD ? next_pid DW ? futex_section section_typ <> patch_sel DW ? timer_spinlock DW ? timer_head DW ? timer_free DW ? timer_entries DB 256 * SIZE timer_struc DUP(?) data ENDS IFDEF __WASM__ .686p .xmm2 ELSE .386p ENDIF code SEGMENT byte public use16 'CODE' assume cs:code ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: Procedure addresses for SMP/non-SMP variants ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; time_diff DD 0,0 update_tics DD 0 system_thread DW 0 lock_list_proc DW OFFSET LockListSingle unlock_list_proc DW OFFSET UnlockListSingle insert_wakeup_proc DW OFFSET InsertWakeupSingle remove_wakeup_proc DW OFFSET RemoveWakeupSingle lock_signal_proc DW OFFSET LockSignalSingle unlock_signal_proc DW OFFSET UnlockSignalSingle lock_kernel_section_proc DW OFFSET LockKernelSectionSingle unlock_kernel_section_proc DW OFFSET UnlockKernelSectionSingle lock_user_section_proc DW OFFSET LockUserSectionSingle unlock_user_section_proc DW OFFSET UnlockUserSectionSingle lock_futex_proc DW OFFSET LockFutexSingle unlock_futex_proc DW OFFSET UnlockFutexSingle flush_tlb_proc DW OFFSET FlushTlb386 update_timer_proc DW OFFSET UpdateCombinedTimer preempt_reload_proc DW OFFSET TimerPreemptReload fpu_exception_proc DW OFFSET FpuExceptionSingle fpu_save_proc DW OFFSET FpuSaveSingle core_count DW 0 core_arr DW MAX_CORES DUP(0) ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: AddLog ; ; DESCRIPTION: Add log entry ; ; PARAMETERS: AX Type ; BX Proc ; EDX Data ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; add_log_name DB 'Add Schedule Log',0 add_log Proc far push es push fs push si ; mov si,core_data_sel mov fs,si mov fs,fs:ps_sel ; mov si,fs:ps_log_sel or si,si jnz alDo ; push eax mov eax,PROC_LOG_ENTRIES SHL 4 AllocateGlobalMem mov fs:ps_log_sel,es mov fs:ps_log_entry,0 mov fs:ps_log_count,0 mov si,es pop eax alDo: mov es,si mov si,fs:ps_log_entry shl si,4 mov es:[si].pls_type,ax mov es:[si].pls_data,edx mov es:[si].pls_proc,bx GetSystemTime mov es:[si].pls_time,eax mov es:[si].pls_time+4,edx ; shr si,4 inc si cmp si,PROC_LOG_ENTRIES jb alSavePos ; xor si,si alSavePos: mov fs:ps_log_entry,si mov si,fs:ps_log_count cmp si,PROC_LOG_ENTRIES je alDone ; inc si mov fs:ps_log_count,si alDone: pop si pop fs pop es retf32 add_log Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: LockTimerGlobal ; ; DESCRIPTION: Lock timer struc, global version ; ; PARAMETERS: DS Task sel ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; LockTimerGlobal Proc near push ax ltigSpinLock: mov ax,ds:timer_spinlock or ax,ax je ltigGet ; sti pause jmp ltigSpinLock ltigGet: cli inc ax xchg ax,ds:timer_spinlock or ax,ax jne ltigSpinLock ; pop ax ret LockTimerGlobal Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: UnlockTimerGlobal ; ; DESCRIPTION: Unlock timer, global version ; ; PARAMETERS: DS task sel ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; UnlockTimerGlobal Proc near mov ds:timer_spinlock,0 sti ret UnlockTimerGlobal Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: LockTimerCore ; ; DESCRIPTION: Lock timer struc, per core version ; ; PARAMETERS: FS Core data sel ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; LockTimerCore Proc near push ax lticSpinLock: mov ax,fs:ps_timer_spinlock or ax,ax je lticGet ; sti pause jmp lticSpinLock lticGet: cli inc ax xchg ax,fs:ps_timer_spinlock or ax,ax jne lticSpinLock ; pop ax ret LockTimerCore Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: UnlockTimerCore ; ; DESCRIPTION: Unlock timer, per core version ; ; PARAMETERS: FS Core data sel ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; UnlockTimerCore Proc near mov fs:ps_timer_spinlock,0 sti ret UnlockTimerCore Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: LocalRemoveTimerGlobal ; ; DESCRIPTION: Remove timer, global version ; ; PARAMETERS: DS Task sel ; FS Core sel ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; LocalRemoveTimerGlobal Proc near mov bx,ds:timer_head mov ax,ds:[bx].timer_next mov ds:timer_head,ax ; push ds push es push fs ; xor eax,eax mov ax,cs push eax mov ax,OFFSET timer_global_return push eax mov ax,ds:[bx].timer_sel push eax push ds:[bx].timer_offset ; mov ax,ds:timer_free mov ds:[bx].timer_next,ax mov ds:timer_free,bx ; mov ecx,ds:[bx].timer_id mov eax,ds:[bx].timer_lsb mov edx,ds:[bx].timer_msb call UnlockTimerGlobal ; xor bx,bx mov ds,bx mov es,bx retf32 timer_global_return: pop fs pop es pop ds ret LocalRemoveTimerGlobal Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: LocalRemoveTimerCore ; ; DESCRIPTION: Remove timer, per core version ; ; PARAMETERS: DS Task sel ; FS Core sel ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; LocalRemoveTimerCore Proc near mov bx,fs:ps_timer_head mov ax,fs:[bx].timer_next mov fs:ps_timer_head,ax ; push es push fs ; xor eax,eax mov ax,cs push eax mov ax,OFFSET timer_core_return push eax mov ax,fs:[bx].timer_sel push eax push fs:[bx].timer_offset ; mov ax,fs:ps_timer_free mov fs:[bx].timer_next,ax mov fs:ps_timer_free,bx ; mov ecx,fs:[bx].timer_id mov eax,fs:[bx].timer_lsb mov edx,fs:[bx].timer_msb call UnlockTimerCore ; xor bx,bx mov ds,bx mov es,bx retf32 timer_core_return: pop fs pop es ret LocalRemoveTimerCore Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: InsertCoreBlock ; ; DESCRIPTION: Insert thread into core list ; ; PARAMETERS: FS:DI List ; ES Thread ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; InsertCoreBlock Proc near mov es:p_sleep_sel,fs mov word ptr es:p_sleep_offset,di mov word ptr es:p_sleep_offset+2,0 push di mov di,fs:[di] or di,di je icbEmpty ; push ds push si mov ds,di mov si,ds:p_prev mov ds:p_prev,es mov ds,si mov ds:p_next,es mov es:p_next,di mov es:p_prev,si pop si pop ds pop di jmp icbDone icbEmpty: mov es:p_next,es mov es:p_prev,es pop di mov fs:[di],es icbDone: ret InsertCoreBlock Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: InsertCoreFirst ; ; DESCRIPTION: Insert thread first into core list ; ; PARAMETERS: FS:DI List ; ES Thread ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; InsertCoreFirst Proc near mov es:p_sleep_sel,fs mov word ptr es:p_sleep_offset,di mov word ptr es:p_sleep_offset+2,0 push di mov di,fs:[di] or di,di je icfEmpty ; push ds push si mov ds,di mov si,ds:p_prev mov ds:p_prev,es mov ds,si mov ds:p_next,es mov es:p_next,di mov es:p_prev,si pop si pop ds pop di jmp icfDone icfEmpty: mov es:p_next,es mov es:p_prev,es pop di icfDone: mov fs:[di],es ret InsertCoreFirst Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: InsertBlock ; ; DESCRIPTION: Insert thread into global list ; ; PARAMETERS: DS:DI List ; ES Thread ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; InsertBlock Proc near mov es:p_sleep_sel,ds mov word ptr es:p_sleep_offset,di mov word ptr es:p_sleep_offset+2,0 push di mov di,[di] or di,di je ibEmpty ; push ds push si mov ds,di mov si,ds:p_prev mov ds:p_prev,es mov ds,si mov ds:p_next,es mov es:p_next,di mov es:p_prev,si pop si pop ds pop di jmp ibDone ibEmpty: mov es:p_next,es mov es:p_prev,es pop di mov [di],es ibDone: ret InsertBlock Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: RemoveCoreBlock ; ; DESCRIPTION: Remove thread from core list ; ; PARAMETERS: FS:SI List ; ES Thread ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; RemoveCoreBlock Proc near push si mov es,fs:[si] push di push ds mov di,es:p_next cmp di,fs:[si] mov fs:[si],di mov si,es:p_prev mov ds,di mov ds:p_prev,si mov ds,si mov ds:p_next,di pop ds pop di pop si jne rcbDone ; mov word ptr fs:[si],0 rcbDone: ret RemoveCoreBlock Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: RemoveBlock ; ; DESCRIPTION: Remove thread from global list ; ; PARAMETERS: DS:SI List ; ES Thread ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; RemoveBlock Proc near push si mov es,[si] push di push ds mov di,es:p_next cmp di,[si] mov [si],di mov si,es:p_prev mov ds,di mov ds:p_prev,si mov ds,si mov ds:p_next,di pop ds pop di pop si jne rblDone ; mov word ptr [si],0 rblDone: ret RemoveBlock Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: InsertBlock ; ; DESCRIPTION: Insert thread into global list ; ; PARAMETERS: DS:EDI List ; ES Thread ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; InsertBlock32 Proc near mov es:p_sleep_sel,ds mov es:p_sleep_offset,edi push di mov di,[edi] or di,di je ib32Empty ; push ds push si mov ds,di mov si,ds:p_prev mov ds:p_prev,es mov ds,si mov ds:p_next,es mov es:p_next,di mov es:p_prev,si pop si pop ds pop di jmp ib32Done ib32Empty: mov es:p_next,es mov es:p_prev,es pop di mov [edi],es ib32Done: ret InsertBlock32 Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: RemoveBlock32 ; ; DESCRIPTION: Remove thread from global list ; ; PARAMETERS: DS:ESI List ; ES Thread ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; RemoveBlock32 Proc near push esi mov es,[esi] push di push ds mov di,es:p_next cmp di,[esi] mov [esi],di mov si,es:p_prev mov ds,di mov ds:p_prev,si mov ds,si mov ds:p_next,di pop ds pop di pop esi jne rb32Done mov word ptr [esi],0 rb32Done: ret RemoveBlock32 Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: FpuExceptionSingle ; ; DESCRIPTION: Notification FPU exception, single core ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; FpuExceptionSingle Proc near push fs ; call LockCore mov ax,fs:ps_curr_thread clts ; test fs:ps_flags,PS_FLAG_FPU jz fpu_exc_saved ; mov bx,fs:ps_math_thread cmp ax,bx je fpu_exc_done ; mov ds,bx mov bx,OFFSET p_math_control db 9Bh, 66h, 0DDh, 37h ; 32-bit fsave [bx] fpu_exc_saved: mov ds,ax mov bx,OFFSET p_math_control db 9Bh, 66h, 0DDh, 27h ; 32-bit frstor [bx] ; mov fs:ps_math_thread,ax lock or fs:ps_flags,PS_FLAG_FPU fpu_exc_done: call UnlockCore pop fs ret FpuExceptionSingle Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: FpuExceptionMultiple ; ; DESCRIPTION: Notification FPU exception, multiple core ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; FpuExceptionMultiple Proc near push fs call LockCore ; mov ax,fs:ps_curr_thread clts mov ds,ax mov bx,OFFSET p_math_control db 9Bh, 66h, 0DDh, 27h ; 32-bit frstor [bx] ; mov fs:ps_math_thread,ax lock or fs:ps_flags,PS_FLAG_FPU ; call UnlockCore pop fs ret FpuExceptionMultiple Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: FpuException ; ; DESCRIPTION: Notification FPU exception ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; fpu_exception_name DB 'Fpu Exception',0 fpu_exception Proc far call cs:fpu_exception_proc retf32 fpu_exception Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: FpuSaveSingle ; ; DESCRIPTION: Save FPU on thread switch, single core ; ; PARAMETERS: FS Locked core ; DS Current thread ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; FpuSaveSingle Proc near test fs:ps_flags,PS_FLAG_FPU jz fpu_save_single_done ; push ebx mov ebx,cr0 or bl,8 mov cr0,ebx pop ebx fpu_save_single_done: ret FpuSaveSingle Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: FpuSaveMultiple ; ; DESCRIPTION: Save FPU on thread switch, multiple core ; ; PARAMETERS: FS Locked core ; DS Current thread ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; FpuSaveMultiple Proc near test fs:ps_flags,PS_FLAG_FPU jz fpu_save_mult_done ; push ebx ; mov bx,OFFSET p_math_control clts db 9Bh, 66h, 0DDh, 37h ; 32-bit fsave [bx] ; lock and fs:ps_flags,NOT PS_FLAG_FPU ; mov ebx,cr0 or bl,8 mov cr0,ebx ; pop ebx fpu_save_mult_done: ret FpuSaveMultiple Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: NotifyTimeDrift ; ; DESCRIPTION: Notification of time drift ; ; PARAMETERS: DS System data sel ; EAX Drift in tics ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; notify_time_drift_name DB 'Notify Time Drift',0 MAX_DRIFT = 250000 notify_time_drift Proc far push es push eax push ecx push edx ; mov cx,SEG data mov es,cx mov es,es:patch_sel ; cdq sub es:time_diff,eax sbb es:time_diff+4,edx ntdDone: pop edx pop ecx pop eax pop es retf32 notify_time_drift Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: HandlePreempt ; ; DESCRIPTION: Handle preempt ; ; PARAMETERS: FS Core selector ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; HandlePreempt Proc near test fs:ps_flags,PS_FLAG_PREEMPT jz hpDone ; mov si,fs:ps_prio_act mov ax,fs:[si] or ax,ax jz hpDone ; cmp ax,fs:ps_last_thread jne hpDone ; mov es,ax mov ax,es:p_next mov fs:[si],ax hpDone: ret HandlePreempt Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: HandlePrio ; ; DESCRIPTION: Handle prio ; ; PARAMETERS: FS Core selector ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; HandlePrio Proc near mov si,fs:ps_prio_act mov ax,fs:[si] or ax,ax jnz hpqInt ; or si,si jz hpqDone ; sub si,2 mov fs:ps_prio_act,si jmp HandlePrio hpqInt: mov es,ax mov es,es:p_process_sel test es:ms_virt_flags,200h jnz hpqDone ; cmp ax,es:ms_cli_thread je hpqDone ; mov ax,es mov si,fs:ps_prio_act call RemoveCoreBlock ; call cs:lock_list_proc push ds mov ds,ax mov di,OFFSET ms_wait_sti call InsertBlock pop ds call cs:unlock_list_proc jmp HandlePrio hpqDone: ret HandlePrio Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: GetPrioThread ; ; DESCRIPTION: Get thread from standard list ; ; PARAMETERS: FS Core selector ; ; RETURNS: ES Thread ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; GetPrioThread Proc near mov si,fs:ps_prio_act mov ax,fs:[si] or ax,ax jz gptNull ; call RemoveCoreBlock gptLoop: mov ax,fs:[si] or ax,ax jnz gptDone ; or si,si jz gptDone ; sub si,2 mov fs:ps_prio_act,si jmp gptLoop gptNull: mov es,fs:ps_null_thread gptDone: ret GetPrioThread Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: SetupPreempt ; ; DESCRIPTION: Setup preempt ; ; PARAMETERS: FS Core selector ; ES Thread ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; SetupPreempt Proc near mov ax,es cmp ax,fs:ps_null_thread je spSet ; cmp ax,fs:ps_last_thread jne spSet ; test fs:ps_flags,PS_FLAG_PREEMPT jz spDone spSet: GetSystemTime mov ecx,eax add eax,1193 adc edx,0 mov fs:ps_preempt_lsb,eax mov fs:ps_preempt_msb,edx spDone: lock and fs:ps_flags,NOT PS_FLAG_PREEMPT ret SetupPreempt Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: GetNextThread ; ; DESCRIPTION: Get next thread to run ; ; PARAMETERS: FS Core selector ; DS Task sel ; ; RETURNS: ES Thread to run next ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; GetNextThread Proc near sti lock and fs:ps_flags,NOT PS_FLAG_PRIO_CHANGE ; call HandlePreempt call HandlePrio call GetPrioThread call SetupPreempt ret GetNextThread Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: ThreadSuspend ; ; DESCRIPTION: Suspend thread callback from scheduler ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; thread_suspend: push dword ptr 0 push ebp mov ebp,esp push eax push ebx mov eax,ds push eax ; NotifyThreadSuspend ; pop eax mov ds,ax pop ebx pop eax pop ebp add sp,4 iretd ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: AddCallback ; ; DESCRIPTION: Add callback to thread ; ; PARAMETERS: DS Thread TSS ; ES Thread block ; BX Callback routine ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; AddCallback Proc near push dx push si push di ; test dword ptr ds:p_rflags,20000h jnz acVm ; test ds:p_cs,3 jnz acPm acKernel: mov si,ss mov edi,esp ; mov dx,ds:p_ss mov ss,dx mov esp,dword ptr ds:p_rsp ; xor dx,dx push dword ptr ds:p_rflags push dx push ds:p_cs push dword ptr ds:p_rip ; mov ds:p_ss,ss mov dword ptr ds:p_rsp,esp mov ss,si mov esp,edi jmp acDone acPm: push es mov dx,ds:p_kernel_ss mov es,dx mov edi,ds:p_kernel_esp mov word ptr es:[edi-2],0 pop es ; mov si,ss mov edi,esp ; mov ss,dx mov esp,ds:p_kernel_esp ; xor dx,dx push dx push ds:p_ss push dword ptr ds:p_rsp push dword ptr ds:p_rflags push dx push ds:p_cs push dword ptr ds:p_rip ; mov ds:p_ss,ss mov dword ptr ds:p_rsp,esp mov ss,si mov esp,edi jmp acDone acVm: mov si,ss mov edi,esp ; mov dx,ds:p_kernel_ss mov ss,dx mov esp,ds:p_kernel_esp ; xor dx,dx push dx push ds:p_gs push dx push ds:p_fs push dx push ds:p_ds push dx push ds:p_es push dx push ds:p_ss push dword ptr ds:p_rsp push dword ptr ds:p_rflags push dx push ds:p_cs push dword ptr ds:p_rip ; mov ds:p_ss,ss mov dword ptr ds:p_rsp,esp and dword ptr ds:p_rflags,NOT 20000h mov ss,si mov esp,edi acDone: mov ds:p_cs,cs movzx ebx,bx mov dword ptr ds:p_rip,ebx ; pop di pop si pop dx ret AddCallback Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: TimerPreemptReload ; ; DESCRIPTION: Timer & preemption reload ; ; PARAMETERS: DS Task sel ; FS Core selector ; ; RETURNS: NC Load a new task ; CY Retry ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; TimerPreemptReload Proc near preempt_reload_loop: GetSystemTime ; call LockTimerCore mov fs:ps_last_lsb,eax add eax,cs:update_tics adc edx,0 mov bx,fs:ps_timer_head mov ecx,fs:ps_preempt_msb cmp ecx,fs:[bx].timer_msb jc preempt_reload_check_preempt ; jnz preempt_reload_check_timer ; mov ecx,fs:ps_preempt_lsb cmp ecx,fs:[bx].timer_lsb jc preempt_reload_check_preempt preempt_reload_check_timer: sub eax,fs:[bx].timer_lsb sbb edx,fs:[bx].timer_msb jc preempt_reload_timer ; call LocalRemoveTimerCore jmp preempt_reload_loop preempt_reload_check_preempt: sub eax,fs:ps_preempt_lsb sbb edx,fs:ps_preempt_msb jc preempt_reload_timer ; call UnlockTimerCore lock or fs:ps_flags,PS_FLAG_PREEMPT stc jmp preempt_timer_reload_done preempt_reload_timer: neg eax ReloadSysPreemptTimer pushf call UnlockTimerCore popf jc preempt_reload_loop preempt_timer_reload_done: ret TimerPreemptReload Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: PreemptReload ; ; DESCRIPTION: Preemption reload ; ; PARAMETERS: DS Task sel ; FS Core selector ; ; RETURNS: NC Load a new task ; CY Retry ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; PreemptReload Proc near GetSystemTime mov fs:ps_last_lsb,eax sub eax,fs:ps_preempt_lsb sbb edx,fs:ps_preempt_msb jc preempt_reload_ok ; lock or fs:ps_flags,PS_FLAG_PREEMPT stc jmp preempt_reload_done preempt_reload_ok: neg eax ReloadPreemptTimer clc preempt_reload_done: ret PreemptReload Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: load_breaks ; ; DESCRIPTION: Load break-points ; ; PARAMETERS: FS Core selector ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; load_breaks Proc near mov eax,dword ptr ds:p_dr0 mov dr0,eax mov eax,dword ptr ds:p_dr1 mov dr1,eax mov eax,dword ptr ds:p_dr2 mov dr2,eax mov eax,dword ptr ds:p_dr3 mov dr3,eax mov eax,dword ptr ds:p_dr7 mov dr7,eax and ax,0FFh jnz load_break_done ; and es:p_flags,NOT THREAD_FLAG_BP load_break_done: ret load_breaks Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: LoadThread ; ; DESCRIPTION: Load a new thread ; ; PARAMETERS: FS Core selector ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; thread_create: push dword ptr 0 push ebp mov ebp,esp push eax push ebx push word ptr 0 push ds push es ; NotifyThreadCreated ; pop es pop ds pop ax pop ebx pop eax pop ebp add esp,4 iretd LoadThread: test fs:ps_flags,PS_FLAG_P_STATE jz load_thread_loop ; lock and fs:ps_flags,NOT PS_FLAG_P_STATE UpdatePState load_thread_loop: mov ax,SEG data mov ds,ax xor ax,ax mov es,ax load_thread_wakeup_loop: cli mov ax,fs:ps_wakeup_list or ax,ax jz load_thread_wakeup_done ; call cs:remove_wakeup_proc ; mov di,es:p_prio call InsertCoreBlock cmp di,fs:ps_prio_act jbe load_thread_wakeup_loop ; mov fs:ps_prio_act,di jmp load_thread_wakeup_loop load_thread_wakeup_done: xor ax,ax mov es,ax sti call GetNextThread ; xor ax,ax xchg ax,es:p_wanted_core or ax,ax jz load_reload_loop ; mov es:p_core,ax mov dx,fs cmp dx,ax jz load_reload_loop ; ; push fs ; mov fs,ax ; movzx edx,fs:ps_id ; pop fs ; mov ax,1 ; mov bx,es ; AddSchedulerLog load_reload_wakeup: call cs:insert_wakeup_proc jmp load_thread_loop load_reload_loop: call cs:preempt_reload_proc jnc load_a_task load_retry: mov ax,SEG data mov ds,ax ; test fs:ps_flags,PS_FLAG_TIMER_EXPIRED jz load_timer_not_expired ; call cs:update_timer_proc load_timer_not_expired: mov ax,es cmp ax,fs:ps_null_thread je load_thread_loop ; mov di,es:p_prio call InsertCoreFirst cmp di,fs:ps_prio_act jbe load_thread_loop ; mov fs:ps_prio_act,di lock or fs:ps_flags,PS_FLAG_PRIO_CHANGE jmp load_thread_loop load_a_task: mov es:p_sleep_type,0 lock or fs:ps_flags,PS_FLAG_LOADING mov ax,gdt_sel mov ds,ax mov bx,es:p_tss_sel or bx,bx jnz load_protected_mode load_long_mode: test fs:ps_flags,PS_FLAG_LONG_MODE jnz load_not_flush ; mov bx,fs:ps_long_tr and byte ptr ds:[bx+5],NOT 2 ltr bx ; mov eax,es:p_cr3 SwitchToLongMode lock or fs:ps_flags,PS_FLAG_LONG_MODE jmp load_not_flush load_protected_mode: test fs:ps_flags,PS_FLAG_LONG_MODE jz load_prot_switch_ok ; mov eax,es:p_cr3 SwitchToProtectedMode lock and fs:ps_flags, NOT PS_FLAG_LONG_MODE load_prot_switch_ok: mov bx,es:p_tss_sel and byte ptr ds:[bx+5],NOT 2 ltr bx load_not_flush: mov edx,io_focus_linear GetSysPageDir ; mov esi,eax mov edi,ebx ; mov edx,io_focus_linear GetPageDir ; cmp esi,eax jne load_reload_cr3 ; cmp edi,ebx jne load_reload_cr3 ; mov eax,cr3 cmp eax,es:p_cr3 je load_cr3_ok load_reload_cr3: GetPageDirAttrib xor edi,edi load_reload_cr3_loop: mov edx,io_focus_linear add edx,edi GetSysPageDir SetPageDir ; add edi,esi loop load_reload_cr3_loop ; mov eax,es:p_cr3 mov cr3,eax mov fs:ps_tlb.pt32_used,0 and ax,0F000h mov fs:ps_cr3,eax load_cr3_ok: mov eax,fs:ps_tlb.pt32_used or eax,eax jz load_cr3_flush_ok ; call cs:flush_tlb_proc load_cr3_flush_ok: mov ax,es mov ds,ax lldt ds:p_ldt ; mov ax,ds:p_flags or ax,ax jz load_bp_done ; test ax,THREAD_FLAG_CREATE jz load_create_done ; and ds:p_flags,NOT THREAD_FLAG_CREATE mov bx,OFFSET thread_create call AddCallback load_create_done: mov ax,ds:p_flags test ax,THREAD_FLAG_SUSPEND jz load_suspend_done ; and ds:p_flags,NOT THREAD_FLAG_SUSPEND mov bx,OFFSET thread_suspend call AddCallback load_suspend_done: mov ax,ds:p_flags test ax,THREAD_FLAG_BP jz load_bp_done ; test fs:ps_flags,PS_FLAG_LONG_MODE jz load_prot_breaks ; mov edx,ds:p_linear LoadLongBreaks jmp load_actions_done load_prot_breaks: call load_breaks jmp load_actions_done load_bp_done: xor eax,eax mov dr7,eax load_actions_done: call LoadUnlockCore mov ax,fs:ps_wakeup_list or ax,ax jnz load_relock ; test fs:ps_flags,PS_FLAG_TIMER_EXPIRED jnz load_relock ; mov eax,fs:ps_tlb.pt32_used or eax,eax jz load_regs load_relock: call LockCore sti jmp load_retry load_regs: test fs:ps_flags,PS_FLAG_LONG_MODE jz load_prot_regs ; mov fs:ps_curr_thread,es mov fs:ps_last_thread,es lock and fs:ps_flags,NOT PS_FLAG_LOADING ; mov eax,es:p_kernel_stack mov fs:ps_syscall_esp,eax ; mov eax,dword ptr es:p_tls_linear mov fs:ps_tls_linear,eax mov eax,dword ptr es:p_tls_linear+4 mov fs:ps_tls_linear+4,eax ; mov edx,es:p_linear mov edi,fs:ps_tr_linear LoadLongRegs load_prot_regs: mov fs:ps_curr_thread,es mov fs:ps_last_thread,es lock and fs:ps_flags,NOT PS_FLAG_LOADING ; test dword ptr ds:p_rflags,20000h jnz load_vm ; test ds:p_cs,3 jnz load_pm_app load_kernel: mov ax,ds:p_ss mov ss,ax mov esp,dword ptr ds:p_rsp ; xor ax,ax push dword ptr ds:p_rflags push ax push ds:p_cs push dword ptr ds:p_rip ; mov ecx,dword ptr ds:p_rcx mov edx,dword ptr ds:p_rdx mov ebx,dword ptr ds:p_rbx mov ebp,dword ptr ds:p_rbp mov esi,dword ptr ds:p_rsi mov edi,dword ptr ds:p_rdi ; mov ax,ds:p_es mov es,ax ; mov ax,ds:p_fs mov fs,ax ; mov ax,ds:p_gs mov gs,ax ; mov ax,ds:p_ds push ax mov eax,dword ptr ds:p_rax pop ds iretd load_pm_app: mov ax,ds:p_kernel_ss mov ss,ax mov esp,ds:p_kernel_esp ; xor ax,ax push ax push ds:p_ss push dword ptr ds:p_rsp push dword ptr ds:p_rflags push ax push ds:p_cs push dword ptr ds:p_rip ; mov ecx,dword ptr ds:p_rcx mov edx,dword ptr ds:p_rdx mov ebx,dword ptr ds:p_rbx mov ebp,dword ptr ds:p_rbp mov esi,dword ptr ds:p_rsi mov edi,dword ptr ds:p_rdi ; mov ax,ds:p_es mov es,ax ; mov ax,ds:p_fs mov fs,ax ; mov ax,ds:p_gs mov gs,ax ; mov ax,ds:p_ds push ax mov eax,dword ptr ds:p_rax pop ds iretd load_vm: mov ax,ds:p_kernel_ss mov ss,ax mov esp,ds:p_kernel_esp ; xor ax,ax push ax push ds:p_gs push ax push ds:p_fs push ax push ds:p_ds push ax push ds:p_es push ax push ds:p_ss push dword ptr ds:p_rsp push dword ptr ds:p_rflags push ax push ds:p_cs push dword ptr ds:p_rip ; mov eax,dword ptr ds:p_rax mov ecx,dword ptr ds:p_rcx mov edx,dword ptr ds:p_rdx mov ebx,dword ptr ds:p_rbx mov ebp,dword ptr ds:p_rbp mov esi,dword ptr ds:p_rsi mov edi,dword ptr ds:p_rdi iretd ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: SaveCurrentThread ; ; DESCRIPTION: Save state of current thread ; ; PARAMETERS: Stack, return IP ; ; RETURNS: SS:SP Core stack ; FS Core selector ; ES, GS Clear ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; SaveCurrentThread Proc near push fs push ds push eax push edx ; call LockCore sti ; GetSystemTime mov ds,fs:ps_curr_thread sub eax,fs:ps_last_lsb add ds:p_lsb_tics,eax adc ds:p_msb_tics,0 add fs:ps_lsb_tics,eax adc fs:ps_msb_tics,0 ; pushfd pop eax or ax,200h mov dword ptr ds:p_rflags,eax ; pushf pop ax and ax,NOT 100h push ax popf ; mov dword ptr ds:p_rcx,ecx mov dword ptr ds:p_rbx,ebx mov dword ptr ds:p_rbp,ebp mov dword ptr ds:p_rsi,esi mov dword ptr ds:p_rdi,edi mov ds:p_es,es mov ds:p_cs,cs mov ds:p_ss,ss mov ds:p_gs,gs ; pop dword ptr ds:p_rdx pop eax mov dword ptr ds:p_rax,eax ; pop ds:p_ds pop ds:p_fs pop bp pop dx movzx edx,dx mov dword ptr ds:p_rip,edx mov dword ptr ds:p_rsp,esp ; lss esp,fword ptr fs:ps_stack_offset call cs:fpu_save_proc mov edx,dword ptr ds:p_rdx push bp ; xor bp,bp mov ds,bp mov es,bp mov gs,bp ret SaveCurrentThread Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: SaveLockedThread ; ; DESCRIPTION: Save state of current thread when lock is already taken ; ; PARAMETERS: Stack, return IP ; FS Core selector ; ; RETURNS: SS:SP Processor stack ; ES, GS Clear ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; SaveLockedThread Proc near push fs push ds push eax push edx ; mov ax,fs:ps_curr_thread or ax,ax jnz sltSave ; add esp,12 pop bp lss esp,fword ptr fs:ps_stack_offset push bp ; xor bp,bp mov ds,bp mov es,bp mov gs,bp ret sltSave: mov ds,ax GetSystemTime sub eax,fs:ps_last_lsb add ds:p_lsb_tics,eax adc ds:p_msb_tics,0 add fs:ps_lsb_tics,eax adc fs:ps_msb_tics,0 ; pushfd pop eax or ax,200h mov dword ptr ds:p_rflags,eax mov dword ptr ds:p_rcx,ecx mov dword ptr ds:p_rbx,ebx mov dword ptr ds:p_rbp,ebp mov dword ptr ds:p_rsi,esi mov dword ptr ds:p_rdi,edi mov ds:p_es,es mov ds:p_cs,cs mov ds:p_ss,ss mov ds:p_gs,gs ; pop dword ptr ds:p_rdx pop eax mov dword ptr ds:p_rax,eax ; pop ds:p_ds pop ds:p_fs pop bp pop dx movzx edx,dx mov dword ptr ds:p_rip,edx mov dword ptr ds:p_rsp,esp ; lss esp,fword ptr fs:ps_stack_offset call cs:fpu_save_proc mov edx,dword ptr ds:p_rdx push bp ; xor bp,bp mov ds,bp mov es,bp mov gs,bp ret SaveLockedThread Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: SkipCurrentThread ; ; DESCRIPTION: Skip current thread (no save of registers) ; ; RETURNS: SS:SP Core stack ; FS Core selector ; ES, GS Clear ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; SkipCurrentThread Proc near call LockCore sti push eax push ds GetSystemTime mov ds,fs:ps_curr_thread sub eax,fs:ps_last_lsb add ds:p_lsb_tics,eax adc ds:p_msb_tics,0 add fs:ps_lsb_tics,eax adc fs:ps_msb_tics,0 pop ds pop eax ; pop bp lss esp,fword ptr fs:ps_stack_offset push bp ; xor bp,bp mov ds,bp mov es,bp mov gs,bp ret SkipCurrentThread Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: ContinueCurrentThread ; ; DESCRIPTION: Continue current thread, by putting it into the ready-list. ; Also releases scheduler lock ; ; PARAMETERS: FS Core selector ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ContinueCurrentThread: mov ax,fs:ps_curr_thread or ax,ax jz cctDone ; push es mov es,ax mov di,es:p_prio or di,di je cctPop ; call InsertCoreBlock cmp di,fs:ps_prio_act jbe cctPop ; mov fs:ps_prio_act,di lock or fs:ps_flags,PS_FLAG_PRIO_CHANGE cctPop: pop es cctDone: mov fs:ps_curr_thread,0 jmp LoadThread ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: RunApCore ; ; DESCRIPTION: Run AP core ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; run_ap_core_name DB 'Run AP Core', 0 run_ap_core: mov eax,cr0 or al,8 mov cr0,eax ; mov ax,core_data_sel mov fs,ax mov fs,fs:ps_sel ; StartSyscall run_core_do: call LockCore sti lock or fs:ps_flags,PS_FLAG_PREEMPT jmp LoadThread ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: DeleteThread ; ; DESCRIPTION: Delete a thread ; ; PARAMETERS: ES Thread block ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; DeleteThread Proc near push es mov es,es:p_kernel_ss FreeMem pop es ; mov bx,es:p_tss_sel FreeGdt ; FreeMem ret DeleteThread Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: DeleteProcess ; ; DESCRIPTION: Delete a process ; ; PARAMETERS: ES Thread block ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; DeleteProcess Proc near mov ax,es:p_app_sel push es mov es,ax FreeMem pop es ; mov ax,es:p_process_sel push es mov es,ax FreeMem pop es ; push es mov es,es:p_kernel_ss FreeMem pop es ; mov bx,es:p_tss_sel FreeGdt ; mov eax,es:p_cr3 FreeMem NotifyDeleteProcess ret DeleteProcess Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: System thread ; ; DESCRIPTION: Cleans up threads & processes ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; system_thread_name DB 'System', 0 system_thread_pr: mov ax,SEG data mov ds,ax mov ds,ds:patch_sel GetThread mov ds:system_thread,ax ; mov ax,SEG data mov ds,ax stLoop: WaitForSignal stThreadLoop: mov si,OFFSET term_thread_list mov ax,[si] or ax,ax jz stThreadOK ; call cs:lock_list_proc call RemoveBlock call cs:unlock_list_proc ; call DeleteThread jmp stThreadLoop stThreadOk: mov si,OFFSET term_proc_list mov ax,[si] or ax,ax jz stLoop ; call cs:lock_list_proc call RemoveBlock call cs:unlock_list_proc ; call DeleteProcess jmp stThreadLoop ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: StartProcessorNullThreads ; ; DESCRIPTION: Start each of the null threads for a processor ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; start_processor_null_threads Proc near mov ax,system_data_sel mov ds,ax mov al,ds:cpu_type cmp al,3 jbe start_locks_ok ; mov ax,SEG data mov ds,ax mov ds,ds:patch_sel mov ds:flush_tlb_proc,OFFSET FlushTlb486 ; GetCoreCount cmp cx,1 jbe start_locks_ok ; mov ds:lock_list_proc,OFFSET LockListMultiple mov ds:unlock_list_proc,OFFSET UnlockListMultiple mov ds:insert_wakeup_proc,OFFSET InsertWakeupMultiple mov ds:remove_wakeup_proc,OFFSET RemoveWakeupMultiple mov ds:lock_signal_proc,OFFSET LockSignalMultiple mov ds:unlock_signal_proc,OFFSET UnlockSignalMultiple mov ds:lock_kernel_section_proc,OFFSET LockKernelSectionMultiple mov ds:unlock_kernel_section_proc,OFFSET UnlockKernelSectionMultiple mov ds:lock_user_section_proc,OFFSET LockUserSectionMultiple mov ds:unlock_user_section_proc,OFFSET UnlockUserSectionMultiple mov ds:lock_futex_proc,OFFSET LockFutexMultiple mov ds:unlock_futex_proc,OFFSET UnlockFutexMultiple mov ds:fpu_exception_proc,OFFSET FpuExceptionMultiple mov ds:fpu_save_proc,OFFSET FpuSaveMultiple start_locks_ok: mov ecx,stack0_size mov ax,cs mov ds,ax mov es,ax mov si,OFFSET system_thread_pr mov di,OFFSET system_thread_name mov ax,1 CreateThread ; mov eax,10 AllocateSmallGlobalMem xor di,di mov eax,cs:dword ptr null_base stosd mov al,' ' stosb mov al,'1' stosb xor al,al stosb ; GetCoreCount mov bx,1 sub cx,1 jz start_processor_free create_null_loop: push cx mov ax,cs mov ds,ax xor ax,ax mov ecx,stack0_size mov si,OFFSET null_thread xor di,di CreateThread pop cx ; mov di,5 inc byte ptr es:[di] inc bx loop create_null_loop start_processor_free: FreeMem ret start_processor_null_threads Endp null_base DB 'Null' null_thread0: mov ax,core_data_sel mov fs,ax mov fs,fs:ps_sel mov ax,fs:ps_curr_thread mov es,ax mov es:p_sleep_sel,fs mov es:p_sleep_offset,0 mov fs:ps_null_thread,ax mov es:p_core,fs lock or fs:ps_flags,PS_FLAG_ACTIVE ; mov ax,start_core_nr IsValidOsGate jc null_ap_ok ; SetupSmpPatch null_ap_ok: push OFFSET null_loop_start call SaveCurrentThread ; mov es,fs:ps_curr_thread mov fs:ps_curr_thread,0 ; mov es:p_sleep_sel,0 mov es:p_sleep_offset,0 jmp LoadThread null_thread: sti mov ax,bx GetCoreNumber GetThread mov es,ax mov es:p_sleep_sel,fs mov es:p_sleep_offset,0 mov fs:ps_null_thread,ax mov es:p_core,fs mov es:p_wanted_core,0 mov es:p_int_count,0 mov es:p_nest_count,0 mov es:p_nest_unwind,0 ; push OFFSET null_loop_start call SaveCurrentThread ; mov es,fs:ps_curr_thread mov fs:ps_curr_thread,0 ; mov es:p_sleep_sel,0 mov es:p_sleep_offset,0 jmp LoadThread null_loop_start: mov ss,es:p_kernel_ss mov esp,es:p_kernel_esp GetCore null_loop: test fs:ps_flags,PS_FLAG_SHUTDOWN jz null_hlt ; push OFFSET null_loop_start call SaveCurrentThread ; call UnlockCore lock and fs:ps_flags,NOT PS_FLAG_SHUTDOWN mov fs:ps_curr_thread,0 ShutdownCore null_hlt: mov ax,fs:ps_nesting cmp ax,-1 je null_nest_ok ; mov ax,fs:ps_curr_thread mov es,ax mov cx,es:p_nest_count mov dx,es:p_nest_unwind ; CrashGate null_nest_ok: hlt jmp null_loop ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: RemoveWakeupSingle ; ; DESCRIPTION: Remove wakeup thread, single core ; ; RETURNS: ES Thread ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; RemoveWakeupSingle PROC near push si cli mov si,OFFSET ps_wakeup_list call RemoveCoreBlock sti pop si ret RemoveWakeupSingle Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: RemoveWakeupMultiple ; ; DESCRIPTION: Remove wakeup thread, multiple core ; ; RETURNS: ES Thread ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; RemoveWakeupMultiple PROC near push si rwmTryLock: mov si,fs:ps_wakeup_spinlock or si,si je rwmGet ; sti pause jmp rwmTryLock rwmGet: cli inc si xchg si,fs:ps_wakeup_spinlock or si,si je rwmLocked ; jmp rwmTryLock rwmLocked: mov si,OFFSET ps_wakeup_list call RemoveCoreBlock mov fs:ps_wakeup_spinlock,0 sti pop si ret RemoveWakeupMultiple Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: InsertWakeupSingle ; ; DESCRIPTION: Insert thread into run list, single core ; ; PARAMETERS: ES Thread ; FS Core ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; InsertWakeupSingle PROC near push di cli mov di,OFFSET ps_wakeup_list call InsertCoreBlock sti pop di ret InsertWakeupSingle Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: InsertWakeupMultiple ; ; DESCRIPTION: Insert thread into run list, multiple core ; ; PARAMETERS: ES Thread ; FS Core ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; InsertWakeupMultiple PROC near push ax push di iwmLogOk: mov ax,fs ; mov di,es:p_core cmp ax,di je iwmTryLockSelf ; push fs mov fs,di iwmTryLockOther: mov ax,fs:ps_wakeup_spinlock or ax,ax je iwmGetOther ; sti pause jmp iwmTryLockOther iwmGetOther: cli inc ax xchg ax,fs:ps_wakeup_spinlock or ax,ax je iwmLockedOther ; jmp iwmTryLockOther iwmLockedOther: mov di,OFFSET ps_wakeup_list call InsertCoreBlock ; mov di,es:p_prio cmp di,fs:ps_prio_act mov fs:ps_wakeup_spinlock,0 sti jbe iwmIntOk ; test fs:ps_flags,PS_FLAG_PREEMPT jnz iwmIntOk ; lock or fs:ps_flags,PS_FLAG_PREEMPT ; mov ax,fs:ps_nesting cmp ax,-1 jne iwmIntOk ; mov al,81h SendInt iwmIntOk: sti pop fs jmp iwmDone iwmTryLockSelf: mov ax,fs:ps_wakeup_spinlock or ax,ax je iwmGetSelf ; sti pause jmp iwmTryLockSelf iwmGetSelf: cli inc ax xchg ax,fs:ps_wakeup_spinlock or ax,ax je iwmLockedSelf ; jmp iwmTryLockSelf iwmLockedSelf: mov di,OFFSET ps_wakeup_list call InsertCoreBlock mov fs:ps_wakeup_spinlock,0 sti iwmDone: pop di pop ax ret InsertWakeupMultiple Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: LockSignalSingle ; ; DESCRIPTION: Lock signal, single core ; ; PARAMETERS: ES Thread ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; LockSignalSingle PROC near cli ret LockSignalSingle Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: LockSignalMultiple ; ; DESCRIPTION: Lock signal, multiple core ; ; PARAMETERS: ES Thread ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; LockSignalMultiple PROC near push ax lsmTryLock: mov ax,es:p_signal_spinlock or ax,ax je lsmGet ; sti pause jmp lsmTryLock lsmGet: cli inc ax xchg ax,es:p_signal_spinlock or ax,ax je lsmLocked ; sti jmp lsmTryLock lsmLocked: pop ax ret LockSignalMultiple Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: UnlockSignalSingle ; ; DESCRIPTION: Unlock signal, single core ; ; PARAMETERS: ES Thread ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; UnlockSignalSingle PROC near sti ret UnlockSignalSingle Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: UnlockSignalMultiple ; ; DESCRIPTION: Unlock signal, multiple core ; ; PARAMETERS: ES Thread ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; UnlockSignalMultiple PROC near mov es:p_signal_spinlock,0 sti ret UnlockSignalMultiple Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: WakeThread ; ; DESCRIPTION: Wake up thread ; ; PARAMETERS: DX:ESI Thread list ; EAX Status to thread ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; WakeThread PROC near push ds push es push fs push bx push di ; mov ds,dx call TryLockCore sti ; mov di,ds:[esi] or di,di jz wtUnlock ; call cs:lock_list_proc call RemoveBlock32 mov es:p_data,eax call cs:unlock_list_proc call cs:insert_wakeup_proc wtUnlock: call TryUnlockCore ; pop di pop bx pop fs pop es pop ds ret WakeThread ENDP ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: DebugBlock ; ; DESCRIPTION: Block thread into debug list ; ; PARAMETERS: FS Core block ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; debug_block_name DB 'Debug Block', 0 debug_block: mov ds,fs:ps_curr_thread lss esp,fword ptr fs:ps_stack_offset call cs:fpu_save_proc ; xor ax,ax mov ds,ax mov es,ax mov gs,ax ; mov es,fs:ps_curr_thread mov eax,es:p_debug_proc or eax,eax jz debug_block_do ; call es:p_debug_proc debug_block_do: mov fs:ps_curr_thread,0 ; mov ax,system_data_sel mov ds,ax mov edi,OFFSET debug_list ; call cs:lock_list_proc call InsertBlock32 call cs:unlock_list_proc ; mov es:p_sleep_type,SLEEP_TYPE_DEBUG mov es:p_sleep_sel,ds mov es:p_sleep_offset,edi jmp LoadThread ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: PreemptNotify ; ; DESCRIPTION: Default preempt procedure ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; preempt_notify_name DB 'Preempt Notify',0 preempt_notify Proc far retf32 preempt_notify Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: CreateLongTss ; ; DESCRIPTION: Create long-mode TSS ; ; RETURNS: BX TSS selector ; EDX Tss linear ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; create_long_tss Proc near push es push eax push ecx ; mov ax,flat_sel mov es,ax ; mov eax,SIZE tss64_seg mov ecx,eax AllocateSmallLinear mov edi,edx ; push ecx push edi xor al,al rep stos byte ptr es:[edi] pop edi pop ecx ; mov es:[edi].tss64_bitmap,OFFSET tss64_io_bitmap ; mov eax,1000h AllocateBigLinear mov dword ptr es:[edi].tss64_ist1,edx ; mov eax,1000h AllocateBigLinear mov dword ptr es:[edi].tss64_ist2,edx ; mov ecx,SIZE tss64_seg mov edx,edi AllocateGdt CreateTssSelector ; pop ecx pop eax pop es ret create_long_tss Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: AddDumpCore ; ; DESCRIPTION: Add dump core ; ; PARAMETERS: ES core sel ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; AddDumpCore Proc near pushad push ds push es ; mov si,core_image_sel mov ds,si mov si,ds:ic_cores cmp si,16 jb adcDo ; xor si,si jmp adcDone adcDo: mov eax,4000h AllocateBigLinear ; inc ds:ic_cores shl si,2 add si,OFFSET ic_linear mov es:ps_dump_offset,si ; mov ds:[si],edx ; mov si,core_save_sel mov ds,si mov si,ds:sc_cores ; inc ds:sc_cores shl si,5 add si,OFFSET sc_phys mov edi,edx ; AllocatePhysical64 mov dword ptr ds:[si].scp_core_phys,eax mov dword ptr ds:[si].scp_core_phys+4,ebx mov al,13h SetPageEntry add edx,1000h ; AllocatePhysical64 mov dword ptr ds:[si].scp_stack_phys,eax mov dword ptr ds:[si].scp_stack_phys+4,ebx mov al,13h SetPageEntry add edx,1000h ; AllocatePhysical64 mov dword ptr ds:[si].scp_log_phys,eax mov dword ptr ds:[si].scp_log_phys+4,ebx mov al,13h SetPageEntry add edx,1000h ; AllocatePhysical64 mov dword ptr ds:[si].scp_log_phys+8,eax mov dword ptr ds:[si].scp_log_phys+12,ebx mov al,13h SetPageEntry ; mov ax,flat_sel mov es,ax mov ecx,1000h xor eax,eax rep stos dword ptr es:[edi] adcDone: pop es pop ds popad ret AddDumpCore Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: CreateCore ; ; DESCRIPTION: Create core ; ; PARAMETERS: EDX Core GDT linear ; ; RETURNS: AX Processor # ; ES Processor sel ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; create_core_name DB 'Create Core',0 create_core Proc far push ds push ebx push ecx push edx push esi push edi ; mov ax,flat_sel mov ds,ax mov ax,gdt_sel mov es,ax ; AllocateGdt mov eax,[edx+core_data_sel] mov es:[bx],eax ; mov eax,[edx+core_data_sel+4] mov es:[bx+4],eax mov es,bx ; mov cx,SIZE processor_seg xor di,di xor al,al rep stosb mov es:ps_sel,es ; HasLongMode jnc cr_flat_stack ; mov eax,1000h AllocateBigLinear AllocateGdt ; mov ecx,1000h CreateDataSelector32 ; mov ds,bx mov ds:[0],dx mov es:ps_stack_offset,ecx mov es:ps_stack_sel,bx jmp cr_stack_ok cr_flat_stack: mov eax,3000h AllocateBigLinear xor ebx,ebx mov eax,4 SetPageEntry add edx,2000h SetPageEntry sub edx,1000h ; mov ds:[edx],dx add edx,1000h mov es:ps_stack_offset,edx mov es:ps_stack_sel,long_kernel_data_sel cr_stack_ok: mov ax,SEG data mov ds,ax mov ds,ds:patch_sel mov ax,ds:core_count mov si,ax add si,si mov ds:[si].core_arr,es inc ds:core_count mov es:ps_id,ax ; xor ax,ax mov ds,ax mov bx,OFFSET ps_ptab mov es:ps_prio_act,bx ; mov cx,256 ptab_init: mov es:[bx],ax add bx,2 loop ptab_init ; mov es:ps_wakeup_list,0 mov es:ps_wakeup_spinlock,0 mov es:ps_nesting,-1 mov es:ps_curr_thread,0 mov es:ps_last_thread,-1 mov es:ps_flags,PS_FLAG_INIT_CLOCK mov es:ps_null_thread,0 mov es:ps_math_thread,0 mov es:ps_apic,-1 mov es:ps_last_lsb,0 mov es:ps_lsb_tics,0 mov es:ps_msb_tics,0 mov es:ps_tlb.pt32_locked,0 mov es:ps_tlb.pt32_used,0 mov eax,cr3 and ax,0F000h mov es:ps_cr3,eax ; mov es:cs_usel,flat_sel mov es:cs_uoffs,0 mov es:cs_fault,-1 mov es:cs_irq,0 ; mov es:ps_timer_spinlock,0 mov bx,OFFSET ps_timer_entries mov es:[bx].timer_next,0 mov es:[bx].timer_msb,0FFFFFFFFh mov es:[bx].timer_lsb,0FFFFFFFFh mov es:ps_timer_head,bx ; mov cx,0FFh add bx,SIZE timer_struc mov es:ps_timer_free,bx core_timer_list_create: mov ax,bx add ax,SIZE timer_struc mov es:[bx].timer_next,ax mov bx,ax loop core_timer_list_create ; mov es:ps_sched_count,0 mov es:ps_log_count,0 mov es:ps_log_sel,0 mov es:ps_log_entry,0 ; call create_long_tss mov es:ps_long_tr,bx mov es:ps_tr_linear,edx ; mov bx,es GetSelectorBaseSize mov es:ps_linear,edx mov es:ps_long_ldt,0 mov es:ps_syscall_esp,0 mov es:ps_syscall_eip,0 call AddDumpCore ; mov ax,es:ps_id ; pop edi pop esi pop edx pop ecx pop ebx pop ds retf32 create_core Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: GetSchedulerLockCounter ; ; DESCRIPTION: Get state of task lock ; ; RETURNS: AX Lock count ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; get_scheduler_lock_counter_name DB 'Get Scheduler Lock Counter',0 get_scheduler_lock_counter Proc far push fs mov ax,core_data_sel mov fs,ax mov ax,fs:ps_nesting pop fs retf32 get_scheduler_lock_counter Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: GetCore ; ; DESCRIPTION: Get current core selector ; ; RETURNS: FS Core sel ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; get_core_name DB 'Get Core',0 get_core Proc far push ax mov ax,core_data_sel mov fs,ax mov fs,fs:ps_sel pop ax retf32 get_core Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: GetCoreNumber ; ; DESCRIPTION: Get core selector for specified processor # ; ; PARAMETERS: AX Core # ; ; RETURNS: FS Core sel ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; get_core_num_name DB 'Get Core Number',0 get_core_num Proc far push ax push bx ; cmp ax,cs:core_count jae gpnFail ; mov bx,ax add bx,bx mov ax,cs:[bx].core_arr mov fs,ax clc jmp gpnDone gpnFail: xor ax,ax mov fs,ax stc gpnDone: pop bx pop ax retf32 get_core_num Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: SetThreadCore ; ; DESCRIPTION: Set core for a thread ; ; PARAMETERS: AX Core # ; ES Thread ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; set_thread_core_name DB 'Set Thread Core',0 set_thread_core Proc far push bx ; cmp ax,cs:core_count jae stcDone ; mov bx,ax add bx,bx mov bx,cs:[bx].core_arr mov es:p_wanted_core,bx stcDone: pop bx retf32 set_thread_core Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: AddThreadInt ; ; DESCRIPTION: Add an int association for a thread ; ; PARAMETERS: AL Int # ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; add_thread_int_name DB 'Add Thread Int',0 add_thread_int Proc far push es push ax GetThread mov es,ax add es:p_int_count,1 pop ax pop es retf32 add_thread_int Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: LockListSingle ; ; DESCRIPTION: Lock list, single processor version ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; LockListSingle Proc near cli ret LockListSingle Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: UnlockListSingle ; ; DESCRIPTION: Unlock list, single processor version ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; UnlockListSingle Proc near sti ret UnlockListSingle Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: LockKernelSectionSingle ; ; DESCRIPTION: Lock kernel critical section, single processor version ; ; PARAMETERS: DS:ESI Section ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; LockKernelSectionSingle Proc near ret LockKernelSectionSingle Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: UnlockKernelSectionSingle ; ; DESCRIPTION: Unlock kernel critical section, single processor version ; ; PARAMETERS: DS:ESI Section ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; UnlockKernelSectionSingle Proc near ret UnlockKernelSectionSingle Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: LockUserSectionSingle ; ; DESCRIPTION: Lock user critical section, single processor version ; ; PARAMETERS: DS:EBX Section ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; LockUserSectionSingle Proc near ret LockUserSectionSingle Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: UnlockUserSectionSingle ; ; DESCRIPTION: Unlock user critical section, single processor version ; ; PARAMETERS: DS:EBX Section ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; UnlockUserSectionSingle Proc near ret UnlockUserSectionSingle Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: LockFutexSingle ; ; DESCRIPTION: Lock futex, single processor version ; ; PARAMETERS: DS:EBX Futex handle data ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; LockFutexSingle Proc near ret LockFutexSingle Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: UnlockFutexSingle ; ; DESCRIPTION: Unlock futex, single processor version ; ; PARAMETERS: DS:EBX Futex handle data ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; UnlockFutexSingle Proc near ret UnlockFutexSingle Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: LockListMultiple ; ; DESCRIPTION: Lock list, multiple processor version ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; LockListMultiple Proc near push ds push ax ; mov ax,SEG data mov ds,ax llSpinLock: sti mov ax,ds:list_lock or ax,ax je llGet ; pause jmp llSpinLock llGet: cli inc ax xchg ax,ds:list_lock or ax,ax je llDone ; jmp llSpinLock llDone: pop ax pop ds ret LockListMultiple Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: UnlockListMultiple ; ; DESCRIPTION: Unlock list, multiple processor version ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; UnlockListMultiple Proc near push ds push ax ; mov ax,SEG data mov ds,ax mov ds:list_lock,0 sti ; pop ax pop ds ret UnlockListMultiple Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: LockKernelSectionMultiple ; ; DESCRIPTION: Lock kernel critical section, multiple processor version ; ; PARAMETERS: DS:ESI Section ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; LockKernelSectionMultiple Proc near push ax lksmSpinLock: mov ax,ds:[esi].cs_lock or ax,ax je lksmGet ; pause jmp lksmSpinLock lksmGet: inc ax xchg ax,ds:[esi].cs_lock or ax,ax je lksmDone ; jmp lksmSpinLock lksmDone: pop ax ret LockKernelSectionMultiple Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: UnlockKernelSectionMultiple ; ; DESCRIPTION: Unlock kernel critical section, multiple processor version ; ; PARAMETERS: DS:ESI Section ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; UnlockKernelSectionMultiple Proc near mov ds:[esi].cs_lock,0 sti ret UnlockKernelSectionMultiple Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: LockUserSectionMultiple ; ; DESCRIPTION: Lock user critical section, multiple processor version ; ; PARAMETERS: DS:EBX Section ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; LockUserSectionMultiple Proc near push ax lusmSpinLock: mov ax,ds:[ebx].us_lock or ax,ax je lusmGet ; pause jmp lusmSpinLock lusmGet: inc ax xchg ax,ds:[ebx].us_lock or ax,ax je lusmDone ; jmp lusmSpinLock lusmDone: pop ax ret LockUserSectionMultiple Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: UnlockUserSectionMultiple ; ; DESCRIPTION: Unlock user critical section, multiple processor version ; ; PARAMETERS: DS:EBX Section ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; UnlockUserSectionMultiple Proc near mov ds:[ebx].us_lock,0 sti ret UnlockUserSectionMultiple Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: LockFutexMultiple ; ; DESCRIPTION: Lock futex, multiple processor version ; ; PARAMETERS: DS:EBX Futex handle data ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; LockFutexMultiple Proc near push ax lfSpinLock: mov ax,ds:[ebx].fh_lock or ax,ax je lfGet ; pause jmp lfSpinLock lfGet: inc ax xchg ax,ds:[ebx].fh_lock or ax,ax je lfDone ; jmp lfSpinLock lfDone: pop ax ret LockFutexMultiple Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: UnlockFutexMultiple ; ; DESCRIPTION: Unlock futex, multiple processor version ; ; PARAMETERS: DS:EBX Futex handle data ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; UnlockFutexMultiple Proc near mov ds:[ebx].fh_lock,0 sti ret UnlockFutexMultiple Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: TryLockCore ; ; DESCRIPTION: Try to lock ; ; RETURNS: CY Owner of section ; FS Core selector ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; TryLockCore Proc near cli push word ptr core_data_sel pop fs mov fs,fs:ps_sel lock add fs:ps_nesting,1 ret TryLockCore Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: LockCore ; ; DESCRIPTION: Lock ; ; RETURNS: CY Owner of section ; FS Core selector ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; LockCore Proc near cli push word ptr core_data_sel pop fs mov fs,fs:ps_sel lock add fs:ps_nesting,1 jc lcDone ; mov ax,fs:ps_nesting or ax,ax je lcDone ; CrashGate lcDone: ret LockCore Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: TryUnlockCore ; ; DESCRIPTION: Try to unlock ; ; PARAMETERS: FS Core selector ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; TryUnlockCore Proc near push eax tucRetry: cli lock sub fs:ps_nesting,1 jnc tucDone ; test fs:ps_flags,PS_FLAG_TIMER_EXPIRED jz tucTimerOk ; lock add fs:ps_nesting,1 jc tucHandleTimer ; mov ax,fs:ps_nesting or ax,ax jz tucHandleTimer ; CrashGate tucHandleTimer: sti pushad call cs:update_timer_proc popad jmp tucRetry tucTimerOk: mov eax,fs:ps_tlb.pt32_used or eax,eax jz tucTlbDone ; lock add fs:ps_nesting,1 jc tucFlush ; mov ax,fs:ps_nesting or ax,ax jz tucFlush ; CrashGate tucFlush: sti call cs:flush_tlb_proc jmp tucRetry tucTlbDone: mov ax,fs:ps_curr_thread or ax,ax jz tucDone ; test fs:ps_flags,PS_FLAG_PREEMPT jnz tucSwap ; mov ax,fs:ps_wakeup_list or ax,ax jz tucDone tucSwap: lock add fs:ps_nesting,1 jc tucSched ; mov ax,fs:ps_nesting or ax,ax jz tucSched ; CrashGate tucSched: sti push OFFSET tucDone call SaveLockedThread jmp ContinueCurrentThread tucDone: sti pop eax ret TryUnlockCore Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: UnlockCore ; ; DESCRIPTION: Unlock ; ; PARAMETERS: FS Core selector ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; UnlockCore Proc near push eax ucRetry: cli lock sub fs:ps_nesting,1 jc ucNestOk ; mov ax,fs:ps_nesting cmp ax,-1 je ucNestOk ; CrashGate ucNestOk: test fs:ps_flags,PS_FLAG_TIMER_EXPIRED jz ucTimerOk ; lock add fs:ps_nesting,1 jc ucHandleTimer ; mov ax,fs:ps_nesting or ax,ax jz ucHandleTimer ; CrashGate ucHandleTimer: sti pushad call cs:update_timer_proc popad jmp ucRetry ucTimerOk: mov eax,fs:ps_tlb.pt32_used or eax,eax jz ucTlbDone ; lock add fs:ps_nesting,1 jc ucFlush ; mov ax,fs:ps_nesting or ax,ax jz ucFlush ; CrashGate ucFlush: sti call cs:flush_tlb_proc jmp ucRetry ucTlbDone: test fs:ps_flags,PS_FLAG_PREEMPT jnz ucSwap ; mov ax,fs:ps_wakeup_list or ax,ax jz ucDone ucSwap: lock add fs:ps_nesting,1 jc ucSched ; mov ax,fs:ps_nesting or ax,ax jz ucSched ; CrashGate ucSched: sti push OFFSET ucDone call SaveLockedThread jmp ContinueCurrentThread ucDone: sti pop eax ret UnlockCore Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: LoadUnlockCore ; ; DESCRIPTION: Thread load unlock ; ; PARAMETERS: FS Core selector ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; LoadUnlockCore Proc near cli lock sub fs:ps_nesting,1 jc lulcDone ; mov ax,fs:ps_nesting or ax,ax jne lulcCrash ; lock sub fs:ps_nesting,1 jc lulcDone lulcCrash: mov ax,fs:ps_nesting cmp ax,-1 je lulcDone ; CrashGate lulcDone: ret LoadUnlockCore Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: IrqSchedule ; ; DESCRIPTION: IRQ scheduling ; ; PARAMETERS: FS Core ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; irq_schedule_name DB 'IRQ Schedule',0 irq_schedule Proc far push OFFSET irqsDone call SaveLockedThread jmp ContinueCurrentThread irqsDone: retf32 irq_schedule Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: TryLockTask ; ; DESCRIPTION: Try lock task ; ; RETURNS: FS Core sel ; NC Not previously locked ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; try_lock_task_name DB 'Try Lock Task',0 try_lock_task Proc far push ds call TryLockCore pop ds retf32 try_lock_task Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: LockTask ; ; DESCRIPTION: Lock task ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; lock_task_name DB 'Lock Task',0 lock_task Proc far push ds push fs ; call LockCore ; pop fs pop ds retf32 lock_task Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: UnlockTask ; ; DESCRIPTION: Unlock task ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; unlock_task_name DB 'Unlock Task',0 unlock_task Proc far push ds push fs push ax ; pushf cli mov ax,core_data_sel mov fs,ax mov fs,fs:ps_sel popf call UnlockCore ; pop ax pop fs pop ds retf32 unlock_task Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: EnterLongInt ; ; DESCRIPTION: Enter long mode int ; ; RETURNS: EAX Locked core selector (high) and thread (low) ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; enter_long_int_name DB 'Enter Long Int',0 enter_long_int Proc far cli mov ax,core_data_sel mov ds,ax lock add ds:ps_nesting,1 mov ax,ds:ps_sel mov fs,ax shl eax,16 ; mov ax,fs:ps_curr_thread mov es,ax or ax,ax jz eliDone ; inc es:p_nest_count mov ax,es:p_nest_count cmp ax,10 jb eliDone ; mov ax,es:p_nest_unwind or ax,ax jnz eliDone ; lock add fs:ps_nesting,1 mov es:p_nest_unwind,1 eliDone: mov ax,es retf32 enter_long_int Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: LeaveLongInt ; ; DESCRIPTION: Leave long mode int ; ; PARAMETERS: EAX Locked core selector (high) and thread (low) ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; leave_long_int_name DB 'Leave Long Int',0 leave_long_int Proc far mov es,ax shr eax,16 mov fs,ax lliRetry: cli lock sub fs:ps_nesting,1 jnc lliStillLocked ; test fs:ps_flags,PS_FLAG_TIMER_EXPIRED jz lliTimerOk ; lock add fs:ps_nesting,1 jc lliHandleTimer ; CrashGate lliHandleTimer: sti pushad call cs:update_timer_proc popad jmp lliRetry lliTimerOk: mov ax,es or ax,ax jz lliDone ; mov eax,fs:ps_tlb.pt32_used or eax,eax jz lliTlbDone ; lock add fs:ps_nesting,1 jc lliFlush ; CrashGate lliFlush: sti call cs:flush_tlb_proc jmp lliRetry lliTlbDone: test fs:ps_flags,PS_FLAG_PREEMPT jnz lliSwap ; mov ax,fs:ps_wakeup_list or ax,ax jz lliDecNest lliSwap: lock add fs:ps_nesting,1 jc lliSched ; CrashGate lliSched: sti push OFFSET lliSwapDone call SaveLockedThread jmp ContinueCurrentThread lliSwapDone: cli mov ax,core_data_sel mov fs,ax mov fs,fs:ps_sel mov es,fs:ps_curr_thread mov ax,fs:ps_nesting cmp ax,-1 je lliDecNest lliStillLocked: mov ax,es or ax,ax jz lliDone ; mov ax,es:p_nest_count cmp ax,1 jne lliDecNest lliLast: xor ax,ax xchg ax,es:p_nest_unwind or ax,ax jnz lliRetry lliDecNest: dec es:p_nest_count lliDone: retf32 leave_long_int Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: FlushTlb386 ; ; DESCRIPTION: Flush TLB entries, 386 processor version ; ; PARAMETERS: FS Core ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; FlushTlb386 Proc near push eax ft386Again: mov eax,fs:ps_tlb.pt32_used or eax,eax jz ft386Done ; mov fs:ps_tlb.pt32_used,0 mov eax,cr3 mov cr3,eax jmp ft386Again ft386Done: pop eax ret FlushTlb386 Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: FlushTlb486 ; ; DESCRIPTION: Flush TLB entries, 486 or higher processor version ; ; PARAMETERS: FS Core ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; FlushTlb486 Proc near ft486Again: mov eax,fs:ps_tlb.pt32_used or eax,eax jz ft486Done ; cmp eax,-1 je ft486All ; push es pushad ; mov cx,fs mov es,cx mov cx,32 mov si,OFFSET ps_tlb.pt32_linear_arr mov di,OFFSET ps_work_tlb.pt32_linear_arr rep movs dword ptr es:[di],es:[si] lock xor fs:ps_tlb.pt32_used,eax ; mov cx,flat_sel mov es,cx mov cx,32 mov di,OFFSET ps_work_tlb.pt32_linear_arr mov esi,1 ft486Loop: test esi,eax jz ft486Next ; mov edx,fs:[di] invlpg es:[edx] ft486Next: add di,4 shl esi,1 sub cx,1 jnz ft486Loop ; popad pop es jmp ft486Again ft486All: mov fs:ps_tlb.pt32_used,0 mov eax,cr3 mov cr3,eax jmp ft486Again ft486Done: ret FlushTlb486 Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: DoFlushTlb ; ; DESCRIPTION: Flush TLB callback from leave int ; ; PARAMETERS: FS Core, locked ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; do_flush_tlb_name DB 'Do Flush TLB', 0 do_flush_tlb Proc far call cs:flush_tlb_proc retf32 do_flush_tlb ENDP ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: AddCoreTlb ; ; DESCRIPTION: Add core TLB ; ; PARAMETERS: EDX Linear address ; FS Core ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; AddCoreTlb Proc near push eax push ecx push di ; cmp edx,system_mem_start jae actTryLock ; mov eax,cr3 and ax,0F000h cmp eax,fs:ps_cr3 jne actDone actTryLock: mov eax,fs:ps_tlb.pt32_used or eax,fs:ps_tlb.pt32_locked not eax bsf ecx,eax jnz actHasEntry ; mov fs:ps_tlb.pt32_used,-1 mov fs:ps_tlb.pt32_locked,0 jmp actDone actHasEntry: cli lock bts fs:ps_tlb.pt32_locked,ecx jnc actLockOk ; sti pause jmp actTryLock actLockOk: mov di,cx shl di,2 mov fs:[di].ps_tlb.pt32_linear_arr,edx ; lock bts fs:ps_tlb.pt32_used,ecx jnc actUnlock ; lock btc fs:ps_tlb.pt32_locked,ecx sti pause jmp actTryLock actUnlock: lock btc fs:ps_tlb.pt32_locked,ecx sti actDone: pop di pop ecx pop eax ret AddCoreTlb Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: FlushTLB ; ; DESCRIPTION: Flush TLB entries ; ; PARAMETERS: CX Number of entries ; EDX Linear base ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; flush_tlb_name DB 'Flush Tlb', 0 flush_tlb Proc far push fs pushad ; or cx,cx jz nfDone ftlbLoop: push cx push edx ; mov cx,cs:core_count or cx,cx jz ateDone ; mov si,OFFSET core_arr ateLoop: mov fs,cs:[si] test fs:ps_flags,PS_FLAG_ACTIVE jz ateNext ; call AddCoreTlb ateNext: add si,2 sub cx,1 jnz ateLoop ateDone: pop edx pop cx ; add edx,1000h sub cx,1 jnz ftlbLoop ; mov cx,cs:core_count or cx,cx jz nfDone ; call TryLockCore sti mov dx,fs mov si,OFFSET core_arr nfLoop: mov bx,cs:[si] cmp dx,bx je nfNext ; mov fs,bx mov eax,fs:ps_tlb.pt32_used or eax,eax jz nfNext ; test fs:ps_flags,PS_FLAG_ACTIVE jz nfNext ; mov ax,fs:ps_nesting cmp ax,-1 jne nfNext ; mov al,81h SendInt nfNext: add si,2 sub cx,1 jnz nfLoop ; mov fs,dx call TryUnlockCore nfDone: popad pop fs retf32 flush_tlb Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: UpdateOwnTimer ; ; DESCRIPTION: Update timers without preempt ; ; PARAMETERS: FS locked core ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; UpdateOwnTimer Proc near push ds ; mov ax,SEG data mov ds,ax lock and fs:ps_flags,NOT PS_FLAG_TIMER_EXPIRED uotCheck: GetSystemTime call LockTimerGlobal add eax,cs:update_tics adc edx,0 mov bx,ds:timer_head sub eax,ds:[bx].timer_lsb sbb edx,ds:[bx].timer_msb jc uotReload ; call LocalRemoveTimerGlobal jmp uotCheck uotReload: neg eax ReloadSysTimer jnc uotDone ; call UnlockTimerGlobal jmp uotCheck uotDone: call UnlockTimerGlobal ; pop ds ret UpdateOwnTimer Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: UpdateCombinedTimer ; ; DESCRIPTION: Update combined timer ; ; PARAMETERS: FS locked core ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; UpdateCombinedTimer Proc near push ds ; mov ax,SEG data mov ds,ax lock and fs:ps_flags,NOT PS_FLAG_TIMER_EXPIRED uctLoop: GetSystemTime call LockTimerCore add eax,cs:update_tics adc edx,0 mov bx,fs:ps_timer_head mov ecx,fs:ps_preempt_msb cmp ecx,fs:[bx].timer_msb jc uctPreempt jnz uctTimer ; mov ecx,fs:ps_preempt_lsb cmp ecx,fs:[bx].timer_lsb jc uctPreempt uctTimer: sub eax,fs:[bx].timer_lsb sbb edx,fs:[bx].timer_msb jc uctReload ; call LocalRemoveTimerCore jmp uctLoop uctPreempt: sub eax,fs:ps_preempt_lsb sbb edx,fs:ps_preempt_msb jc uctReload ; call UnlockTimerCore lock or fs:ps_flags,PS_FLAG_PREEMPT ; GetSystemTime add eax,1193 adc edx,0 mov fs:ps_preempt_lsb,eax mov fs:ps_preempt_msb,edx jmp uctLoop uctReload: neg eax ReloadSysPreemptTimer call UnlockTimerCore ; pop ds ret UpdateCombinedTimer Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: UseCombinedTimer ; ; DESCRIPTION: Use combined timer ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; use_own_preempt_timer_name DB 'Use Own Preempt Timer', 0 use_own_preempt_timer Proc far push ds push ax ; mov ax,SEG data mov ds,ax mov ds,ds:patch_sel mov ds:update_timer_proc,OFFSET UpdateOwnTimer ; pop ax pop ds retf32 use_own_preempt_timer Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: TimerExpired ; ; DESCRIPTION: Timer expired notification from normal IRQ ; ; PARAMETERS: FS Locked core ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; timer_expired_name DB 'Timer Expired', 0 timer_expired Proc far call cs:update_timer_proc retf32 timer_expired Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: PreemptExpired ; ; DESCRIPTION: Preemption expired notification ; ; PARAMETERS: ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; preempt_expired_name DB 'Preempt Expired', 0 preempt_expired Proc far call TryLockCore sti lock or fs:ps_flags,PS_FLAG_PREEMPT call TryUnlockCore retf32 preempt_expired Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: START_TIMER ; ; DESCRIPTION: Start a timer, global version ; ; PARAMETERS: EDX:EAX Timeout time ; ES:EDI Callback ; BX Owner (selector ID) ; ECX ID passed to callback ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; start_global_timer_name DB 'Start Timer',0 start_global_timer PROC far push ds push fs push bx push si ; mov si,SEG data mov ds,si ; call LockTimerGlobal ; mov si,ds:timer_free mov ds:[si].timer_owner,bx mov bx,ds:[si].timer_next mov ds:timer_free,bx mov ds:[si].timer_lsb,eax mov ds:[si].timer_msb,edx mov ds:[si].timer_id,ecx mov ds:[si].timer_offset,edi mov ds:[si].timer_sel,es mov bx,OFFSET timer_head push si mov si,bx mov bx,ds:[bx].timer_next cmp edx,ds:[bx].timer_msb jc start_global_insert_first jnz start_global_try_next cmp eax,ds:[bx].timer_lsb jnc start_global_try_next start_global_insert_first: pop ds:[si].timer_next mov si,ds:[si].timer_next mov ds:[si].timer_next,bx ; call TryLockCore call UnlockTimerGlobal lock or fs:ps_flags,PS_FLAG_TIMER_EXPIRED call TryUnlockCore jmp start_global_done start_global_try_next: mov si,bx mov bx,ds:[bx].timer_next cmp edx,ds:[bx].timer_msb jc start_global_insert jnz start_global_try_next cmp eax,ds:[bx].timer_lsb jnc start_global_try_next start_global_insert: pop ds:[si].timer_next mov si,ds:[si].timer_next mov ds:[si].timer_next,bx call UnlockTimerGlobal start_global_done: pop si pop bx pop fs pop ds retf32 start_global_timer ENDP ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: STOP_TIMER ; ; DESCRIPTION: Stop timer, global version ; ; PARAMETERS: BX Owner ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; stop_global_timer_name DB 'Stop Timer',0 stop_global_timer PROC far push ds push ax push bx push cx push si ; mov ax,SEG data mov ds,ax ; call LockTimerGlobal ; mov cx,bx mov bx,OFFSET timer_head timer_global_stop_next: mov si,bx mov bx,ds:[bx].timer_next or bx,bx je timer_global_stop_done cmp cx,ds:[bx].timer_owner jne timer_global_stop_next timer_global_stop_this: mov ax,ds:[bx].timer_next mov ds:[si].timer_next,ax mov ax,ds:timer_free mov ds:[bx].timer_next,ax mov ds:timer_free,bx timer_global_stop_done: call UnlockTimerGlobal ; pop si pop cx pop bx pop ax pop ds retf32 stop_global_timer ENDP ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: START_TIMER ; ; DESCRIPTION: Start a timer, per core version ; ; PARAMETERS: EDX:EAX Timeout time ; ES:EDI Callback ; BX Owner (selector ID) ; ECX ID passed to callback ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; start_core_timer_name DB 'Start Timer',0 start_core_timer PROC far push ds push es push fs pushad ; call TryLockCore sti call LockTimerCore ; mov si,fs:ps_timer_free mov fs:[si].timer_owner,bx mov bx,fs:[si].timer_next mov fs:ps_timer_free,bx mov fs:[si].timer_lsb,eax mov fs:[si].timer_msb,edx mov fs:[si].timer_id,ecx mov fs:[si].timer_offset,edi mov fs:[si].timer_sel,es mov bx,OFFSET ps_timer_head push si start_core_try_next: mov si,bx mov bx,fs:[bx].timer_next cmp edx,fs:[bx].timer_msb jc start_core_insert jnz start_core_try_next cmp eax,fs:[bx].timer_lsb jnc start_core_try_next start_core_insert: pop fs:[si].timer_next mov si,fs:[si].timer_next mov fs:[si].timer_next,bx ; call UnlockTimerCore lock or fs:ps_flags,PS_FLAG_TIMER_EXPIRED call TryUnlockCore ; popad pop fs pop es pop ds retf32 start_core_timer ENDP ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: STOP_TIMER ; ; DESCRIPTION: Stop timer, per core version ; ; PARAMETERS: BX Owner ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; stop_core_timer_name DB 'Stop Timer',0 stop_core_timer PROC far push fs push ax push bx push cx push si ; mov cx,cs:core_count mov si,OFFSET core_arr stop_core_loop: push bx push cx push si ; mov fs,cs:[si] call LockTimerCore ; mov cx,bx mov bx,OFFSET ps_timer_head core_timer_stop_next: mov si,bx mov bx,fs:[bx].timer_next or bx,bx je core_timer_stop_done cmp cx,fs:[bx].timer_owner jne core_timer_stop_next core_timer_stop_this: mov ax,fs:[bx].timer_next mov fs:[si].timer_next,ax mov ax,fs:ps_timer_free mov fs:[bx].timer_next,ax mov fs:ps_timer_free,bx core_timer_stop_done: call UnlockTimerCore ; pop si pop cx pop bx add si,2 loop stop_core_loop ; pop si pop cx pop bx pop ax pop fs retf32 stop_core_timer ENDP ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: init_first_thread ; ; DESCRIPTION: Init first thread ; ; PARAMETERS: ES Thread control block ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; init_first_thread: mov eax,cr0 or al,8 mov cr0,eax StartSyscall ; mov ax,es mov ds,ax mov ax,ds:p_ss mov ds:p_kernel_ss,ax mov eax,dword ptr ds:p_rsp mov ds:p_kernel_esp,eax ; call LockCore sti mov di,es:p_prio mov fs:ps_prio_act,di call InsertCoreBlock ; mov bx,core_data_sel mov fs,bx mov fs,fs:ps_sel ; mov ax,start_preempt_timer_nr IsValidOsGate jc preempt_timer_combined ; StartSysTimer mov bx,SEG data mov ds,bx mov ds,ds:patch_sel mov ds:update_tics,eax xor ax,ax mov ds,ax ; StartPreemptTimer mov bx,SEG data mov ds,bx mov ds,ds:patch_sel mov ds:preempt_reload_proc,OFFSET PreemptReload ; mov bx,SEG data mov ds,bx GetSystemTime mov ds:sys_lsb_tics_base,eax mov ds:sys_msb_tics_base,edx ; mov ax,cs mov ds,ax mov es,ax ; mov si,OFFSET start_global_timer mov di,OFFSET start_global_timer_name xor cl,cl mov ax,start_timer_nr RegisterOsGate ; mov si,OFFSET stop_global_timer mov di,OFFSET stop_global_timer_name xor cl,cl mov ax,stop_timer_nr RegisterOsGate jmp LoadThread preempt_timer_combined: StartSysPreemptTimer mov bx,SEG data mov ds,bx mov ds,ds:patch_sel mov ds:update_tics,eax ; mov ax,cs mov ds,ax mov es,ax ; mov si,OFFSET start_core_timer mov di,OFFSET start_core_timer_name xor cl,cl mov ax,start_timer_nr RegisterOsGate ; mov si,OFFSET stop_core_timer mov di,OFFSET stop_core_timer_name xor cl,cl mov ax,stop_timer_nr RegisterOsGate jmp LoadThread ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: WAKE_NEW ; ; DESCRIPTION: Wake-up a newly create thread ; ; PARAMETERS: ES Thread ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; wake_new PROC near pushf mov dx,es push OFFSET wake_new_done call SaveCurrentThread ; mov es,dx call cs:insert_wakeup_proc jmp ContinueCurrentThread wake_new_other_core: CrashGate wake_new_done: popf ret wake_new ENDP ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: CleanupThread ; ; DESCRIPTION: Free resources for current thread ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; cleanup_thread: call SkipCurrentThread mov ax,fs:ps_curr_thread cmp ax,fs:ps_math_thread jne cleanup_thread_math_ok ; lock and fs:ps_flags,NOT PS_FLAG_FPU mov fs:ps_math_thread,0 cleanup_thread_math_ok: mov bx,cs:system_thread Signal ; mov es,fs:ps_curr_thread mov fs:ps_curr_thread,0 ; mov ax,SEG data mov ds,ax mov edi,OFFSET term_thread_list ; call cs:lock_list_proc call InsertBlock32 call cs:unlock_list_proc ; xor ax,ax mov es,ax jmp LoadThread ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: CleanupProcess ; ; DESCRIPTION: Free resources for current process ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; cleanup_process: call SkipCurrentThread mov ax,fs:ps_curr_thread cmp ax,fs:ps_math_thread jne cleanup_process_math_ok ; lock and fs:ps_flags,NOT PS_FLAG_FPU mov fs:ps_math_thread,0 cleanup_process_math_ok: mov bx,cs:system_thread Signal ; mov es,fs:ps_curr_thread mov fs:ps_curr_thread,0 ; mov ax,SEG data mov ds,ax mov edi,OFFSET term_proc_list ; call cs:lock_list_proc call InsertBlock32 call cs:unlock_list_proc ; xor ax,ax mov es,ax jmp LoadThread ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: WAKE ; ; DESCRIPTION: Wake up thread ; ; PARAMETERS: DS:SI Thread list ; EAX Status to thread ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; wake_thread_name DB 'Wake',0 wake_thread PROC far push dx push esi ; mov dx,ds movzx esi,si call WakeThread ; pop esi pop dx retf32 wake_thread ENDP ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: SLEEP ; ; DESCRIPTION: Suspend thread ; ; PARAMETERS: DS:DI Suspend list ; ; RETURNS: EAX Wake-up status ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; sleep_thread_name DB 'Sleep',0 sleep_thread PROC far push ds pushf mov ax,ds ; push OFFSET sleep_thread_done call SaveCurrentThread ; mov ds,ax movzx edi,di mov es,fs:ps_curr_thread mov fs:ps_curr_thread,0 ; call cs:lock_list_proc call InsertBlock32 call cs:unlock_list_proc ; mov es:p_sleep_type,SLEEP_TYPE_SUSPEND mov es:p_sleep_sel,ds mov es:p_sleep_offset,edi jmp LoadThread sleep_thread_done: GetThread mov ds,ax mov eax,ds:p_data popf pop ds retf32 sleep_thread ENDP ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: ClearSignal ; ; DESCRIPTION: Set status to unsignaled ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; clear_signal_name DB 'Clear Signal',0 clear_signal PROC far push ds push ax ; mov ax,core_data_sel mov ds,ax mov ds,ds:ps_curr_thread mov ds:p_signal,0 ; pop ax pop ds retf32 clear_signal ENDP ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: SIGNAL_THREAD ; ; DESCRIPTION: Send a signal to a thread ; ; PARAMETERS: BX Thread to signal ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; signal_thread_name DB 'Signal',0 signal_thread PROC far push ax push es push fs ; or bx,bx jz signal_done ; mov es,bx call TryLockCore sti call cs:lock_signal_proc mov es:p_signal,1 ; mov ax,es:p_sleep_type cmp ax,SLEEP_TYPE_SIGNAL jne signal_unlock_signal ; mov es:p_sleep_type,0 mov es:p_signal,0 call cs:unlock_signal_proc call cs:insert_wakeup_proc jmp signal_unlock_core signal_unlock_signal: call cs:unlock_signal_proc signal_unlock_core: call TryUnlockCore signal_done: pop fs pop es pop ax retf32 signal_thread ENDP ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: WaitForSignal ; ; DESCRIPTION: Wait for a signal ; ; PARAMETERS: ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; wait_for_signal_name DB 'Wait For Signal',0 wait_for_signal PROC far push es push fs push ax ; call LockCore sti mov es,fs:ps_curr_thread call cs:lock_signal_proc ; xor al,al xchg al,es:p_signal or al,al jnz wait_for_signal_unlock ; mov es:p_sleep_type,SLEEP_TYPE_SIGNAL call cs:unlock_signal_proc ; push OFFSET wait_for_signal_done call SaveLockedThread xor ax,ax mov es,ax mov fs:ps_curr_thread,ax jmp LoadThread wait_for_signal_unlock: call cs:unlock_signal_proc call UnlockCore wait_for_signal_done: pop ax pop fs pop es retf32 wait_for_signal ENDP ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: signal_timeout ; ; DESCRIPTION: Signal timeout handler ; ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; signal_timeout PROC far mov bx,cx Signal retf32 signal_timeout Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: WaitForSignalWithTimeout ; ; DESCRIPTION: Wait for a signal with timeout ; ; PARAMETERS: EDX:EAX Timeout ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; wait_for_signal_timeout_name DB 'Wait For Signal With Timeout',0 wait_for_signal_timeout PROC far push es push fs push ax push bx push cx push edi ; call LockCore sti mov es,fs:ps_curr_thread call cs:lock_signal_proc ; xor al,al xchg al,es:p_signal or al,al jnz wait_for_signal_timeout_unlock ; mov es:p_sleep_type,SLEEP_TYPE_SIGNAL call cs:unlock_signal_proc ; mov bx,es mov cx,cs mov es,cx mov edi,OFFSET signal_timeout mov cx,bx StartTimer ; push OFFSET wait_for_signal_timeout_clear call SaveLockedThread xor ax,ax mov es,ax mov fs:ps_curr_thread,ax jmp LoadThread wait_for_signal_timeout_unlock: call cs:unlock_signal_proc call UnlockCore jmp wait_for_signal_timeout_done wait_for_signal_timeout_clear: GetThread mov bx,ax StopTimer wait_for_signal_timeout_done: pop edi pop cx pop bx pop ax pop fs pop es retf32 wait_for_signal_timeout ENDP ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: enter_section ; ; DESCRIPTION: Enter section ; ; PARAMETERS: DS:ESI ADDRESS OF SECTION ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; enter_section_name DB 'Enter Critical Section',0 enter_section PROC far push ax push dx push fs ; call LockCore sti call cs:lock_kernel_section_proc mov dx,ds:[esi].cs_list cmp dx,-1 jne ecsBlock ; mov ds:[esi].cs_list,0 jmp ecsUnlock ecsBlock: mov ax,ds push OFFSET ecsDone call SaveLockedThread ; mov ds,ax lea edi,[esi].cs_list mov es,fs:ps_curr_thread mov fs:ps_curr_thread,0 ; call InsertBlock32 call cs:unlock_kernel_section_proc ; mov es:p_sleep_type,SLEEP_TYPE_SECTION mov es:p_sleep_sel,ds mov es:p_sleep_offset,esi jmp LoadThread ecsUnlock: call cs:unlock_kernel_section_proc call UnlockCore ecsDone: pop fs pop dx pop ax retf32 enter_section ENDP ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: LeaveSection ; ; DESCRIPTION: Leave section ; ; PARAMETERS: DS:ESI ADDRESS OF SECTION ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; leave_section_name DB 'Leave Critical Section',0 leave_section PROC far push ax push dx push fs ; call LockCore sti call cs:lock_kernel_section_proc mov ax,ds:[esi].cs_list cmp ax,-1 je lcsUnlockList ; or ax,ax jnz lcsUnblock ; mov ds:[esi].cs_list,-1 jmp lcsUnlockList lcsUnblock: push es push esi push di ; add esi,OFFSET cs_list call RemoveBlock32 mov es:p_data,0 sub esi,OFFSET cs_list call cs:unlock_kernel_section_proc ; call cs:insert_wakeup_proc lcsUnblocked: pop di pop esi pop es jmp lcsUnlock lcsUnlockList: call cs:unlock_kernel_section_proc lcsUnlock: call UnlockCore lcsDone: pop fs pop dx pop ax retf32 leave_section ENDP ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: cond_section_timeout ; ; DESCRIPTION: Timeout on enter section ; ; PARAMETERS: CX thread to remove from list ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; cond_section_timeout Proc far push ds push es push fs push esi ; mov es,cx call TryLockCore sti ; mov eax,es:p_data cmp eax,-1 jne cstDone ; mov ds,es:p_sleep_sel mov esi,es:p_sleep_offset ; sub esi,OFFSET cs_list call cs:lock_kernel_section_proc ; add esi,OFFSET cs_list mov ax,[esi] or ax,ax jz cstUnlock ; cmp ax,-1 je cstUnlock ; call RemoveBlock32 cstUnlock: sub esi,OFFSET cs_list call cs:unlock_kernel_section_proc cstDone: call TryUnlockCore ; pop esi pop fs pop es pop ds retf32 cond_section_timeout Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: cond_enter_section ; ; DESCRIPTION: Conditional enter section ; ; PARAMETERS: DS:ESI ADDRESS OF SECTION ; EAX Timeout in milliseconds ; ; RETURNS: CY Entered ok ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; cond_enter_section_name DB 'Conditional Enter Critical Section',0 cond_enter_section PROC far push eax push edx push fs ; call LockCore sti call cs:lock_kernel_section_proc push ds mov ds,fs:ps_curr_thread mov ds:p_data,-1 pop ds ; mov dx,ds:[esi].cs_list cmp dx,-1 jne cecsBlock ; mov ds:[esi].cs_list,0 jmp cecsUnlockOk cecsBlock: or eax,eax jz cecsUnlockFail ; push es push bx push ecx push edi ; mov eax,1193 mul ecx push edx push eax GetSystemTime pop ecx add eax,ecx pop ecx adc edx,ecx ; push ds mov cx,SEG data mov ds,cx mov cx,cs mov es,cx mov edi,OFFSET cond_section_timeout mov bx,fs:ps_curr_thread mov cx,bx StartTimer pop ds ; pop edi pop ecx pop bx pop es ; mov ax,ds push OFFSET cecsDone call SaveLockedThread ; mov ds,ax lea edi,[esi].cs_list mov es,fs:ps_curr_thread mov fs:ps_curr_thread,0 ; call InsertBlock32 call cs:unlock_kernel_section_proc ; mov es:p_sleep_sel,ds mov es:p_sleep_offset,edi jmp LoadThread cecsUnlockOk: call cs:unlock_kernel_section_proc call UnlockCore stc jmp cecsLeave cecsUnlockFail: call cs:unlock_kernel_section_proc call UnlockCore clc jmp cecsLeave cecsDone: call LockCore sti push ds push bx ; mov bx,fs:ps_curr_thread StopTimer ; mov ds,bx mov eax,ds:p_data call UnlockCore ; pop bx pop ds or eax,eax stc jz cecsLeave ; clc cecsLeave: pop fs pop edx pop eax retf32 cond_enter_section ENDP ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: create_user_section ; ; DESCRIPTION: Create user section ; ; RETURNS: BX Section handle ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; create_user_section_name DB 'Create User Section',0 create_user_section PROC far push ds push eax push cx ; mov cx,SIZE section_handle_seg AllocateHandle mov ds:[ebx].us_value,0 mov ds:[ebx].us_list,0 mov ds:[ebx].us_owner,0 mov ds:[ebx].us_count,0 mov ds:[ebx].us_lock,0 mov [ebx].hh_sign,SECTION_HANDLE mov bx,[ebx].hh_handle clc ; pop cx pop eax pop ds retf32 create_user_section ENDP ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: create_named_user_section ; ; DESCRIPTION: Create named user section ; ; PARAMETERS: ES:(E)DI Section name ; ; RETURNS: BX Section handle ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; create_named_user_section_name DB 'Create Named User Section',0 create_named_user_section PROC near push ds push eax push cx ; mov cx,SIZE section_handle_seg AllocateHandle mov ds:[ebx].us_value,0 mov ds:[ebx].us_list,0 mov ds:[ebx].us_owner,0 mov ds:[ebx].us_count,0 mov ds:[ebx].us_lock,0 mov [ebx].hh_sign,SECTION_HANDLE mov bx,[ebx].hh_handle clc ; pop cx pop eax pop ds ret create_named_user_section ENDP create_named_user_section16 Proc far push edi ; movzx edi,di call create_named_user_section ; pop edi retf32 create_named_user_section16 Endp create_named_user_section32 Proc far call create_named_user_section retf32 create_named_user_section32 Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: create_blocked_user_section ; ; DESCRIPTION: Create blocked user section ; ; RETURNS: BX Section handle ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; create_blocked_user_section_name DB 'Create Blocked User Section',0 create_blocked_user_section PROC far push ds push eax push cx ; mov cx,SIZE section_handle_seg AllocateHandle mov ds:[ebx].us_value,-1 mov ds:[ebx].us_list,0 mov ds:[ebx].us_owner,-1 mov ds:[ebx].us_count,1 mov [ebx].hh_sign,SECTION_HANDLE mov bx,[ebx].hh_handle clc ; pop cx pop eax pop ds retf32 create_blocked_user_section ENDP ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: delete_user_section ; ; DESCRIPTION: Delete user section ; ; PARAMETERS: BX Section handle ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; delete_user_section_name DB 'Delete User Section',0 delete_user_section PROC far push ds push ebx ; mov ax,SECTION_HANDLE DerefHandle jc free_section_done ; FreeHandle clc free_section_done: pop ebx pop ds retf32 delete_user_section ENDP ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: enter_user_section ; ; DESCRIPTION: Enter user section ; ; PARAMETERS: BX Section handle ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; enter_user_section_name DB 'Enter User Section',0 enter_user_section PROC far push ax push ebx push dx push ds push fs ; mov ax,SECTION_HANDLE DerefHandle jc eusEnd ; lock sub ds:[ebx].us_value,1 jc eusDone ; mov ax,core_data_sel mov fs,ax mov ax,fs:ps_curr_thread cmp ax,ds:[ebx].us_owner je eusDone ; call LockCore sti call cs:lock_user_section_proc mov ax,ds:[ebx].us_list cmp ax,-1 jne eusBlock ; mov ds:[ebx].us_list,0 jmp eusUnlock eusBlock: mov ax,ds push OFFSET eusDone call SaveLockedThread ; mov ds,ax lea edi,[ebx].us_list mov es,fs:ps_curr_thread mov fs:ps_curr_thread,0 ; call InsertBlock32 call cs:unlock_user_section_proc ; mov es:p_sleep_type,SLEEP_TYPE_SECTION mov es:p_sleep_sel,ds mov es:p_sleep_offset,edi jmp LoadThread eusUnlock: call cs:unlock_user_section_proc call UnlockCore eusDone: mov ax,core_data_sel mov fs,ax mov ax,fs:ps_curr_thread mov ds:[ebx].us_owner,ax inc ds:[ebx].us_count eusEnd: pop fs pop ds pop dx pop ebx pop ax retf32 enter_user_section ENDP ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: leave_user_section ; ; DESCRIPTION: Leave user section ; ; PARAMETERS: BX Section handle ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; leave_user_section_name DB 'Leave User Section',0 leave_user_section PROC far push ax push ebx push dx push ds push fs ; mov ax,SECTION_HANDLE DerefHandle jc lusDone ; mov ax,core_data_sel mov fs,ax mov ax,fs:ps_curr_thread cmp ax,ds:[ebx].us_owner jne lusDone ; sub ds:[ebx].us_count,1 jz lusFreeOwner ; lock add ds:[ebx].us_value,1 jnc lusNotValueError ; int 3 lusNotValueError: jmp lusDone lusFreeOwner: jnc lusNotCountError ; int 3 lusNotCountError: mov ds:[ebx].us_owner,0 lock add ds:[ebx].us_value,1 jc lusDone ; call LockCore sti call cs:lock_user_section_proc mov ax,ds:[ebx].us_list or ax,ax jnz lusUnblock ; mov ds:[ebx].us_list,-1 jmp lusUnlockList lusUnblock: push es push esi push di ; lea esi,ds:[ebx].us_list call RemoveBlock32 mov es:p_data,0 call cs:unlock_user_section_proc ; call cs:insert_wakeup_proc lusUnblocked: pop di pop esi pop es jmp lusUnlock lusUnlockList: call cs:unlock_user_section_proc lusUnlock: call UnlockCore lusDone: pop fs pop ds pop dx pop ebx pop ax retf32 leave_user_section ENDP ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: SetFutexId ; ; DESCRIPTION: Set futex ID ; ; PARAMS: AX Futex ID ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; set_futex_id_name DB 'Set Futex ID',0 set_futex_id Proc far push es push bx ; push ax GetThread mov es,ax pop ax mov es:p_futex_id,ax ; pop bx pop es retf32 set_futex_id Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: acquire_futex ; ; DESCRIPTION: Acquire futex ; ; PARAMS: ES:(E)BX address to futex struct ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; acquire_futex_name DB 'Acquire Futex',0 acquire_futex Proc near push ds push fs push eax push ebx push cx push esi ; mov esi,ebx mov ebx,es:[esi].fs_handle mov ax,FUTEX_HANDLE DerefHandle jnc acquire_no_sect ; mov ax,SEG data mov ds,ax EnterSection ds:futex_section ; mov ebx,es:[esi].fs_handle mov ax,FUTEX_HANDLE DerefHandle jnc acquire_handle_ok ; mov cx,SIZE futex_handle_seg AllocateHandle mov ds:[ebx].fh_list,0 mov ds:[ebx].fh_lock,0 mov [ebx].hh_sign,FUTEX_HANDLE ; movzx eax,[ebx].hh_handle mov es:[esi].fs_handle,eax acquire_handle_ok: push ds mov ax,SEG data mov ds,ax LeaveSection ds:futex_section pop ds acquire_no_sect: call LockCore sti call cs:lock_futex_proc mov ax,1 xchg ax,es:[esi].fs_val cmp ax,-1 je acquire_take ; mov ax,ds push OFFSET acquire_done call SaveLockedThread mov ds,ax ; lea edi,[ebx].fh_list mov es,fs:ps_curr_thread mov fs:ps_curr_thread,0 ; call InsertBlock32 call cs:unlock_futex_proc ; mov es:p_sleep_type,SLEEP_TYPE_FUTEX mov es:p_sleep_sel,ds mov es:p_sleep_offset,esi jmp LoadThread acquire_take: push es mov es,fs:ps_curr_thread mov ax,es:p_futex_id pop es mov es:[esi].fs_owner,ax mov es:[esi].fs_counter,1 ; call cs:unlock_futex_proc call UnlockCore acquire_done: pop esi pop cx pop ebx pop eax pop fs pop ds ret acquire_futex Endp acquire_futex16 Proc far push ebx movzx ebx,bx call acquire_futex pop ebx retf32 acquire_futex16 Endp acquire_futex32 Proc far call acquire_futex retf32 acquire_futex32 Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: acquire_named_futex ; ; DESCRIPTION: Acquire named futex ; ; PARAMS: ES:(E)BX address to futex struct ; ES:(E)DI name of section ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; acquire_named_futex_name DB 'Acquire Named Futex',0 acquire_named_futex Proc near push ds push fs push eax push ebx push ecx push esi push edi ; GetThread mov fs,ax mov esi,OFFSET p_list_name mov ecx,31 acquire_named_copy: mov al,es:[edi] mov fs:[esi],al inc esi inc edi or al,al jz acquire_named_copied ; loop acquire_named_copy acquire_named_copied: xor al,al mov fs:[esi],al ; mov esi,ebx mov ebx,es:[esi].fs_handle mov ax,FUTEX_HANDLE DerefHandle jnc acquire_named_no_sect ; mov ax,SEG data mov ds,ax EnterSection ds:futex_section ; mov ebx,es:[esi].fs_handle mov ax,FUTEX_HANDLE DerefHandle jnc acquire_named_handle_ok ; mov cx,SIZE futex_handle_seg AllocateHandle mov ds:[ebx].fh_list,0 mov ds:[ebx].fh_lock,0 mov [ebx].hh_sign,FUTEX_HANDLE ; movzx eax,[ebx].hh_handle mov es:[esi].fs_handle,eax acquire_named_handle_ok: push ds mov ax,SEG data mov ds,ax LeaveSection ds:futex_section pop ds acquire_named_no_sect: call LockCore sti call cs:lock_futex_proc mov ax,1 xchg ax,es:[esi].fs_val cmp ax,-1 je acquire_named_take ; mov ax,ds push OFFSET acquire_named_done call SaveLockedThread mov ds,ax ; lea edi,[ebx].fh_list mov es,fs:ps_curr_thread mov fs:ps_curr_thread,0 ; call InsertBlock32 call cs:unlock_futex_proc ; mov es:p_sleep_type,SLEEP_TYPE_FUTEX mov es:p_sleep_sel,ds mov es:p_sleep_offset,esi jmp LoadThread acquire_named_take: push es mov es,fs:ps_curr_thread mov ax,es:p_futex_id pop es mov es:[esi].fs_owner,ax mov es:[esi].fs_counter,1 ; call cs:unlock_futex_proc call UnlockCore acquire_named_done: pop edi pop esi pop ecx pop ebx pop eax pop fs pop ds ret acquire_named_futex Endp acquire_named_futex16 Proc far push ebx push edi movzx edi,di movzx ebx,bx call acquire_named_futex pop edi pop ebx retf32 acquire_named_futex16 Endp acquire_named_futex32 Proc far call acquire_named_futex retf32 acquire_named_futex32 Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: release_futex ; ; DESCRIPTION: Release futex ; ; PARAMS: ES:(E)BX address to futex struct ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; release_futex_name DB 'Release Futex',0 release_futex Proc near push ds push fs push eax push ebx push cx push esi ; mov esi,ebx mov ebx,es:[esi].fs_handle mov ax,FUTEX_HANDLE DerefHandle jc release_done ; call LockCore sti call cs:lock_futex_proc ; mov ax,ds:[ebx].fh_list or ax,ax jz release_unlock ; mov ax,1 xchg ax,es:[esi].fs_val cmp ax,-1 jne release_unlock ; push di ; push es push esi lea esi,ds:[ebx].fh_list call RemoveBlock32 mov es:p_data,0 mov cx,es mov di,es:p_futex_id pop esi pop es ; mov es:[esi].fs_owner,di mov es:[esi].fs_counter,1 call cs:unlock_futex_proc ; push es mov es,cx call cs:insert_wakeup_proc pop es ; pop di ; call UnlockCore jmp release_done release_unlock: call cs:unlock_futex_proc call UnlockCore release_done: pop esi pop cx pop ebx pop eax pop fs pop ds ret release_futex Endp release_futex16 Proc far push ebx movzx ebx,bx call release_futex pop ebx retf32 release_futex16 Endp release_futex32 Proc far call release_futex retf32 release_futex32 Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: cleanup_futex ; ; DESCRIPTION: Cleanup futex ; ; PARAMS: ES:(E)BX address to futex struct ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; cleanup_futex_name DB 'Cleanup Futex',0 cleanup_futex Proc near push ds push eax push ebx ; mov ebx,es:[ebx].fs_handle mov ax,FUTEX_HANDLE DerefHandle jc cleanup_done ; FreeHandle cleanup_done: pop ebx pop eax pop ds ret cleanup_futex Endp cleanup_futex16 Proc far push ebx movzx ebx,bx call cleanup_futex pop ebx retf32 cleanup_futex16 Endp cleanup_futex32 Proc far call cleanup_futex retf32 cleanup_futex32 Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: set_thread_action ; ; DESCRIPTION: Set thread action text ; ; PARAMS: ES:(E)DI Action buffer ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; set_thread_action_name DB 'Set Thread Action',0 set_thread_action Proc near push ds push eax push ecx push esi push edi ; mov esi,OFFSET p_action_text mov ecx,32 GetThread mov ds,ax staLoop: mov al,es:[edi] mov ds:[esi],al or al,al jz staDone ; inc esi inc edi loop staLoop staDone: pop edi pop esi pop ecx pop eax pop ds ret set_thread_action Endp set_thread_action16 Proc far push edi movzx edi,di call set_thread_action pop edi retf32 set_thread_action16 Endp set_thread_action32 Proc far call set_thread_action retf32 set_thread_action32 Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: get_thread ; ; DESCRIPTION: Get current thread control block ; ; PARAMETERS: AX Thread ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; get_thread PROC far push ds mov ax,SEG data verr ax jz get_thread_norm get_thread_pre_tasking: mov ax,virt_thread_sel jmp get_thread_done get_thread_norm: mov ax,core_data_sel mov ds,ax mov ax,ds:ps_curr_thread get_thread_done: pop ds ret get_thread ENDP ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: GetThread ; ; DESCRIPTION: Get current thread ; ; PARAMETERS: AX Thread ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; get_thread_name DB 'Get Thread',0 get_thread_pr PROC far push ds mov ax,core_data_sel mov ds,ax mov ax,ds:ps_curr_thread pop ds retf32 get_thread_pr ENDP ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: IsLongThread ; ; DESCRIPTION: Check for long mode thread ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; is_long_thread_name DB 'Is Long Thread', 0 is_long_thread Proc far push ds push ax ; mov ax,core_data_sel mov ds,ax mov ds,ds:ps_curr_thread mov ax,ds:p_tss_sel or ax,ax clc jz iltDone ; stc iltDone: pop ax pop ds retf32 is_long_thread Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: GetCore ; ; DESCRIPTION: Get current core # ; ; RETURNS: AX Core ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; get_core_id_name DB 'Get Core ID',0 get_core_id PROC far push ds mov ax,core_data_sel mov ds,ax mov ax,ds:ps_id pop ds retf32 get_core_id ENDP ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: GetCoreCount ; ; DESCRIPTION: Get number of cores ; ; RETURNS: CX Core count ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; get_core_count_name DB 'Get Core Count',0 get_core_count PROC far mov cx,cs:core_count retf32 get_core_count ENDP ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: GetCoreLoad ; ; DESCRIPTION: Get core load ; ; PARAMETERS: AX Core # ; ; RETURNS: EDX:EAX core tics ; ECX:EBX null tics ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; get_core_load_name DB 'Get Core Load',0 get_core_load PROC far cmp ax,cs:core_count jae gclFail ; push ds push si ; mov si,ax add si,si mov ds,cs:[si].core_arr gclCoreRetry: mov edx,ds:ps_msb_tics mov eax,ds:ps_lsb_tics cmp edx,ds:ps_msb_tics jne gclCoreRetry ; mov ds,ds:ps_null_thread gclNullRetry: mov ecx,ds:p_msb_tics mov ebx,ds:p_lsb_tics cmp ecx,ds:p_msb_tics jne gclNullRetry ; pop si pop ds clc jmp gclDone gclFail: stc gclDone: retf32 get_core_load ENDP ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: GetCoreDuty ; ; DESCRIPTION: Get core duty cycle ; ; PARAMETERS: AX Core # ; ; RETURNS: EDX:EAX core tics ; ECX:EBX total tics ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; get_core_duty_name DB 'Get Core Duty',0 get_core_duty PROC far cmp ax,cs:core_count jae gcdFail ; push ds push si ; mov si,ax add si,si mov ds,cs:[si].core_arr GetSystemTime mov ecx,edx mov ebx,eax gcdRetry: mov edx,ds:ps_msb_tics mov eax,ds:ps_lsb_tics cmp edx,ds:ps_msb_tics jne gcdRetry ; pop si pop ds clc jmp gcdDone gcdFail: stc gcdDone: retf32 get_core_duty ENDP ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: SoftReset ; ; DESCRIPTION: Trigger a CPU soft reset ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; soft_reset_name DB 'Soft Reset',0 soft_reset PROC far IsLongThread cli pushf ; wait_gate1: in al,64h and al,2 jnz wait_gate1 mov al,0D1h out 64h,al wait_gate2: in al,64h and al,2 jnz wait_gate2 mov al,0FEh out 60h,al ; popf jc prot_reset ; xor eax,eax mov cr3,eax ; LongModeReset prot_reset: mov ax,idt_sel mov ds,ax ; mov bx,13 * 8 xor eax,eax mov [bx],eax mov [bx+4],eax ; mov bx,8 * 8 mov [bx],eax mov [bx+4],eax ; mov ax,-1 mov ds,ax reset_wait: jmp reset_wait retf32 soft_reset ENDP ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: HardReset ; ; DESCRIPTION: Trigger a CPU hard reset ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; hard_reset_name DB 'Hard Reset',0 hard_reset PROC far SoftReset retf32 hard_reset ENDP ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: PowerFailure ; ; DESCRIPTION: Power failure indication ; ; RETURNS: AX 0 = power ok ; 1 = power failure ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; power_failure_name DB 'Power Failure',0 power_failure PROC far xor ax,ax retf32 power_failure Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: GetCpuTime ; ; DESCRIPTION: Get used CPU time ; ; PARAMETERS: BX Thread ; ; RETURNS: EDX:EAX Used CPU time ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; get_cpu_time_name DB 'Get CPU time',0 get_cpu_time PROC far push ds mov ds,bx mov eax,ds:p_lsb_tics mov edx,ds:p_msb_tics pop ds retf32 get_cpu_time ENDP ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: WakeUntil ; ; DESCRIPTION: Timer callback for wait time ; ; PARAMETERS: CX Thread to wake up ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; wake_until PROC far call TryLockCore sti mov es,cx call cs:insert_wakeup_proc call TryUnlockCore retf32 wake_until ENDP ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: WAIT_UNTIL ; ; DESCRIPTION: Wait until time occurs. ; ; PARAMETERS: EDX:EAX System time to wait until ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; wait_until_name DB 'Wait Until',0 wait_until PROC far pushf push OFFSET wait_until_done call SaveCurrentThread ; mov cx,SEG data mov ds,cx mov cx,fs:ps_curr_thread mov bx,cs mov es,bx mov edi,OFFSET wake_until xor bx,bx StartTimer ; mov es,fs:ps_curr_thread mov fs:ps_curr_thread,0 ; mov es:p_sleep_type,SLEEP_TYPE_WAIT mov es:p_sleep_sel,0 mov es:p_sleep_offset,0 jmp LoadThread wait_until_done: popf retf32 wait_until ENDP ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: WAIT_MILLI_SEC ; ; DESCRIPTION: Wait a number of ms ; ; PARAMETERS: AX Value ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; wait_milli_name DB 'Wait Milli Sec',0 wait_milli_sec PROC far pushf push OFFSET wait_milli_done call SaveCurrentThread ; mov es,fs:ps_curr_thread mov bx,1193 mul bx push dx push ax pop ebx GetSystemTime ; add eax,ebx adc edx,0 ; mov cx,SEG data mov ds,cx mov cx,es mov bx,cs mov es,bx mov edi,OFFSET wake_until xor bx,bx StartTimer ; mov es,fs:ps_curr_thread mov fs:ps_curr_thread,0 ; mov es:p_sleep_type,SLEEP_TYPE_WAIT mov es:p_sleep_sel,0 mov es:p_sleep_offset,0 jmp LoadThread wait_milli_done: popf retf32 wait_milli_sec ENDP ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: WAIT_MICRO_SEC ; ; DESCRIPTION: Wait a number of us ; ; PARAMETERS: AX Value ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; wait_micro_name DB 'Wait Micro Seconds',0 wait_micro_sec PROC far pushf push OFFSET wait_micro_done call SaveCurrentThread ; mov es,fs:ps_curr_thread movzx eax,ax mov ebx,78184 mul ebx push dx push eax pop ax pop ebx GetSystemTime add eax,ebx adc edx,0 ; mov cx,SEG data mov ds,cx mov cx,es mov bx,cs mov es,bx mov edi,OFFSET wake_until xor bx,bx StartTimer ; mov es,fs:ps_curr_thread mov fs:ps_curr_thread,0 ; mov es:p_sleep_type,SLEEP_TYPE_WAIT mov es:p_sleep_sel,0 mov es:p_sleep_offset,0 jmp LoadThread wait_micro_done: popf retf32 wait_micro_sec ENDP ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: CHECK_LIST ; ; DESCRIPTION: Check if thread is in list ; ; PARAMETERS: BX Thread selector ; ES:EDI Buffer ; ; RETURNS: NC processed ; CX:EDX List ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; Wait_state DB 'Wait',0 Ready_state DB 'Ready',0 Signal_state DB 'Signal',0 Debug_state DB 'Debug',0 Futex_state DB 'User Section',0 Section_state DB 'Kernel Section',0 Suspend_state DB 'Suspend',0 Wakeup_state DB 'Wakeup ',0 Run_state DB 'Run ', 0 Ready_cpu_state DB 'Ready ',0 check_list Proc far push ds push fs push ax push si ; GetCoreCount xor dx,dx check_cpu_loop: mov ax,dx GetCoreNumber jc check_not_cpu ; cmp bx,fs:ps_curr_thread je check_curr_ok ; cmp bx,fs:ps_null_thread je check_null_ok ; mov ax,fs mov fs,bx cmp ax,fs:p_sleep_sel jne check_cpu_next ; mov eax,fs:p_sleep_offset cmp ax,OFFSET ps_wakeup_list je check_wakeup_ok ; mov ax,fs:p_sleep_type or ax,ax jz check_cpu_ready_ok check_cpu_next: inc dx add si,2 loop check_cpu_loop jmp check_not_cpu check_curr_ok: mov si,OFFSET Run_state jmp check_copy_id check_null_ok: mov si,OFFSET Ready_cpu_state jmp check_copy_id check_wakeup_ok: mov si,OFFSET Wakeup_state jmp check_copy_id check_cpu_ready_ok: mov si,OFFSET Ready_cpu_state jmp check_copy_id check_copy_id: mov al,cs:[si] or al,al jz check_copy_id_done ; inc si stos byte ptr es:[edi] jmp check_copy_id check_copy_id_done: mov al,'0' add al,dl stos byte ptr es:[edi] xor al,al stos byte ptr es:[edi] ; mov ds,bx mov cx,ds:p_cs mov edx,dword ptr ds:p_rip clc jmp check_done check_not_cpu: mov fs,bx mov ax,fs:p_sleep_type cmp ax,SLEEP_TYPE_WAIT jne check_not_wait ; mov si,OFFSET Wait_state jmp check_copy_zero check_not_wait: cmp ax,SLEEP_TYPE_SIGNAL jne check_not_signal ; mov si,OFFSET Signal_state jmp check_copy_zero check_not_signal: cmp ax,SLEEP_TYPE_DEBUG jne check_not_debug ; mov si,OFFSET Debug_state jmp check_copy_cpu check_not_debug: cmp ax,SLEEP_TYPE_SECTION jne check_not_kernel ; mov si,OFFSET Section_state jmp check_copy_sleep check_not_kernel: cmp ax,SLEEP_TYPE_SUSPEND jne check_not_suspend ; mov si,OFFSET Suspend_state jmp check_copy_sleep check_not_suspend: cmp ax,SLEEP_TYPE_FUTEX jne check_not_futex ; mov si,OFFSET p_list_name mov cx,32 check_futex_copy: mov al,fs:[si] or al,al jz check_futex_done inc si stos byte ptr es:[edi] loop check_futex_copy check_futex_done: xor al,al stos byte ptr es:[edi] ; xor cx,cx xor edx,edx clc jmp check_done check_not_futex: cmp ax,SEG data jne check_failed ; mov si,OFFSET Ready_state jmp check_copy_cpu check_copy_zero: xor cx,cx xor edx,edx jmp check_copy check_copy_cpu: mov ds,bx mov cx,ds:p_cs mov edx,dword ptr ds:p_rip jmp check_copy check_copy_sleep: mov ds,bx mov cx,ds:p_sleep_sel mov edx,ds:p_sleep_offset check_copy: mov al,cs:[si] or al,al jz check_copy_done inc si stos byte ptr es:[edi] jmp check_copy check_copy_done: xor al,al stos byte ptr es:[edi] clc jmp check_done check_failed: stc check_done: pop si pop ax pop fs pop ds retf32 check_list Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: DebugBreak ; ; DESCRIPTION: Put thread in debug list ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; debug_exc_break_name DB 'Debug Exc Break',0 debug_exc_break: DebugException ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: GetTime ; ; DESCRIPTION: Return user time ; ; PARAMETERS: EDX:EAX Binary user time ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; get_time_name DB 'Get Time',0 get_time PROC far GetSystemTime add eax,cs:time_diff adc edx,cs:time_diff+4 retf32 get_time ENDP ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: TimeToSystemTime ; ; DESCRIPTION: Converts real time to system time ; ; PARAMETERS: EDX:EAX Real time ; ; RETURNS: EDX:EAX System time ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; time_to_system_time_name DB 'Time To System Time',0 time_to_system_time PROC far cli sub eax,cs:time_diff sbb edx,cs:time_diff+4 sti retf32 time_to_system_time ENDP ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: SystemTimeToTime ; ; DESCRIPTION: Converts system time to real time ; ; PARAMETERS: EDX:EAX System time ; ; RETURNS: EDX:EAX Real time ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; system_time_to_time_name DB 'System Time To Time',0 system_time_to_time PROC far cli add eax,cs:time_diff adc edx,cs:time_diff+4 sti retf32 system_time_to_time ENDP ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: UpdateTime ; ; DESCRIPTION: Sets difference between real time and system time ; ; PARAMETERS: EDX:EAX Difference ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; update_time_name DB 'Update Time',0 update_time PROC far push ds push bx push esi push edi ; mov bx,SEG data mov ds,bx mov ds,ds:patch_sel cli mov esi,ds:time_diff mov edi,ds:time_diff+4 mov ds:time_diff,eax mov ds:time_diff+4,edx sti ; sub esi,eax sbb edi,edx test edi,80000000h jz utSignOk ; not esi not edi utSignOk: or edi,edi jnz utSetRtc ; cmp esi,1193000 jb utDone utSetRtc: mov ax,update_rtc_nr IsValidOsGate jc utDone ; UpdateRtc utDone: pop edi pop esi pop bx pop ds retf32 update_time ENDP ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: CreateProcHandle ; ; DESCRIPTION: Create a process handle ; ; PARAMETERS: AX Lib selector ; DX Process descriptor ; ; RETURNS: BX Process handle ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; create_proc_handle_name DB 'Create Process Handle',0 create_proc_handle PROC far push ds push cx mov cx,SIZE proc_handle_seg AllocateHandle mov [ebx].ph_lib_sel,ax mov [ebx].ph_proc_sel,dx mov [ebx].hh_sign,PROCESS_HANDLE mov bx,[ebx].hh_handle ; mov ds,dx inc ds:pd_ref_count ; pop cx pop ds retf32 create_proc_handle Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: DerefProcHandle ; ; DESCRIPTION: Deref a process handle ; ; PARAMETERS: BX Process handle ; ; RETURNS: AX Lib selector ; DX Process descriptor ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; deref_proc_handle_name DB 'Deref Process Handle',0 deref_proc_handle PROC far push ds push ebx ; mov ax,PROCESS_HANDLE DerefHandle jc deref_proc_handle_done ; mov ax,[ebx].ph_lib_sel mov dx,[ebx].ph_proc_sel clc deref_proc_handle_done: pop ebx pop ds retf32 deref_proc_handle Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: FreeProcHandle ; ; DESCRIPTION: Free a process handle ; ; PARAMETERS: BX Process handle ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; free_proc_handle_name DB 'Free Process Handle',0 free_proc_handle PROC far push ds push ax push ebx push dx ; mov ax,PROCESS_HANDLE DerefHandle jc free_proc_handle_done ; mov dx,[ebx].ph_proc_sel FreeHandle ; mov ds,dx sub ds:pd_ref_count,1 jnz free_proc_handle_done ; push es mov ax,ds mov es,ax xor ax,ax mov ds,ax FreeMem pop es clc free_proc_handle_done: pop dx pop ebx pop ax pop ds retf32 free_proc_handle Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: GetProcExitCode ; ; DESCRIPTION: Get process exit code ; ; PARAMETERS: BX Process handle ; ; RETURNS: AX Exit code ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; get_proc_exit_code_name DB 'Get Process Exit Code',0 get_proc_exit_code PROC far push ds push ebx ; mov ax,PROCESS_HANDLE DerefHandle mov ax,-1 jc get_proc_exit_done ; mov ds,[ebx].ph_proc_sel mov ax,ds:pd_exit_code clc get_proc_exit_done: pop ebx pop ds retf32 get_proc_exit_code Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: StartWaitForProcEnd ; ; DESCRIPTION: Start a wait for process end event ; ; PARAMETERS: ES Wait object ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; start_wait_for_proc_end PROC far push ds push eax ; ClearSignal mov ax,es:pew_proc_sel mov ds,ax mov ds:pd_wait,es ; mov ax,ds:pd_proc_sel or ax,ax jnz start_wait_done ; mov ds:pd_wait,0 SignalWait start_wait_done: pop eax pop ds ret start_wait_for_proc_end Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: StopWaitForProcEnd ; ; DESCRIPTION: Stop a wait for process end event ; ; PARAMETERS: ES Wait object ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; stop_wait_for_proc_end PROC far push ds push eax ; mov ax,es:pew_proc_sel mov ds,ax mov ds:pd_wait,0 ; pop eax pop ds ret stop_wait_for_proc_end Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: DummyClearProcEnd ; ; DESCRIPTION: Clear process end event ; ; PARAMETERS: ES Wait object ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; dummy_clear_proc_end PROC far ret dummy_clear_proc_end Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: IsProcEndIdle ; ; DESCRIPTION: Check if proc end is idle ; ; PARAMETERS: ES Wait object ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; is_proc_end_idle PROC far push ds push eax ; mov ax,es:pew_proc_sel mov ds,ax mov ax,ds:pd_proc_sel or ax,ax clc jne is_idle_done ; stc is_idle_done: pop eax pop ds ret is_proc_end_idle Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: AddWaitForProcEnd ; ; DESCRIPTION: Add a wait for process end ; ; PARAMETERS: AX Process handle ; BX Wait handle ; ECX Signalled ID ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; add_wait_for_proc_end_name DB 'Add Wait For Process End',0 add_wait_tab: aw0 DD OFFSET start_wait_for_proc_end, task_code_sel aw1 DD OFFSET stop_wait_for_proc_end, task_code_sel aw2 DD OFFSET dummy_clear_proc_end, task_code_sel aw3 DD OFFSET is_proc_end_idle, task_code_sel add_wait_for_proc_end PROC far push ds push es push eax push dx push edi ; push bx mov bx,ax DerefProcHandle pop bx jc add_wait_done ; push ax mov ax,cs mov es,ax mov ax,SIZE proc_end_wait_header - SIZE wait_obj_header mov edi,OFFSET add_wait_tab AddWait pop ax jc add_wait_done ; mov es:pew_proc_sel,dx add_wait_done: pop edi pop dx pop eax pop es pop ds retf32 add_wait_for_proc_end ENDP ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: ALLOCATE_THREAD_BLOCK ; ; DESCRIPTION: Allocate thread control block ; ; PARAMETERS: ES Thread control block ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; allocate_thread_block PROC near push ebx push ecx push edx ; mov eax,SIZE thread_seg AllocateSmallGlobalMem mov es:p_thread_sel,es ; mov bx,es GetSelectorBaseSize mov es:p_linear,edx ; pop edx pop ecx pop ebx ret allocate_thread_block ENDP ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: INIT_THREAD_BLOCK ; ; DESCRIPTION: Init thread content ; ; PARAMETERS: ES Thread ; DX Priority ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; init_thread_block PROC near GetThread mov ds,ax ; push fs mov ax,ds:p_process_sel mov fs,ax inc fs:ms_thread_count mov es:p_process_sel,ax ; GetCore mov es:p_core,fs pop fs ; mov es:p_signal_spinlock,0 mov es:p_wanted_core,0 mov es:p_int_count,0 mov es:p_nest_count,0 mov es:p_nest_unwind,0 mov ax,ds:p_app_sel mov es:p_app_sel,ax mov ax,ds:p_ldt_sel mov es:p_ldt_sel,ax mov ax,ds:p_lib_sel mov es:p_lib_sel,ax mov eax,ds:p_debug_proc mov es:p_debug_proc,eax mov es:p_signal,0 mov es:p_parent_switch,0 mov es:p_wait_list,0 mov es:p_kill,0 mov es:p_ref_count,0 mov es:p_is_waiting,0 mov es:p_flags,0 mov es:p_sleep_sel,0 ; add dx,dx mov es:p_prio,dx xor eax,eax mov es:p_msb_tics,eax mov es:p_lsb_tics,eax mov es:p_vm_deb_sel,ax mov es:p_vm_deb_offs,eax mov es:p_pm_deb_sel,ax mov es:p_pm_deb_offs,eax mov es:p_events,0 ; mov es:p_sleep_sel,0 mov es:p_sleep_offset,0 push ds mov ax,SEG data mov ds,ax cli mov ax,ds:next_pid mov es:p_id,ax inc ax mov ds:next_pid,ax sti pop ds ret init_thread_block ENDP ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: INIT_PROCESS_BLOCK ; ; DESCRIPTION: Init process content ; ; PARAMETERS: ES Thread ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; init_process_block PROC near push es mov eax,SIZE process_seg AllocateSmallGlobalMem mov es:ms_virt_flags,7200h mov es:ms_wait_sti,0 mov es:ms_thread_count,1 mov es:ms_iopl,0 mov es:ms_cli_thread,0 mov bx,es ; mov eax,SIZE proc_descr_seg AllocateSmallGlobalMem mov es:pd_proc_sel,bx mov es:pd_exit_code,0 mov es:pd_ref_count,1 mov es:pd_wait,0 ; push ds mov ax,es mov ds,ax InitSection ds:pd_section pop ds ; mov ax,es mov es,bx mov es:ms_pd_sel,ax ; pop es mov es:p_process_sel,bx ; push es mov eax,SIZE app_seg AllocateSmallGlobalMem mov bx,es mov es:app_next,0 ; pop es mov es:p_app_sel,bx ; mov es:p_ldt_sel,0 ret init_process_block ENDP ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: INIT_PROT_THREAD ; ; DESCRIPTION: Init protected mode thread ; ; PARAMETERS: ES Thread ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; init_prot_thread PROC near push ds lds esi,[ebp].cr_name mov di,OFFSET thread_name mov cx,30 pm_move_thread_name: lods byte ptr [esi] or al,al jz pm_move_pad_name stosb loop pm_move_thread_name jmp pm_move_pad_done pm_move_pad_name: mov al,' ' rep stosb pm_move_pad_done: pop ds ret init_prot_thread ENDP ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: INIT_VIRT_THREAD ; ; DESCRIPTION: Init V86 mode thread ; ; PARAMETERS: ES Thread ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; init_virt_thread PROC near push ds xor eax,eax mov ax,[ebp].cr_name xor edx,edx mov dx,[ebp+4].cr_name shl edx,4 add edx,eax mov ax,flat_sel mov ds,ax mov di,OFFSET thread_name mov cx,30 vm_move_thread_name: mov al,[edx] or al,al jz vm_move_pad_name inc edx stosb loop vm_move_thread_name jmp vm_move_pad_done vm_move_pad_name: mov al,' ' rep stosb vm_move_pad_done: pop ds ret init_virt_thread ENDP ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: CreateTss32 ; ; DESCRIPTION: Create 32-bit TSS ; ; PARAMETERS: DS Thread ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; create_tss32 PROC near push es push eax push ebx push ecx push edx ; mov ax,flat_sel mov es,ax ; mov eax,SIZE tss32_seg mov ecx,eax AllocateSmallLinear mov edi,edx ; push ecx push edi xor al,al rep stos byte ptr es:[edi] pop edi pop ecx ; mov es:[edi].tss32_bitmap,OFFSET tss32_io_bitmap ; mov eax,stack0_size AllocateBigLinear AllocateGdt mov ecx,eax CreateDataSelector32 ; mov es:[edi].tss32_esp0,stack0_size mov es:[edi].tss32_ess0,bx ; mov ds:p_kernel_esp,stack0_size mov ds:p_kernel_ss,bx mov es,bx mov es:[0],bx ; add edx,stack0_size mov ds:p_kernel_stack,edx ; mov ecx,SIZE tss32_seg mov edx,edi AllocateGdt CreateTssSelector mov ds:p_tss_sel,bx mov ds:p_futex_id,bx ; sldt dx mov ds:p_ldt,dx ; pop edx pop ecx pop ebx pop eax pop es ret create_tss32 Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: CreateInitialTss ; ; DESCRIPTION: Create initial TSS ; ; PARAMETERS: DS Thread ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; create_initial_tss PROC near push es push eax push ebx push ecx push edx ; mov ax,flat_sel mov es,ax ; mov eax,OFFSET tss32_io_bitmap + 2000h mov ecx,eax AllocateSmallLinear mov edi,edx ; push ecx push edi xor al,al rep stos byte ptr es:[edi] pop edi pop ecx ; mov es:[edi].tss32_bitmap,OFFSET tss32_io_bitmap ; mov eax,stack0_size AllocateBigLinear AllocateGdt mov ecx,eax CreateDataSelector32 ; mov es:[edi].tss32_esp0,stack0_size mov es:[edi].tss32_ess0,bx ; mov ds:p_kernel_esp,stack0_size mov ds:p_kernel_ss,bx mov es,bx mov es:[0],bx ; add edx,stack0_size mov ds:p_kernel_stack,edx ; mov ecx,OFFSET tss32_io_bitmap + 2000h mov edx,edi AllocateGdt CreateTssSelector mov ds:p_tss_sel,bx mov ds:p_futex_id,bx ; sldt dx mov ds:p_ldt,dx ; pop edx pop ecx pop ebx pop eax pop es ret create_initial_tss Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: CreateTss64 ; ; DESCRIPTION: Create 64-bit TSS ; ; PARAMETERS: DS Thread ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; create_tss64 PROC near push es push eax push ebx push ecx push edx ; mov ax,flat_sel mov es,ax ; mov eax,stack0_size AllocateBigLinear mov es:[edx],eax ; add edx,stack0_size mov ds:p_kernel_stack,edx ; mov ds:p_kernel_esp,edx mov ds:p_kernel_ss,long_kernel_data_sel ; mov ds:p_tss_sel,0 mov ds:p_ldt_sel,0 mov ds:p_ldt,long_ldt_sel mov ds:p_futex_id,0 ; pop edx pop ecx pop ebx pop eax pop es ret create_tss64 Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: INIT_DEFAULT_REGS ; ; DESCRIPTION: Setup default register state ; ; PARAMETERS: DS Thread block ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; init_default_regs PROC near mov edx,cr3 mov es:p_cr3,edx ; mov edx,[ebp].cr_offs mov dword ptr ds:p_rip,edx ; mov edx,[ebp].cr_eax mov dword ptr ds:p_rax,edx ; mov edx,[ebp].cr_ecx mov dword ptr ds:p_rcx,edx ; mov edx,[ebp].cr_edx mov dword ptr ds:p_rdx,edx ; mov edx,[ebp].cr_ebx mov dword ptr ds:p_rbx,edx ; mov edx,[ebp].cr_ebp mov dword ptr ds:p_rbp,edx ; mov edx,[ebp].cr_esi mov dword ptr ds:p_rsi,edx ; mov edx,[ebp].cr_edi mov dword ptr ds:p_rdi,edx ; xor edx,edx mov dword ptr ds:p_rip+4,edx mov dword ptr ds:p_rsp+4,edx mov dword ptr ds:p_rflags+4,edx mov dword ptr ds:p_rax+4,edx mov dword ptr ds:p_rcx+4,edx mov dword ptr ds:p_rdx+4,edx mov dword ptr ds:p_rbx+4,edx mov dword ptr ds:p_rbp+4,edx mov dword ptr ds:p_rsi+4,edx mov dword ptr ds:p_rdi+4,edx ; mov dword ptr ds:p_r8,edx mov dword ptr ds:p_r8+4,edx ; mov dword ptr ds:p_r9,edx mov dword ptr ds:p_r9+4,edx ; mov dword ptr ds:p_r10,edx mov dword ptr ds:p_r10+4,edx ; mov dword ptr ds:p_r11,edx mov dword ptr ds:p_r11+4,edx ; mov dword ptr ds:p_r12,edx mov dword ptr ds:p_r12+4,edx ; mov dword ptr ds:p_r13,edx mov dword ptr ds:p_r13+4,edx ; mov dword ptr ds:p_r14,edx mov dword ptr ds:p_r14+4,edx ; mov dword ptr ds:p_r15,edx mov dword ptr ds:p_r15+4,edx ; mov dx,[ebp].cr_seg mov ds:p_cs,dx ; ; dr0 - dr7 ; xor edx,edx mov dword ptr ds:p_dr0,edx mov dword ptr ds:p_dr0+4,edx mov dword ptr ds:p_dr1,edx mov dword ptr ds:p_dr1+4,edx mov dword ptr ds:p_dr2,edx mov dword ptr ds:p_dr2+4,edx mov dword ptr ds:p_dr3,edx mov dword ptr ds:p_dr3+4,edx mov dword ptr ds:p_dr7,edx mov dword ptr ds:p_dr7+4,edx ; ; 387 status ; mov ds:p_math_control,37Fh mov ds:p_math_status,0 mov ds:p_math_tag,0FFFFh mov ds:p_math_eip,0 mov ds:p_math_cs,0 mov ds:p_math_data_offs,0 mov ds:p_math_data_sel,0 ; ; thread control ; mov ds:p_fault_vector,-1 mov ds:p_fault_code,0 mov ds:p_action_text,0 ret init_default_regs ENDP ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: INIT_PROT_TSS ; ; DESCRIPTION: Init protected mode TSS ; ; PARAMETERS: DS TSS ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; init_prot_tss PROC near push fs ; GetThread mov fs,ax mov fs,fs:p_app_sel ; mov dx,[ebp].cr_flags or dx,200h and dx,NOT 7000h movzx edx,dx mov dword ptr ds:p_rflags,edx ; mov es:p_free_proc,0 mov es:p_free_proc+4,0 mov ax,[ebp].cr_seg test ax,3 jz init_kernel_tss ; mov eax,fs:app_init_thread_proc or eax,fs:app_init_thread_proc+4 jz init_prot_tss_default ; mov eax,fs:app_free_thread_proc mov es:p_free_proc,eax mov eax,fs:app_free_thread_proc+4 mov es:p_free_proc+4,eax ; mov eax,[ebp].cr_stack mov es:p_stack_sel,0 call fword ptr fs:app_init_thread_proc pop fs ret init_prot_tss_default: push es mov eax,[ebp].cr_stack AllocateLocalMem sub eax,6 mov dword ptr ds:p_rsp,eax mov ds:p_ss,es mov es:[eax+4],dx mov word ptr es:[eax+2],term_code_sel mov word ptr es:[eax],0 mov bx,es pop es mov es:p_stack_sel,bx jmp init_prot_tss_com init_kernel_tss: push es mov ax,ds:p_kernel_ss mov ds:p_ss,ax mov bx,stack0_size - 6 mov es,ax mov es:[bx+4],dx mov es:[bx+2],cs mov word ptr es:[bx],OFFSET terminate_thread pop es mov dword ptr ds:p_rsp,stack0_size - 6 mov es:p_stack_sel,0 init_prot_tss_com: mov ax,[ebp].cr_es mov ds:p_es,ax ; mov ax,[ebp].cr_ds mov ds:p_ds,ax ; mov ax,[ebp].cr_fs mov ds:p_fs,ax ; mov ax,[ebp].cr_gs mov ds:p_gs,ax pop fs ret init_prot_tss ENDP ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: INIT_LONG_TSS ; ; DESCRIPTION: Init long mode TSS ; ; PARAMETERS: DS TSS ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; init_long_tss PROC near push fs ; mov dx,[ebp].cr_flags or dx,200h and dx,NOT 7000h movzx edx,dx mov dword ptr ds:p_rflags,edx ; mov ax,[ebp].cr_seg test ax,3 jz init_long_kernel_tss ; int 3 jmp init_long_tss_com init_long_kernel_tss: mov ax,ds:p_kernel_ss mov ds:p_ss,ax mov ebx,ds:p_kernel_stack mov dword ptr ds:p_rsp,ebx mov es:p_stack_sel,0 init_long_tss_com: mov ax,[ebp].cr_es mov ds:p_es,ax ; mov ax,[ebp].cr_ds mov ds:p_ds,ax ; mov ax,[ebp].cr_fs mov ds:p_fs,ax ; mov ax,[ebp].cr_gs mov ds:p_gs,ax pop fs ret init_long_tss ENDP ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: INIT_VIRT_TSS ; ; DESCRIPTION: Init V86 mode TSS ; ; PARAMETERS: DS TSS ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; init_virt_tss PROC near push es xor eax,eax mov ax,[ebp].cr_stack AllocateDosMem push eax mov ax,es SelectorToSegment mov ds:p_ss,ax pop eax sub eax,6 mov dword ptr ds:p_rsp,eax mov dx,[ebp].cr_flags movzx edx,dx or edx,20200h and dx,NOT 7000h mov dword ptr ds:p_rflags,edx mov es:[eax+4],dx ; mov dx,SEG code mov es:[eax+2],dx ; mov dx,OFFSET terminate_thread_pr mov es:[eax],dx mov ax,es pop es mov es:p_stack_sel,ax ; mov ax,[ebp].cr_es SelectorToSegment mov ds:p_es,ax ; mov ax,[ebp].cr_ds SelectorToSegment mov ds:p_ds,ax ; mov ax,[ebp].cr_fs SelectorToSegment mov ds:p_fs,ax ; mov ax,[ebp].cr_gs SelectorToSegment mov ds:p_gs,ax ret init_virt_tss ENDP ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: CREATE_THREAD ; ; DESCRIPTION: Create a thread ; ; PARAMETERS: AL Priority ; AH Mode, 0=PM, 1=VM, 2=long ; ECX Stack size ; DS:(E)SI Start address ; ES:(E)DI Thread name ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; create_thread_name DB 'Create Thread',0 create_thread PROC near sub esp,30 push ebp mov ebp,esp pushf push ds push es push fs push gs push eax push ebx push ecx push edx push esi push edi mov [ebp].cr_seg,ds mov [ebp].cr_offs,esi xor dx,dx mov dl,al mov [ebp].cr_prio,dx mov [ebp].cr_stack,ecx mov dl,ah mov [ebp].cr_mode,dx mov [ebp].cr_name,edi mov [ebp+4].cr_name,es call allocate_thread_block mov dx,[ebp].cr_prio call init_thread_block mov ax,es mov ds,ax mov ax,[ebp].cr_mode cmp ax,2 jne create_t32 create_t64: call create_tss64 call init_default_regs jmp create_prot create_t32: call create_tss32 call init_default_regs mov ax,[ebp].cr_mode test ax,1 jz create_prot call init_virt_thread call init_virt_tss jmp create_tss_done create_prot: call init_prot_thread call init_prot_tss create_tss_done: ; or es:p_flags,THREAD_FLAG_CREATE call wake_new pop edi pop esi pop edx pop ecx pop ebx pop eax pop gs pop fs pop es pop ds popf pop ebp add esp,30 ret create_thread ENDP create_thread16 Proc far push ecx push esi push edi ; movzx ecx,cx movzx esi,si movzx edi,di call create_thread ; pop edi pop esi pop ecx retf32 create_thread16 Endp create_thread32 Proc far call create_thread retf32 create_thread32 Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: TERMINATE_THREAD ; ; DESCRIPTION: Terminate thread ; ; PARAMETERS: ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; terminate_thread_name DB 'Terminate Thread',0 terminate_thread: GetThread mov ds,ax mov al,ds:p_parent_switch or al,al jz terminate_focus_ok ; SetFocus terminate_focus_ok: mov es,ds:p_thread_sel mov bx,es:p_stack_sel verr bx jnz no_free_ss ; mov es,bx FreeMem no_free_ss: GetThread mov ds,ax mov ds,ds:p_process_sel sub ds:ms_thread_count,1 jz terminate_proc ; GetThread mov ds,ax mov eax,ds:p_free_proc or eax,ds:p_free_proc+4 jz terminate_app_handled ; call fword ptr ds:p_free_proc terminate_app_handled: NotifyThreadExit jmp cleanup_thread terminate_proc: GetThread mov ds,ax mov ds,ds:p_process_sel mov ds,ds:ms_pd_sel sub ds:pd_ref_count,1 jz terminate_free_pd ; EnterSection ds:pd_section ; mov ds:pd_proc_sel,0 mov ax,ds:pd_wait or ax,ax jz terminate_proc_sig_done ; mov es,ax SignalWait terminate_proc_sig_done: LeaveSection ds:pd_section xor ax,ax mov ds,ax jmp terminate_pd_done terminate_free_pd: mov ax,ds mov es,ax xor ax,ax mov ds,ax FreeMem terminate_pd_done: NotifyThreadExit NotifyProcessExit jmp cleanup_process ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: INIT_PROCESS_REGS ; ; DESCRIPTION: Init process regs ; ; PARAMETERS: DS Thread ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; init_process_regs PROC near mov ax,[ebp].cr_mode cmp ax,2 je init_long_proc_esp ; mov eax,OFFSET create_process_callback mov dword ptr ds:p_rip,eax mov ds:p_cs,cs mov ax,ds:p_kernel_ss mov ds:p_ss,ax mov eax,stack0_size mov dword ptr ds:p_rsp,eax jmp init_proc_esp_ok init_long_proc_esp: mov eax,OFFSET create_process_callback mov dword ptr ds:p_rip,eax mov ds:p_cs,cs mov ax,ds:p_kernel_ss mov ds:p_ss,ax mov eax,ds:p_kernel_stack mov dword ptr ds:p_rsp,eax init_proc_esp_ok: mov ax,[ebp].cr_mode cmp ax,1 jnz init_regs_prot_iopl ; mov ax,[ebp].cr_flags or dx,200h and dx,NOT 7000h movzx edx,dx mov dword ptr ds:p_rflags,eax jmp init_regs_iopl_done init_regs_prot_iopl: mov ax,[ebp].cr_flags or ax,200h and ax,NOT 7000h movzx eax,ax mov dword ptr ds:p_rflags,eax init_regs_iopl_done: xor ax,ax mov ds:p_es,ax mov ds:p_ds,ax mov ds:p_fs,ax mov ds:p_gs,ax ret init_process_regs ENDP ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: INIT_PROCESS_CALLBACK ; ; DESCRIPTION: Init process data ; ; PARAMETERS: ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; init_process_callback PROC near push es mov eax,1000h AllocateGlobalMem mov ds:p_ds,es ; mov ax,[ebp].cr_mode mov es:cm_mode,ax mov eax,[ebp].cr_stack mov es:cm_stack,eax mov es:cm_process,fs mov ax,[ebp].cr_flags mov es:cm_flags,ax mov ax,[ebp].cr_seg mov es:cm_cs,ax mov eax,[ebp].cr_offs mov es:cm_eip,eax mov eax,[ebp].cr_eax mov es:cm_eax,eax mov eax,[ebp].cr_ebx mov es:cm_ebx,eax mov eax,[ebp].cr_ecx mov es:cm_ecx,eax mov eax,[ebp].cr_edx mov es:cm_edx,eax mov eax,[ebp].cr_esi mov es:cm_esi,eax mov eax,[ebp].cr_edi mov es:cm_edi,eax mov eax,[ebp].cr_ebp mov es:cm_ebp,eax pop es ret init_process_callback ENDP ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: CREATE_PROCESS ; ; DESCRIPTION: Create process ; ; PARAMETERS: AL Priority ; AH Mode, 0=Protected mode, 1=V86 mode, 2=Long mode ; ECX Stack size ; DS:ESI Start address ; ES:EDI Thread name ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; create_process_name DB 'Create Process',0 create_process PROC far sub esp,30 push ebp mov ebp,esp pushf push ds push es push fs push gs push eax push ebx push ecx push edx push esi push edi mov [ebp].cr_seg,ds mov [ebp].cr_offs,esi xor dx,dx mov dl,al mov [ebp].cr_prio,dx mov [ebp].cr_stack,ecx mov dl,ah mov [ebp].cr_mode,dx mov [ebp].cr_name,edi mov [ebp+4].cr_name,es xor ax,ax mov fs,ax mov gs,ax call allocate_thread_block mov dx,[ebp].cr_prio call init_thread_block mov es:p_debug_proc,0 call init_process_block mov ax,es mov ds,ax mov ax,[ebp].cr_mode cmp ax,2 jne create_mod32 ; call create_tss64 call init_default_regs NotifyCreateLongProcess jmp create_mod_prot create_mod32: call create_tss32 call init_default_regs NotifyCreateProcess ; mov ax,[ebp].cr_mode cmp ax,1 jne create_mod_prot ; call init_virt_thread jmp create_mod_tss_done create_mod_prot: call init_prot_thread create_mod_tss_done: call init_process_regs call init_process_callback ; call wake_new pop edi pop esi pop edx pop ecx pop ebx pop eax ; pop gs pop fs pop es; pop ds popf pop ebp add esp,30 retf32 create_process ENDP ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: INIT_VIRT_CALLBACK_FRAME ; ; DESCRIPTION: Create V86 mode IRETD stack frame ; ; PARAMETERS: ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; init_virt_callback_frame PROC near pop bp xor eax,eax push eax push eax push eax push eax mov eax,ds:cm_stack AllocateDosMem mov ax,es SelectorToSegment push 0 push ax ; mov eax,ds:cm_stack push eax ; push 2 mov ax,ds:cm_flags or ax,3200h and ax,NOT 4000h push ax ; mov ax,ds:cm_cs push eax ; mov eax,ds:cm_eip push eax ; push bp ret init_virt_callback_frame ENDP ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: INIT_PROT_CALLBACK_FRAME ; ; DESCRIPTION: Create protected mode IRETD stack frame ; ; PARAMETERS: ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; init_prot_callback_frame PROC near pop bp xor eax,eax mov eax,ds:cm_stack AllocateLocalMem ; push 0 push es ; push eax ; push 0 mov ax,ds:cm_flags or ax,200h and ax,NOT 7000h push ax ; mov ax,ds:cm_cs push eax ; mov eax,ds:cm_eip push eax ; push bp ret init_prot_callback_frame ENDP ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: INIT_LONG_CALLBACK_FRAME ; ; DESCRIPTION: Create long mode IRETD stack frame ; ; PARAMETERS: ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; init_long_callback_frame PROC near pop bp xor eax,eax mov eax,ds:cm_stack AllocateGlobalMem ; push 0 push es ; push eax ; push 0 mov ax,ds:cm_flags or ax,200h and ax,NOT 7000h push ax ; mov ax,ds:cm_cs push eax ; mov eax,ds:cm_eip push eax ; push bp ret init_long_callback_frame ENDP ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: CREATE_PROCESS_CALLBACK ; ; DESCRIPTION: Process startup (in new address space) ; ; PARAMETERS: DS Process data ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; create_process_callback: GetThread mov fs,ax ; push ds NotifyProcessCreated pop ds ; mov es,ds:cm_process mov ax,ds:cm_mode or ax,ax jz create_callback_prot ; cmp ax,1 je create_callback_virt ; call init_long_callback_frame jmp create_callback_frame_done create_callback_virt: call init_virt_callback_frame jmp create_callback_frame_done create_callback_prot: call init_prot_callback_frame create_callback_frame_done: mov eax,ds:cm_eax mov ebx,ds:cm_ebx mov ecx,ds:cm_ecx mov edx,ds:cm_edx mov esi,ds:cm_esi mov edi,ds:cm_edi mov ebp,ds:cm_ebp push ax mov ax,ds mov es,ax xor ax,ax mov ds,ax mov fs,ax mov gs,ax FreeMem pop ax iretd ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: CreateFirstThread ; ; DESCRIPTION: Create first thread control block ; ; PARAMETERS: ES Thread ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; null_name DB 'Null 0',0 create_first_thread PROC near xor eax,eax mov es:p_prio,ax mov es:p_msb_tics,eax mov es:p_lsb_tics,eax mov es:p_vm_deb_sel,ax mov es:p_vm_deb_offs,eax mov es:p_pm_deb_sel,ax mov es:p_pm_deb_offs,eax mov es:p_app_sel,ax mov es:p_ldt_sel,ax mov es:p_lib_sel,ax ; mov es:p_debug_proc,0 mov es:p_flags,0 mov es:p_signal_spinlock,0 mov es:p_wanted_core,0 mov es:p_int_count,0 mov es:p_nest_count,0 mov es:p_nest_unwind,0 mov es:p_signal,0 mov es:p_parent_switch,0 mov es:p_wait_list,0 mov es:p_kill,0 mov es:p_ref_count,0 mov es:p_is_waiting,0 ; mov ax,cs mov ds,ax mov si,OFFSET null_name mov di,OFFSET thread_name mov cx,30 first_move_name: lodsb or al,al jz first_move_pad stosb loop first_move_name jmp first_move_done first_move_pad: mov al,' ' rep stosb first_move_done: mov es:p_sleep_sel,0 mov es:p_sleep_offset,0 push ds mov ax,SEG data mov ds,ax cli mov ax,ds:next_pid mov es:p_id,ax inc ax mov ds:next_pid,ax sti pop ds ret create_first_thread ENDP ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: INIT_FIRST_TSS ; ; DESCRIPTION: Init first TSS ; ; PARAMETERS: DS TSS ; ES Thread ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; init_first_tss PROC near xor edx,edx ; ; dr0 - dr7 ; mov dword ptr ds:p_dr0,edx mov dword ptr ds:p_dr0+4,edx mov dword ptr ds:p_dr1,edx mov dword ptr ds:p_dr1+4,edx mov dword ptr ds:p_dr2,edx mov dword ptr ds:p_dr2+4,edx mov dword ptr ds:p_dr3,edx mov dword ptr ds:p_dr3+4,edx mov dword ptr ds:p_dr7,edx mov dword ptr ds:p_dr7+4,edx ; ; 387 status ; mov ds:p_math_control,37Fh mov ds:p_math_status,0 mov ds:p_math_tag,0FFFFh mov ds:p_math_eip,0 mov ds:p_math_cs,0 mov ds:p_math_data_offs,0 mov ds:p_math_data_sel,0 ; ; thread control ; mov ds:p_fault_vector,-1 mov ds:p_fault_code,0 mov ds:p_action_text,0 ; mov ds:p_cr3,edx mov dword ptr ds:p_rax,edx mov dword ptr ds:p_rcx,edx mov dword ptr ds:p_rdx,edx mov dword ptr ds:p_rbx,edx mov dword ptr ds:p_rbp,edx mov dword ptr ds:p_rsi,edx mov dword ptr ds:p_rdi,edx mov ds:p_es,dx mov ds:p_ds,dx mov ds:p_fs,dx mov ds:p_gs,dx mov ds:p_ldt,0 ; mov eax,OFFSET init_first_process_callback mov dword ptr ds:p_rip,eax mov ds:p_cs,cs ; push es mov eax,stack0_size AllocateSmallGlobalMem mov ax,es pop es mov ds:p_ss,ax mov eax,stack0_size mov dword ptr ds:p_rsp,eax ; pushfd pop eax and ax,NOT 7000h mov dword ptr ds:p_rflags,eax ret init_first_tss ENDP ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: INIT_FIRST_PROCESS ; ; DESCRIPTION: Init first process (system process) ; ; PARAMETERS: ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; init_first_process Proc near mov ax,0002h push ax popf ; mov ax,virt_thread_sel mov es,ax call allocate_thread_block ; mov ax,es mov ds,ax call create_initial_tss ; call create_first_thread call init_process_block mov ax,es mov ds,ax call init_first_tss NotifyCreateProcess mov ds:p_es,0 GetCore mov fs:ps_null_thread,es mov es:p_signal_spinlock,0 mov es:p_wanted_core,0 mov es:p_int_count,0 mov es:p_nest_count,0 mov es:p_nest_unwind,0 mov es:p_core,fs mov es:p_sleep_sel,0 ret init_first_process Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: INIT_FIRST_PROCESS_CALLBACK ; ; DESCRIPTION: Startup code of system process ; ; PARAMETERS: ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; init_first_process_callback: NotifyInitProcess call start_processor_null_threads NotifyInitTasking sti jmp null_thread0 ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: StartTasking ; ; DESCRIPTION: Start tasikng ; ; PARAMETERS: ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; start_tasking_name DB 'Start Tasking', 0 start_tasking: call init_first_process jmp init_first_thread ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: CreateCoreDump ; ; DESCRIPTION: Create core dump ; ; PARAMETERS: ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; CreateCoreDump Proc near mov ax,system_data_sel mov ds,ax mov ds:core_dump_sel,0 ; mov eax,1000h AllocateBigLinear ; xor ebx,ebx mov eax,core_save_phys and eax,NOT 0FFFh mov al,13h SetPageEntry ; mov bx,core_save_sel mov eax,core_save_phys and eax,0FFFh add edx,eax mov ecx,SIZE save_core_struc CreateDataSelector16 mov ds,bx mov eax,ds:sc_sign cmp eax,SAVE_CORE_SIGN jnz ccdSetup ; mov cx,ds:sc_cores cmp cx,MAX_SAVE_CORES ja ccdSetup ; or cx,cx jz ccdSetup ; mov eax,1000h AllocateBigLinear ; mov eax,SIZE image_core_struc AllocateSmallGlobalMem mov bx,core_save_sel mov ds,bx mov cx,ds:sc_cores mov es:ic_cores,cx mov si,OFFSET sc_phys mov di,OFFSET ic_linear ccdLoop: push es push cx push di ; push edx mov eax,4000h AllocateBigLinear mov es:[di],edx mov edi,edx pop edx ; mov ax,flat_sel mov es,ax ; mov eax,dword ptr ds:[si].scp_core_phys mov ebx,dword ptr ds:[si].scp_core_phys+4 mov al,13h SetPageEntry ; push si mov esi,edx mov ecx,400h rep movs dword ptr es:[edi],es:[esi] pop si ; mov eax,dword ptr ds:[si].scp_stack_phys mov ebx,dword ptr ds:[si].scp_stack_phys+4 mov al,13h SetPageEntry ; push si mov esi,edx mov ecx,400h rep movs dword ptr es:[edi],es:[esi] pop si ; mov eax,dword ptr ds:[si].scp_log_phys mov ebx,dword ptr ds:[si].scp_log_phys+4 mov al,13h SetPageEntry ; push si mov esi,edx mov ecx,400h rep movs dword ptr es:[edi],es:[esi] pop si ; mov eax,dword ptr ds:[si].scp_log_phys+8 mov ebx,dword ptr ds:[si].scp_log_phys+12 mov al,13h SetPageEntry ; push si mov esi,edx mov ecx,400h rep movs dword ptr es:[edi],es:[esi] pop si ; pop di pop cx pop es ; add si,SIZE save_core_phys_struc add di,4 sub cx,1 jnz ccdLoop ; xor eax,eax xor ebx,ebx SetPageEntry mov ecx,1000h FreeLinear ; mov ax,system_data_sel mov ds,ax mov ds:core_dump_sel,es ccdSetup: mov bx,core_save_sel mov ds,bx mov ds:sc_sign,0 mov ds:sc_cores,0 ; mov eax,SIZE image_core_struc mov bx,core_image_sel AllocateFixedSystemMem mov es:ic_cores,0 ret CreateCoreDump Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: HasCrashInfo ; ; DESCRIPTION: Check for crash info ; ; PARAMETERS: ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; has_crash_info_name DB 'Has Crash Info',0 has_crash_info Proc far push ds push ax ; mov ax,system_data_sel mov ds,ax mov ax,ds:core_dump_sel or ax,ax stc jz hciDone ; clc hciDone: pop ax pop ds retf32 has_crash_info Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: GetCrashCoreInfo ; ; DESCRIPTION: Get crash info for core ; ; PARAMETERS: AX Core # ; ES:EDI Buffer ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; get_crash_core_name DB 'Get Crash Core Info',0 get_crash_core Proc near push ds push eax push ecx push esi push edi ; mov si,system_data_sel mov ds,si mov si,ds:core_dump_sel or si,si jz gcciFail ; mov ds,si cmp ax,ds:ic_cores jae gcciFail ; mov si,ax shl si,2 mov esi,ds:[si].ic_linear mov ax,flat_sel mov ds,ax ; mov eax,ds:[esi].cls_sign cmp eax,LOG_CORE_SIGN jne gcciFail ; mov ecx,1000h rep movs dword ptr es:[edi],ds:[esi] clc jmp gcciDone gcciFail: stc gcciDone: pop edi pop esi pop ecx pop eax pop ds ret get_crash_core Endp get_crash_core16: push edi movzx edi,di call get_crash_core pop edi retf32 get_crash_core32: call get_crash_core retf32 ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: INIT_TASK ; ; DESCRIPTION: Init module ; ; PARAMETERS: ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; init PROC far pusha push ds ; mov ax,SEG data mov ds,ax mov ds:term_thread_list,0 mov ds:term_proc_list,0 mov ds:list_lock,0 mov ds:next_pid,0 ; InitSection ds:futex_section mov ds:timer_spinlock,0 mov bx,OFFSET timer_entries mov ds:[bx].timer_next,0 mov ds:[bx].timer_msb,0FFFFFFFFh mov ds:[bx].timer_lsb,0FFFFFFFFh mov ds:timer_head,bx ; mov cx,0FFh add bx,SIZE timer_struc mov ds:timer_free,bx timer_free_list_create: mov ax,bx add ax,SIZE timer_struc mov ds:[bx].timer_next,ax mov bx,ax loop timer_free_list_create ; mov bx,cs GetSelectorBaseSize AllocateGdt CreateDataSelector32 mov ds:patch_sel,bx ; mov ax,cs mov ds,ax mov es,ax xor ebx,ebx xor esi,esi xor edi,edi ; mov si,OFFSET start_tasking mov di,OFFSET start_tasking_name xor cl,cl mov ax,start_tasking_nr RegisterOsGate ; mov si,OFFSET create_core mov di,OFFSET create_core_name xor cl,cl mov ax,create_core_nr RegisterOsGate ; mov si,OFFSET get_core mov di,OFFSET get_core_name xor cl,cl mov ax,get_core_nr RegisterOsGate ; mov si,OFFSET get_core_count mov di,OFFSET get_core_count_name xor cl,cl mov ax,get_core_count_nr RegisterOsGate ; mov si,OFFSET get_core_num mov di,OFFSET get_core_num_name xor cl,cl mov ax,get_core_num_nr RegisterOsGate ; mov si,OFFSET run_ap_core mov di,OFFSET run_ap_core_name xor cl,cl mov ax,run_ap_core_nr RegisterOsGate ; mov si,OFFSET use_own_preempt_timer mov di,OFFSET use_own_preempt_timer_name xor cl,cl mov ax,use_own_preempt_timer_nr RegisterOsGate ; mov si,OFFSET preempt_expired mov di,OFFSET preempt_expired_name xor cl,cl mov ax,preempt_expired_nr RegisterOsGate ; mov si,OFFSET timer_expired mov di,OFFSET timer_expired_name xor cl,cl mov ax,timer_expired_nr RegisterOsGate ; mov esi,OFFSET flush_tlb mov edi,OFFSET flush_tlb_name xor cl,cl mov ax,flush_tlb_nr RegisterOsGate ; mov si,OFFSET irq_schedule mov di,OFFSET irq_schedule_name xor cl,cl mov ax,irq_schedule_nr RegisterOsGate ; mov si,OFFSET get_scheduler_lock_counter mov di,OFFSET get_scheduler_lock_counter_name xor cl,cl mov ax,get_scheduler_lock_counter_nr RegisterOsGate ; mov si,OFFSET try_lock_task mov di,OFFSET try_lock_task_name xor cl,cl mov ax,try_lock_task_nr RegisterOsGate ; mov si,OFFSET lock_task mov di,OFFSET lock_task_name xor cl,cl mov ax,lock_task_nr RegisterOsGate ; mov si,OFFSET unlock_task mov di,OFFSET unlock_task_name xor cl,cl mov ax,unlock_task_nr RegisterOsGate ; mov si,OFFSET set_futex_id mov di,OFFSET set_futex_id_name xor cl,cl mov ax,set_futex_id_nr RegisterOsGate ; mov si,OFFSET enter_long_int mov di,OFFSET enter_long_int_name xor cl,cl mov ax,enter_long_int_nr RegisterOsGate ; mov si,OFFSET leave_long_int mov di,OFFSET leave_long_int_name xor cl,cl mov ax,leave_long_int_nr RegisterOsGate ; mov si,OFFSET set_thread_core mov di,OFFSET set_thread_core_name xor cl,cl mov ax,set_thread_core_nr RegisterOsGate ; mov si,OFFSET add_thread_int mov di,OFFSET add_thread_int_name xor cl,cl mov ax,add_thread_int_nr RegisterOsGate ; mov si,OFFSET debug_block mov di,OFFSET debug_block_name xor cl,cl mov ax,debug_block_nr RegisterOsGate ; mov si,OFFSET wake_thread mov di,OFFSET wake_thread_name xor cl,cl mov ax,wake_thread_nr RegisterOsGate ; mov si,OFFSET sleep_thread mov di,OFFSET sleep_thread_name xor cl,cl mov ax,sleep_thread_nr RegisterOsGate ; mov si,OFFSET clear_signal mov di,OFFSET clear_signal_name xor cl,cl mov ax,clear_signal_nr RegisterOsGate ; mov si,OFFSET signal_thread mov di,OFFSET signal_thread_name xor cl,cl mov ax,signal_nr RegisterOsGate ; mov si,OFFSET wait_for_signal mov di,OFFSET wait_for_signal_name xor cl,cl mov ax,wait_for_signal_nr RegisterOsGate ; mov si,OFFSET wait_for_signal_timeout mov di,OFFSET wait_for_signal_timeout_name xor cl,cl mov ax,wait_for_signal_timeout_nr RegisterOsGate ; mov si,OFFSET create_proc_handle mov di,OFFSET create_proc_handle_name xor cl,cl mov ax,create_proc_handle_nr RegisterOsGate ; mov si,OFFSET deref_proc_handle mov di,OFFSET deref_proc_handle_name xor cl,cl mov ax,deref_proc_handle_nr RegisterOsGate ; mov si,OFFSET fpu_exception mov di,OFFSET fpu_exception_name xor cl,cl mov ax,fpu_exception_nr RegisterOsGate ; mov si,OFFSET do_flush_tlb mov di,OFFSET do_flush_tlb_name xor cl,cl mov ax,do_flush_tlb_nr RegisterOsGate ; mov si,OFFSET add_log mov di,OFFSET add_log_name xor cl,cl mov ax,add_scheduler_log_nr RegisterOsGate ; mov si,OFFSET soft_reset mov di,OFFSET soft_reset_name xor dx,dx mov ax,fault_reset_nr RegisterOsGate ; mov si,OFFSET free_proc_handle mov di,OFFSET free_proc_handle_name xor dx,dx mov ax,free_proc_handle_nr RegisterBimodalUserGate ; mov si,OFFSET get_proc_exit_code mov di,OFFSET get_proc_exit_code_name xor dx,dx mov ax,get_proc_exit_code_nr RegisterBimodalUserGate ; mov si,OFFSET add_wait_for_proc_end mov di,OFFSET add_wait_for_proc_end_name xor dx,dx mov ax,add_wait_for_proc_end_nr RegisterBimodalUserGate ; mov bx,OFFSET create_thread16 mov si,OFFSET create_thread32 mov di,OFFSET create_thread_name mov dx,virt_es_in mov ax,create_thread_nr RegisterUserGate ; mov si,OFFSET terminate_thread mov di,OFFSET terminate_thread_name xor dx,dx mov ax,terminate_thread_nr RegisterBimodalUserGate ; mov si,OFFSET create_process mov di,OFFSET create_process_name xor cl,cl mov ax,create_process_nr RegisterOsGate ; mov si,OFFSET soft_reset mov di,OFFSET soft_reset_name xor dx,dx mov ax,soft_reset_nr RegisterBimodalUserGate ; mov si,OFFSET hard_reset mov di,OFFSET hard_reset_name xor dx,dx mov ax,hard_reset_nr RegisterBimodalUserGate ; mov si,OFFSET power_failure mov di,OFFSET power_failure_name xor dx,dx mov ax,power_failure_nr RegisterBimodalUserGate ; mov si,OFFSET get_thread_pr mov di,OFFSET get_thread_name xor dx,dx mov ax,get_thread_nr RegisterBimodalUserGate ; mov si,OFFSET get_core_id mov di,OFFSET get_core_id_name xor dx,dx mov ax,get_core_id_nr RegisterBimodalUserGate ; mov si,OFFSET get_cpu_time mov di,OFFSET get_cpu_time_name xor dx,dx mov ax,get_cpu_time_nr RegisterBimodalUserGate ; mov si,OFFSET wait_milli_sec mov di,OFFSET wait_milli_name xor dx,dx mov ax,wait_milli_nr RegisterBimodalUserGate ; mov si,OFFSET wait_micro_sec mov di,OFFSET wait_micro_name xor dx,dx mov ax,wait_micro_nr RegisterBimodalUserGate ; mov si,OFFSET wait_until mov di,OFFSET wait_until_name xor dx,dx mov ax,wait_until_nr RegisterBimodalUserGate ; mov si,OFFSET notify_time_drift mov di,OFFSET notify_time_drift_name xor cl,cl mov ax,notify_time_drift_nr RegisterOsGate ; mov si,OFFSET get_time mov di,OFFSET get_time_name xor dx,dx mov ax,get_time_nr RegisterBimodalUserGate ; mov si,OFFSET time_to_system_time mov di,OFFSET time_to_system_time_name xor dx,dx mov ax,time_to_system_time_nr RegisterBimodalUserGate ; mov si,OFFSET system_time_to_time mov di,OFFSET system_time_to_time_name xor dx,dx mov ax,system_time_to_time_nr RegisterBimodalUserGate ; mov si,OFFSET get_core_load mov di,OFFSET get_core_load_name xor dx,dx mov ax,get_core_load_nr RegisterBimodalUserGate ; mov si,OFFSET get_core_duty mov di,OFFSET get_core_duty_name xor dx,dx mov ax,get_core_duty_nr RegisterBimodalUserGate ; mov si,OFFSET debug_exc_break mov di,OFFSET debug_exc_break_name xor cl,cl mov ax,debug_exc_break_nr RegisterOsGate ; mov si,OFFSET enter_section mov di,OFFSET enter_section_name xor cl,cl mov ax,enter_section_nr RegisterOsGate ; mov si,OFFSET leave_section mov di,OFFSET leave_section_name xor cl,cl mov ax,leave_section_nr RegisterOsGate ; mov si,OFFSET cond_enter_section mov di,OFFSET cond_enter_section_name xor cl,cl mov ax,cond_enter_section_nr RegisterOsGate ; mov esi,OFFSET is_long_thread mov edi,OFFSET is_long_thread_name xor cl,cl mov ax,is_long_thread_nr RegisterOsGate ; mov si,OFFSET create_user_section mov di,OFFSET create_user_section_name xor dx,dx mov ax,create_user_section_nr RegisterBimodalUserGate ; mov si,OFFSET create_named_user_section mov di,OFFSET create_named_user_section_name xor dx,dx mov ax,create_named_user_section_nr RegisterBimodalUserGate ; mov si,OFFSET create_blocked_user_section mov di,OFFSET create_blocked_user_section_name xor dx,dx mov ax,create_blocked_user_section_nr RegisterBimodalUserGate ; mov si,OFFSET delete_user_section mov di,OFFSET delete_user_section_name xor dx,dx mov ax,delete_user_section_nr RegisterBimodalUserGate ; mov si,OFFSET enter_user_section mov di,OFFSET enter_user_section_name xor dx,dx mov ax,enter_user_section_nr RegisterBimodalUserGate ; mov si,OFFSET leave_user_section mov di,OFFSET leave_user_section_name xor dx,dx mov ax,leave_user_section_nr RegisterBimodalUserGate ; mov si,OFFSET update_time mov di,OFFSET update_time_name xor cl,cl mov ax,update_time_nr RegisterBimodalUserGate ; mov ebx,OFFSET acquire_futex16 mov esi,OFFSET acquire_futex32 mov edi,OFFSET acquire_futex_name mov dx,virt_es_in mov ax,acquire_futex_nr RegisterUserGate ; mov ebx,OFFSET acquire_named_futex16 mov esi,OFFSET acquire_named_futex32 mov edi,OFFSET acquire_named_futex_name mov dx,virt_es_in mov ax,acquire_named_futex_nr RegisterUserGate ; mov ebx,OFFSET release_futex16 mov esi,OFFSET release_futex32 mov edi,OFFSET release_futex_name mov dx,virt_es_in mov ax,release_futex_nr RegisterUserGate ; mov ebx,OFFSET cleanup_futex16 mov esi,OFFSET cleanup_futex32 mov edi,OFFSET cleanup_futex_name mov dx,virt_es_in mov ax,cleanup_futex_nr RegisterUserGate ; mov ebx,OFFSET set_thread_action16 mov esi,OFFSET set_thread_action32 mov edi,OFFSET set_thread_action_name mov dx,virt_es_in mov ax,set_thread_action_nr RegisterUserGate ; mov si,OFFSET has_crash_info mov di,OFFSET has_crash_info_name xor cl,cl mov ax,has_crash_info_nr RegisterBimodalUserGate ; mov ebx,OFFSET get_crash_core16 mov esi,OFFSET get_crash_core32 mov edi,OFFSET get_crash_core_name mov dx,virt_es_in mov ax,get_crash_core_info_nr RegisterUserGate ; mov edi,OFFSET check_list HookState ; call CreateCoreDump ; mov eax,4000h AllocateBigLinear mov edi,edx mov ax,flat_sel mov es,ax mov cx,1000h xor eax,eax rep stos dword ptr es:[edi] ; mov edx,gdt_linear CreateCore or es:ps_flags,PS_FLAG_ACTIVE ; pop ds popa ret init ENDP code ENDS END init