home *** CD-ROM | disk | FTP | other *** search
- page 60,126
- ;
- ; Name iscalint -- Simulate software-interrupt-style call to
- ; interrupt service routine.
- ;
- ; Synopsis ercode = iscalint(pintvec,preg);
- ;
- ; int ercode Error code: 0 if okay;
- ; 1 if interrupt vector is invalid.
- ; ADS *pintvec Address of ADS structure containing
- ; four-byte address to call
- ; ALLREG *preg Standard registers
- ;
- ; Description This function invokes the interrupt service routine
- ; whose address is contained in *pintvec and returns the
- ; results in the most general fashion. All values
- ; specified in *preg are passed to the interrupt service
- ; routine in the corresponding registers except for CS,
- ; IP, SS, and SP. All the resulting register contents are
- ; returned in *preg (again, except for CS, IP, SS, and
- ; SP).
- ;
- ; The interrupt state (that is, whether interrupts are
- ; enabled or disabled) is preserved.
- ;
- ; Use ISRETVEC to obtain the address of an interrupt
- ; service routine which is installed in one of the
- ; standard interrupt vectors in low memory.
- ;
- ; Use the function called DOS for invoking DOS functions
- ; because it protects against certain DOS bugs.
- ;
- ; Returns ercode Error code: 0 if okay, 1 if interrupt
- ; vector is invalid.
- ; *preg The values of the registers may be altered
- ; by the interrupt handler.
- ;
- ; Version 3.0 (C)Copyright Blaise Computing Inc. 1983,1984,1985,1986
- ;
-
- popff macro ;; Simulate POPF instruction
- local do_call,do_iret
- jmp short do_call
- do_iret: iret ;; Pop IP, CS, flags.
- do_call: push cs ;; Push CS
- call do_iret ;; Push IP & jump.
- endm
-
- name iscalint
-
- LONGPROG = 0 ; initialize constants for
- LONGDATA = 0 ; Pass1 of the assembler
-
- include compiler.mac ; Specifies the C compiler
-
- if LAT200 or LAT210 or LAT300
- include dos.mac
- LONGPROG = LPROG
- LONGDATA = LDATA
-
- pseg
- public iscalint
- if LPROG
- x equ 6 ; parameter offset
- iscalint proc far
- else
- x equ 4
- iscalint proc near
- endif
- endif
-
- if CI201A
- include model.h
- include prologue.h
- LONGPROG = @bigmodel
- LONGDATA = @bigmodel
-
- public iscalint
- if @bigmodel
- x equ 6 ; parameter offset
- iscalint proc far
- else
- x equ 4
- iscalint proc near
- endif
- endif
-
- if MSC300
- include dos.mac
- LONGPROG = LPROG
- LONGDATA = LDATA
-
- pseg iscalint
- public _iscalint
- if LPROG
- x equ 6 ; parameter offset
- _iscalint proc far
- else
- x equ 4
- _iscalint proc near
- endif
- endif
-
- ERR_BADVEC equ 1 ; Error code indicating invalid
- ; interrupt vector.
-
- if LONGDATA
- pintvec equ dword ptr [bp + x]
- preg equ dword ptr [bp + x + 4]
- else
- pintvec equ word ptr [bp + x]
- preg equ word ptr [bp + x + 2]
- endif
-
- ALLREG struc ; ALLREG structure
- reg_ax dw ? ; (This must match declaration in
- reg_bx dw ? ; BUTILITY.H).
- reg_cx dw ?
- reg_dx dw ?
- reg_si dw ?
- reg_di dw ?
- reg_ds dw ?
- reg_es dw ?
- reg_ss dw ?
- reg_cs dw ?
- reg_flags dw ?
- reg_bp dw ?
- reg_sp dw ?
- reg_ip dw ?
- ALLREG ends
-
-
- push bp
- mov bp,sp
-
- push si ; Save registers that are important
- push di ; to the C environment.
- push ds
- push es
- ; Obtain interrupt vector
- if LONGDATA
- lds bx,pintvec
- assume ds:nothing
- else
- mov bx,pintvec
- endif
- mov ax,ds ; Quit if pintvec is 0:0.
- or ax,bx
- jz bad_vector
- les si,[bx]
- assume es:nothing ; Now ES:SI is address of interrupt
- ; handler.
- mov ax,es ; Quit if *pintvec is 0:0.
- or ax,si
- jnz ok_vector
-
- bad_vector:
- mov ax,ERR_BADVEC
- jmp short exit
-
- ok_vector:
- ; Obtain contents of preg.
- if LONGDATA
- lds bx,preg
- assume ds:nothing
- else
- mov bx,preg
- endif
- ; Now DS:BX points to *preg
- ; (ALLREG structure is at DS:BX).
-
- ; Begin setting up for the call:
-
- push ds ; Step 1: Save stack frame pointer
- push bp ; so we can restore it in Step 10.
-
- pushf ; Step 2: Save our own flags
- ; for restoration in Step 9.
-
- ; Step 3: Push the stack items that
- ; the ISR will see: a copy of its
- ; flags and its return address
- ; (CS:ret_addr).
-
- mov ax,[bx.reg_flags] ; Push the flags
- push ax ; without clearing IF and TF.
- push cs ; Push CS.
- mov cx,offset ret_addr
- push cx ; Push ret_addr.
-
- ; Step 4:
- ; We will enter the handler by doing
- ; an IRET, so push the flags and the
- ; vector.
-
- and ah,0fch ; Push the flags
- push ax ; with IF and TF cleared.
- push es ; Push the vector.
- push si
-
- mov ax,[bx.reg_ax] ; Step 5: Load the easy registers.
- mov cx,[bx.reg_cx]
- mov dx,[bx.reg_dx]
- mov si,[bx.reg_si]
- mov di,[bx.reg_di]
- mov es,[bx.reg_es]
- assume es:nothing
- mov bp,[bx.reg_bp]
-
- push [bx.reg_bx] ; Step 6: Load DS and BX
- mov ds,[bx.reg_ds] ; by temporarily using the stack.
- assume ds:nothing
- pop bx
-
- iret ; Step 7: Load the flags and jump
- ; to the handler. (This IRET uses &
- ; removes what was pushed in Step 4.)
-
- ret_addr:
- assume es:nothing,ds:nothing
- ; The interrupt handler brought us here
- ; by performing an IRET, thus popping
- ; what was pushed in Step 3.
- ; (However, the interrupt handler
- ; may choose to modify the flags
- ; before popping them.)
-
- push bp ; Step 8: Temporarily preserve
- push ds ; returned values of BP, DS, BX,
- push bx ; and flags.
- pushf
-
- mov bp,sp ; Step 9: Restore our own flags
- mov bx,[bp+8] ; (which were pushed in Step 2).
- push bx ; (They may contain special values
- popff ; required by the C environment).
-
- lds bp,[bp+10] ; Step 10: Restore stack frame pointer
- assume ds:nothing ; (pushed in Step 1) so we can find preg.
-
- ; Obtain contents of preg.
- if LONGDATA
- lds bx,preg
- assume ds:nothing
- else
- mov bx,preg
- endif
- ; Now DS:BX points to *preg
- ; (ALLREG structure is at DS:BX).
-
- mov [bx.reg_ax],ax ; Store the easy registers in
- mov [bx.reg_cx],cx ; the *preg structure.
- mov [bx.reg_dx],dx
- mov [bx.reg_si],si
- mov [bx.reg_di],di
- mov [bx.reg_es],es
-
- pop [bx.reg_flags] ; Transfer values pushed in Step 8
- pop [bx.reg_bx] ; to the *preg structure.
- pop [bx.reg_ds]
- pop [bx.reg_bp]
-
- add sp,6 ; Discard values pushed in
- ; Steps 2 and 1.
-
- xor ax,ax ; Success code.
-
- exit:
- ; Now AX contains the error/success code.
-
- pop es ; Restore registers that are important
- pop ds ; to the C environment.
- pop di
- pop si
-
- pop bp
- ret
-
- if MSC300
- _iscalint endp
- else
- iscalint endp
- endif
-
- if LAT200 or LAT210 or LAT300
- endps
- endif
-
- if CI201A
- include epilogue.h
- endif
-
- if MSC300
- endps iscalint
- endif
-
- end