;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; 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 ; ; ELF.ASM ; ELF (LINUX executable) loader ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; NAME elf ;;;;;;;;; INTERNAL PROCEDURES ;;;;;;;;;;; GateSize = 16 INCLUDE system.def INCLUDE protseg.def INCLUDE driver.def INCLUDE int.def INCLUDE user.def INCLUDE virt.def INCLUDE os.def INCLUDE user.inc INCLUDE virt.inc INCLUDE os.inc INCLUDE exec.def INCLUDE elf.def INCLUDE system.inc code SEGMENT byte public 'CODE' .386p assume cs:code ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: CreateLib ; ; DESCRIPTION: Create lib ; ; PARAMETERS: FS:ESI Image name ; BX File handle ; EDX File offset of program header ; AX Object header size ; CX Object count ; ; RETURNS: ES Lib handle ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; CreateLib Proc near push esi push edi ; push eax push ecx xor cx,cx push esi create_lib_size_loop: mov al,fs:[esi] or al,al jz create_lib_size_ok ; cmp al,'.' jz create_lib_size_ok inc cx inc esi jmp create_lib_size_loop create_lib_size_ok: pop esi movzx eax,cx mov edi,OFFSET lib_name add eax,edi inc eax AllocateSmallGlobalMem ; rep movs byte ptr es:[edi],fs:[esi] mov byte ptr es:[edi],0 mov es:lib_usage_count,1 InitSection es:lib_section mov es:lib_file_handle,bx pop ecx pop eax mov es:lib_file_offset,edx mov es:lib_object_size,ax mov es:lib_object_count,cx mov es:lib_proc_entries,0 mov es:lib_offset_table,0 mov es:lib_reloc_entries,0 mov es:lib_init,0 mov es:lib_fini,0 ; pop edi pop esi ret CreateLib Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: InsertApp ; ; DESCRIPTION: Insert App image into module list ; ; PARAMETERS: ES Lib handle ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; elf_loader_name DB 'ELF',0 InsertApp Proc near push ds push ax ; mov ax,elf_app_sel mov ds,ax mov ds:elf_app,es ; mov ax,thread_app_sel mov ds,ax mov word ptr ds:app_loader_name,OFFSET elf_loader_name mov word ptr ds:app_loader_name+2,cs ; pop ax pop ds ret InsertApp Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: FindLib ; ; DESCRIPTION: Search for a DLL or app module ; ; PARAMETERS: EDX Virtual adress ; ; RETURNS: ES Lib ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; FindLib Proc near push ds push eax push ecx push si ; mov ax,elf_app_sel mov ds,ax EnterSection ds:elf_section mov ax,ds:elf_dlls or ax,ax jz find_lib_try_app ; mov si,ax find_lib_dll_loop: mov es,ax mov ecx,edx sub ecx,es:lib_base jc find_lib_dll_next cmp ecx,es:lib_size jc find_lib_ok find_lib_dll_next: mov ax,es:lib_next cmp ax,si jne find_lib_dll_loop find_lib_try_app: mov ax,ds:elf_app or ax,ax jz find_lib_fail ; mov es,ax mov ecx,edx sub ecx,es:lib_base jc find_lib_fail ; cmp ecx,es:lib_size jc find_lib_ok find_lib_fail: LeaveSection ds:elf_section stc jmp find_lib_done find_lib_ok: LeaveSection ds:elf_section clc find_lib_done: pop si pop ecx pop eax pop ds ret FindLib Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: load_object ; ; DESCRIPTION: Demand load object ; ; PARAMETERS: EDX LINEAR ADDRESS ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; load_object Proc far push ds push es pushad mov eax,1 UnhookPage sub edx,local_page_linear and dx,0F000h ; call FindLib jc load_object_done ; mov ds,es:lib_objects mov cx,es:lib_object_count xor si,si load_object_loop: push cx cmp edx,[si].ep_va ja load_object_next ; mov eax,[si].ep_va add eax,[si].ep_mem_size sub eax,1000h cmp edx,eax jb load_object_next ; mov bx,es:lib_file_handle mov edi,edx cmp edi,[si].ep_va jae load_object_start_ok ; mov edi,[si].ep_va load_object_start_ok: mov eax,edi sub eax,[si].ep_va mov ecx,[si].ep_file_size sub ecx,eax cmp ecx,1000h jc load_object_size_ok ; mov ecx,1000h load_object_size_ok: add eax,[si].ep_offset SetFilePos ; push es mov ax,flat_data_sel mov es,ax UserGateForce32 read_file_nr add edi,eax mov ecx,1000h sub ecx,eax mov eax,[si].ep_mem_size sub eax,[si].ep_file_size cmp ecx,eax jc pad_object_size_ok ; mov ecx,eax pad_object_size_ok: push ecx shr ecx,2 xor eax,eax rep stos dword ptr es:[edi] pop ecx and ecx,3 rep stos byte ptr es:[edi] pop es load_object_next: pop cx add si,es:lib_object_size sub cx,1 jnz load_object_loop load_object_done: int 3 popad pop es pop ds ret load_object Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: CreateImage ; ; DESCRIPTION: Create image ; ; PARAMETERS: ES Lib handle ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; CreateImage Proc near push ds push es push fs push eax push cx push edx push si push edi ; mov ax,es mov fs,ax mov ax,fs:lib_object_size mul fs:lib_object_count push dx push ax pop eax AllocateSmallGlobalMem mov fs:lib_objects,es xor di,di mov cx,ax ReadFile cmp ax,cx jne create_image_fail ; mov cx,fs:lib_object_count or cx,cx jz create_image_fail ; mov edx,es:ep_va mov eax,es:ep_mem_size add eax,edx sub cx,1 jz create_image_size_ok ; mov si,fs:lib_object_size create_image_size_loop: mov edi,es:[si].ep_va cmp edi,edx ja create_image_size_not_before ; mov edx,edi create_image_size_not_before: add edi,es:[si].ep_mem_size cmp edi,eax jb create_image_size_not_after ; mov eax,edi create_image_size_not_after: add si,fs:lib_object_size loop create_image_size_loop create_image_size_ok: and dx,0F000h sub eax,edx dec eax and ax,0F000h add eax,1000h push edx add edx,local_page_linear ReserveLocalLinear jnc create_image_alloced ; AllocateLocalLinear create_image_alloced: sub edx,local_page_linear SetFlatLinearInvalid ; mov fs:lib_base,edx mov fs:lib_size,eax pop edx sub edx,fs:lib_base neg edx mov fs:lib_reloc,edx ; mov ax,es mov ds,ax mov ax,cs mov es,ax mov di,OFFSET load_object mov cx,fs:lib_object_count xor si,si hook_object_loop: mov edx,[si].ep_va add edx,fs:lib_reloc mov [si].ep_va,edx add edx,local_page_linear mov eax,[si].ep_mem_size HookPage add si,fs:lib_object_size loop hook_object_loop ; clc jmp create_image_done create_image_fail: stc create_image_done: pop edi pop si pop edx pop cx pop eax pop fs pop es pop ds ret CreateImage Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: InitStack ; ; DESCRIPTION: Init stack ; ; PARAMETERS: ES Lib handle ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; InitStack Proc near push eax push edx ; mov eax,100000h AllocateLocalLinear sub edx,local_page_linear mov word ptr [bp].load_ss,flat_data_sel mov [bp].load_esp,edx ; pop edx pop eax ret InitStack Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: InitDynamic ; ; DESCRIPTION: Init dynamic section ; ; PARAMETERS: ES Lib handle ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; DummyDynamic Proc near ret DummyDynamic Endp SetupRelocTable Proc near mov eax,[edx+4] add eax,es:lib_reloc mov es:lib_reloc_table,eax ret SetupRelocTable Endp SetupRelocEntries Proc near mov eax,[edx+4] mov es:lib_reloc_entries,eax ret SetupRelocEntries Endp SetupRelocSize Proc near mov eax,[edx+4] mov es:lib_reloc_size,eax ret SetupRelocSize Endp SetupProcTable Proc near mov eax,[edx+4] add eax,es:lib_reloc mov es:lib_proc_table,eax ret SetupProcTable Endp SetupProcEntries Proc near mov eax,[edx+4] mov es:lib_proc_entries,eax ret SetupProcEntries Endp SetupOffsetTable Proc near mov eax,[edx+4] add eax,es:lib_reloc mov es:lib_offset_table,eax ret SetupOffsetTable Endp SetupInit Proc near mov eax,[edx+4] add eax,es:lib_reloc mov es:lib_init,eax ret SetupInit Endp SetupFini Proc near mov eax,[edx+4] add eax,es:lib_reloc mov es:lib_fini,eax ret SetupFini Endp dynamic_tab: dt00 DW OFFSET DummyDynamic dt01 DW OFFSET DummyDynamic dt02 DW OFFSET SetupProcEntries dt03 DW OFFSET SetupOffsetTable dt04 DW OFFSET DummyDynamic dt05 DW OFFSET DummyDynamic dt06 DW OFFSET DummyDynamic dt07 DW OFFSET DummyDynamic dt08 DW OFFSET DummyDynamic dt09 DW OFFSET DummyDynamic dt10 DW OFFSET DummyDynamic dt11 DW OFFSET DummyDynamic dt12 DW OFFSET SetupInit dt13 DW OFFSET SetupFini dt14 DW OFFSET DummyDynamic dt15 DW OFFSET DummyDynamic dt16 DW OFFSET DummyDynamic dt17 DW OFFSET SetupRelocTable dt18 DW OFFSET SetupRelocEntries dt19 DW OFFSET SetupRelocSize dt20 DW OFFSET DummyDynamic dt21 DW OFFSET DummyDynamic dt22 DW OFFSET DummyDynamic dt23 DW OFFSET SetupProcTable InitDynamic Proc near int 3 push ds push eax push ebx push cx push edx push si ; mov ds,es:lib_objects mov cx,es:lib_object_count xor si,si init_dynamic_loop: mov eax,[si].ep_type cmp eax,PT_DYNAMIC jne init_dynamic_next ; push ds push ecx push edx mov edx,[si].ep_va mov ecx,[si].ep_mem_size mov ax,flat_data_sel mov ds,ax init_dynamic_entry_loop: mov ebx,[edx] cmp ebx,23 ja init_dynamic_entry_next ; add bx,bx call word ptr cs:[bx].dynamic_tab init_dynamic_entry_next: add edx,8 sub ecx,8 ja init_dynamic_entry_loop ; pop edx pop ecx pop ds init_dynamic_next: add si,es:lib_object_size sub cx,1 jnz hook_object_loop ; pop si pop edx pop cx pop ebx pop eax pop ds ret InitDynamic Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: RunImage ; ; DESCRIPTION: Setup registers to run image ; ; PARAMETERS: EDX Entry point ; ES Lib handle ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; RunImage Proc near push eax push ebx push edx ; mov dword ptr [bp].load_cs,flat_code_sel mov [bp].load_eip,edx mov dword ptr [bp].load_esi,0 mov dword ptr [bp].load_edi,0 mov word ptr [bp+2].load_eflags,0 mov ax,7202h SetFlags mov [bp].load_eflags,ax mov dword ptr [bp].load_ds,flat_data_sel mov dword ptr [bp].load_es,flat_data_sel ; pop edx pop ebx pop eax ret RunImage Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: load_elf ; ; DESCRIPTION: Load ELF executable file ; ; PARAMETERS: BX file handle ; DS:ESI File name ; ES:EDI Command line ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; load_elf Proc far push ds push es push fs push esi push edi ; mov ax,ds mov fs,ax mov ax,flat_data_sel mov ds,ax ; xor eax,eax SetFilePos mov eax,SIZE elf_header AllocateSmallGlobalMem mov cx,ax xor di,di ReadFile jc load_elf_fail ; cmp ax,SIZE elf_header jne load_elf_fail ; mov eax,dword ptr es:elf_magic cmp al,7Fh jne load_elf_fail ; shr eax,8 cmp ax,'LE' jne load_elf_fail ; shr eax,16 cmp al,'F' jne load_elf_fail ; mov al,es:elf_file_class cmp al,EFC32 jne load_elf_fail ; mov al,es:elf_endian cmp al,EELSB jne load_elf_fail ; mov ax,es:elf_type cmp ax,ET_EXEC jne load_elf_fail ; mov ax,es:elf_machine cmp ax,EM_386 jne load_elf_fail ; push es:elf_entry mov edx,es:elf_phoff mov ax,es:elf_phentsize mov cx,es:elf_phnum FreeMem call CreateLib call CreateImage pop edx jc load_elf_fail mov eax,es:lib_reloc or eax,eax jnz load_elf_fail ; mov al,32 SetBitness call InsertApp call InitStack call InitDynamic add edx,es:lib_reloc call RunImage ; clc jmp load_elf_done load_elf_fail: FreeMem stc load_elf_done: pop edi pop esi pop fs pop es pop ds ret load_elf Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: OpenApp ; ; DESCRIPTION: open app callback ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; open_app Proc far push ds mov ax,elf_app_sel mov ds,ax mov ds:elf_app,0 mov ds:elf_dlls,0 InitSection ds:elf_section pop ds ret open_app Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: close_app ; ; DESCRIPTION: Close app ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; close_app Proc far mov ax,elf_app_sel mov ds,ax close_app_done: ret close_app Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: init ; ; DESCRIPTION: Init elf loader module ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; init PROC far push ds push es pusha ; mov bx,elf_code_sel InitDevice ; mov bx,elf_app_sel mov eax,SIZE elf_data_seg AllocateFixedAppMem ; mov ax,cs mov ds,ax mov es,ax ; mov di,OFFSET load_elf HookLoadExe ; mov di,OFFSET open_app HookOpenApp ; mov di,OFFSET close_app HookCloseApp ; popa pop es pop ds ret init ENDP code ENDS END init