mirror of
				https://github.com/microsoft/MS-DOS.git
				synced 2025-10-26 18:13:14 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			1036 lines
		
	
	
		
			29 KiB
		
	
	
	
		
			NASM
		
	
	
	
	
	
			
		
		
	
	
			1036 lines
		
	
	
		
			29 KiB
		
	
	
	
		
			NASM
		
	
	
	
	
	
| SUBTTL $exec - load/go a program
 | |
| PAGE
 | |
| ;
 | |
| ; Assembler usage:
 | |
| ;           LDS     DX, name
 | |
| ;           LES     BX, blk
 | |
| ;           MOV     AH, Exec
 | |
| ;           MOV     AL, func
 | |
| ;           INT     int_command
 | |
| ;
 | |
| ;       AL  Function
 | |
| ;       --  --------
 | |
| ;        0  Load and execute the program.
 | |
| ;        1  Load, create  the  program  header  but  do  not
 | |
| ;           begin execution.
 | |
| ;        3  Load overlay. No header created.
 | |
| ;
 | |
| ;           AL = 0 -> load/execute program
 | |
| ;
 | |
| ;           +---------------------------+
 | |
| ;           | WORD segment address of   |
 | |
| ;           | environment.              |
 | |
| ;           +---------------------------+
 | |
| ;           | DWORD pointer to ASCIZ    |
 | |
| ;           | command line at 80h       |
 | |
| ;           +---------------------------+
 | |
| ;           | DWORD pointer to default  |
 | |
| ;           | FCB to be passed at 5Ch   |
 | |
| ;           +---------------------------+
 | |
| ;           | DWORD pointer to default  |
 | |
| ;           | FCB to be passed at 6Ch   |
 | |
| ;           +---------------------------+
 | |
| ;
 | |
| ;           AL = 1 -> load program
 | |
| ;
 | |
| ;           +---------------------------+
 | |
| ;           | WORD segment address of   |
 | |
| ;           | environment.              |
 | |
| ;           +---------------------------+
 | |
| ;           | DWORD pointer to ASCIZ    |
 | |
| ;           | command line at 80h       |
 | |
| ;           +---------------------------+
 | |
| ;           | DWORD pointer to default  |
 | |
| ;           | FCB to be passed at 5Ch   |
 | |
| ;           +---------------------------+
 | |
| ;           | DWORD pointer to default  |
 | |
| ;           | FCB to be passed at 6Ch   |
 | |
| ;           +---------------------------+
 | |
| ;           | DWORD returned value of   |
 | |
| ;           | CS:IP                     |
 | |
| ;           +---------------------------+
 | |
| ;           | DWORD returned value of   |
 | |
| ;           | SS:IP                     |
 | |
| ;           +---------------------------+
 | |
| ;
 | |
| ;           AL = 3 -> load overlay
 | |
| ;
 | |
| ;           +---------------------------+
 | |
| ;           | WORD segment address where|
 | |
| ;           | file will be loaded.      |
 | |
| ;           +---------------------------+
 | |
| ;           | WORD relocation factor to |
 | |
| ;           | be applied to the image.  |
 | |
| ;           +---------------------------+
 | |
| ;
 | |
| ; Returns:
 | |
| ;           AX = exec_invalid_function
 | |
| ;              = exec_bad_format
 | |
| ;              = exec_bad_environment
 | |
| ;              = exec_not_enough_memory
 | |
| ;              = exec_file_not_found
 | |
| ;
 | |
| 
 | |
| IF IBM
 | |
| ZEXEC_DATA  SEGMENT PUBLIC BYTE
 | |
| ZERO =   $
 | |
| ENDIF
 | |
| 
 | |
| exec_blk            DD  ?
 | |
| exec_func           DB  ?
 | |
| exec_fh             DW  ?
 | |
| exec_rel_fac        DW  ?
 | |
| exec_res_len_para   DW  ?
 | |
| exec_init_IP        DW  ?
 | |
| exec_init_CS        DW  ?
 | |
| exec_init_SP        DW  ?
 | |
| exec_init_SS        DW  ?
 | |
| exec_environ        DW  ?
 | |
| exec_size           DW  ?
 | |
| exec_load_block     DW  ?
 | |
| 
 | |
| exec_load_high      DB  ?
 | |
| 
 | |
| exec_internal_buffer    EQU $
 | |
| exec_signature      DW  ?               ; must contain 4D5A  (yay zibo!)
 | |
| exec_len_mod_512    DW  ?               ; low 9 bits of length
 | |
| exec_pages          DW  ?               ; number of 512b pages in file
 | |
| exec_rle_count      DW  ?               ; count of reloc entries
 | |
| exec_par_dir        DW  ?               ; number of paragraphs before image
 | |
| exec_min_BSS        DW  ?               ; minimum number of para of BSS
 | |
| exec_max_BSS        DW  ?               ; max number of para of BSS
 | |
| exec_SS             DW  ?               ; stack of image
 | |
| exec_SP             DW  ?               ; SP of image
 | |
| exec_chksum         DW  ?               ; checksum  of file (ignored)
 | |
| exec_IP             DW  ?               ; IP of entry
 | |
| exec_CS             DW  ?               ; CS of entry
 | |
| exec_rle_table      DW  ?               ; byte offset of reloc table
 | |
| exec_iov            DW  ?               ; overlay number (0 for root)
 | |
| exec_dma            DW  ?
 | |
| exec_internal_buffer_size   EQU $-exec_internal_buffer
 | |
| 
 | |
| IF IBM
 | |
| exec_ctrlc          DB  ?               ; state of users ctrlc flag
 | |
| Exec_low_seg        DW  ?
 | |
| CurrentPDB          DW  ?
 | |
| NUMIO               DB  ?
 | |
| ZEXECDATASIZ    =       $-ZERO
 | |
| ZEXECDATAEND    LABEL   BYTE
 | |
|         PUBLIC  ZEXECDATAEND
 | |
