home *** CD-ROM | disk | FTP | other *** search
/ Liren Large Software Subsidy 5 / 05.iso / a / a003 / 1.ddi / OVUSERM.ASM < prev    next >
Encoding:
Assembly Source File  |  1986-09-29  |  27.2 KB  |  732 lines

  1. ;*
  2. ;**************************************************************************
  3. ;*                                                                        *
  4. ;*          OVUSER.ASM: Overlay Loader User Module for PLINK86plus        *
  5. ;*        Copyright (C) 1985, 1986 by Phoenix Software Associates Ltd.    *
  6. ;*                                                                        *
  7. ;**************************************************************************
  8. ;*
  9. ;*
  10.      title     Runtime Overlay Loader User Module for PLINK86
  11.      subttl    Phoenix Software Associates, Ltd.
  12.      .sfcond
  13. ;
  14. ;
  15. ;    Users may modify this routine and link it into their programs to 
  16. ;customize the way the PLINK86 overlay loader operates.  All messages 
  17. ;generated at runtime, as well as any communications carried out with the 
  18. ;operator, are handled here.  This module also controls the process by which
  19. ;the linkage editor searches for the overlays.  Modifications should be made
  20. ;with care:  Phoenix will probably be unsympathetic to those who make 
  21. ;changes and then expect us to help with the debugging.  However, virtually
  22. ;any modifications needed to make your application run more smoothly may be 
  23. ;accomplished by making minor changes to this code.  Also, see the file
  24. ;"SAMPLE.ASM" for a much simpler version of this module.
  25. ;
  26. ;    As the program begins execution the $OVLYIU routine is called before
  27. ;the application code executes.  Initialization is performed here.  The DS
  28. ;register is set to enable local data to be accessed, but the ES register
  29. ;points to the MSDOS program segment prefix.  The standard user module uses
  30. ;this to obtain the address of the environment (specifically the PATH 
  31. ;string).  These strings are directory names used by the system to find 
  32. ;executable files, and they are used in the same way here to find overlay 
  33. ;files.
  34. ;
  35. ;    The overlay loader calls symbol $OVLYxU each time it is about to open a
  36. ;new overlay file (keep in mind that a file can contain several overlays), 
  37. ;or whenever a fatal error is encountered while reading an overlay.  The 
  38. ;name of the overlay file being requested is provided in a string area 
  39. ;labled by public symbol $OVLYFN, and may be modified if desired.  The file 
  40. ;name must be terminated by a NUL, and the string area must be at least 13 
  41. ;bytes long. As initialized by the overlay loader, the file name consists of
  42. ;a name and type only, of maximum length 8 and 3 chars respectively, 
  43. ;separated by a period.
  44. ;
  45. ;    A result code is provided in the AL register describing what happened
  46. ;during the previous attempt to load an current overlay from the current 
  47. ;file:
  48. ;
  49. ;   0 - The first attempt to load an overlay has not been made yet.
  50. ;   1 - The overlay file was not found.
  51. ;   2 - The overlay file couldn't be opened due to a disk error.  These 
  52. ;       errors are things like no disk in drive, drive not ready, or media 
  53. ;       errors such as seek and CRC problems.
  54. ;   3 - The overlay couldn't be read completely (premature EOF).  The file
  55. ;       is probably smashed.
  56. ;   4 - The overlay couldn't be read because of a disk error (drive not 
  57. ;       ready, media error, etc).
  58. ;   5 - An "invalid handle" error was returned when the file was accessed.
  59. ;       Possible the file was closed because another task was terminated 
  60. ;       (due to the strange way MSDOS 2 tries to support multi-tasking).
  61. ;
  62. ;    After possibly making modifications to the overlay file name, the 
  63. ;$OVLYxU routine must return, in the AL register, a function code 
  64. ;instructing the overlay loader how to handle the next attempt to load the 
  65. ;overlay:
  66. ;
  67. ;   0 - Try to load the overlay from the file name given in $OVLYFN.
  68. ;   1 - Give up.  The program is normally terminated when this code is 
  69. ;       received.  However, if the user program called the overlay loader 
  70. ;       itself (at the LOAD$ routine, see PLINK86 user manual), an error 
  71. ;       code may be returned to the user program at its option.
  72. ;
  73. ;Standard Search Sequence for Overlay Load Request:
  74. ;   a) try the unmodified file name
  75. ;   b) if running under MSDOS 3.0 or above, try prepending the directory
  76. ;           portion of the original DOS command (part of the environment)
  77. ;   c) try prepending each string from the MSDOS 2.0 PATH variable, if any
  78. ;   d) prepend a default string to the file name (our default is "A:")
  79. ;   e) prompt the operator for a string to prepend to the file name
  80. ;           at this point, the operator may terminate the program (standard
  81. ;           mechanism is to input a '.').  code 1 is returned to the
  82. ;           overlay loader.
  83. ;
  84.      name OVUSERM
  85. ;
  86. ;
  87. ;
  88. ;   The following symbols may be used to customize versions of OVUSER.
  89.     ;
  90.     ;
  91. MULTICOPY    equ 0    ;    set to true for shared overlays
  92. VIANET        equ 0    ;    set to true to include Vianet support
  93. MYDEBUG        equ 0    ;    set to true to debug this module
  94. ;
  95. ;
  96. ; The following public symbols are declared here in order to force the
  97. ;   assembler to order the segments as OVcode, OVdata.  MASM 3.0+ outputs
  98. ;   the segments in the order they are defined, whereas previous versions
  99. ;   output the segments in alphabetical order.
  100. ;
  101. ;
  102. OVcode  segment  para  public   'OVLOADER'
  103.      public    $OVLYIU            ;entry point to initialize user module
  104.      public    $OVLYFN            ;overlay file name string
  105.      public    $OVEXIT            ;overlay manager exit routine
  106.      public    $OVLEND            ;subroutine invoked at the end of a 
  107.                                   ;   each overlay load request
  108.      if MYDEBUG
  109.          public   $$get_prefix
  110.          public   $$is_separator
  111.          public   $$prepend
  112.          public   $$error_msg
  113.      endif
  114.          extrn     $OVCLOSE: far  ;close open overlay file, if any
  115.          public    $OVLYMU        ;entry point for MSDOS user routine
  116.                                   
  117.                                   ;fatal messages
  118.          public    $needdos2      ;   need dos 2.0 i/o capability
  119.          public   $roverflow  ;   overlay loader reload stack overflow
  120.          public   $runderflow ;   overlay loader reload stack undeflow
  121.          if MYDEBUG
  122.              public   $$try_path_strings
  123.              public   $$try_exe_path
  124.          endif
  125. OVcode  ends
  126. ;
  127. ;
  128. ;********
  129. ;* Data *
  130. ;********
  131. ;
  132. CR              equ  0Dh
  133. LF              equ  0Ah
  134. STD_OPN         equ  20h     ;NETBIOS standard shared open mode
  135. STD_SEP         equ  '\'     ;char to separate Msdos 2.0 directory names
  136. ABORT_SEARCH    equ  '.'     ;operator request to abort search for an ovly
  137. ;
  138. ;
  139. OVdata    segment   word public 'OVLOADER'
  140.     extrn  $OVRERR$: byte    ;1 => return error to user
  141.     extrn  $OVOM$: byte      ;open mode for overlay files - 
  142.                              ;    network may be running => shared files
  143.     extrn  $OVDOS2: byte     ;-1 => MSDOS >= 2.0 running
  144.  
  145.     DosVer    db   0    ;MSDOS version number.
  146.     EnvPar    dw   0    ;paragraph address of environment
  147.     EnvOff    dw   0    ;offset to PATH string
  148.     TryEnv    dw   0    ;current offset to PATH string
  149.     EPN       dw   0    ;Execution Path Name (MSDOS version >= 3 only).
  150.     Path db   "PATH"    ;environment name for executable files path string
  151.     Psize =   4         ;length of Path
  152.     ExeTyp db ".EXE"    ;Executable file type name
  153.     ExeLen    dw   0    ;Length of path portion of executable name (DOS 3).
  154.     MustAsk   db   0    ;1=>must ask operator for overlay location.
  155.     TriedExe db    0    ;0=>haven't tried EPN to find overlay file (DOS 3).
  156.     Pflag     db   0    ;1=>DspFN should display file name prefix w/ file name
  157.     FNlen     equ 80
  158.     $OVLYFN   db FNlen dup (?) ;overlay file name work area
  159.     KeyBuf    dw   66   ;Buffer size for buffered keyboard input O.S. call.
  160.     Prefix    db   "A:" ;default prefix for file names
  161.               db   64 dup (?)
  162.     is_net_active  db  0  ;true => NETBIOS is active: close overlay files
  163.                           ;        if configured for multiple copy support
  164.     ;
  165.     ;
  166.     ;    All messages are here.
  167.     ;
  168.         $runderflow    db   "PLINK86 Overlay Loader: stack underflow $"
  169.         $roverflow     db   "PLINK86 Overlay Loader: stack overflow $"
  170.     $needdos2 db   "PLINK86 Overlay Loader: dos 2.0 required $"
  171.     SignOn    db   CR,LF,"PLINK86 Overlay Loader - $"
  172.     AskPrf    db   ".",CR,LF,"Enter file name prefix (X: or path name/) "
  173.               db   "or '.' to quit=>$"
  174.     Err       db   "Fatal error - $"
  175.     Err1      db   "Can't find file $"
  176.     Err2      db   "Disk I/O error in $"
  177.     NewLin    db   CR,LF,"$"
  178. OVdata    ends
  179. ;
  180. ;
  181. ;********
  182. ;* Code *
  183. ;********
  184. ;
  185. OVcode    segment    para   public   'OVLOADER'
  186.      assume cs:OVcode, ds:OVdata, es:OVdata
  187. ;
  188. ;
  189. ;*******    Macro for operating system calls.
  190. ;* Sys *
  191. ;*******
  192. ;
  193. ;
  194. LodOff macro Reg, Var   ;;macro to load variable offset into register.
  195.         mov  Reg, offset Var        ;;MSDOS, offsets are from segment
  196. endm
  197. ;
  198. Sys macro Fun, Arg      ;;macro for invoking Operating system functions
  199.          mov  AH, Fun
  200.      ifnb <Arg>
  201.          LodOff DX, Arg
  202.      endif
  203.          int  21H
  204. endm
  205. ;
  206. ;
  207. ;
  208. ;
  209. ; $$get_prefix ------------------------------------------------------------
  210. ;
  211. ;       This subroutine prompts the operator for a file name prefix.  This
  212. ;           prefix is then used to try to find the file with the requested
  213. ;           overlay.  If the operator complies, the string is left in the
  214. ;           Prefix buffer.  The operator can abort the search by entering
  215. ;           the ABORT_SEARCH character specified above.
  216. ;
  217. ;       Returns: carry clear iff string left in Prefix
  218. ;                carry set   iff operator entered ABORT_SEARCH 
  219. ;       
  220. ;
  221. $$get_prefix proc near
  222.      Sys  9, AskPrf                  ;prompt for path/file 
  223.      Sys  10, KeyBuf                 ;get new prefix
  224.      Sys  9, NewLin
  225.      mov  AL, byte ptr Prefix        ;get first char from buffer
  226.      cmp  AL, ABORT_SEARCH           ;
  227.      jz   gp_abort                   ;if '.' abort
  228.  
  229.      mov  BL, byte ptr KeyBuf + 1    ;BL = # chars read.
  230.      xor  BH, BH
  231.      mov  byte ptr Prefix[ BX ], 0   ;set last byte to NUL
  232.      xor  SI, SI                     ;use SI as pointer to buffer
  233.   gp_convert:                        ;start at zero
  234.      mov  AL, byte ptr Prefix[ SI ]  ;get a character from buffer
  235.      test AL, -1                     ;exit if done
  236.      jz   gp_ok
  237.      cmp  AL, 'z' + 1                ;if not lowercase
  238.      jnc  gp_next
  239.      cmp  AL, 'a'                    ;ASCII a-z
  240.      jc   gp_next                    ;get the next char
  241.      and  AL, 0DFH                   ;else force to uppercase
  242.      mov  byte ptr Prefix[ SI ], AL  ;and re-save to keyboard buffer
  243.   gp_next:                           ;
  244.      inc  SI                         ;point to next char
  245.      jmp  gp_convert                 ;and go convert it
  246.  
  247.   gp_ok:                             ;everything is ok 
  248.      clc                             ;   string is in Prefix
  249.      ret
  250.   gp_abort:                          ;operator wants to quit
  251.      stc                             ;   notify overlay loader
  252.      ret
  253. $$get_prefix endp
  254. ;
  255. ;
  256. ;
  257. ; $$is_separator ----------------------------------------------------------
  258. ;
  259. ;       The zero flag is set iff the character in AL is a valid path name
  260. ;           separator or a valid drive-pathname separator.  If it is not,
  261. ;           AL is changed to the standard path name separator.
  262. ;
  263. ;
  264. $$is_separator proc near
  265.      cmp  AL, ':'             ;a valid separator?
  266.      jz   got_it
  267.      cmp  AL, '/'
  268.      jz   got_it
  269.      cmp  AL, '\'
  270.      jz   got_it
  271.      mov  AL, STD_SEP         ;No, use standard one.
  272.   got_it:
  273.      ret
  274. $$is_separator endp
  275. ;
  276. ;
  277. ;
  278. ; $$prepend ---------------------------------------------------------------
  279. ;
  280. ;   The file name prefix pointed to by BP:SI is appended to the front of 
  281. ;       the file name string.  Much of the code here involves moving 
  282. ;       segment registers around, because the prefix may exist in a 
  283. ;       different segment if it is coming from the PATH in the environment 
  284. ;       area.  At the end of this routine SI points to the end of the 
  285. ;       source prefix string.  If there is no separator at the end of the 
  286. ;       prefix one is added (STD_SEP).
  287. ;
  288. $$prepend proc near
  289.      mov  DS, BP            ;DS:SI points to prefix
  290.      push SI
  291.      xor  DX, DX            ;DX := length of file name prefix
  292.   Count:    
  293.      lodsb
  294.      inc  DX
  295.      or   AL, AL
  296.      jz   CntEnd
  297.      cmp  AL, ' '
  298.      jz   CntEnd
  299.      cmp  AL, ';'
  300.      jnz  Count
  301.   CntEnd:   
  302.      dec  DX
  303.      pop  SI
  304.      ;fall through
  305.  
  306.   Appnd2:
  307.      mov  BX, ES            ;save current DS and ES  in BX.
  308.      mov  DS, BP            ;DS:SI points to prefix
  309.      push SI
  310.      add  SI, DX            ;get to end of prefix, look at last char.
  311.      dec  SI
  312.      lodsb                  ;save it in AL.
  313.      mov  DS, BX            ;restore my DS.
  314.      call $$is_separator    ;see if   last char is separator,  save
  315.      mov  AH, 0             ;separator in AL.
  316.      jnz  Shift             ;no, but now have it in  AL
  317.      inc  AH                ;AH is # of chars not used in prefix.
  318.      dec  DX                ;leave out separator, will put back later.
  319.   Shift:    
  320.      LodOff  SI,$OVLYFN     ;DS:SI -> end of file name
  321.      push SI                ;ES:DI -> end of file name area
  322.      add  SI, FNlen - 1
  323.      mov  DI, SI
  324.      inc  DX
  325.      sub  SI, DX
  326.      mov  CX, FNlen         ;shift file name up by length of prefix.
  327.      sub  CX, DX            ;plus separator byte.
  328.      dec  DX
  329.      std
  330.      rep  movsb
  331.      cld
  332.      pop  DI                ;move prefix to file name area.
  333.      pop  SI
  334.      mov  DS, BP
  335.      mov  CX, DX
  336.      rep  movsb
  337.      mov  DS, BX            ;restore my DS
  338.      mov  byte ptr [DI],AL  ;add file name separator.
  339.      mov  AL, AH            ;get SI to end of prefix
  340.      mov  AH, 0
  341.      add  SI, AX
  342.      ret
  343. $$prepend endp
  344. ;
  345. ;
  346. ;
  347.     ;
  348.     ;
  349.     ; $$try_path_strings --------------------------------------------------
  350.     ;
  351.     ;   If MSDOS 2.0+ is running, the file name prefix is set up as the 
  352.     ;       next directory name from the PATH string in the environment, 
  353.     ;       if any exist.
  354.     ;
  355.     $$try_path_strings proc near
  356.          cmp  DosVer, 1           ;if running 2.0,
  357.          jbe  NoPath
  358.          mov  SI, TryEnv          ;try next path
  359.          or   SI, SI
  360.          jz   NoPath
  361.          push DS                  ;DS:SI->next environment string
  362.          mov  AX, EnvPar
  363.          mov  DS, AX
  364.  
  365.       TryP1:    
  366.          lodsb                    ;get to start of next directory path
  367.          cmp  AL, ';'
  368.          jz   TryP1
  369.          cmp  AL, ' '
  370.          jz   TryP1
  371.          dec  SI
  372.  
  373.          test byte ptr [SI], -1   ;anything here?
  374.          mov  BP, DS              ;(BP -> prefix paragraph)
  375.          pop  DS                  ;(restore DS)
  376.          jz   NoPath
  377.          call $$prepend           ;append to file name.
  378.          mov  TryEnv, SI          ;save new ptr for next time.
  379.          clc                      ;fn ready to use now.
  380.          ret
  381.  
  382.       NoPath:
  383.          stc
  384.          ret
  385.     $$try_path_strings endp
  386.     ;
  387.     ;
  388.     ;
  389.     ; $$try_exe_path ------------------------------------------------------
  390.     ;
  391.     ;   If MSDOS 3.0+ is running "EnvOff" points to the complete path name
  392.     ;       the program was executed from.  If the overlay file name in 
  393.     ;       $OVLYFN has a type of ".EXE" this path name is substituted for 
  394.     ;       the $OVLYFN name.  That way someone can rename the .EXE file 
  395.     ;       and the loader will still be able to find the .EXE file.  For 
  396.     ;       auxilliary overlay files (not of type .EXE) the path portion of 
  397.     ;       the file name is preppended to the front of $OVLYFN:  it is 
  398.     ;       assumed that the auxilliary files will reside in the same 
  399.     ;       directory as the EXE file.
  400.     ;
  401.     ;
  402.     $$try_exe_path proc near
  403.          LodOff    DI,$OvlyFN     ;get to end of name.
  404.          mov  CX, -1
  405.          mov  AL, 0
  406.          repnz     scasb
  407.          sub  DI, 5               ;look at last 4 chars.
  408.          LodOff    SI,ExeTyp      ;is it .EXE file?
  409.          mov  CX, 4
  410.          repz cmpsb
  411.          jz   Try1
  412.          mov  SI, EPN             ;No, append path portion of   
  413.          mov  DX, ExeLen          ;.EXE name to given name.
  414.          mov  BP, EnvPar
  415.          mov  DS, BP
  416.          jmp  Appnd2
  417.  
  418.       Try1:     
  419.          mov  SI, EPN             ;Yes, move in entire name as is.
  420.          push DS
  421.          mov  DS, EnvPar
  422.          LodOff    DI,$OvlyFN
  423.          mov  CX, FNlen
  424.          rep  movsb
  425.          pop  DS
  426.          ret
  427.     $$try_exe_path endp
  428. ;
  429. ;
  430. ;
  431. ; $$error_msg -------------------------------------------------------------
  432. ;
  433. ;   An error message is generated from the code in AL.
  434. ;
  435. ;
  436. $$error_msg proc near
  437.      push AX
  438.      Sys  9, SignOn
  439.      pop  AX
  440.      LodOff    DX, Err1       ;This msg for errs 1 and 2
  441.      cmp  AL,3
  442.      jc   EM
  443.      LodOff    DX, Err2       ;this one for errs 3 and 4.
  444.   EM:
  445.      Sys  9
  446.      test Pflag, -1           ;want prefix printed?
  447.      jz   DspFN1
  448.      LodOff    SI, Prefix     ;yes, SI points to it.
  449.      call near ptr DspFN2     ;print it.
  450.      dec  SI                  ;was last character a separator?
  451.      mov  AL, [SI]
  452.      call $$is_separator
  453.      jz   DspFN1
  454.      mov  DL, STD_SEP         ;no, better print one
  455.      Sys  2
  456.   DspFN1:
  457.      LodOff    SI,$OVLYFN     ;now print file name.
  458.   DspFN2:
  459.      mov  DL, [SI]            ;display text from SI until NUL found.
  460.      or   DL,DL
  461.      jz   DspFN3
  462.      push SI
  463.      Sys  2
  464.      pop  SI
  465.      inc  SI
  466.      jmp  DspFN2
  467.   DspFN3:
  468.      ret
  469. $$error_msg endp
  470. ;
  471. ;
  472. ;
  473. ; $OVLYxU -----------------------------------------------------------------
  474. ;
  475. ;   This routine is driven  off of the MustAsk flag.  It is initialized to 
  476. ;       zero as the overlay loader is allowed to use the file name as is.  
  477. ;       On subsequent tries the various ways of modifying the file name in 
  478. ;       an effort to find the overlay are attempted.  These all involve 
  479. ;       adding a prefix to the file name, like a drive specifier or a 
  480. ;       directory name.  The last method tried is to ask the operator to 
  481. ;       enter a prefix.  If the operator so instructs, the overlay loader 
  482. ;       is told to give up and terminate the program.
  483. ;
  484. ;
  485. $OVLYMU   proc near
  486.      cld                           ;all increments forward.
  487.      or   AL, AL                   ;first attempt on this overlay file?
  488.      jz   First
  489.      cmp  AL, 5                    ;If handle error, try starting over.
  490.      jnz  Error
  491.  
  492.   First:  
  493.      mov  MustAsk,  0              ;Start with internal methods.
  494.      mov  TriedExe, 0              ;try MSDOS 3 EPN
  495.      mov  Pflag,    0              ;don't display prefix yet on error.
  496.      mov  BX, EnvOff            ;set initial environment ptr
  497.      mov  TryEnv, BX
  498.      jmp  short Return             ;try file name as is.
  499.  
  500.   Error:  
  501.      test MustAsk, -1              ;no, is it time to ask operator?
  502.      jnz  Prf
  503.      cmp  DosVer, 3             ;No.
  504.      jc   Try2
  505.      test TriedExe, -1          ;Tried this yet?
  506.      jnz  Try2
  507.      call $$try_exe_path        ;Try MSDOS 3 executable name.
  508.      inc  TriedExe              ;come through here once only
  509.      jmp  short Return
  510.      Try2:   
  511.      call $$try_path_strings    ;Try next MSDOS 2 PATH
  512.      jnc  Return                ;Any left?
  513.      inc  MustAsk                  ;No, try default prefix, once.
  514.      jmp  short Prf2
  515.   Prf:    
  516.      test $OVRERR$, -1
  517.      jnz  GiveUp
  518.      call $$error_msg              ;Tell operator what problem is,
  519.      mov  Pflag, 1                 ;display file name prefixes from now on,
  520.      call $$get_prefix             ;and ask for new file name prefix.
  521.      jc   GiveUp
  522.   Prf2:
  523.      mov  BP,DS                    ;BP:SI points to prefix.
  524.      LodOff    SI, Prefix          ;append current prefix to file name.
  525.      call $$prepend
  526.  
  527.   Return:
  528.      mov  AL, 0                    ;Tell loader to try current file name.
  529.      ret
  530.   GiveUp: 
  531.      mov  AL, 1                    ;Operator wants to terminate program.
  532.      ret
  533.  
  534. $OVLYMU   endp
  535. ;
  536. ;
  537. ;
  538. ; $OVLYIU -----------------------------------------------------------------
  539. ;
  540. ;   This is the initialization routine for the user module.  If we are 
  541. ;       running under MSDOS 2 the environment is searched for the PATH 
  542. ;       string.  If MSDOS 3 is running the path name used to execute the 
  543. ;       program appears at the end of the environment and is used instead 
  544. ;       of the PATH.
  545. ;
  546. ;    The environment paragraph address is given in the program prefix 
  547. ;       (pointed to by ES as the program begins execution) at offset 2CH.  
  548. ;       The environment consists of strings separated by nuls, with an 
  549. ;       extra nul at the end.  Each string has a name at the front and a 
  550. ;       value at the end, separated by an equal sign. The string searched 
  551. ;       for here has a name of "PATH".  The value portion of this string is
  552. ;       a list of directory names separated by semi-colons. The command 
  553. ;       processor looks in these directories to find executable files, so
  554. ;       program overlays might also be found there.  If the "PATH" string 
  555. ;       is found, its address is saved for later use.
  556. ;
  557. ;    MSDOS 3 passes the name the program was executed under as a string
  558. ;       following the end of the environment.  There's an extra word in 
  559. ;       front of it: I don't know what it is.  The MSDOS ref. says it's a 
  560. ;       "word count" but it always seems to be set to a one.  This routine 
  561. ;       ignores it.
  562. ;
  563. ;
  564.     Space proc near      ;scan ES:DI until a non-blank is found.
  565.          mov  AL," "
  566.          repz scasb
  567.          dec  DI         ;get ES:DI back to non-blank
  568.          ret
  569.     Space endp
  570. ;
  571. ;
  572. ;
  573. $OVLYIU   proc near
  574.      Sys  30H                 ;AL := DOS version #
  575.      mov  DosVer, AL          ;Save for later.
  576.      cmp  AL, 2               ;go away if earlier than 2.0
  577.      jnc  IU2
  578.      jmp  IUend
  579.   IU2:                        ;DOS 2.x or later
  580.      mov  $OVDOS2, -1         ;   -1 => use dos 2.0 file handles
  581.      cmp  AL, 3               ;check if on 3.x
  582.      jnc  IUdos3
  583.      jmp  short IUdos2
  584.  
  585.   IUdos3:
  586.      if MULTICOPY             ;NETBIOS test described in 
  587.          mov  ah, 0           ;    "IBM PC Network Program Manual"
  588.          int  2ah    
  589.          cmp  ah, 0           ;    ah != 0  =>  NETBIOS is active
  590.          je   IUmulti
  591.          mov  is_net_active, -1
  592.       IUmulti:
  593.      endif
  594.      mov  AL, STD_OPN         ;NETBIOS shared file open mode
  595.      jmp  short IUcontinue
  596.  
  597.   IUdos2:
  598.      xor  AL, AL              ;vanilla open mode
  599.  
  600.   IUcontinue:
  601.      if VIANET
  602.          mov  BP, 8AH         ;   check if VIANET installed
  603.          xor  BX, BX
  604.          Sys  0EFH            ;   sets BX: bit 0 => installed
  605.                               ;            bit 4 => 3.0 emulation
  606.          and  BX, 11H
  607.          cmp  BX, 1           ;   installed?
  608.          jb   NoPax               
  609.          mov  AL, 30H         ;   read only: net but no 3.0 emulation
  610.      endif
  611.   NoPax:    
  612.      mov  $OVOM$, AL          ;   set my open mode
  613.      mov  BX, ES:2CH          ;Save environment paragraph address.
  614.      mov  EnvPar, BX
  615.      mov  ES, BX              ;set ES to environment
  616.      xor  DI, DI              ;DI is offset into it.
  617.      cld                      ;auto-increment forward.
  618.   Srch:
  619.      test byte ptr ES:[DI],-1 ;end of environment?
  620.      jz   E1
  621.      mov  SI, offset Path     ;no, compare next string
  622.      mov  CX, Psize
  623.      repe cmpsb
  624.      mov  CX, -1              ;for REP instructions.
  625.      jz   GotIt               ;Is it a match?
  626.  
  627.   Skip:     
  628.      mov  AL, 0               ;No, get past string and
  629.      repnz     scasb          ;the nul terminating it.
  630.      jmp  Srch
  631.  
  632.   GotIt:
  633.      call Space               ;Got it:  skip blanks.
  634.      cmp  byte ptr ES:[DI],'='     ;at end   of "PATH"?
  635.      jnz  Skip                ;no, had match on prefix.
  636.      inc  DI                  ;yes, get over =.
  637.      call Space               ;and spaces.
  638.      mov  EnvOff, DI          ;save pointer to it.
  639.   E1:
  640.      cmp  DosVer, 2           ;Is it version 3 or more?
  641.      jna  IUend
  642.                               ;yes, find execution path name.
  643.      xor  DI, DI              ;DI->front of environment.
  644.      mov  CX, -1              ;for REP instructions.
  645.      mov  AL, 0               ;strings end with NUL.
  646.   Its3:
  647.      cmp  byte ptr ES:[DI],AL ;end of   environment?
  648.      jz   EnvEnd
  649.      repnz     scasb          ;no:  skip next environment string.
  650.      jmp  Its3 
  651.   EnvEnd:   
  652.      add  DI, 3               ;get over NUL and mystery word.
  653.      mov  EPN, DI             ;save pointer to it.
  654.      xor  DX, DX
  655.   GetExL:
  656.      mov  AL,byte ptr ES:[DI] ;get length of path name portion
  657.      inc  DI
  658.      inc  DX
  659.      cmp  AL, 0               ;of this name.
  660.      jz   IUend
  661.      call $$is_separator
  662.      jnz  GetExL
  663.      mov  ExeLen, DX
  664.      jmp  GetExL
  665.   IUend:
  666.      ret
  667. $OVLYIU   endp
  668. ;
  669. ;
  670. ;
  671. ; $OVLEND -----------------------------------------------------------------
  672. ;
  673. ;    $OVLEND is invoked at the end of a request for an overlay load.  By
  674. ;         default, it determines whether to close the current overlay file.
  675. ;         However, it may be customized for any post-overlay-load use:
  676. ;         for example, if you are overlaying code and data, after an
  677. ;         overlay is loaded, parts of the data segment could be initialized
  678. ;         according to some run-time criterion.
  679. ;
  680. ;    By default, the file remains open until the user explicitly closes it
  681. ;         via $OVCLOSE, or the next load operation determines a different
  682. ;         overlay file must be used.
  683. ;
  684. ;    This module may be configured to allow multiple copies of a single
  685. ;         executable (and its overlays) to be running concurrently across a
  686. ;         network conforming to the NETBIOS standard.  PLINK86 will open
  687. ;         overlay files in "shared" mode;  however, DOS 3.0 uses read/write
  688. ;         mode to initially load a program: this restriction means that 
  689. ;         the overlay loader should automatically close all open files 
  690. ;         after a load, if NETBIOS is active.
  691. ;
  692. ;              Some NETBIOS standard vendors: IBM Network
  693. ;                                             AT&T Starlan
  694. ;                                             Novell Advanced Network
  695. ;
  696. ;         If your application does not put any overlays into the DOS loaded
  697. ;              .EXE file, you do not need to turn MULTICOPY on.  
  698. ;
  699. ;
  700. $OVLEND proc near
  701.     if   MULTICOPY 
  702.          pushf                    ;save flags: they give status of load
  703.          cmp  is_net_active, -1   ;check if running under NETBIOS standard
  704.          jne  lendend             ;
  705.          call $OVCLOSE            ;    close any open files
  706.       lendend:
  707.          popf                     ;restore load status
  708.     endif
  709.     ret
  710. $OVLEND endp
  711. ;
  712. ;
  713. ;
  714. ; $OVEXIT -----------------------------------------------------------------
  715. ;
  716. ;   $OVEXIT is called by the overlay loader when a fatal condition has been
  717. ;       found: it is included in this file to allow users to cleanup their
  718. ;       open files, etc. before the application terminates.
  719. ;
  720. ;
  721. $OVEXIT proc near
  722.         call  $OVCLOSE
  723.         mov al, 0ffh
  724.         mov ah, 4ch
  725.         int 21h
  726.     ret
  727. $OVEXIT endp
  728.  
  729. OVcode    ends
  730.      end
  731.