home *** CD-ROM | disk | FTP | other *** search
/ Liren Large Software Subsidy 7 / 07.iso / c / c005 / 5.ddi / ISDISPAT.ASM < prev    next >
Encoding:
Assembly Source File  |  1987-04-13  |  14.7 KB  |  543 lines

  1.     page 60,126
  2. ;
  3. ; Name        isdispat -- Interrupt service routine dispatcher.
  4. ;
  5. ; Synopsis    This routine is NOT to be called from any user program!
  6. ;        It is designed to be called from the "Far Call"
  7. ;        instruction at the beginning of an ISR control block.
  8. ;
  9. ; Description    C TOOLS PLUS interrupt service routines (ISRs) are each
  10. ;        governed by their own ISR control block which is set up
  11. ;        by ISINSTAL.  When an ISR is installed, the address of
  12. ;        its control block is put into the relevant interrupt
  13. ;        vector.  The first item in each control block is a far
  14. ;        call instruction to this routine.  Therefore, when the
  15. ;        interrupt occurs, the next address in the control block
  16. ;        is pushed onto the stack and ISDISPAT is called.
  17. ;
  18. ;        This routine then performs the remaining operations
  19. ;        needed to invoke the user's ISR (which is assumed to be
  20. ;        written in C).    This consists of the following:
  21. ;
  22. ;            (1) saving all registers;
  23. ;            (2) finding the ISR control block and using
  24. ;            the information therein;
  25. ;            (3) deciding whether a new stack may be set up
  26. ;            for this invocation of the ISR (if not,
  27. ;            then ISDISPAT just returns);
  28. ;            (4) setting up and using a new stack;
  29. ;            (5) preparing copies of the ISR's arguments,
  30. ;            including an ALLREG structure containing
  31. ;            the values of the machine registers that
  32. ;            were in effect at the time of the interrupt;
  33. ;            (6) calling the ISR as a normal C function;
  34. ;            (7) retrieving the register values in the ALLREG
  35. ;            structure;
  36. ;            (8) restoring the original stack;
  37. ;            (9) discarding the stack that was created for
  38. ;            this invocation of the ISR (as well as any
  39. ;            stacks left by intervening invocations that
  40. ;            did not return);
  41. ;            (10) restoring the machine registers using
  42. ;             values from the ALLREG structure;
  43. ;            (11) performing an IRET or RETF, as selected
  44. ;             by the ISR.
  45. ;
  46. ;        The interrupt state (that is, whether interrupts are
  47. ;        enabled or disabled) is preserved unless the ISR
  48. ;        requests otherwise.
  49. ;
  50. ; Returns    ercode          Error code:  0 if okay, 1 if interrupt
  51. ;                  vector is invalid.
  52. ;        *preg          The values of the registers may be altered
  53. ;                  by the interrupt handler.
  54. ;
  55. ; Version    3.0 (C)Copyright Blaise Computing Inc. 1986
  56. ;
  57. ; Version    3.02 March 23, 1987
  58. ;        Forced STK_GROWTH_LIMIT variable references to use DS
  59. ;            register.  (The only effect of this is to make the
  60. ;            source code more reliable in case of modification.
  61. ;            The object code produced by this version is the same
  62. ;            as in C TOOLS PLUS version 3.00.)
  63. ;
  64.  
  65. popff     macro                ;; Simulate POPF instruction
  66.      local    do_call,do_iret
  67.      jmp    short do_call
  68. do_iret: iret                ;; Pop IP, CS, flags.
  69. do_call: push    cs            ;; Push CS
  70.      call    do_iret         ;; Push IP & jump.
  71.      endm
  72.  
  73.      name       isdispat
  74.  
  75.      LONGPROG  = 0               ; initialize constants for
  76.      LONGDATA  = 0               ; Pass1 of the assembler
  77.  
  78.      include   compiler.mac        ; Specifies the C compiler
  79.  
  80.      if LAT200 or LAT210 or LAT300
  81.      include  dos.mac
  82.      LONGPROG = LPROG
  83.      LONGDATA = LDATA
  84.  
  85.      extrn      _BASE:word           ; Stack growth limit.
  86. STK_GROWTH_LIMIT  equ    ds:_BASE
  87.  
  88.      pseg
  89.      public   isdispat
  90. isdispat proc      far
  91.      endif
  92.  
  93.      if MSC300
  94.      include  dos.mac
  95.      LONGPROG = LPROG
  96.      LONGDATA = LDATA
  97.  
  98.      extrn      STKHQQ:word           ; Stack growth limit.
  99. STK_GROWTH_LIMIT  equ    ds:STKHQQ
  100.  
  101.      pseg      isdispat
  102.      public   _isdispat
  103. _isdispat proc       far
  104.      endif
  105.  
  106. ISRCTRL struc                ; ISRCTRL:    ISR control block
  107.                     ; (This must match declaration in
  108.                     ; BISR.H.)
  109.                     ;
  110. icb_fcall_opcode    dw    ?        ; NOP + Far call opcode
  111.                     ;    (0x9A90)
  112. icb_isrdisp        dd    ?        ; Address of ISR dispatcher
  113. icb_iret_opcode     dw    ?        ; IRET + RETF opcodes (0xcbcf)
  114.                     ;    (Offset of icb_iret_opcode
  115.                     ;    is on stack on entry to ISDISPAT.)
  116.                     ;
  117. icb_isrstk_r        dw    ?        ; ISR stack region:  offset
  118. icb_isrstk_s        dw    ?        ;             segment
  119. icb_isrstksize        dw    ?        ; ISR stack size
  120. icb_isrsp        dw    ?        ; ISR SP value at start of
  121.                     ;    current ISR call
  122.                     ;
  123. icb_isrds        dw    ?        ; DS value required by ISR
  124. icb_isres        dw    ?        ; ES value required by ISR
  125. icb_isr         dd    ?        ; Address of ISR itself
  126. icb_isrpsp        dw    ?        ; PSP of program containing ISR
  127.                     ;
  128. icb_prev_vec        dd    ?        ; Previous value of vector
  129.                     ;
  130. icb_level        dw    ?        ; Number of calls in progress
  131.                     ;    (0 if not in progress)
  132. icb_limit        dw    ?        ; Max number of nested calls
  133. icb_signature        dw    ?        ; Signature identifying this
  134.                     ;    structure
  135. icb_sign2        dw    ?        ; One's complement of
  136.                     ;    "signature"
  137. icb_ident        db    16 dup (?)  ; Identifying name
  138. icb_control        dw    ?        ; Bit fields to control
  139.                     ;    dispatcher options
  140. icb_status        dw    ?        ; Status info left by
  141.                     ;    dispatcher
  142. icb_scratch        db    10 dup (?)  ; Scratch space for use by
  143.                     ;    dispatcher & related programs
  144. ISRCTRL ends
  145.  
  146. ;   Exit style codes -- these must match BISR.H.
  147.  
  148. IEXIT_NORMAL    equ    0
  149. IEXIT_RETF    equ    1
  150.  
  151. ALLREG     struc                ; ALLREG structure
  152. reg_ax     dw    ?            ; (This must match declaration in
  153. reg_bx     dw    ?            ; BUTILITY.H).
  154. reg_cx     dw    ?
  155. reg_dx     dw    ?
  156. reg_si     dw    ?
  157. reg_di     dw    ?
  158. reg_ds     dw    ?
  159. reg_es     dw    ?
  160. reg_ss     dw    ?
  161. reg_cs     dw    ?
  162. reg_flags dw    ?
  163. reg_bp     dw    ?
  164. reg_sp     dw    ?
  165. reg_ip     dw    ?
  166.  
  167. allreg_size    db  ?
  168. ALLREG     ends
  169.  
  170. ; ****************   BEGIN CODE HERE   ******************
  171.  
  172.      assume ds:nothing,es:nothing,ss:nothing
  173.  
  174. ;   Save initial register values so we can have some workspace.
  175.  
  176.      push    bp            ; Save original BP (used only here >--\ )
  177.      mov    bp,sp            ;                      |
  178.      push    bp            ; (Entry value of SP) - 2          |
  179.      push    [bp]            ; Original BP             <----/
  180.      push    ax            ; Original AX
  181.      push    bx            ; Original BX
  182.      push    cx            ; Original CX
  183.      push    dx            ; Original DX
  184.      push    ds            ; Original DS
  185.      pushf                ; Original flags
  186.      push    es            ; Original ES
  187.      push    si            ; Original SI
  188.  
  189.      add    word ptr [bp-2],6   ; Convert entry SP-2 to "Caller's SP"
  190.                     ;    thus ignoring the first BP we pushed
  191.                     ;    and segment and offset pushed by
  192.                     ;    the far call in the control block.
  193.  
  194. ;   Set up DS:BX to address these values we just saved on caller's stack.
  195.  
  196.      mov    bx,ss
  197.      mov    ds,bx
  198.      assume ds:nothing
  199.      mov    bx,bp
  200.  
  201.                     ; Current stack contents
  202.                     ; (high addresses to low):
  203.                     ;
  204. caller_flags equ word ptr [bx+10]   ;    Caller's flags
  205. caller_cs   equ word ptr [bx+8]     ;    Caller's CS
  206.                     ; "Caller's SP" points to . . .
  207. caller_ip   equ word ptr [bx+6]     ;    Caller's IP (see note below)
  208. ret_seg     equ word ptr [bx+4]     ;    Segment of ISR Control Block
  209.                     ; Entry SP pointed to . . .
  210. ret_offset  equ word ptr [bx+2]     ;    Offset of icb_iret_opcode in
  211.                     ;      ISR Control Block
  212.                     ; BX and BP point to . . .
  213.                     ;    Original BP (won't be used again)
  214. caller_sp   equ word ptr [bx-2]     ;  "Caller's SP value"
  215. orig_bp     equ word ptr [bx-4]     ;    Original BP again
  216. orig_ax     equ word ptr [bx-6]     ;    Original AX
  217. orig_bx     equ word ptr [bx-8]     ;    Original BX
  218. orig_cx     equ word ptr [bx-10]    ;    Original CX
  219. orig_dx     equ word ptr [bx-12]    ;    Original DX
  220. orig_ds     equ word ptr [bx-14]    ;    Original DS
  221. orig_flags  equ word ptr [bx-16]    ;    Original flags
  222. orig_es     equ word ptr [bx-18]    ;    Original ES
  223. orig_si     equ word ptr [bx-20]    ;    Original SI
  224.  
  225.     ; Note:  the value called "caller's SP" is the SP value
  226.     ;   immediately after the INT instruction which invoked
  227.     ;   this interrupt.  Any adjustments to "caller's SP" should
  228.     ;   be RELATIVE (e.g., to discard some stack items just before
  229.     ;   returning to caller, add a fixed value to SP in the
  230.     ;   ALLREG structure).
  231.  
  232. ;   Set up ES:SI to point to beginning of ISR control block.
  233.  
  234.      mov    es,ret_seg
  235.      assume es:nothing
  236.      mov    si,ret_offset
  237.      sub    si,(icb_iret_opcode-icb_fcall_opcode)
  238.  
  239. ;   Make sure nesting depth isn't over limit.
  240.  
  241.      mov    ax,es:[si].icb_level
  242.      cmp    ax,es:[si].icb_limit
  243.      jb    level_ok
  244.  
  245.      jmp    exit            ; Nested too deep:    just exit.
  246.  
  247.                     ; Check that stack size is adequate.
  248.  
  249. ;   Generate a new stack for the ISR:  increment level & icb_isrsp.
  250.  
  251. level_ok:                ; Nesting depth is okay.
  252.  
  253.      inc    ax            ; Increment nesting depth.
  254.      cli
  255.      mov    es:[si].icb_level,ax
  256.      push    ax            ; Local copy of nesting depth
  257.                     ;    (will be popped later).
  258.  
  259.      mov    ax,es:[si].icb_isrsp; Former ISR stack pointer (plus 6
  260.      mov    dx,ax            ;    for safety) will serve as
  261.      add    dx,6            ;    stack growth limit.
  262.      add    ax,es:[si].icb_isrstksize  ; Raise stack pointer
  263.      mov    es:[si].icb_isrsp,ax       ;   for a new stack.
  264.  
  265. ;   Switch to the ISR's own stack (which we just generated).
  266. ;   Interrupts must be disabled during this.
  267.  
  268.      mov    ax,sp            ; Save old SP (old SS is in DS register)
  269.      mov    ss,es:[si].icb_isrstk_s   ; Switch to ISR's own stack.
  270.      assume ss:nothing
  271.      mov    sp,es:[si].icb_isrsp
  272.  
  273.      push    orig_flags        ; Restore original flags to
  274.      popff                ;  (possibly) re-enable interrupts.
  275.  
  276. ;   Now create an ALLREG structure on the ISR's own stack.
  277.  
  278.      push    caller_ip
  279.      push    caller_sp
  280.      push    orig_bp
  281.      push    caller_flags
  282.      push    caller_cs
  283.      push    ds            ; Caller's SS
  284.      push    orig_es
  285.      push    orig_ds
  286.      push    di
  287.      push    orig_si
  288.      push    orig_dx
  289.      push    orig_cx
  290.      push    orig_bx
  291.      push    orig_ax
  292.      mov    bp,sp            ; Now BP points to ALLREG structure.
  293. regs     equ    [bp]
  294.  
  295. ;   Create two-word area for messages between dispatcher (us) and ISR.
  296.  
  297.      push    orig_flags        ; "Working" flags -- depending on how
  298.                     ;    dispatcher exits, these may persist
  299.                     ;    on return to caller.
  300.      mov    cx,IEXIT_NORMAL
  301.      push    cx            ; Default exit style.
  302. message  equ    dword ptr [bp-4]
  303.  
  304. ;   Create local vector to the ISR
  305.  
  306.      if    LONGPROG
  307.      push    word ptr es:[si].icb_isr+2
  308.      push    word ptr es:[si].icb_isr
  309. vector_to_isr    equ    dword ptr [bp-8]
  310.      else
  311.      push    word ptr es:[si].icb_isr
  312. vector_to_isr    equ    word ptr [bp-6]
  313.      endif
  314.  
  315. ;   Save items needed locally by dispatcher (us)
  316.  
  317.      push    ds            ; Caller's SS
  318.      push    ax            ; SP in caller's stack just before
  319.                     ;    stack switch
  320.      push    bx            ; Frame pointer in caller's stack.
  321.      push    es            ; Segment of ISR control block.
  322.      push    si            ; Offset  of ISR control block.
  323.  
  324. ;   Point to ISR's data segment
  325.  
  326.      mov    ds,es:[si].icb_isrds
  327.      assume ds:nothing
  328.  
  329. ;   Save former stack growth limit and establish new value therefor.
  330. ;   (The stack growth limit variable is in the ISR's data segment.)
  331.  
  332.      push    STK_GROWTH_LIMIT
  333.      mov    STK_GROWTH_LIMIT,dx
  334.  
  335. ;   Push ISR's arguments.
  336.  
  337.      if    LONGDATA
  338.  
  339.      push    ss            ; Address of message area.
  340.      lea    ax,message
  341.      push    ax
  342.  
  343.      push    es            ; Address of ISR control block.
  344.      push    si
  345.  
  346.      push    ss            ; Address of ALLREG structure.
  347.      lea    ax,regs
  348.      push    ax
  349.  
  350.      else
  351.  
  352.      lea    ax,message        ; Address of message area.
  353.      push    ax
  354.  
  355.      push    si            ; Address of ISR control block.
  356.                     ;  (For S, P, and M model programs
  357.                     ;  this must be in ISR's default
  358.                     ;  data segment.)
  359.  
  360.      lea    ax,regs         ; Address of ALLREG structure.
  361.      push    ax
  362.  
  363.      endif
  364.  
  365. ;   Set up remaining registers required by ISR.
  366.  
  367.      cld
  368.      mov    es,es:[si].icb_isres
  369.      assume es:nothing
  370.  
  371. ;   Actually call the ISR.
  372.  
  373.      call    vector_to_isr
  374.  
  375. ;   Discard the ISR's arguments in usual C fashion.
  376.  
  377.      if    LONGDATA
  378.      add    sp,12
  379.      else
  380.      add    sp,6
  381.      endif
  382.  
  383. ;   Restore former stack growth limit.
  384.  
  385.      pop    STK_GROWTH_LIMIT
  386.  
  387. ;   Restore dispatcher's various pointers.
  388.  
  389.      pop    si
  390.      pop    es            ; Now ES:SI point to ISR control block.
  391.      assume es:nothing
  392.      pop    bx
  393.      pop    ax
  394.      pop    ds            ; DS:AX is SS:SP in caller's stack
  395.      assume ds:nothing        ;    just before stack switch.
  396.                     ; DS:BX is frame pointer in caller's
  397.                     ;    stack.
  398.  
  399. ;   Discard the local vector to the ISR.
  400.  
  401.      if    LONGPROG
  402.      add    sp,4
  403.      else
  404.      inc    sp
  405.      inc    sp
  406.      endif
  407.  
  408. ;   Read the message area & react to it.
  409.  
  410.      pop    dx            ; "Exit style"
  411.      pop    orig_flags        ; "Working flags"
  412.  
  413. ;   Transfer ALLREG structure values into proper registers or into
  414. ;   holding locations.    These values may have been modified by the ISR.
  415.  
  416.      pop    orig_ax
  417.      pop    orig_bx
  418.      pop    orig_cx
  419.      pop    orig_dx
  420.      pop    orig_si
  421.      pop    di
  422.      pop    orig_ds
  423.      pop    orig_es
  424.      inc    sp            ; Discard SS value from structure.
  425.      inc    sp
  426.      pop    caller_cs
  427.      pop    caller_flags
  428.      pop    orig_bp
  429.      pop    caller_sp
  430.      pop    caller_ip
  431.  
  432. ;   Switch back to caller's stack.
  433.  
  434.      mov    cx,ds
  435.      cli
  436.      mov    ss,cx
  437.      assume ss:nothing
  438.      mov    sp,ax
  439.                     ; Now both DS and SS refer to
  440.                     ;  caller's stack segment.
  441.  
  442.                     ; Leave interrupts disabled.
  443.  
  444. ;   Discard the ISR stack created above, i.e. reset icb_level and icb_isrsp.
  445. ;
  446. ;   If icb_level exceeds our local (stacked) copy of level, then
  447. ;     this ISR invocation returned BEFORE a deeper-nested invocation returned.
  448. ;     In this case, assume that deeper invocations will never return.
  449. ;     Adjust icb_level and icb_isrsp to discard deeper stacks.
  450. ;
  451. ;   Interrupts are NOT okay during this operation.
  452.  
  453.      mov    cx,es:[si].icb_level
  454.      pop    ax            ; Local copy of nesting depth
  455.      dec    ax
  456.      mov    es:[si].icb_level,ax
  457.      sub    cx,ax            ; CX = number of stacks to discard
  458.                     ;       (normally 1)
  459.  
  460.      jbe    stacks_discarded
  461.      mov    ax,es:[si].icb_isrstksize
  462. discard_stack:
  463.      sub    es:[si].icb_isrsp,ax    ; Decrement beginning_SP once
  464.      loop    discard_stack        ;   for each stack discarded.
  465. stacks_discarded:
  466.                     ; Interrupts now okay.
  467.  
  468. ;   Consider optional exit style
  469.  
  470.      cmp    dx,IEXIT_RETF
  471.      je    retf_exit
  472.  
  473.                     ; If exit style code not recognized,
  474.                     ;    fall through to normal exit.
  475.  
  476. ;   Normal exit:  restore everything and do IRET.
  477. ;
  478. ;   Some of the values we restore were copied from the ALLREG structure,
  479. ;     so they may have been modified by the ISR.
  480.  
  481. exit:
  482.      pop    si
  483.      pop    es
  484.  
  485.      popff                ; Install "original" flags.
  486.                     ; These may have been modified via
  487.                     ;    the "WORKING_FLAGS" item in the
  488.                     ;    message area.
  489.  
  490.      pop    ds
  491.      pop    dx
  492.      pop    cx
  493.      pop    bx
  494.      pop    ax
  495.      pop    bp
  496.      pop    sp            ; Unless modified by ISR,
  497.                     ;    SP now contains the value
  498.                     ;    it had after the caller
  499.                     ;    performed an INT.
  500.      iret
  501.  
  502. ;   Special exit:  restore everything and do a RETF.
  503. ;
  504. ;   Some of the values we restore were copied from the ALLREG structure,
  505. ;     so they may have been modified by the ISR.
  506.  
  507. retf_exit:
  508.      pop    si
  509.      pop    es
  510.  
  511.      popff                ; Install "original" flags.
  512.                     ; These may have been modified via
  513.                     ;    the "working flags" item in the
  514.                     ;    message area.
  515.  
  516.      pop    ds
  517.      pop    dx
  518.      pop    cx
  519.      pop    bx
  520.      pop    ax
  521.      pop    bp
  522.      pop    sp            ; Unless modified by ISR,
  523.                     ;    SP now contains the value
  524.                     ;    it had after the caller
  525.                     ;    performed an INT.
  526.      ret
  527.  
  528.      if MSC300
  529. _isdispat endp
  530.      else
  531. isdispat endp
  532.      endif
  533.  
  534.      if LAT200 or LAT210 or LAT300
  535.      endps
  536.      endif
  537.  
  538.      if MSC300
  539.      endps      isdispat
  540.      endif
  541.  
  542.      end
  543.