| ZEXEC_DATA  ENDS
 | |
| ZEXEC_CODE  SEGMENT PUBLIC PARA
 | |
|         PUBLIC  $EXEC
 | |
| ZERO =   $
 | |
|         procedure   $EXEC,FAR
 | |
|         ASSUME  CS:EGROUP,SS:RESGROUP,ES:NOTHING,DS:NOTHING
 | |
| ENDIF
 | |
| IF NOT IBM
 | |
|         procedure   $Exec,NEAR
 | |
|         ASSUME  DS:NOTHING, ES:NOTHING
 | |
| ENDIF
 | |
| ;
 | |
| ; validate function
 | |
| ;
 | |
| 
 | |
| IF IBM
 | |
|         PUSH    CS
 | |
|         POP     DS
 | |
|         ASSUME  DS:EGROUP
 | |
| 
 | |
|         MOV     AX,(Set_Ctrl_C_Trapping SHL 8) + 0      ; Save current ctrl-c
 | |
|         INT     int_command
 | |
|         MOV     exec_ctrlc,DL
 | |
|         XOR     DX,DX
 | |
|         MOV     AX,(Set_Ctrl_C_Trapping SHL 8) + 1      ; Turn it off!
 | |
|         INT     int_command
 | |
| 
 | |
|         MOV     AH,Get_current_PDB
 | |
|         INT     int_command
 | |
|         MOV     [CurrentPDB],BX
 | |
| ;
 | |
| ; set up user return stack info
 | |
| ;
 | |
|         MOV     ES,BX
 | |
|         LES     BX,DWORD PTR [user_sp]
 | |
|         MOV     WORD PTR ES:[PDB_user_stack+2],ES
 | |
|         MOV     WORD PTR ES:[PDB_user_stack],BX
 | |
| 
 | |
|         MOV     AH,Get_Default_Drive
 | |
|         INT     int_command
 | |
|         MOV     DL,AL
 | |
|         MOV     AH,Set_default_drive
 | |
|         INT     int_command
 | |
|         MOV     [NUMIO],AL
 | |
| ;
 | |
| ; determine lowest seg address for overwrite problem (round DOWN)
 | |
| ;
 | |
|         MOV     CL,4
 | |
|         MOV     AX,OFFSET ZEXEC_CODE:exec_check
 | |
|         SHR     AX,CL
 | |
|         PUSH    CS
 | |
|         POP     BX
 | |
|         ADD     AX,BX
 | |
|         MOV     [exec_low_seg],AX
 | |
| 
 | |
|         CALL    get_user_stack
 | |
|         ASSUME  DS:NOTHING
 | |
|         MOV     AX,[SI.user_AX]
 | |
|         MOV     BX,[SI.user_BX]
 | |
|         MOV     DX,[SI.user_DX]
 | |
|         MOV     ES,[SI.user_ES]
 | |
|         MOV     DS,[SI.user_DS]
 | |
| ENDIF
 | |
| 
 | |
|         CMP     AL,3                    ; only 0, 1 or 3 are allowed
 | |
|         JNA     exec_check_2
 | |
| 
 | |
| exec_bad_fun:
 | |
|         error   error_invalid_function
 | |
| 
 | |
| exec_ret_err:
 | |
|         transfer    SYS_RET_ERR
 | |
| 
 | |
| exec_check_2:
 | |
|         CMP     AL,2
 | |
|         JZ      exec_bad_fun
 | |
| 
 | |
|         MOV     WORD PTR [exec_blk],BX  ; stash args
 | |
|         MOV     WORD PTR [exec_blk+2],ES
 | |
|         MOV     BYTE PTR [exec_func],AL
 | |
|         MOV     BYTE PTR [exec_load_high],0
 | |
| IF IBM
 | |
|         MOV     AX,(OPEN SHL 8) + 0
 | |
|         INT     int_command
 | |
| ENDIF
 | |
| IF NOT IBM
 | |
|         XOR     AL,AL                   ; open for reading
 | |
|         invoke  $OPEN                   ; is the file there?
 | |
| ENDIF
 | |
|         JC      exec_ret_err
 | |
|         MOV     [exec_fh],AX
 | |
|         MOV     BX,AX
 | |
| IF IBM
 | |
|         MOV     AX,(ioctl SHL 8)        ; get device information
 | |
|         INT     int_command
 | |
| ENDIF
 | |
| IF NOT IBM
 | |
|         XOR     AL,AL
 | |
|         invoke  $IOCTL
 | |
| ENDIF
 | |
|         TEST    DL,devid_ISDEV
 | |
|         JZ      exec_check_environ
 | |
|         MOV     AL,exec_file_not_found
 | |
|         transfer    SYS_RET_ERR
 | |
| 
 | |
| exec_check_environ:
 | |
|         MOV     [exec_load_block],0
 | |
| 
 | |
|         TEST    BYTE PTR [exec_func],exec_func_overlay   ; overlays... no environment
 | |
|         JNZ     exec_read_header
 | |
|         LDS     SI,DWORD PTR [exec_blk] ; get block
 | |
|         MOV     AX,[SI].Exec1_environ   ; address of environ
 | |
|         OR      AX,AX
 | |
|         JNZ     exec_scan_env
 | |
|         MOV     DS,[CurrentPDB]
 | |
|         MOV     AX,DS:[PDB_environ]
 | |
|         MOV     [exec_environ],AX
 | |
|         OR      AX,AX
 | |
|         JZ      exec_read_header
 | |
| 
 | |
| exec_scan_env:
 | |
|         CLD
 | |
|         MOV     ES,AX
 | |
|         XOR     DI,DI
 | |
|         MOV     CX,07FFFh               ; at most 32k of environment
 | |
|         XOR     AL,AL
 | |
| 
 | |
| exec_get_environ_len:
 | |
|         REPNZ   SCASB                   ; find that nul byte
 | |
