home *** CD-ROM | disk | FTP | other *** search
/ Liren Large Software Subsidy 9 / 09.iso / l / l170 / 3.ddi / INTRPT.ASM < prev    next >
Encoding:
Assembly Source File  |  1991-01-29  |  14.5 KB  |  359 lines

  1.        TITLE   INTERRUPT - BASCOM software interrupt calling routine
  2.         PAGE    56,132
  3. ;***
  4. ; INTERRUPT - BASCOM software interrupt calling routine
  5. ;
  6. ;       Copyright <C> 1986, 1987 Microsoft Corporation
  7. ;
  8. ;Purpose:
  9. ;   Allows a BASIC program to invoke an interrupt through a CALL statement.
  10. ;
  11. ;   INTERRUPT allows BASIC to set AX,BX,CX,DX,BP,SI,DI, and the flags
  12. ;   before the call.  INTERRUPTX also allows DS and ES to be set.
  13. ;   Both routines will return the values of the registers upon the
  14. ;   completion of a successful call.  If the interrupt could not
  15. ;   be generated (due to a bad interrupt number or an illegal array)
  16. ;   then the interrupt number will be set to -1 to indicate an error.
  17. ;
  18. ;******************************************************************************
  19. ;
  20. ;Note:
  21. ;   The DOSSEG, .MODEL, .CODE, and .DATA? directives used in this program
  22. ;   are part of the simplified segment system of MASM 5.0. If you have
  23. ;   an earlier version of MASM, you must modify the source to define
  24. ;   the segments required by Microsoft high-level languages. These
  25. ;   segments are discussed in Appendix C of "Learning and Using QuickBASIC."
  26. ;
  27.  
  28.         DOSSEG                  ;requires MASM 5.0 or higher
  29.         .MODEL  medium
  30.  
  31. ;       Define all publicly accessible routines.
  32.  
  33.         PUBLIC  INTERRUPT, INTERRUPTX
  34.  
  35. ;       Frame structure definition
  36.  
  37. ARG1    =       0AH             ;pointer to first of three arguments
  38. ARG2    =       08H             ;pointer to second of three arguments
  39. ARG3    =       06H             ;pointer to third of three arguments
  40.  
  41. ;       Frame temp variables
  42.  
  43. UCODE_FLGS =    -02H            ;user code flag register value
  44. UCODE_DS =      -04H            ;user code DS register value
  45. REG_NUM =       -06H            ;number of regs used (INTERRUPT=8, INTERRUPTX=10)
  46. INT_ES  =       -08H            ;INT ES register value
  47. INT_DS  =       -0AH            ;INT DS register value
  48. INT_FLGS =      -0CH            ;INT flags register value
  49. INT_DI  =       -1EH            ;INT DI register value
  50. INT_SI  =       -10H            ;INT SI register value
  51. INT_BP  =       -12H            ;INT BP register value
  52. INT_DX  =       -14H            ;INT DX register value
  53. INT_CX  =       -16H            ;INT CX register value
  54. INT_BX  =       -18H            ;INT BX register value
  55. INT_AX  =       -1AH            ;INT AX register value
  56. OLD_SI  =       -1CH            ;save old SI for interpreter
  57. OLD_DI  =       -1EH            ;save old DI for interpreter
  58.  
  59. FRM_SIZ =       -1EH            ;negative size of frame temporaries
  60.  
  61. ;       Locations past frame allocation used to recover post-INT BP value.
  62.  
  63. INT_BP_TMP =    -22H            ;temp location for INT BP register value
  64.  
  65. ;***
  66. ; INTERRUPT, and INTERRUPTX - BASCOM software interrupt calling interface
  67. ;
  68. ; Purpose:
  69. ;       To allow a BASIC Compiler program to perform any software
  70. ;       interrupt.  The interrupt is executed with the registers
  71. ;       set to values specified in a register variable.  The post-
  72. ;       interrupt values of the registers are then stored in
  73. ;       another register variable.
  74. ;
  75. ;       CALL INTERRUPT[X] (int_no AS INTEGER,
  76. ;                          inreg AS RegType[X],
  77. ;                          outreg AS RegType[X])
  78. ;
  79. ; Inputs:
  80. ;       int_no = interrupt number (range 0 to 255) to execute
  81. ;       inreg and outreg are register variables of type RegType[X]
  82. ;       defined as follows;
  83. ;
  84. ; TYPE RegType
  85. ;     ax    AS INTEGER
  86. ;     bx    AS INTEGER
  87. ;     cx    AS INTEGER
  88. ;     dx    AS INTEGER
  89. ;     bp    AS INTEGER
  90. ;     si    AS INTEGER
  91. ;     di    AS INTEGER
  92. ;     flags AS INTEGER
  93. ; END TYPE
  94. ;
  95. ;
  96. ; TYPE RegTypeX
  97. ;     ax    AS INTEGER
  98. ;     bx    AS INTEGER
  99. ;     cx    AS INTEGER
  100. ;     dx    AS INTEGER
  101. ;     bp    AS INTEGER
  102. ;     si    AS INTEGER
  103. ;     di    AS INTEGER
  104. ;     flags AS INTEGER
  105. ;     ds    AS INTEGER
  106. ;     es    AS INTEGER
  107. ; END TYPE
  108. ;
  109. ; Outputs:
  110. ;       If no error:
  111. ;               int_no = unchanged (range 0 to 255)
  112. ;               outreg: This array will be set to the post-interrupt
  113. ;                       register values.  It has the same structure
  114. ;                       as inreg.
  115. ;       If error:
  116. ;               int_no = -1
  117. ;               outreg unchanged.  INT call is not performed.
  118. ;               error occurs:
  119. ;                       first argument not 0 to 255 (2^8-1)
  120. ;                       second or third arguments not  0 to 1048575 (2^20-1)
  121. ;                               (VARPTR will always be in this range)
  122. ;
  123. ; Modifies:
  124. ;       All, except BP, DS, and flags.
  125. ;       Also, possible side effects of INT call.
  126. ;
  127. ; Exceptions:
  128. ;       INT 24H call may result from some INT 21H MS-DOS calls.
  129. ;
  130. ;******************************************************************************
  131.  
  132.         .CODE
  133.  
  134. INTERRUPT PROC  FAR
  135.  
  136.         PUSH    BP              ;save BASCOM frame pointer on stack
  137.         MOV     BP,SP           ;establish program frame reference
  138.         ADD     SP,FRM_SIZ      ;allocate working space for frame
  139.         MOV     WORD PTR [BP].REG_NUM,08H ;eight regs used (not DS or ES)
  140.         JMP     SHORT INTERRUPT_COMMON ;jump to common code
  141.  
  142. INTERRUPT ENDP
  143.  
  144.  
  145. INTERRUPTX PROC FAR
  146.  
  147.         PUSH    BP              ;save BASCOM frame pointer on stack
  148.         MOV     BP,SP           ;establish program frame reference
  149.         ADD     SP,FRM_SIZ      ;allocate working space for frame
  150.         MOV     WORD PTR [BP].REG_NUM,0AH ;ten regs used (including DS and ES)
  151.  
  152. ;       Save a copy of the processor flags, SI, DI, and DS in the stack frame.
  153.  
  154. INTERRUPT_COMMON:
  155.         MOV     [BP].OLD_SI,SI  ;save old SI for interpreter
  156.         MOV     [BP].OLD_DI,DI  ;save old DI for interpreter
  157.         MOV     [BP].UCODE_DS,DS;save DS for interpreter
  158.         PUSHF                   ;push the flags on the stack
  159.         POP     [BP].UCODE_FLGS ;put value in the stack frame
  160.  
  161. ;       Move eight or ten words (depending if executing INTERRUPT or INTERRUPTX)
  162. ;       of the integer input array from the far pointer computed to the frame.
  163.  
  164.         MOV     SI,[BP].ARG2    ;and array offset - pointer in DS:SI
  165.         LEA     DI,[BP].INT_AX  ;get start of temporary register storage.
  166.         MOV     CX,[BP].REG_NUM ;eight or ten words to move
  167.         CLD                     ;movement is to higher memory
  168.         PUSH    SS
  169.         POP     ES
  170.         REP     MOVSW           ;move the array into the stack frame
  171.  
  172. ;       Save stack frame pointer to recover its value after the INT call.
  173.  
  174.         PUSH    BP              ;saved to first word past the stack frame
  175.  
  176. ;       Create a two-instruction program on the stack to execute the
  177. ;       INT call requested and return with stack cleanup.
  178. ;
  179. ;       INT     XX      (hex: CD XX)
  180. ;       RETF    06      (hex: CA 06 00)
  181. ;
  182. ;       In the case of INT 25 and 26 (which leave a word of flags on the stack)
  183. ;       We generate:
  184. ;
  185. ;       INT     XX      (hex: CD XX)
  186. ;       ADD     SP,2    (hex: 83 C4 02)
  187. ;       RETF    08      (hex: CA 08 00)
  188. ;
  189.         MOV     SI,[BP].ARG1    ;[SI] = ptr to first CALL arg - interrupt #
  190.         MOV     BX,[SI]         ;[BL] = get integer value of INT type
  191.         OR      BH,BH           ;test if in range, 00 to FFH is legal
  192.         JZ      NO_INT_ERROR    ;if not, then error - jump
  193.         JMP     INT_ERROR       ;long jump to error routine
  194. NO_INT_ERROR:
  195.  
  196.         CMP     BL,25H          ;Interrupt 25 request?
  197.         JZ      Int2526         ;Jump if so
  198.         CMP     BL,26H          ;Interrupt 26 request?
  199.         JNZ     IntNorm         ;Jump if other, "normal" interrupt
  200. Int2526:
  201.         MOV     AX,8            ;[AX] = argument of RETF instruction
  202.         PUSH    AX
  203.         MOV     AX,0CA02H       ;[AX] = RETF opcode, & arg to ADD SP
  204.         PUSH    AX
  205.         MOV     AX,0C483H       ;[AX] = ADD SP, opcode
  206.         PUSH    AX
  207.         JMP     SHORT IntInstruct
  208.  
  209. IntNorm:
  210.         XOR     AX,AX           ;value of second word past frame
  211.         PUSH    AX              ;put on stack - 00 byte of RETF and filler
  212.         MOV     AX,06CAH        ;value of third word past frame
  213.         PUSH    AX              ;put on stack - CA 06 bytes of RETF
  214. IntInstruct:
  215.         MOV     AH,BL           ;move interrupt number to upper byte of AX
  216.         MOV     AL,0CDH         ;value of fourth word past frame
  217.         PUSH    AX              ;put on stack - CD XX bytes of INT XX
  218.  
  219. ;       Push far pointer of return address after the stack program
  220. ;       executes, which is INT_RET in this code segment.
  221.  
  222.         PUSH    CS              ;push current code segment for return segment
  223.         MOV     AX,OFFSET INT_RET ;offset just after stack program call
  224.         PUSH    AX              ;push value for return offset
  225.  
  226. ;       Push far pointer pointer to the start of the stack program.
  227. ;       The stack program will be entered by executing a RETF after the
  228. ;       registers are set up.
  229.  
  230.         PUSH    SS              ;push current stack segment for starting ptr
  231.         MOV     AX,SP           ;get current stack offset
  232.         ADD     AX,6            ;move past the last three stack entries
  233.         PUSH    AX              ;push offset for starting ptr of stack program
  234.  
  235. ;       Move the input array values from the stack to their actual registers.
  236.  
  237.         MOV     AX,[BP].INT_FLGS ;get input flag register value
  238.         AND     AX,0000111111010101B ;mask out undefined 8086 flags
  239.         PUSH    AX              ;push masked flag register value
  240.  
  241.         MOV     AX,[BP].INT_AX  ;set up input AX value
  242.         MOV     BX,[BP].INT_BX  ;set up input BX value
  243.         MOV     CX,[BP].INT_CX  ;set up input CX value
  244.         MOV     DX,[BP].INT_DX  ;set up input DX value
  245.  
  246.         MOV     SI,[BP].INT_SI  ;set up input SI value
  247.         MOV     DI,[BP].INT_DI  ;set up input DI value
  248.  
  249. ;       For DS and ES, leave in the compiler data segment values if:
  250. ;       executing INTERRUPT; or executing INTERRUPTX with array values of -1.
  251.  
  252.         CMP     WORD PTR [BP].REG_NUM,08H ;test if executing INTERRUPT
  253.         JE      INT_ES_DEF      ;if so, then use both default values
  254.  
  255.         CMP     [BP].INT_DS,0FFFFH ;test if default DS to be used
  256.         JE      INT_DS_DEF      ;if so, then leave it unchanged
  257.         MOV     DS,[BP].INT_DS  ;set up input DS value
  258. INT_DS_DEF:
  259.         CMP     [BP].INT_ES,0FFFFH ;test if default ES to be used
  260.         JE      INT_ES_DEF      ;if so, then leave it unchanged
  261.         MOV     ES,[BP].INT_ES  ;set up input ES value
  262. INT_ES_DEF:
  263.  
  264.         MOV     BP,[BP].INT_BP  ;set up input BP value
  265.                                 ;must be last move using BP
  266.  
  267.         POPF                    ;set up input flag register value
  268.  
  269. ;       With all registers set according to the input array, execute the
  270. ;       stack program.
  271. ;
  272. ;       The following RETF pops the last two stack entries, which are
  273. ;       interpreted as a far pointer to the stack program.
  274. ;
  275. ;       The stack program executes the INT XX call which changes the
  276. ;       registers (flags included) to the values to be put into the
  277. ;       output array.
  278. ;
  279. ;       The stack program then executes the RETF 06 instruction which
  280. ;       does two operations.  First, the next two entries on stack are
  281. ;       popped and interpreted as a far ptr return address, which points
  282. ;       the code at INT_RET in this code segment.  Second, the stack
  283. ;       pointer is then adjusted by six bytes to remove the six-byte
  284. ;       program from the stack.
  285.  
  286.         RET                     ;far return to execute stack program, etc.
  287. INT_RET:
  288.  
  289. ;       The stack should now contain only the first entry past the
  290. ;       frame, the value of the stack frame pointer itself.  First
  291. ;       save the BP value from the INT call, then get the old value
  292. ;       to reference the frame.
  293.  
  294.         PUSH    BP              ;save post-INT value of BP
  295.         MOV     BP,SP           ;temporary frame is second word past frame
  296.         MOV     BP,[BP+02H]     ;get real frame reference value
  297.  
  298. ;       Put post-INT value of all registers into the frame variables
  299. ;       to be subsequently written into the output array.
  300.  
  301.         PUSHF                   ;put flags on the stack
  302.         POP     [BP].INT_FLGS   ;put in post-INT flag register value
  303.  
  304.         PUSH    [BP].UCODE_FLGS ;get old copy of flags from frame
  305.         POPF                    ;and restore the old flag values
  306.  
  307.         MOV     [BP].INT_AX,AX  ;put in post-INT AX value
  308.         MOV     [BP].INT_BX,BX  ;put in post-INT BX value
  309.         MOV     [BP].INT_CX,CX  ;put in post-INT CX value
  310.         MOV     [BP].INT_DX,DX  ;put in post-INT DX value
  311.  
  312.         MOV     AX,[BP].INT_BP_TMP ;get post-INT BP value (one entry past frame)
  313.         MOV     [BP].INT_BP,AX  ;put in post-INT BP value
  314.  
  315.         MOV     [BP].INT_SI,SI  ;put in post-INT SI value
  316.         MOV     [BP].INT_DI,DI  ;put in post-INT DI value
  317.  
  318.         MOV     [BP].INT_DS,DS  ;put in post-INT DS value
  319.         MOV     [BP].INT_ES,ES  ;put in post-INT ES value
  320.  
  321. ;       Move frame register values to the output array whose
  322. ;       far pointer is in the frame.
  323.  
  324.         MOV     DS,[BP].UCODE_DS;replace original DS value
  325.  
  326.         LEA     SI,[BP].INT_AX  ;get start of register area in frame
  327.  
  328.         PUSH    DS
  329.         POP     ES
  330.         MOV     DI,[BP].ARG3    ;get output array offset
  331.         MOV     CX,[BP].REG_NUM ;eight or ten words to move
  332.         CLD                     ;movement is toward upper memory
  333.         REP     MOVSW           ;perform the transfer
  334.  
  335. ;       Clean up stack to remove frame.  Remove CALL arguments with RETF.
  336.  
  337.         MOV     SI,[BP].OLD_SI  ;replace old SI for interpreter
  338.         MOV     DI,[BP].OLD_DI  ;replace old DI for interpreter
  339.         MOV     SP,BP           ;deallocate temporary frame variables
  340.         POP     BP              ;return compiler frame pointer
  341.         RET     06              ;remove three CALL arguments and far return
  342.  
  343. ;       If error, then restore DS, set int_no to -1 to report error,
  344. ;       clean up, and exit.
  345.  
  346. INT_ERROR:
  347.         MOV     SI,[BP].ARG1    ;ptr to first CALL arg - interrupt number
  348.         MOV     [SI],0FFFFH     ;set interrupt number to -1 for error
  349.         MOV     SI,[BP].OLD_SI  ;replace old SI for interpreter
  350.         MOV     DI,[BP].OLD_DI  ;replace old DI for interpreter
  351.         MOV     DS,[BP].UCODE_DS;replace original DS value
  352.         MOV     SP,BP           ;deallocate temporary frame variables
  353.         POP     BP              ;return compiler frame pointer
  354.         RET     06              ;remove three CALL arguments and far return
  355.  
  356. INTERRUPTX ENDP
  357.  
  358.         END
  359.