home *** CD-ROM | disk | FTP | other *** search
/ Liren Large Software Subsidy 7 / 07.iso / c / c005 / 5.ddi / DOS.ASM < prev    next >
Encoding:
Assembly Source File  |  1986-08-05  |  6.8 KB  |  258 lines

  1. ;
  2. ; Name        dos -- Access to DOS functions
  3. ;
  4. ; Synopsis    ercode = dos(preg);
  5. ;
  6. ;        int ercode      Error code returned by DOS function
  7. ;                  calls which set the carry flag in case
  8. ;                  of error.
  9. ;        DOSREG *preg      Standard registers
  10. ;
  11. ; Description    dos() is the gate to the DOS functions.  All registers
  12. ;        may be set.  All register values are returned in the
  13. ;        structure pointed to by preg except for the DS register
  14. ;        (whose value remains unchanged).  Any actual machine
  15. ;        registers may be changed by the function except for SS,
  16. ;        SP, DS, ES, and any other registers which the C
  17. ;        environment requires to be preserved.
  18. ;
  19. ; Limitation    This function takes special steps to avoid a bug in DOS
  20. ;        function 0x4b (Load or Execute a Program).  These steps
  21. ;        involve preserving the state of the stack in the local
  22. ;        variables savess and savesp.  Therefore this function is
  23. ;        NOT RE-ENTRANT in the following limited sense:    If
  24. ;        function 0x4b is in progress, then the same copy of
  25. ;        dos() should not be used to perform an additional 0x4b
  26. ;        call because that would corrupt the original values
  27. ;        stored in savess and savesp.
  28. ;
  29. ;        This limitation does not apply to other DOS functions.
  30. ;
  31. ; Returns    ercode          Error code returned by DOS function
  32. ;                  calls which set the carry flag in case
  33. ;                  of error.
  34. ;        *preg          The values of the registers may be altered
  35. ;                  by some of the DOS function calls.
  36. ;
  37. ; Version    3.0 (C)Copyright Blaise Computing Inc. 1983,1984,1985,1986
  38. ;
  39.  
  40. popff     macro                ;; Simulate POPF instruction
  41.      local    do_call,do_iret
  42.      jmp    short do_call
  43. do_iret: iret                ;; Pop IP, CS, flags.
  44. do_call: push    cs            ;; Push CS
  45.      call    do_iret         ;; Push IP & jump.
  46.      endm
  47.  
  48.      name       dosgate
  49.  
  50.      LONGPROG  = 0               ; initialize constants for
  51.      LONGDATA  = 0               ; Pass1 of the assembler
  52.  
  53.      include   compiler.mac        ; Specifies the C compiler
  54.  
  55.      if LAT200 or LAT210 or LAT300
  56.      include  dos.mac
  57.      LONGPROG = LPROG
  58.      LONGDATA = LDATA
  59.  
  60.      pseg
  61.      public   dos
  62.      if      LPROG
  63.      x   equ  6             ; parameter offset
  64. dos     proc      far
  65.      else
  66.      x   equ  4
  67. dos     proc      near
  68.      endif
  69.      endif
  70.  
  71.      if MSC300
  72.      include  dos.mac
  73.      LONGPROG = LPROG
  74.      LONGDATA = LDATA
  75.  
  76.      pseg      dos
  77.      public   _dos
  78.      if      LPROG
  79.      x   equ  6             ; parameter offset
  80. _dos     proc      far
  81.      else
  82.      x   equ  4
  83. _dos     proc      near
  84.      endif
  85.      endif
  86.  
  87.                        ; Begin the code.
  88.  
  89.      jmp       short begin_dos
  90.  
  91. savess     dw       0               ; Storage for stack segment
  92. savesp     dw       0               ; Storage for stack pointer
  93.  
  94. begin_dos:
  95.      push       bp
  96.      mov       bp,sp
  97.      if       MSC300
  98.      push       di               ; Save register variables
  99.      push       si
  100.      endif
  101.      push       es               ; Save data segment DS and
  102.      push       ds               ; ES
  103.  
  104.      if       LONGDATA
  105.      les       si,dword ptr [bp + x]  ; Retrieve the parameter
  106.      mov       ax,es:[si]          ; address information.
  107.      mov       bx,es:[si+2]
  108.      mov       cx,es:[si+4]
  109.      mov       dx,es:[si+6]
  110.      mov       di,es:[si+10]
  111.      push       di
  112.      mov       di,si
  113.      mov       si,es:[di+8]
  114.      mov       ds,es:[di+12]
  115.      mov       es,es:[di+14]
  116.      pop       di
  117.      else
  118.      mov       si,[bp+x]
  119.      mov       ax,[si]
  120.      mov       bx,[si+2]
  121.      mov       cx,[si+4]
  122.      mov       dx,[si+6]
  123.      mov       di,[si+10]
  124.      push       di
  125.      mov       di,si
  126.      mov       si,[di+8]
  127.      mov       es,[di+14]
  128.      mov       ds,[di+12]
  129.      pop       di
  130.      endif
  131.      cmp       ah,4bh           ; Check for the EXEC call
  132.      jne       start           ; Different call so begin
  133.  
  134. ;  The EXEC function call under DOS 2.00 can trash part of the data
  135. ;  segment.  The purpose of the following code is to change the
  136. ;  stack segment to one paragraph "above" the current stack.  Then
  137. ;  a 256 byte stack is allocated (which is sufficient for DOS).
  138. ;  Giving DOS its own stack avoids the problem.  It does mean that
  139. ;  the calling program must have sufficient stack space; worst case
  140. ;  begin 256 + 16 bytes.  First we check if 2.00 is the current version;
  141. ;  the bug has been fixed in DOS 2.10 and greater.
  142.  
  143.      mov       cs:savess,ss        ; Save location of stack
  144.      mov       cs:savesp,sp
  145.      push       ax               ; EXEC needs AX and BX so we
  146.      push       bx               ; save them.
  147.      mov       ah,30h
  148.      int       21h
  149.      cmp       al,2            ; Problem only if version 2.00
  150.      jne       stack_ok           ; It is not 2.x, so continue
  151.      cmp       ah,10           ; Is it version 2.10 or greater?
  152.      jae       stack_ok           ; It is, so continue
  153.  
  154.                        ; Now we know it's version 2.00.
  155.      pop       bx               ; Restore BX and AX
  156.      pop       ax
  157.  
  158.      cli                   ; No interrupts while constructing
  159.      mov       cl,4            ; new stack segment
  160.      shr       bp,cl           ; Turn into segment address, and
  161.      sub       bp,17           ; one more for safety
  162.      mov       di,ss
  163.      add       di,bp
  164.      mov       ss,di           ; New stack segment,
  165.      mov       sp,256           ; and allow for 256 byte stack
  166.      jmp       short do_4bh
  167.  
  168. stack_ok:
  169.      pop       bx               ; Restore BX and AX
  170.      pop       ax
  171.  
  172. do_4bh:
  173.      clc                   ; Clear carry flag for error ret
  174.      cld                   ; Some DOS functions allegedly
  175.                        ; require that DF == 0.
  176.      int       21h
  177.      cli
  178.      mov       ss,cs:savess        ; Restore normal stack.
  179.      mov       sp,cs:savesp
  180.      sti
  181.  
  182.      jmp       short finish
  183.  
  184. start:
  185.      clc                   ; Clear carry flag for error ret
  186.      cld                   ; Some DOS functions allegedly
  187.                        ; require that DF == 0.
  188.      int       21h               ; Invoke DOS function (AH)
  189.  
  190. finish:                    ; Common exit code:
  191.  
  192.      pop       ds               ; Recover the data segment, DS
  193.      pushf                   ; Save the carry flag
  194.      mov       bp,sp
  195.      if       MSC300
  196.      add       bp,8            ; ES, DI, SI, and flags are on stack
  197.      else
  198.      add       bp,4            ; ES and flags are on the stack
  199.      endif
  200.      push       si
  201.  
  202.      if       LONGDATA           ; Recover the register information
  203.      push       ax               ; Used to store ES
  204.      push       es
  205.      les       si,dword ptr [bp + x]
  206.      mov       es:[si],ax
  207.      mov       es:[si+2],bx
  208.      mov       es:[si+4],cx
  209.      mov       es:[si+6],dx
  210.      mov       es:[si+10],di
  211.      pop       ax
  212.      mov       es:[si+14],ax       ; Return the value of ES
  213.      pop       ax               ; Recover value of AX
  214.      pop       si
  215.      les       di,dword ptr [bp + x]
  216.      mov       es:[di+8],si
  217.      else
  218.      mov       si,[bp+x]
  219.      mov       [si],ax
  220.      mov       [si+2],bx
  221.      mov       [si+4],cx
  222.      mov       [si+6],dx
  223.      mov       [si+10],di
  224.      mov       [si+14],es
  225.      pop       si               ; Restore SI and use DI to address
  226.      mov       di,[bp+x]           ; the return structure
  227.      mov       [di+8],si
  228.      endif
  229.  
  230.      popff                   ; Restore the flags and check for
  231.      jc       error           ; a return error code.
  232.      mov       ax,0            ; Returned function value is 0
  233. error:                       ; otherwise value of AX register
  234.      pop       es               ; Recover ES
  235.      if       MSC300
  236.      pop       si               ; Recover index registers
  237.      pop       di               ; (used for register variables)
  238.      cld                   ; MSC 3 expects DF cleared
  239.      endif
  240.  
  241.      pop       bp
  242.      ret
  243.      if MSC300
  244. _dos     endp
  245.      else
  246. dos     endp
  247.      endif
  248.  
  249.      if LAT200 or LAT210 or LAT300
  250.      endps
  251.      endif
  252.  
  253.      if MSC300
  254.      endps      dos
  255.      endif
  256.  
  257.      end
  258.