|         JZ      exec_check              ; CX is out... bad environment
 | |
|         MOV     AL,exec_bad_environment
 | |
|         JMP     exec_bomb
 | |
| 
 | |
| exec_check:
 | |
|         SCASB                           ; is there another nul byte?
 | |
|         JNZ     exec_get_environ_len    ; no, scan some more
 | |
|         PUSH    DI
 | |
|         MOV     BX,DI                   ; AX <- length of environment
 | |
|         ADD     BX,0Fh
 | |
|         MOV     CL,4
 | |
|         SHR     BX,CL                   ; number of paragraphs needed
 | |
|         PUSH    ES
 | |
| IF IBM
 | |
|         MOV     AH,ALLOC
 | |
|         INT     int_command
 | |
| ENDIF
 | |
| IF NOT IBM
 | |
|         invoke  $ALLOC                  ; can we get the space?
 | |
| ENDIF
 | |
|         POP     DS
 | |
|         POP     CX
 | |
|         JNC     exec_save_environ
 | |
|         JMP     exec_no_mem             ; nope... cry and sob
 | |
| 
 | |
| exec_save_environ:
 | |
|         MOV     ES,AX
 | |
|         MOV     [exec_environ],AX       ; save him for a rainy day
 | |
| IF IBM
 | |
|         PUSH    CX
 | |
|         MOV     CX,ES
 | |
|         ADD     CX,BX
 | |
|         CMP     BX,[exec_low_seg]
 | |
|         POP     CX
 | |
|         JA      exec_no_mem
 | |
| ENDIF
 | |
|         XOR     SI,SI
 | |
|         XOR     DI,DI
 | |
|         REP     MOVSB                   ; copy the environment
 | |
| 
 | |
| exec_read_header:
 | |
| ;
 | |
| ; We read in the program header into the above data area and determine
 | |
| ; where in this memory the image will be located.
 | |
| ;
 | |
| IF IBM
 | |
|         PUSH    CS
 | |
|         POP     DS                      ; and put it in DS:DX
 | |
|         ASSUME  DS:EGROUP
 | |
| ENDIF
 | |
| IF NOT IBM
 | |
|         PUSH    SS
 | |
|         POP     DS                      ; and put it in DS:DX
 | |
|         ASSUME  DS:DOSGROUP
 | |
| ENDIF
 | |
|         MOV     CX,exec_internal_buffer_size; header size
 | |
|         MOV     BX,[exec_fh]            ; from the handle
 | |
| IF IBM
 | |
|         MOV     DX,OFFSET EGROUP:exec_signature
 | |
| ENDIF
 | |
| IF NOT IBM
 | |
|         MOV     DX,OFFSET DOSGROUP:exec_signature
 | |
| ENDIF
 | |
|         PUSH    ES
 | |
|         PUSH    DS
 | |
|         CALL    exec_dealloc
 | |
| IF IBM
 | |
|         MOV     AH,READ
 | |
|         INT     int_command
 | |
| ENDIF
 | |
| IF NOT IBM
 | |
|         invoke  $READ
 | |
| ENDIF
 | |
|         CALL    exec_alloc
 | |
|         POP     DS
 | |
|         POP     ES
 | |
|         JC      exec_bad_file
 | |
|         CMP     AX,exec_internal_buffer_size; did we read the right number?
 | |
|         JNZ     exec_com_filej          ; yep... continue
 | |
|         CMP     [exec_max_BSS],0
 | |
|         JNZ     exec_check_sig
 | |
|         MOV     [exec_load_high],-1
 | |
| exec_check_sig:
 | |
|         MOV     AX,[exec_signature]
 | |
|         CMP     AX,exe_valid_signature  ; zibo arises!
 | |
|         JZ      exec_save_start         ; assume com file if no signature
 | |
|         CMP     AX,exe_valid_old_signature  ; zibo arises!
 | |
|         JZ      exec_save_start         ; assume com file if no signature
 | |
| 
 | |
| exec_com_filej:
 | |
|         JMP     exec_com_file
 | |
| 
 | |
| ;
 | |
| ; We have the program header... determine memory requirements
 | |
| ;
 | |
| exec_save_start:
 | |
|         MOV     AX,[exec_pages]         ; get 512-byte pages
 | |
|         MOV     CL,5                    ; convert to paragraphs
 | |
|         SHL     AX,CL
 | |
|         SUB     AX,[exec_par_dir]       ; AX = size in paragraphs
 | |
|         MOV     [exec_res_len_para],AX
 | |
| 
 | |
| ;
 | |
| ; Do we need to allocate memory?  Yes if function is not load-overlay
 | |
| ;
 | |
|         TEST    BYTE PTR [exec_func],exec_func_overlay
 | |
|         JZ      exec_allocate           ; allocation of space
 | |
| ;
 | |
| ; get load address from block
 | |
| ;
 | |
|         LES     DI,DWORD PTR [exec_blk]
 | |
|         MOV     AX,ES:[DI].exec3_load_addr
 | |
|         MOV     [exec_dma],AX
 | |
|         MOV     AX,ES:[DI].exec3_reloc_fac
 | |
|         MOV     [exec_rel_fac],AX
 | |
| IF IBM
 | |
|         JMP     exec_find_res
 | |
| ENDIF
 | |
| IF NOT IBM
 | |
|         JMP     SHORT exec_find_res
 | |
| ENDIF
 | |
| 
 | |
| exec_no_mem:
 | |
|         MOV     AL,exec_not_enough_memory
 | |
|         JMP     SHORT exec_bomb             ; AX should be set by $ALLOC
 | |
| 
 | |
| exec_bad_file:
 | |
|         MOV     AL,exec_bad_format
 | |
| 
 | |
| exec_bomb:
 | |
|         ASSUME  DS:NOTHING,ES:NOTHING
 | |
|         PUSH    AX
 | |
|         MOV     BX,[exec_fh]
 | |
|         CALL    exec_dealloc
 | |
