;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; 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 ; ; V86BIOS.ASM ; V86 mode code emulation in boot-time environment ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; INCLUDE ..\driver.def INCLUDE ..\os\protseg.def INCLUDE ..\user.def INCLUDE ..\os.def INCLUDE ..\user.inc INCLUDE ..\os.inc INCLUDE ..\os\system.def INCLUDE ..\os\system.inc INCLUDE ..\os\int.def list_entry STRUC list_link DW ? list_thread DW ? list_eax DD ? list_ebx DD ? list_ecx DD ? list_edx DD ? list_esi DD ? list_edi DD ? list_ebp DD ? list_ds DW ? list_es DW ? list_flags DW ? list_int DB ? list_entry ENDS data SEGMENT byte public 'DATA' bios_thread DW ? bios_list DW ? bios_section section_typ <> data ENDS code SEGMENT byte public 'CODE' .386p assume cs:code ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: DecodeSelector ; ; DESCRIPTION: Decode a selector ; ; PARAMETERS: BX Selector ; ; RETURNS: EDX Base ; ECX Limit ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; DecodeSelector Proc near push ds ; test bx,7 jnz decode_sel_fail ; GetSelectorBaseSize jc decode_sel_fail ; sub edx,global_page_linear jc decode_sel_fail ; cmp edx,global_page_size jae decode_sel_fail ; test dx,0FFFh jnz decode_sel_fail ; add edx,global_page_linear clc jmp decode_sel_done decode_sel_fail: stc decode_sel_done: pop ds ret DecodeSelector Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: HandleInputSel ; ; DESCRIPTION: Handle input selectors ; ; PARAMETERS: FS List ; ; RETURNS: DS,ES BIOS selectors ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; HandleInputSel Proc near mov bx,fs:list_ds call DecodeSelector jc handle_input_ds_fail ; mov esi,edx mov edi,10000h dec ecx shr ecx,12 inc ecx cmp ecx,10h jbe handle_input_ds_move ; mov ecx,10h handle_input_ds_move: MovePageEntries ; mov ax,v86_bios_ds_sel mov ds,ax jmp handle_input_es handle_input_ds_fail: xor ax,ax mov ds,ax handle_input_es: mov bx,fs:list_es call DecodeSelector jc handle_input_es_fail ; mov esi,edx mov edi,20000h dec ecx shr ecx,12 inc ecx cmp ecx,10h jbe handle_input_es_move ; mov ecx,10h handle_input_es_move: MovePageEntries ; mov ax,v86_bios_es_sel mov es,ax jmp handle_input_done handle_input_es_fail: xor ax,ax mov es,ax handle_input_done: ret HandleInputSel Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: HandleOutputSel ; ; DESCRIPTION: Handle output selectors ; ; PARAMETERS: FS List ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; HandleOutputSel Proc near mov ax,ds or ax,ax jz handle_output_es ; cmp ax,v86_bios_ds_sel je handle_output_ds_same ; mov bx,fs:list_ds call DecodeSelector jc handle_output_es ; mov edi,edx ; mov bx,ds push ecx GetSelectorBaseSize pop ecx mov esi,edx dec ecx shr ecx,12 inc ecx cmp ecx,10h jbe handle_output_ds_copy ; mov ecx,10h handle_output_ds_copy: CopyPageEntries jmp handle_output_es handle_output_ds_same: mov bx,fs:list_ds call DecodeSelector jc handle_output_es ; mov edi,edx mov esi,10000h dec ecx shr ecx,12 inc ecx cmp ecx,10h jbe handle_output_ds_move ; mov ecx,10h handle_output_ds_move: MovePageEntrieshandle_output_es: handle_output_es: mov ax,es or ax,ax jz handle_output_done ; cmp ax,v86_bios_es_sel je handle_output_es_same ; mov bx,fs:list_es call DecodeSelector jc handle_output_done ; mov edi,edx ; mov bx,es push ecx GetSelectorBaseSize pop ecx mov esi,edx dec ecx shr ecx,12 inc ecx cmp ecx,10h jbe handle_output_es_copy ; mov ecx,10h handle_output_es_copy: CopyPageEntries jmp handle_output_done handle_output_es_same: mov bx,fs:list_es call DecodeSelector jc handle_output_done ; mov edi,edx mov esi,20000h dec ecx shr ecx,12 inc ecx cmp ecx,10h jbe handle_output_es_move ; mov ecx,10h handle_output_es_move: MovePageEntries handle_output_done: mov edi,10000h mov cx,20h handle_output_zero: mov edx,edi GetPageEntry ; push eax push ebx xor ebx,ebx mov eax,2 SetPageEntry pop ebx pop eax ; and ax,0F000h or eax,eax jz handle_output_zero_next ; FreePhysical handle_output_zero_next: add edi,1000h loop handle_output_zero ; ret HandleOutputSel Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: bios_process ; ; DESCRIPTION: BIOS process ; ; PARAMETERS: ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; bios_name DB 'V86 BIOS',0 bios_process: CreatePrivateLdt ; mov al,16 SetBitness ; xor edx,edx xor ebx,ebx mov eax,7 SetPageEntry ; xor ebx,ebx mov eax,0A0007h mov edx,0A0000h mov cx,40h rom_loop: SetPageEntry add eax,1000h add edx,1000h loop rom_loop ; xor ebx,ebx mov eax,0F0007h mov edx,0F0000h mov cx,10h bios_loop: SetPageEntry add eax,1000h add edx,1000h loop bios_loop ; GetThread mov ds,ax ; mov ax,flat_sel mov es,ax ; mov eax,OFFSET tss32_io_bitmap + 2000h mov ecx,eax AllocateSmallLinear mov edi,edx ; push ds push ecx push esi ; push ecx mov ecx,OFFSET tss32_io_bitmap xor al,al rep stos byte ptr es:[edi] ; mov ax,io_bitmap_sel mov ds,ax xor esi,esi mov ecx,80h rep movs byte ptr es:[edi],ds:[esi] pop ecx ; sub ecx,80h + OFFSET tss32_io_bitmap xor al,al rep stos byte ptr es:[edi] ; pop esi pop ecx pop ds ; mov eax,cr3 mov es:[edx].tss32_cr3,eax ; mov es:[edx].tss32_esp0,esp mov es:[edx].tss32_ess0,ss mov es:[edx].tss32_bitmap, OFFSET tss32_io_bitmap ; AllocateGdt CreateTssSelector xchg bx,ds:p_tss_sel FreeGdt ; mov ax,1 WaitMilliSec bios_bitmap_done: mov ax,flat_sel mov ds,ax xor al,al mov cx,8 xor si,si bios_int_loop1: mov bx,[si] add si,2 mov dx,[si] add si,2 SetVMInt inc al loop bios_int_loop1 ; mov al,10h mov cx,68h mov si,10h * 4 bios_int_loop2: mov bx,[si] add si,2 mov dx,[si] add si,2 SetVMInt inc al loop bios_int_loop2 ; mov al,80h mov cx,80h mov si,80h * 4 bios_int_loop3: mov bx,[si] add si,2 mov dx,[si] add si,2 SetVMInt inc al loop bios_int_loop3 ; mov ax,SEG data mov ds,ax GetThread mov ds:bios_thread,ax jmp bios_proc_check bios_proc_loop: WaitForSignal mov ax,SEG data mov ds,ax bios_proc_check: mov ax,ds:bios_list or ax,ax jz bios_proc_loop ; EnterSection ds:bios_section mov fs,ds:bios_list mov ax,fs:list_link mov ds:bios_list,ax LeaveSection ds:bios_section ; mov ax,flat_sel mov ds,ax movzx bx,fs:list_int shl bx,2 push dword ptr [bx] ; call HandleInputSel push fs:list_flags mov eax,cr3 mov cr3,eax mov eax,fs:list_eax mov ebx,fs:list_ebx mov ecx,fs:list_ecx mov edx,fs:list_edx mov esi,fs:list_esi mov edi,fs:list_edi mov ebp,fs:list_ebp popf CallVm ; pushf pop fs:list_flags mov fs:list_eax,eax mov fs:list_ebx,ebx mov fs:list_ecx,ecx mov fs:list_edx,edx mov fs:list_esi,esi mov fs:list_edi,edi mov fs:list_ebp,ebp call HandleOutputSel ; mov bx,fs:list_thread xor ax,ax mov fs,ax Signal jmp bios_proc_loop ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: V86BiosInt ; ; DESCRIPTION: Run int in real BIOS ; ; PARAMETERS: Stack Int # ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; v86_bios_int_name DB 'V86 BIOS Int',0 V86_bios_int Proc far push ds push es ; push eax pushf mov eax,SIZE list_entry AllocateSmallGlobalMem pop es:list_flags pop es:list_eax pop es:list_es pop es:list_ds mov es:list_ebx,ebx mov es:list_ecx,ecx mov es:list_edx,edx mov es:list_esi,esi mov es:list_edi,edi mov es:list_ebp,ebp mov bp,sp mov al,[bp+8] mov es:list_int,al ; GetThread mov es:list_thread,ax mov ax,SEG data mov ds,ax ; ClearSignal EnterSection ds:bios_section mov ax,ds:bios_list mov es:list_link,ax mov ds:bios_list,es LeaveSection ds:bios_section ; mov bx,ds:bios_thread Signal WaitForSignal ; mov eax,es:list_eax mov ebx,es:list_ebx mov ecx,es:list_ecx mov edx,es:list_edx mov esi,es:list_esi mov edi,es:list_edi mov ebp,es:list_ebp push es:list_ds push es:list_es push es:list_flags FreeMem popf ; pop es pop ds Retf32Pop 2 v86_bios_int Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: init_process ; ; DESCRIPTION: init BIOS process ; ; PARAMETERS: ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; init_process PROC far push ds push es pusha mov ax,cs mov ds,ax mov es,ax mov esi,OFFSET bios_process mov edi,OFFSET bios_name mov ecx,stack0_size mov bx,1 mov ax,5 CreateProcess popa pop es pop ds retf32 init_process ENDP public init_v86_bios init_v86_bios PROC near mov bx,SEG data mov ds,bx InitSection ds:bios_section mov ds:bios_list,0 mov ds:bios_thread,0 ; mov bx,v86_bios_ds_sel mov edx,10000h mov ecx,10000h CreateDataSelector16 ; mov bx,v86_bios_es_sel mov edx,20000h mov ecx,10000h CreateDataSelector16 ; mov ax,cs mov ds,ax mov es,ax mov edi,OFFSET init_process HookInitTasking ; mov esi,OFFSET v86_bios_int mov edi,OFFSET v86_bios_int_name mov cl,1 mov ax,v86_bios_int_nr RegisterOsGate ret init_v86_bios ENDP code ENDS END