;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; 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 ; ; SCHBASE.ASM ; Scheduler support functions ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; INCLUDE protseg.def INCLUDE ..\user.def INCLUDE state.def INCLUDE ..\os.def INCLUDE ..\user.inc INCLUDE ..\os.inc INCLUDE ..\driver.def INCLUDE system.def INCLUDE proc.inc .686p data SEGMENT byte public 'DATA' state_hooks DW ? state_arr DD 2*32 DUP(?) data ENDS _TEXT SEGMENT byte public 'CODE' assume cs:_TEXT extrn IdToHandle:near extrn MoveThread:near ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: CreateThread ; ; DESCRIPTION: Create thread callback ; ; PARAMETERS: ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; extrn ThreadCreated:near create_thread Proc far push es pushad ; GetThread movzx eax,ax mov es,eax movzx edx,es:p_id movzx ecx,es:p_prio call ThreadCreated ; popad pop es ret create_thread Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: TerminateThread ; ; DESCRIPTION: Terminate thread callback ; ; PARAMETERS: ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; extrn ThreadTerminated:near terminate_thread Proc far pushad ; GetThread movzx eax,ax call ThreadTerminated ; popad ret terminate_thread Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: DEFAULT_STATE ; ; DESCRIPTION: Default (unknown) state ; ; PARAMETERS: BX Thread selector ; ES:EDI Buffer ; ; RETURNS: NC processed ; CX:EDX List ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; unknown_state DB 'Unknown State',0 default_state Proc far push ds push eax push esi ; mov esi,OFFSET unknown_state default_copy: mov al,cs:[esi] or al,al jz default_copy_done ; inc esi stos byte ptr es:[edi] jmp default_copy default_copy_done: xor al,al stos byte ptr es:[edi] ; mov ds,ebx mov cx,ds:p_cs mov edx,dword ptr ds:p_rip clc ; pop esi pop eax pop ds ret default_state Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: HOOK_STATE ; ; DESCRIPTION: Add a state hook ; ; PARAMETERS: ES:EDI Callback ; ; CALLED WITH: BX Thread selector ; ES:EDI Buffer ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; hook_state_name DB 'Hook State',0 hook_state Proc far push ds push ebx ; mov bx,SEG data mov ds,ebx movzx ebx,ds:state_hooks shl ebx,3 mov ds:[ebx].state_arr,edi mov ds:[ebx+4].state_arr,es inc ds:state_hooks ; pop ebx pop ds ret hook_state Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: ThreadToSel ; ; DESCRIPTION: Convert thread # (p_id) to selector ; ; PARAMETERS: BX Thread # ; ; RETURNS: BX Thread sel ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; thread_to_sel_name DB 'Thread To Sel',0 thread_to_sel Proc far push eax push ecx push edx push esi push edi ; movzx eax,bx call IdToHandle or eax,eax stc jz thread_to_sel_done ; mov bx,ax clc thread_to_sel_done: pop edi pop esi pop edx pop ecx pop eax ret thread_to_sel Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: GetThreadState ; ; DESCRIPTION: Get state of a thread ; ; PARAMETERS: ES:(E)DI BUFFER TO PUT STATE IN ; AX THREAD # ; NC THREAD EXISTS ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; get_thread_state_name DB 'Get Thread State',0 get_thread_state Proc near push ds pushad ; movzx eax,ax call IdToHandle or eax,eax stc jz get_state_done ; mov ds,ax mov ax,ds:p_id mov es:[edi].st_id,ax mov esi,OFFSET thread_name mov ecx,32 push edi add edi,OFFSET st_name rep movs byte ptr es:[edi],ds:[esi] pop edi ; mov eax,ds:p_msb_tics mov es:[edi].st_time,eax mov eax,ds:p_lsb_tics mov es:[edi].st_time+4,eax ; push edi add edi,OFFSET st_list mov bx,ds ; mov ax,SEG data mov ds,ax mov esi,OFFSET state_arr get_state_loop: call fword ptr [esi] jnc get_state_found ; add esi,8 jmp get_state_loop get_state_found: pop edi ; mov es:[edi].st_sel,cx mov es:[edi].st_offs,edx clc get_state_done: popad pop ds ret get_thread_state Endp get_thread_state16 Proc far push edi movzx edi,di call get_thread_state pop edi ret get_thread_state16 Endp get_thread_state32 Proc far call get_thread_state ret get_thread_state32 Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: ReadFlatAppDword ; ; DESCRIPTION: Read flat app dword ; ; PARAMETERS: DS Thread ; ESI Offset ; ; RETURNS: NC ; EAX Data ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ReadFlatAppDword Proc near push ebx push ecx push edx push esi ; add esi,3 xor ecx,ecx mov edx,flat_data_sel mov bx,ds ReadThreadSelector jc rfadDone ; dec esi mov cl,al ReadThreadSelector jc rfadDone ; shl ecx,8 mov cl,al ; dec esi mov cl,al ReadThreadSelector jc rfadDone ; shl ecx,8 mov cl,al ; dec esi mov cl,al ReadThreadSelector jc rfadDone ; shl ecx,8 mov cl,al mov eax,ecx clc rfadDone: pop esi pop edx pop ecx pop ebx ret ReadFlatAppDword Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: ProbeFlatAppCode ; ; DESCRIPTION: Proble flat app code ; ; PARAMETERS: DS Thread ; ESI Offset ; ; RETURNS: NC OK ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ProbeFlatAppCode Proc near push eax push ebx push edx ; mov edx,flat_data_sel mov bx,ds ReadThreadSelector jc pfacDone ; GetThreadSelectorPage jc pfacDone ; test al,2 jz pfacDone ; stc pfacDone: pop edx pop ebx pop eax ret ProbeFlatAppCode Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: GetThreadActionState ; ; DESCRIPTION: Get action state of a thread ; ; PARAMETERS: ES:(E)DI BUFFER TO PUT STATE IN ; AX THREAD # ; NC THREAD EXISTS ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; get_thread_action_state_name DB 'Get Thread Action State',0 get_thread_action_state Proc near push ds push fs pushad ; movzx eax,ax call IdToHandle or eax,eax stc jz get_action_state_done ; mov ds,ax mov ax,ds:p_id mov es:[edi].ast_id,ax mov esi,OFFSET thread_name mov ecx,32 push edi add edi,OFFSET ast_name rep movs byte ptr es:[edi],ds:[esi] pop edi ; mov esi,OFFSET p_action_text mov ecx,32 push edi add edi,OFFSET ast_action rep movs byte ptr es:[edi],ds:[esi] pop edi ; mov eax,ds:p_msb_tics mov es:[edi].ast_time,eax mov eax,ds:p_lsb_tics mov es:[edi].ast_time+4,eax ; push edi add edi,OFFSET ast_list mov bx,ds ; mov ax,SEG data mov ds,ax mov esi,OFFSET state_arr get_action_state_loop: call fword ptr [esi] jnc get_action_state_found ; add esi,8 jmp get_action_state_loop get_action_state_found: pop edi ; mov es:[edi].ast_pos.sep_sel,cx mov dword ptr es:[edi].ast_pos.sep_offs,edx mov dword ptr es:[edi].ast_pos.sep_offs+4,0 mov es:[edi].ast_count,0 ; mov ds,ebx test word ptr ds:p_rflags+2,2 jnz get_action_user_done ; mov ax,ds:p_cs cmp ax,flat_code_sel jne get_action_not_app ; mov edx,edi add edx,OFFSET ast_user mov eax,dword ptr ds:p_rbp jmp get_action_user_loop get_action_not_app: test ax,7 jnz get_action_user_done ; mov ax,ds:p_ss mov fs,ax mov ecx,dword ptr ds:p_rsp cmp ecx,stack0_size jae get_action_user_done ; mov ecx,stack0_size mov eax,fs:[ecx-4] cmp eax,flat_data_sel jne get_action_user_done ; mov eax,fs:[ecx-12] cmp eax,flat_code_sel jne get_action_user_done ; mov edx,edi add edx,OFFSET ast_user mov eax,fs:[ecx-12] mov es:[edx].sep_sel,ax mov eax,fs:[ecx-16] mov dword ptr es:[edx].sep_offs,eax mov dword ptr es:[edx].sep_offs+4,0 add edx,SIZE state_ep inc es:[edi].ast_count ; mov esi,fs:[ecx-8] call ReadFlatAppDword jc get_action_user_done get_action_user_loop: mov esi,eax push esi add esi,24 call ReadFlatAppDword pop esi jc get_action_user_done ; push esi mov esi,eax call ProbeFlatAppCode pop esi jnc get_action_user_save ; push esi add esi,20 call ReadFlatAppDword pop esi jc get_action_user_done ; push esi mov esi,eax call ProbeFlatAppCode pop esi jnc get_action_user_save ; xor eax,eax get_action_user_save: mov es:[edx].sep_sel,flat_code_sel mov dword ptr es:[edx].sep_offs,eax mov dword ptr es:[edx].sep_offs+4,0 add edx,SIZE state_ep mov ax,es:[edi].ast_count inc ax mov es:[edi].ast_count,ax cmp ax,64 jae get_action_user_done ; call ReadFlatAppDword or eax,eax jnz get_action_user_loop get_action_user_done: mov ax,es:[edi].ast_count cmp ax,2 jb get_action_user_ok ; sub edx,SIZE state_ep mov eax,dword ptr es:[edx].sep_offs or eax,dword ptr es:[edx].sep_offs+4 jnz get_action_user_ok ; dec es:[edi].ast_count get_action_user_ok: clc get_action_state_done: popad pop fs pop ds ret get_thread_action_state Endp get_thread_action_state16 Proc far push edi movzx edi,di call get_thread_action_state pop edi ret get_thread_action_state16 Endp get_thread_action_state32 Proc far call get_thread_action_state ret get_thread_action_state32 Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: SuspendThread ; ; DESCRIPTION: Suspend thread (put it in debugger) ; ; PARAMETER: AX Thread ID ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; suspend_thread_name DB 'Suspend Thread',0 suspend_thread PROC far push ds push es pushad ; movzx eax,ax call IdToHandle or eax,eax stc jz suspend_thread_done ; mov es,ax or es:p_flags,THREAD_FLAG_SUSPEND clc suspend_thread_done: popad pop es pop ds ret suspend_thread ENDP ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: SuspendAndSignalThread ; ; DESCRIPTION: Suspend and signal thread (put it in debugger) ; ; PARAMETER: AX Thread # ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; suspend_and_signal_thread_name DB 'Suspend and Signal Thread',0 suspend_and_signal_thread PROC far push ds push es pushad ; movzx eax,ax call IdToHandle or eax,eax stc jz suspend_signal_done ; mov bx,ax mov es,ax or es:p_flags,THREAD_FLAG_SUSPEND Signal clc suspend_signal_done: popad pop es pop ds ret suspend_and_signal_thread ENDP ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: MoveToCore ; ; DESCRIPTION: Move current thread to new core ; ; PARAMETER: AX Core # ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; move_to_core_name DB 'Move To Core',0 move_to_core PROC far push es push eax push ebx ; movzx eax,ax push eax GetThread mov es,eax movzx ebx,es:p_id pop eax call MoveThread ; pop ebx pop eax pop es ret move_to_core ENDP ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: MoveThreadToCore ; ; DESCRIPTION: Move thread to new core ; ; PARAMETER: AX Core # ; BX Thread ID ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; move_thread_to_core_name DB 'Move Thread To Core',0 move_thread_to_core PROC far push eax push ebx ; movzx eax,ax movzx ebx,bx call MoveThread ; pop ebx pop eax ret move_thread_to_core ENDP ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: SetThreadCore ; ; DESCRIPTION: Set new core for thread ; ; PARAMETER: AX Core # ; DX Thread handle ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; public SetThreadCore_ SetThreadCore_ PROC near push es mov es,dx SetThreadCore pop es ret SetThreadCore_ ENDP ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: GetThreadTics ; ; DESCRIPTION: Get thread tics ; ; PARAMETER: AX Thread handle ; ; RETURNS: EDX:EAX Tics ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; public GetThreadTics_ GetThreadTics_ PROC near push es mov es,ax mov edx,es:p_msb_tics mov eax,es:p_lsb_tics pop es ret GetThreadTics_ ENDP ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: GetThreadIntCount ; ; DESCRIPTION: Get thread int count ; ; PARAMETER: AX Thread handle ; ; RETURNS: EAX Int count ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; public GetThreadIntCount_ GetThreadIntCount_ PROC near push es mov es,ax movzx eax,es:p_int_count pop es ret GetThreadIntCount_ ENDP ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: InitScheduler ; ; DESCRIPTION: Initialize scheduler ; ; PARAMETERS: ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; public InitScheduler_ InitScheduler_ Proc near push ds push es pushad ; mov bx,SEG data mov es,ebx mov es:state_hooks,0 mov ecx,32 mov edi,OFFSET state_arr init_state_hooks: mov dword ptr es:[edi],OFFSET default_state mov es:[edi+4],cs add edi,8 loop init_state_hooks ; mov ax,cs mov ds,ax mov es,ax ; mov edi,OFFSET create_thread HookCreateThread ; mov edi,OFFSET terminate_thread HookTerminateThread ; mov esi,OFFSET hook_state mov edi,OFFSET hook_state_name xor cl,cl mov ax,hook_state_nr RegisterOsGate ; mov esi,OFFSET thread_to_sel mov edi,OFFSET thread_to_sel_name xor cl,cl mov ax,thread_to_sel_nr RegisterOsGate ; mov ebx,OFFSET get_thread_state16 mov esi,OFFSET get_thread_state32 mov edi,OFFSET get_thread_state_name mov dx,virt_es_in mov ax,get_thread_state_nr RegisterUserGate ; mov ebx,OFFSET get_thread_action_state16 mov esi,OFFSET get_thread_action_state32 mov edi,OFFSET get_thread_action_state_name mov dx,virt_es_in mov ax,get_thread_action_state_nr RegisterUserGate ; mov esi,OFFSET suspend_thread mov edi,OFFSET suspend_thread_name xor dx,dx mov ax,suspend_thread_nr RegisterBimodalUserGate ; mov esi,OFFSET suspend_and_signal_thread mov edi,OFFSET suspend_and_signal_thread_name xor dx,dx mov ax,suspend_and_signal_thread_nr RegisterBimodalUserGate ; mov esi,OFFSET move_to_core mov edi,OFFSET move_to_core_name xor dx,dx mov ax,move_to_core_nr RegisterBimodalUserGate ; mov esi,OFFSET move_thread_to_core mov edi,OFFSET move_thread_to_core_name xor dx,dx mov ax,move_thread_to_core_nr RegisterBimodalUserGate ; popad pop es pop ds ret InitScheduler_ Endp _TEXT ENDS END