| IF IBM
 | |
|         MOV     AH,CLOSE
 | |
|         INT     int_command
 | |
| ENDIF
 | |
| IF NOT IBM
 | |
|         invoke  $CLOSE
 | |
| ENDIF
 | |
|         POP     AX
 | |
|         transfer    SYS_RET_ERR
 | |
| 
 | |
| exec_allocate:
 | |
| IF IBM
 | |
|         ASSUME  DS:EGROUP
 | |
| ENDIF
 | |
| IF NOT IBM
 | |
|         ASSUME  DS:DOSGROUP
 | |
| ENDIF
 | |
|         PUSH    AX
 | |
|         MOV     BX,0FFFFh               ; see how much room in arena
 | |
|         PUSH    DS
 | |
| IF IBM
 | |
|         MOV     AH,ALLOC
 | |
|         INT     int_command
 | |
| ENDIF
 | |
| IF NOT IBM
 | |
|         invoke  $ALLOC                  ; should have carry set and BX has max
 | |
| ENDIF
 | |
|         POP     DS
 | |
|         POP     AX
 | |
|         ADD     AX,10h                  ; room for header
 | |
|         CMP     BX,11h                  ; enough room for a header
 | |
|         JB      exec_no_mem
 | |
|         CMP     AX,BX                   ; is there enough for bare image?
 | |
|         JA      exec_no_mem
 | |
|         CMP     [exec_load_high],0      ; if load high, use max
 | |
|         JNZ     exec_BX_max             ; use max
 | |
|         ADD     AX,[exec_min_BSS]       ; go for min allocation
 | |
|         JC      exec_no_mem             ; oops! carry
 | |
|         CMP     AX,BX                   ; enough space?
 | |
|         JA      exec_no_mem             ; nope...
 | |
|         SUB     AX,[exec_min_BSS]
 | |
|         ADD     AX,[exec_max_BSS]       ; go for the MAX
 | |
|         JC      exec_BX_max
 | |
|         CMP     AX,BX
 | |
|         JBE     exec_got_block
 | |
| 
 | |
| exec_BX_max:
 | |
|         MOV     AX,BX
 | |
| 
 | |
| exec_got_block:
 | |
|         PUSH    DS
 | |
|         MOV     BX,AX
 | |
|         MOV     [exec_size],BX
 | |
| IF IBM
 | |
|         MOV     AH,ALLOC
 | |
|         INT     int_command
 | |
| ENDIF
 | |
| IF NOT IBM
 | |
|         invoke  $ALLOC                  ; get the space
 | |
| ENDIF
 | |
|         POP     DS
 | |
|         JC      exec_no_mem
 | |
|         MOV     [exec_load_block],AX
 | |
|         ADD     AX,10h
 | |
|         CMP     [exec_load_high],0
 | |
|         JZ      exec_use_ax             ; use ax for load info
 | |
|         ADD     AX,[exec_size]          ; go to end
 | |
|         SUB     AX,[exec_res_len_para]  ; drop off header
 | |
|         SUB     AX,10h                  ; drop off pdb
 | |
| exec_use_ax:
 | |
|         MOV     [exec_rel_fac],AX       ; new segment
 | |
|         MOV     [exec_dma],AX           ; beginning of dma
 | |
| IF IBM
 | |
|         CMP     AX,[exec_low_seg]       ; below loader
 | |
|         JA      exec_no_mem_try
 | |
|         ADD     AX,[exec_res_len_para]  ; go to end
 | |
|         CMP     Ax,[exec_low_seg]       ; above loader
 | |
|         JBE     exec_find_res
 | |
| exec_try_high:
 | |
|         CMP     [exec_load_high],0
 | |
|         JZ      exec_no_memj1
 | |
| exec_try_just_below:
 | |
|         MOV     DX,AX
 | |
|         SUB     DX,[exec_size]          ; get beginning
 | |
|         ADD     DX,[exec_res_len_para]  ; no space
 | |
|         CMP     DX,[exec_low_seg]       ; room there?
 | |
|         JA      exec_no_memj1
 | |
|         MOV     AX,[exec_low_seg]
 | |
|         SUB     AX,[exec_res_len_para]
 | |
|         JMP     exec_use_ax
 | |
| exec_no_mem_try:
 | |
|         MOV     DX,CS
 | |
|         ADD     DX,(zexecdatasiz+zexeccodesize+15)/16
 | |
|         CMP     AX,DX
 | |
|         JAE     exec_try_high
 | |
|         JMP     exec_try_just_below
 | |
| exec_no_memj1:
 | |
|         JMP     exec_no_mem
 | |
| ENDIF
 | |
| 
 | |
| ;
 | |
| ; Determine the location in the file of the beginning of the resident
 | |
| ;
 | |
| exec_find_res:
 | |
|         MOV     DX,[exec_par_dir]
 | |
|         PUSH    DX
 | |
|         MOV     CL,4
 | |
|         SHL     DX,CL                   ; low word of location
 | |
|         POP     AX
 | |
|         MOV     CL,12
 | |
|         SHR     AX,CL                   ; high word of location
 | |
|         MOV     CX,AX                   ; CX <- high
 | |
| 
 | |
| ;
 | |
| ; Read in the resident image (first, seek to it)
 | |
| ;
 | |
|         MOV     BX,[exec_fh]
 | |
|         PUSH    DS
 | |
| IF IBM
 | |
|         MOV     AX,(LSEEK SHL 8) + 0
 | |
|         INT     int_command
 | |
| ENDIF
 | |
| IF NOT IBM
 | |
|         XOR     AL,AL
 | |
|         invoke  $LSEEK                  ; seek to resident
 | |
| ENDIF
 | |
|         POP     DS
 | |
| 
 | |
| exec_big_read:                          ; Read resident into memory
 | |
|         MOV     BX,[exec_res_len_para]
 | |
|         CMP     BX,1000h                ; too many bytes to read?
 | |
|         JB      exec_read_ok
 | |
|         MOV     BX,0FE0h                ; max in one chunk FE00 bytes
 | |
| 
 | |
| exec_read_ok:
 | |
|         SUB     [exec_res_len_para],BX  ; we read (soon) this many
 | |
|         PUSH    BX
 | |
|         MOV     CL,4
 | |
|         SHL     BX,CL                   ; get count in bytes from paras
 | |
|         MOV     CX,BX                   ; count in correct register
 | |
|         MOV     BX,[exec_fh]            ; handle in correct register
 | |
|         PUSH    DS
 | |
|         MOV     DS,[exec_dma]           ; Set up read buffer
 | |
|         ASSUME  DS:NOTHING
 | |
|         XOR     DX,DX
 | |
|         PUSH    CX                      ; save our count
 | |
|         CALL    exec_dealloc
 | |
| IF IBM
 | |
|         MOV     AH,READ
 | |
|         INT     int_command
 | |
| ENDIF
 | |
| IF NOT IBM
 | |
|         invoke  $READ                   ; WOMP!
 | |
| ENDIF
 | |
|         CALL    exec_alloc
 | |
|         POP     CX                      ; get old count to verify
 | |
|         POP     DS
 | |
| IF IBM
 | |
|         ASSUME  DS:EGROUP
 | |
| ENDIF
 | |
| IF NOT IBM
 | |
|         ASSUME  DS:DOSGROUP
 | |
| ENDIF
 | |
|         CMP     CX,AX                   ; did we read enough?
 | |
|         POP     BX                      ; get paragraph count back
 | |
|         JNZ     exec_do_reloc           ; and do reloc if no more to read
 | |
| ;
 | |
| ; We've read in CX bytes... bump DTA location
 | |
| ;
 | |
| 
 | |
|         ADD     [exec_dma],BX           ; bump dma address
 | |
|         CMP     [exec_res_len_para],0
 | |
|         JNZ     exec_big_read
 | |
| 
 | |
| ;
 | |
| ; The image has now been read in.  We must perform relocation to
 | |
| ; the current location.
 | |
| ;
 | |
| 
 | |
| exec_do_reloc:
 | |
|         MOV     CX,[exec_rel_fac]
 | |
|         MOV     AX,[exec_SS]            ; get initial SS
 | |
|         ADD     AX,CX                   ; and relocate him
 | |
|         MOV     [exec_init_SS],AX
 | |
| 
 | |
|         MOV     AX,[exec_SP]            ; initial SP
 | |
|         MOV     [exec_init_SP],AX
 | |
| 
 | |
|         LES     AX,DWORD PTR [exec_IP]
 | |
|         MOV     [exec_init_IP],AX
 | |
|         MOV     AX,ES
 | |
|         ADD     AX,CX                   ; relocated...
 | |
|         MOV     [exec_init_CS],AX
 | |
| 
 | |
|         XOR     CX,CX
 | |
|         MOV     DX,[exec_rle_table]
 | |
|         MOV     BX,[exec_fh]
 | |
|         PUSH    DS
 | |
| IF IBM
 | |
|         MOV     AX,(LSEEK SHL 8) + 0
 | |
|         INT     int_command
 | |
| ENDIF
 | |
| IF NOT IBM
 | |
|         XOR     AX,AX
 | |
|         invoke  $LSEEK
 | |
| ENDIF
 | |
|         POP     DS
 | |
| 
 | |
|         JNC     exec_get_entries
 | |
| exec_bad_filej:
 | |
|         JMP     exec_bad_file
 | |
| 
 | |
| exec_get_entries:
 | |
|         MOV     DX,[exec_rle_count]     ; Number of entries left
 | |
| 
 | |
| exec_read_reloc:
 | |
|         ASSUME  DS:NOTHING
 | |
|         PUSH    DX
 | |
| IF IBM
 | |
|         MOV     DX,OFFSET EGROUP:exec_signature
 | |
| ENDIF
 | |
| IF NOT IBM
 | |
|         MOV     DX,OFFSET DOSGROUP:exec_signature
 | |
| ENDIF
 | |
|         MOV     CX,((exec_internal_buffer_size)/4)*4
 | |
|         MOV     BX,[exec_fh]
 | |
|         PUSH    DS
 | |
|         CALL    exec_dealloc
 | |
| IF IBM
 | |
|         MOV     AH,READ
 | |
|         INT     int_command
 | |
| ENDIF
 | |
| IF NOT IBM
 | |
|         invoke  $READ
 | |
| ENDIF
 | |
|         CALL    exec_alloc
 | |
|         POP     ES
 | |
|         POP     DX
 | |
|         JC      exec_bad_filej
 | |
|         MOV     CX,(exec_internal_buffer_size)/4
 | |
| IF IBM
 | |
|         MOV     DI,OFFSET EGROUP:exec_signature   ; Pointer to byte location in header
 | |
| ENDIF
 | |
| IF NOT IBM
 | |
|         MOV     DI,OFFSET DOSGROUP:exec_signature   ; Pointer to byte location in header
 | |
| ENDIF
 | |
| ;
 | |
| ; Relocate a single address
 | |
| ;
 | |
|         MOV     SI,[exec_rel_fac]
 | |
| 
 | |
| exec_reloc_one:
 | |
|         CMP     DX,0                    ; Any more entries?
 | |
|         JNE     exec_get_addr
 | |
|         JMP     Exec_set_PDB
 | |
| 
 | |
| exec_get_addr:
 | |
|         LDS     BX,DWORD PTR ES:[DI]    ; Get ra/sa of entry
 | |
|         MOV     AX,DS                   ; Relocate address of item
 | |
|         ADD     AX,SI
 | |
|         MOV     DS,AX
 | |
|         MOV     AX,WORD PTR DS:[BX]     ; Relocate item
 | |
|         ADD     AX,SI
 | |
|         MOV     WORD PTR DS:[BX],AX
 | |
|         ADD     DI,4
 | |
|         DEC     DX
 | |
|         LOOP    exec_reloc_one              ; End of internal buffer?
 | |
| 
 | |
| ;
 | |
| ; We've exhausted a single buffer's worth.  Read in the next piece
 | |
| ; of the relocation table.
 | |
| ;
 | |
| 
 | |
|         PUSH    ES
 | |
|         POP     DS
 | |
|         JMP     exec_read_reloc
 | |
| 
 | |
| exec_no_memj:
 | |
|         JMP     exec_no_mem
 | |
| 
 | |
| ;
 | |
| ; we have a .COM file.  First, determine if we are merely loading an overlay.
 | |
| ;
 | |
| exec_com_file:
 | |
|         TEST    BYTE PTR [exec_func],exec_func_overlay
 | |
|         JZ      exec_alloc_com_file
 | |
|         LDS     SI,DWORD PTR [exec_blk]           ; get arg block
 | |
|         LODSW                           ; get load address
 | |
|         MOV     [exec_dma],AX
 | |
|         JMP     SHORT exec_64k          ; read it all!
 | |
| 
 | |
| ; We must allocate the max possible size block (ick!)  and set up
 | |
| ; CS=DS=ES=SS=PDB pointer, IP=100, SP=max size of block.
 | |
| ;
 | |
| exec_alloc_com_file:
 | |
|         MOV     BX,0FFFFh
 | |
| IF IBM
 | |
|         MOV     AH,ALLOC
 | |
|         INT     int_command
 | |
| ENDIF
 | |
| IF NOT IBM
 | |
|         invoke  $ALLOC                  ; largest piece available as error
 | |
| ENDIF
 | |
|         OR      BX,BX
 | |
|         JZ      exec_no_memj
 | |
|         MOV     [exec_size],BX          ; save size of allocation block
 | |
| IF IBM
 | |
|         MOV     AH,ALLOC
 | |
|         INT     int_command
 | |
| ENDIF
 | |
| IF NOT IBM
 | |
|         PUSH    BX
 | |
|         invoke  $ALLOC                  ; largest piece available as error
 | |
|         POP     BX                      ; get size of block...
 | |
| ENDIF
 | |
|         MOV     [exec_load_block],AX
 | |
|         ADD     AX,10h                  ; increment for header
 | |
|         MOV     [exec_dma],AX
 | |
|         SUB     BX,10h                  ; remember header
 | |
| IF IBM
 | |
| ;
 | |
| ; need to read up to exec_low_seg (at most)
 | |
| ;
 | |
|         MOV     CX,[exec_low_seg]
 | |
|         CMP     AX,CX                   ; is base of allocation above spot
 | |
|         JA      exec_check_64k
 | |
|         SUB     CX,AX
 | |
|         CMP     CX,BX
 | |
|         JA      exec_check_64k
 | |
|         MOV     BX,CX
 | |
| 
 | |
| exec_check_64k:
 | |
| ENDIF
 | |
|         CMP     BX,1000h                ; 64k or more?
 | |
|         JAE     exec_64k                ; yes, read only 64k
 | |
|         MOV     AX,BX                   ; convert size to bytes
 | |
|         MOV     CL,4
 | |
|         SHL     AX,CL
 | |
|         JMP     SHORT exec_read_com
 | |
| 
 | |
| exec_64k:
 | |
|         MOV     AX,0FFFFh               ; 64k-1 bytes
 | |
| 
 | |
| exec_read_com:
 | |
|         PUSH    AX                      ; save number to read
 | |
|         MOV     BX,[exec_fh]            ; of com file
 | |
|         XOR     CX,CX                   ; but seek to 0:0
 | |
|         MOV     DX,CX
 | |
| IF IBM
 | |
|         MOV     AX,(LSEEK SHL 8) + 0
 | |
|         INT     int_command
 | |
| ENDIF
 | |
| IF NOT IBM
 | |
|         XOR     AX,AX                   ; seek relative to beginning
 | |
|         invoke  $LSEEK                  ; back to beginning of file
 | |
| ENDIF
 | |
|         MOV     BX,[exec_fh]
 | |
|         POP     CX                      ; number to read
 | |
|         MOV     DS,[exec_dma]
 | |
|         XOR     DX,DX
 | |
|         PUSH    CX
 | |
|         CALL    exec_dealloc
 | |
| IF IBM
 | |
|         MOV     AH,READ
 | |
|         INT     int_command
 | |
| ENDIF
 | |
| IF NOT IBM
 | |
|         invoke  $READ                   ; read in com file
 | |
| ENDIF
 | |
|         CALL    exec_alloc
 | |
|         POP     SI                      ; get number of bytes to read
 | |
|         CMP     AX,SI                   ; did we read them all?
 | |
| IF IBM
 | |
|         JNZ     exec_skip               ; exactly the wrong number... no memory
 | |
|         JMP     exec_no_mem
 | |
| exec_skip:
 | |
| ENDIF
 | |
| IF NOT IBM
 | |
|         JZ      exec_no_memj            ; exactly the wrong number... no memory
 | |
| ENDIF
 | |
|         TEST    BYTE PTR [exec_func],exec_func_overlay
 | |
|         JNZ     exec_set_PDB            ; no starto, chumo!
 | |
|         MOV     AX,[exec_DMA]
 | |
|         SUB     AX,10h
 | |
|         MOV     [exec_init_CS],AX
 | |
|         MOV     [exec_init_IP],100h     ; initial IP is 100
 | |
|         ; SI is at most FFFFh
 | |
|         DEC     SI                      ; make room for stack
 | |
|         ; SI is at most FFFEh, room for a 0!
 | |
|         MOV     [exec_init_SP],SI       ; max value for read is also SP!
 | |
|         MOV     [exec_init_SS],AX
 | |
|         MOV     DS,AX
 | |
|         MOV     WORD PTR DS:[SI],0      ; 0 for return
 | |
| 
 | |
| exec_set_PDB:
 | |
|         MOV     BX,[exec_fh]            ; we are finished with the file.
 | |
|         CALL    exec_dealloc
 | |
| IF IBM
 | |
|         MOV     AH,CLOSE
 | |
|         INT     int_command
 | |
| ENDIF
 | |
| IF NOT IBM
 | |
|         invoke  $CLOSE                  ; release the jfn
 | |
| ENDIF
 | |
|         CALL    exec_alloc
 | |
|         TEST    BYTE PTR [exec_func],exec_func_overlay
 | |
|         JZ      exec_build_header
 | |
|         transfer    SYS_RET_OK          ; overlay load -> done
 | |
| 
 | |
| exec_build_header:
 | |
|         MOV     DX,[exec_load_block]
 | |
| ;
 | |
| ; assign the space to the process
 | |
| ;
 | |
| 
 | |
|         MOV     SI,arena_owner          ; pointer to owner field
 | |
| 
 | |
|         MOV     AX,[exec_environ]       ; get environ pointer
 | |
|         OR      AX,AX
 | |
|         JZ      NO_OWNER                ; no environment
 | |
|         DEC     AX                      ; point to header
 | |
|         MOV     DS,AX
 | |
|         MOV     DS:[SI],DX              ; assign ownership
 | |
| NO_OWNER:
 | |
|         MOV     AX,[exec_load_block]    ; get load block pointer
 | |
|         DEC     AX
 | |
|         MOV     DS,AX                   ; point to header
 | |
|         MOV     DS:[SI],DX              ; assign ownership
 | |
| 
 | |
|         PUSH    DX
 | |
| IF IBM
 | |
|         MOV     AH,DUP_PDB
 | |
|         INT     int_command
 | |
|         MOV     ES,DX
 | |
|         MOV     [CurrentPDB],DX
 | |
| ENDIF
 | |
| IF NOT IBM
 | |
|         MOV     BYTE PTR [CreatePDB], 0FFH  ; indicate a new process
 | |
|         invoke  $Dup_PDB                    ; ES is now PDB
 | |
| ENDIF
 | |
|         POP     DX
 | |
|         PUSH    [exec_environ]
 | |
|         POP     ES:[PDB_environ]
 | |
|         MOV     SI,[exec_size]
 | |
|         ADD     SI,DX
 | |
|         MOV     ES:[PDB_block_len],SI
 | |
| ;
 | |
| ; set up proper command line stuff
 | |
| ;
 | |
|         LDS     SI,DWORD PTR [exec_blk]           ; get the block
 | |
|         PUSH    DS                      ; save its location
 | |
|         PUSH    SI
 | |
|         LDS     SI,DS:[SI.exec0_5C_FCB] ; get the 5c fcb
 | |
|         MOV     CX,12                   ; copy drive, name and ext
 | |
|         PUSH    CX
 | |
|         MOV     DI,5Ch
 | |
|         MOV     BL,DS:[SI]
 | |
|         REP     MOVSB
 | |
|         XOR     AX,AX                   ; zero extent, etc for CPM
 | |
|         STOSW
 | |
|         STOSW
 | |
|         POP     CX
 | |
|         POP     SI                      ; get block
 | |
|         POP     DS
 | |
|         PUSH    DS                      ; save (again)
 | |
|         PUSH    SI
 | |
|         LDS     SI,DS:[SI.exec0_6C_FCB] ; get 6C FCB
 | |
|         MOV     DI,6Ch                  ; do same as above
 | |
|         MOV     BH,DS:[SI]
 | |
|         REP     MOVSB
 | |
|         STOSW
 | |
|         STOSW
 | |
|         POP     SI                      ; get block (last time)
 | |
|         POP     DS
 | |
|         LDS     SI,DS:[SI.exec0_com_line]   ; command line
 | |
|         MOV     CX,80h
 | |
|         MOV     DI,CX
 | |
|         REP     MOVSB                   ; Wham!
 | |
| 
 | |
| ;
 | |
| ; Process BX into default AX (validity of drive specs on args)
 | |
| ;
 | |
|         DEC     CL                      ; get 0FFh in CX
 | |
|         CMP     BH,[NUMIO]
 | |
|         JBE     exec_BH_good
 | |
|         MOV     BH,CL
 | |
|         JMP     SHORT exec_BL
 | |
| exec_BH_good:
 | |
|         XOR     BH,BH
 | |
| exec_BL:
 | |
|         CMP     BL,[NUMIO]
 | |
|         JBE     exec_BL_good
 | |
|         MOV     BL,CL
 | |
|         JMP     SHORT exec_set_return
 | |
| exec_BL_good:
 | |
|         XOR     BL,BL
 | |
| exec_set_return:
 | |
|         invoke  get_user_stack          ; get his return address
 | |
|         PUSH    [SI.user_CS]            ; suck out the CS and IP
 | |
|         PUSH    [SI.user_IP]
 | |
|         PUSH    [SI.user_CS]            ; suck out the CS and IP
 | |
|         PUSH    [SI.user_IP]
 | |
|         POP     WORD PTR ES:[PDB_Exit]
 | |
|         POP     WORD PTR ES:[PDB_Exit+2]
 | |
|         XOR     AX,AX
 | |
|         MOV     DS,AX
 | |
|         POP     DS:[addr_int_terminate] ; save them where we can get them later
 | |
|         POP     DS:[addr_int_terminate+2]   ; when the child exits.
 | |
| IF NOT IBM
 | |
|         MOV     WORD PTR [DMAADD],80h
 | |
|         MOV     DS,[CurrentPDB]
 | |
|         MOV     WORD PTR [DMAADD+2],DS
 | |
| ENDIF
 | |
| IF IBM
 | |
|         PUSH    DX
 | |
|         PUSH    DS
 | |
|         MOV     DS,[CurrentPDB]
 | |
|         MOV     DX,80h
 | |
|         MOV     AH,SET_DMA
 | |
|         INT     int_command
 | |
|         POP     DS
 | |
|         POP     DX
 | |
| ENDIF
 | |
|         TEST    BYTE PTR [exec_func],exec_func_no_execute
 | |
|         JZ      exec_go
 | |
| 
 | |
|         LDS     SI,DWORD PTR [exec_init_SP] ; get stack
 | |
|         LES     DI,DWORD PTR [exec_blk]           ; and block for return
 | |
|         MOV     ES:[DI].exec1_SS,DS     ; return SS
 | |
| 
 | |
|         DEC     SI                      ; 'push' default AX
 | |
|         DEC     SI
 | |
|         MOV     DS:[SI],BX              ; save default AX reg
 | |
|         MOV     ES:[DI].exec1_SP,SI     ; return 'SP'
 | |
| 
 | |
|         LDS     AX,DWORD PTR [exec_init_IP]
 | |
|         MOV     ES:[DI].exec1_CS,DS     ; initial entry stuff
 | |
| 
 | |
|         MOV     ES:[DI].exec1_IP,AX
 | |
|         transfer    SYS_RET_OK
 | |
| 
 | |
| exec_go:
 | |
| IF IBM
 | |
|         CALL    restore_ctrlc               ; restore value of ctrl-c checker
 | |
| ENDIF
 | |
|         LDS     SI,DWORD PTR [exec_init_IP] ; get entry point
 | |
|         CLI
 | |
| IF NOT IBM
 | |
|         MOV     BYTE PTR INDOS,0
 | |
| ENDIF
 | |
|         MOV     SS,[exec_init_SS]       ; set up user's stack
 | |
|         ASSUME  SS:NOTHING
 | |
|         MOV     SP,[exec_init_SP]       ; and SP
 | |
|         STI
 | |
|         PUSH    DS                      ; fake long call to entry
 | |
|         PUSH    SI
 | |
|         MOV     ES,DX                   ; set up proper seg registers
 | |
|         MOV     DS,DX
 | |
|         MOV     AX,BX                   ; set up proper AX
 | |
|         procedure   exec_long_ret,FAR
 | |
|         RET
 | |
| exec_long_ret   ENDP
 | |
| 
 | |
| $Exec   ENDP
 | |
| 
 | |
|         procedure   exec_dealloc,near
 | |
|         ASSUME      DS:NOTHING,ES:NOTHING
 | |
|         PUSH        BX
 | |
|         MOV         BX,arena_owner_system
 | |
|         CALL        exec_do_change_owner
 | |
|         POP         BX
 | |
|         return
 | |
| exec_dealloc  ENDP
 | |
| 
 | |
|         procedure   exec_alloc,near
 | |
|         PUSH        BX
 | |
|         MOV         BX,[CurrentPDB]
 | |
|         CALL        exec_do_change_owner
 | |
|         POP         BX
 | |
|         return
 | |
| exec_alloc  ENDP
 | |
| 
 | |
|         procedure   exec_do_change_owner,NEAR
 | |
|         PUSH    DS
 | |
|         PUSH    AX
 | |
|         MOV     AX,[exec_environ]
 | |
|         OR      AX,AX
 | |
|         JZ      exec_alloc_try_load
 | |
|         DEC     AX
 | |
|         MOV     DS,AX
 | |
|         MOV     DS:[arena_owner],BX
 | |
| exec_alloc_try_load:
 | |
|         MOV     AX,[exec_load_block]
 | |
|         OR      AX,AX
 | |
|         JZ      exec_alloc_done
 | |
|         DEC     AX
 | |
|         MOV     DS,AX
 | |
|         MOV     DS:[arena_owner],BX
 | |
| exec_alloc_done:
 | |
|         POP     AX
 | |
|         POP     DS
 | |
|         RET
 | |
| exec_do_change_owner    ENDP
 | |
| 
 | |
| IF IBM
 | |
| SYS_RET_ERR:
 | |
|         CALL    get_user_stack
 | |
|         PUSH    [SI.user_f]
 | |
|         XOR     AH,AH
 | |
|         MOV     [SI.user_AX],AX
 | |
|         POPF
 | |
|         STC
 | |
|         JMP SYS_RET
 | |
| SYS_RET_OK:
 | |
|         CALL    get_user_stack
 | |
|         PUSH    [SI.user_f]
 | |
|         POPF
 | |
|         CLC
 | |
| SYS_RET:
 | |
|         PUSHF
 | |
|         CALL    restore_ctrlc
 | |
|         POP     [SI.user_f]
 | |
|         JMP     exec_long_ret
 | |
| 
 | |
| ;
 | |
| ; get_user_stack returns the user's stack (and hence registers) in DS:SI
 | |
| ;
 | |
|         procedure   get_user_stack,NEAR
 | |
|         PUSH    SS
 | |
|         POP     DS
 | |
|         ASSUME  DS:RESGROUP
 | |
|         LDS     SI,DWORD PTR [user_SP]
 | |
|         RET
 | |
| get_user_stack  ENDP
 | |
| ;
 | |
| ; restore value of the ctrl-c checker
 | |
| ;
 | |
|         procedure    restore_ctrlc
 | |
|         PUSH    AX
 | |
|         PUSH    DX
 | |
|         MOV     DL,CS:[exec_ctrlc]
 | |
|         MOV     AX,(Set_Ctrl_C_Trapping SHL 8) + 1      ; Put it back
 | |
|         INT     int_command
 | |
|         POP     DX
 | |
|         POP     AX
 | |
|         RET
 | |
| restore_ctrlc   ENDP
 | |
| 
 | |
| ZEXECCODESIZE   EQU     $-ZERO
 | |
| ZEXECCODEEND    LABEL BYTE
 | |
|         PUBLIC  ZEXECCODEEND
 | |
| ZEXEC_CODE      ENDS
 | |
| ENDIF
 |