home *** CD-ROM | disk | FTP | other *** search
/ Computerworld 1996 March / Computerworld_1996-03_cd.bin / idg_cd3 / grafika / fraktaly / frasr192 / general.asm < prev    next >
Assembly Source File  |  1995-02-15  |  65KB  |  2,293 lines

  1. ;    Generic assembler routines that have very little at all
  2. ;    to do with fractals.
  3. ;
  4. ;    (NOTE:    The video routines have been moved over to VIDEO.ASM)
  5. ;
  6. ; ---- Overall Support
  7. ;
  8. ;    initasmvars()
  9. ;
  10. ; ---- Quick-copy to/from Extraseg support
  11. ;
  12. ;    toextra()
  13. ;    fromextra()
  14. ;    cmpextra()
  15. ;
  16. ; ---- far memory allocation support
  17. ;
  18. ;    farmemalloc()
  19. ;    farmemfree()
  20. ;    erasesegment()
  21. ;
  22. ; ---- General Turbo-C (artificial) support
  23. ;
  24. ;    disable()
  25. ;    enable()
  26. ;
  27. ; ---- 32-bit Multiply/Divide Routines (includes 16-bit emulation)
  28. ;
  29. ;    multiply()
  30. ;    divide()
  31. ;
  32. ; ---- Keyboard, audio (and, hidden inside them, Mouse) support
  33. ;
  34. ;    keypressed()
  35. ;    getakey()
  36. ;    buzzer()
  37. ;    delay()
  38. ;    tone()
  39. ;    snd()
  40. ;    nosnd()
  41. ;
  42. ; ---- Expanded Memory Support
  43. ;
  44. ;    emmquery()
  45. ;    emmgetfree()
  46. ;    emmallocate()
  47. ;    emmdeallocate()
  48. ;    emmgetpage()
  49. ;    emmclearpage()
  50. ;
  51. ; ---- Extended Memory Support
  52. ;
  53. ;    xmmquery()
  54. ;    xmmlongest()
  55. ;    xmmallocate()
  56. ;    xmmreallocate()
  57. ;    xmmdeallocate()
  58. ;    xmmmoveextended()
  59. ;
  60. ; ---- CPU, FPU Detectors
  61. ;
  62. ;    cputype()
  63. ;    fputype()
  64. ;
  65. ; ---- IIT FPU Support
  66. ;
  67. ;    load_mat()
  68. ;    mult_vec_iit()
  69. ;    IITCoPro()
  70. ;
  71.  
  72.  
  73. ;             required for compatibility if Turbo ASM
  74. IFDEF ??version
  75.     MASM51
  76.     QUIRKS
  77. ENDIF
  78.  
  79.     .MODEL    medium,c
  80.  
  81.     .8086
  82.  
  83.     ; these must NOT be in any segment!!
  84.     ; this get's rid of TURBO-C fixup errors
  85.  
  86.     extrn    help:far        ; help code (in help.c)
  87.     extrn    tab_display:far     ; TAB display (in fractint.c)
  88.       ; extrn    restore_active_ovly:far
  89.     extrn    edit_text_colors:far
  90.     extrn    adapter_init:far    ; video adapter init (in video.asm)
  91.     extrn    slideshw:far
  92.     extrn    recordshw:far
  93.     extrn    stopslideshow:far
  94.  
  95. .DATA
  96.  
  97. ; ************************ External variables *****************************
  98.  
  99.     extrn    soundflag:word        ; if 0, supress sounds
  100.     extrn    debugflag:word        ; for debugging purposes only
  101.     extrn    helpmode:word        ; help mode (AUTHORS is special)
  102.     extrn    tabmode:word        ; tab key enabled?
  103.     extrn    sxdots:word        ; horizontal pixels
  104.     extrn    timedsave:word        ; triggers autosave
  105.     extrn    calc_status:word    ; in calcfrac.c
  106.     extrn    got_status:word     ; in calcfrac.c
  107.     extrn    currow:word        ; in calcfrac.c
  108.     extrn    slides:word        ; in cmdfiles.c
  109.  
  110.  
  111. ; ************************ Public variables *****************************
  112.  
  113. public        cpu            ; used by 'calcmand'
  114. public        fpu            ; will be used by somebody someday
  115. public        iit            ; IIT fpu?
  116. public        lookatmouse        ; used by 'calcfrac'
  117. public        saveticks        ; set by fractint
  118. public        savebase        ; set by fractint
  119. public        finishrow        ; set by fractint
  120. public        _dataseg_xx        ; used by TARGA, other Turbo Code
  121.  
  122. public        extraseg        ; extra 64K segment
  123.  
  124. public        overflow        ; Mul, Div overflow flag: 0 means none
  125.  
  126. ;        arrays declared here, used elsewhere
  127. ;        arrays not used simultaneously are deliberately overlapped
  128.  
  129. public        keybuffer            ; needed for ungetakey
  130. public        prefix, suffix, dstack, decoderline ; for the Decoder
  131. public        tstack                ; for the prompting routines
  132. public        strlocn, teststring, block    ; used by the Encoder
  133. public        boxx, boxy, boxvalues        ; zoom-box arrays
  134. public        olddacbox            ; temporary DAC saves
  135. public        rlebuf                ; Used ty the TARGA En/Decoder
  136. public        paldata, stbuff         ; 8514A arrays, (FR8514A.ASM)
  137.  
  138. ; ************************* "Shared" array areas **************************
  139.  
  140. ; Shared near arrays are discussed below. First some comments about "extraseg".
  141. ; It is a 96k far area permanently allocated during fractint runup.
  142. ; The first part is used for:
  143. ;   64k coordinate arrays during image calculation (initialized in fracsubr.c)
  144. ;   64k create gif save file, encoder.c
  145. ;   64k 3d transforms, line3d.c
  146. ;   64k credits screen, intro.c
  147. ;   22k video mode selection (for fractint.cfg, loadfdos, miscovl)
  148. ;    2k .frm .ifs .l .par entry selection and file selection (prompts.c)
  149. ;   ??k printing, printer.c
  150. ;   ??k fractint.doc creation, help.c, not important cause it saves/restores
  151. ;   64K arbitrary precision - critical variables at top, safe from encoder
  152. ;   10K cmdfiles.c input buffer
  153. ;   4K  miscovl.c PAR file buffer
  154. ;
  155. ; The high 32k is used for graphics image save during text mode; video.asm
  156. ; and realdos.c.
  157.  
  158. ; Short forms used in subsequent comments:
  159. ;   name........duration of use......................modules....................
  160. ;   encoder    "s"aving an image                    encoder.c
  161. ;   decoder    "r"estoring an image                 decoder.c, gifview.c
  162. ;   zoom    zoom box is visible             zoom.c, video.asm
  163. ;   vidswitch    temp during video mode setting         video.asm
  164. ;   8514a    8514a is in use (graphics only?)     fr8514a.asm
  165. ;   tgaview    restore of tga image             tgaview.c
  166. ;   solidguess    image gen with "g", not to disk      calcfrac.c
  167. ;   btm     image gen with "b", not to disk      calcfrac.c
  168. ;   editpal    palette editor "heap"                editpal.c
  169. ;   cellular    cellular fractal type generation     misfrac.c
  170. ;   browse    browsing                             loadfile.c
  171. ;   lsystem    lsystem fractal type generation      lsys.c, lsysf.c, lsysa.asm, lsysaf.asm
  172. ; Several arrays used in cmdfiles.c and miscovl.c, but are saved and
  173. ; restored to extraseg so not critical.
  174. ;
  175. ; Note that decoder using an 8514a is worst case, uses all arrays at once.
  176. ; Keep db lengths even so that word alignment is preserved for speed.
  177.  
  178. block        label    byte        ; encoder(266)
  179. suffix        dw    2048 dup(0)    ; decoder(4k), vidswitch(256),
  180.                     ; savegraphics/restoregraphics(4k)
  181.  
  182. tstack        label    byte        ; prompts(4k), ifsload(4k),
  183.                     ; make_batch(4k)
  184. teststring    label    byte        ; encoder(100)
  185. olddacbox    label    byte        ; fractint(768), prompts(768)
  186. dstack        dw    2048 dup(0)    ; decoder(4k), solidguess(4k), btm(2k)
  187.                     ;   zoom(2k), printer(2400)
  188.                     ;   loadfdos(2000), cellular(2025)
  189.  
  190. strlocn     label    word        ; encoder(10k), editpal(10k)
  191. prefix        label    word        ; decoder(8k), solidguess(6k)
  192. boxx        dw    2048 dup(0)    ; zoom(4k), tgaview(4k), prompts(9k)
  193.                     ;   make_batch(8k), parser(8k)
  194.                     ;   browse(?)
  195. boxy        dw    2048 dup(0)    ; zoom(4k), cellular(2025), browse(?)
  196.                     ;   lsystem(1000)
  197. boxvalues    label    byte        ; zoom(2k), browse(?)
  198. decoderline    db    2050 dup(0)    ; decoder(2049), btm(2k)
  199.  
  200. rlebuf        label    byte        ; f16.c(258) .tga save/restore?
  201. paldata     db    1024 dup(0)    ; 8514a(1k)
  202. stbuff        db    415 dup(0)    ; 8514a(415)
  203.  
  204. ; ************************ Internal variables *****************************
  205.  
  206.         align    2
  207. cpu        dw    0        ; cpu type: 86, 186, 286, or 386
  208. fpu        dw    0        ; fpu type: 0, 87, 287, 387
  209. iit        dw    0        ; iit fpu:  0=no, 1=yes
  210. _dataseg_xx    dw    0        ; our "near" data segment
  211.  
  212. overflow    dw    0        ; overflow flag
  213.  
  214. kbd_type    db    0        ; type of keyboard
  215.         align    2
  216. keybuffer    dw    0        ; real small keyboard buffer
  217.  
  218. delayloop    dw    32        ; delay loop value
  219. delaycount    dw    0        ; number of delay "loops" per ms.
  220.  
  221. extraseg    dw    0        ; extra 64K segment (allocated by init)
  222.  
  223. ; ********************** Mouse Support Variables **************************
  224.  
  225. lookatmouse    dw    0        ; see notes at mouseread routine
  226. prevlamouse    dw    0        ; previous lookatmouse value
  227. mousetime    dw    0        ; time of last mouseread call
  228. mlbtimer    dw    0        ; time of left button 1st click
  229. mrbtimer    dw    0        ; time of right button 1st click
  230. mhtimer     dw    0        ; time of last horiz move
  231. mvtimer     dw    0        ; time of last vert  move
  232. mhmickeys    dw    0        ; pending horiz movement
  233. mvmickeys    dw    0        ; pending vert    movement
  234. mbstatus    db    0        ; status of mouse buttons
  235. mouse        db    0        ; == -1 if/when a mouse is found.
  236. mbclicks    db    0        ; had 1 click so far? &1 mlb, &2 mrb
  237.  
  238.         align    2
  239. ; timed save variables, handled by readmouse:
  240. savechktime    dw    0        ; time of last autosave check
  241. savebase    dw    2 dup(0)    ; base clock ticks
  242. saveticks    dw    2 dup(0)    ; save after this many ticks
  243. finishrow    dw    0        ; save when this row is finished
  244.  
  245.  
  246. .CODE
  247.  
  248. ; *************** Function toextra(tooffset,fromaddr, fromcount) *********
  249.  
  250. toextra proc    uses es di si, tooffset:word, fromaddr:word, fromcount:word
  251.     cld                ; move forward
  252.     mov    ax,extraseg        ; load ES == extra segment
  253.     mov    es,ax            ;  ..
  254.     mov    di,tooffset        ; load to here
  255.     mov    si,fromaddr        ; load from here
  256.     mov    cx,fromcount        ; this many bytes
  257.     rep    movsb            ; do it.
  258.     ret                ; we done.
  259. toextra endp
  260.  
  261.  
  262. ; *************** Function fromextra(fromoffset, toaddr, tocount) *********
  263.  
  264. fromextra proc    uses es di si, fromoffset:word, toaddr:word, tocount:word
  265.     push    ds            ; save DS for a tad
  266.     pop    es            ; restore it to ES
  267.     cld                ; move forward
  268.     mov    si,fromoffset        ; load from here
  269.     mov    di,toaddr        ; load to here
  270.     mov    cx,tocount        ; this many bytes
  271.     mov    ax,extraseg        ; load DS == extra segment
  272.     mov    ds,ax            ;  ..
  273.     rep    movsb            ; do it.
  274.     push    es            ; save ES again.
  275.     pop    ds            ; restore DS
  276.     ret                ; we done.
  277. fromextra endp
  278.  
  279.  
  280. ; *************** Function cmpextra(cmpoffset,cmpaddr, cmpcount) *********
  281.  
  282. cmpextra proc    uses es di si, cmpoffset:word, cmpaddr:word, cmpcount:word
  283.     cld                ; move forward
  284.     mov    ax,extraseg        ; load ES == extra segment
  285.     mov    es,ax            ;  ..
  286.     mov    di,cmpoffset        ; load to here
  287.     mov    si,cmpaddr        ; load from here
  288.     mov    cx,cmpcount        ; this many bytes
  289.     rep    cmpsb            ; do it.
  290.     jnz    cmpbad            ; failed.
  291.     sub    ax,ax            ; 0 == true
  292.     jmp    short cmpend
  293. cmpbad:
  294.     mov    ax,1            ; 1 == false
  295. cmpend:
  296.     ret                ; we done.
  297. cmpextra    endp
  298.  
  299.  
  300. ; =======================================================
  301. ;
  302. ;    32-bit integer multiply routine with an 'n'-bit shift.
  303. ;    Overflow condition returns 0x7fffh with overflow = 1;
  304. ;
  305. ;    long x, y, z, multiply();
  306. ;    int n;
  307. ;
  308. ;    z = multiply(x,y,n)
  309. ;
  310. ;    requires the presence of an external variable, 'cpu'.
  311. ;        'cpu' == 386 if a 386 is present.
  312.  
  313. .8086
  314.  
  315. .DATA
  316.  
  317. temp    dw    5 dup(0)        ; temporary 64-bit result goes here
  318. sign    db    0            ; sign flag goes here
  319.  
  320. .CODE
  321.  
  322. FRAME    MACRO regs
  323.     push    bp
  324.     mov    bp, sp
  325.     IRP    reg, <regs>
  326.       push    reg
  327.       ENDM
  328.     ENDM
  329.  
  330. UNFRAME MACRO regs
  331.     IRP    reg, <regs>
  332.       pop reg
  333.       ENDM
  334.     pop bp
  335.     ENDM
  336.  
  337.  
  338. multiply    proc    x:dword, y:dword, n:word
  339.  
  340.     cmp    cpu,386         ; go-fast time?
  341.     jne    slowmultiply        ; no.  yawn...
  342.  
  343. .386                    ; 386-specific code starts here
  344.  
  345.     mov    eax,x            ; load X into EAX
  346.     imul    y            ; do the multiply
  347.     mov    cx,n            ; set up the shift
  348.     cmp    cx,32            ; ugly klooge:    check for 32-bit shift
  349.     jb    short fastm1        ;  < 32 bits:  no problem
  350.     mov    eax,edx         ;  >= 32 bits:    manual shift
  351.     mov    edx,0            ;  ...
  352.     sub    cx,32            ;  ...
  353. fastm1: shrd    eax,edx,cl        ; shift down 'n' bits
  354.     js    fastm3
  355.     sar    edx,cl
  356.     jne    overmf
  357.     shld    edx,eax,16
  358.     ret
  359. fastm3: sar    edx,cl
  360.     inc    edx
  361.     jne    overmf
  362.     shld    edx,eax,16
  363.     ret
  364. overmf:
  365.     mov    ax,0ffffh        ; overflow value
  366.     mov    dx,07fffh        ; overflow value
  367.     mov    overflow,1        ; flag overflow
  368.     ret
  369.  
  370. .8086                    ; 386-specific code ends here
  371.  
  372. slowmultiply:                ; (sigh)  time to do it the hard way...
  373.     push    di
  374.     push    si
  375.     push    es
  376.  
  377.     mov    ax,0
  378.     mov    temp+4,ax        ; first, zero out the (temporary)
  379.     mov    temp+6,ax        ;  result
  380.     mov    temp+8,ax
  381.  
  382.     les    bx,x            ; move X to SI:BX
  383.     mov    si,es            ;  ...
  384.     les    cx,y            ; move Y to DI:CX
  385.     mov    di,es            ;  ...
  386.  
  387.     mov    sign,0            ; clear out the sign flag
  388.     cmp    si,0            ; is X negative?
  389.     jge    mults1            ;  nope
  390.     not    sign            ;  yup.  flip signs
  391.     not    bx            ;   ...
  392.     not    si            ;   ...
  393.     stc                ;   ...
  394.     adc    bx,ax            ;   ...
  395.     adc    si,ax            ;   ...
  396. mults1: cmp    di,0            ; is DI:CX negative?
  397.     jge    mults2            ;  nope
  398.     not    sign            ;  yup.  flip signs
  399.     not    cx            ;   ...
  400.     not    di            ;   ...
  401.     stc                ;   ...
  402.     adc    cx,ax            ;   ...
  403.     adc    di,ax            ;   ...
  404. mults2:
  405.  
  406.     mov    ax,bx            ; perform BX x CX
  407.     mul    cx            ;  ...
  408.     mov    temp,ax         ;  results in lowest 32 bits
  409.     mov    temp+2,dx        ;  ...
  410.  
  411.     mov    ax,bx            ; perform BX x DI
  412.     mul    di            ;  ...
  413.     add    temp+2,ax        ;  results in middle 32 bits
  414.     adc    temp+4,dx        ;  ...
  415.     jnc    mults3            ;  carry bit set?
  416.     inc    word ptr temp+6     ;  yup.  overflow
  417. mults3:
  418.  
  419.     mov    ax,si            ; perform SI * CX
  420.     mul    cx            ;  ...
  421.     add    temp+2,ax        ;  results in middle 32 bits
  422.     adc    temp+4,dx        ;  ...
  423.     jnc    mults4            ;  carry bit set?
  424.     inc    word ptr temp+6     ;  yup.  overflow
  425. mults4:
  426.  
  427.     mov    ax,si            ; perform SI * DI
  428.     mul    di            ;  ...
  429.     add    temp+4,ax        ; results in highest 32 bits
  430.     adc    temp+6,dx        ;  ...
  431.  
  432.     mov    cx,n            ; set up for the shift loop
  433.     cmp    cx,24            ; shifting by three bytes or more?
  434.     jl    multc1            ;  nope.  check for something else
  435.     sub    cx,24            ; quick-shift 24 bits
  436.     mov    ax,temp+3        ; load up the registers
  437.     mov    dx,temp+5        ;  ...
  438.     mov    si,temp+7        ;  ...
  439.     mov    bx,0            ;  ...
  440.     jmp    short multc4        ; branch to common code
  441. multc1: cmp    cx,16            ; shifting by two bytes or more?
  442.     jl    multc2            ;  nope.  check for something else
  443.     sub    cx,16            ; quick-shift 16 bits
  444.     mov    ax,temp+2        ; load up the registers
  445.     mov    dx,temp+4        ;  ...
  446.     mov    si,temp+6        ;  ...
  447.     mov    bx,0            ;  ...
  448.     jmp    short multc4        ; branch to common code
  449. multc2: cmp    cx,8            ; shifting by one byte or more?
  450.     jl    multc3            ;  nope.  check for something else
  451.     sub    cx,8            ; quick-shift 8 bits
  452.     mov    ax,temp+1        ; load up the registers
  453.     mov    dx,temp+3        ;  ...
  454.     mov    si,temp+5        ;  ...
  455.     mov    bx,temp+7        ;  ...
  456.     jmp    short multc4        ; branch to common code
  457. multc3: mov    ax,temp         ; load up the regs
  458.     mov    dx,temp+2        ;  ...
  459.     mov    si,temp+4        ;  ...
  460.     mov    bx,temp+6        ;  ...
  461. multc4: cmp    cx,0            ; done shifting?
  462.     je    multc5            ;  yup.  bail out
  463.  
  464. multloop:
  465.     shr    bx,1            ; shift down 1 bit, cascading
  466.     rcr    si,1            ;  ...
  467.     rcr    dx,1            ;  ...
  468.     rcr    ax,1            ;  ...
  469.     loop    multloop        ; try the next bit, if any
  470. multc5:
  471.     cmp    si,0            ; overflow time?
  472.     jne    overm1            ; yup.    Bail out.
  473.     cmp    bx,0            ; overflow time?
  474.     jne    overm1            ; yup.    Bail out.
  475.     cmp    dx,0            ; overflow time?
  476.     jl    overm1            ; yup.    Bail out.
  477.  
  478.     cmp    sign,0            ; should we negate the result?
  479.     je    mults5            ;  nope.
  480.     not    ax            ;  yup.  flip signs.
  481.     not    dx            ;   ...
  482.     mov    bx,0            ;   ...
  483.     stc                ;   ...
  484.     adc    ax,bx            ;   ...
  485.     adc    dx,bx            ;   ...
  486. mults5:
  487.     jmp    multiplyreturn
  488.  
  489. overm1:
  490.     mov    ax,0ffffh        ; overflow value
  491.     mov    dx,07fffh        ; overflow value
  492.     mov    overflow,1        ; flag overflow
  493.  
  494. multiplyreturn:             ; that's all, folks!
  495.     pop    es
  496.     pop    si
  497.     pop    di
  498.     ret
  499. multiply    endp
  500.  
  501.  
  502. ; =======================================================
  503. ;
  504. ;    32-bit integer divide routine with an 'n'-bit shift.
  505. ;    Overflow condition returns 0x7fffh with overflow = 1;
  506. ;
  507. ;    long x, y, z, divide();
  508. ;    int n;
  509. ;
  510. ;    z = divide(x,y,n);    /* z = x / y; */
  511. ;
  512. ;    requires the presence of an external variable, 'cpu'.
  513. ;        'cpu' == 386 if a 386 is present.
  514.  
  515.  
  516. .8086
  517.  
  518. divide        proc    uses di si es, x:dword, y:dword, n:word
  519.  
  520.     cmp    cpu,386         ; go-fast time?
  521.     jne    slowdivide        ; no.  yawn...
  522.  
  523. .386                    ; 386-specific code starts here
  524.  
  525.     mov    edx,x            ; load X into EDX (shifts to EDX:EAX)
  526.     mov    ebx,y            ; load Y into EBX
  527.  
  528.     mov    sign,0            ; clear out the sign flag
  529.     cmp    edx,0            ; is X negative?
  530.     jge    short divides1        ;  nope
  531.     not    sign            ;  yup.  flip signs
  532.     neg    edx            ;   ...
  533. divides1:
  534.     cmp    ebx,0            ; is Y negative?
  535.     jge    short divides2        ;  nope
  536.     not    sign            ;  yup.  flip signs
  537.     neg    ebx            ;   ...
  538. divides2:
  539.  
  540.     mov    eax,0            ; clear out the low-order bits
  541.     mov    cx,32            ; set up the shift
  542.     sub    cx,n            ; (for large shift counts - faster)
  543. fastd1: cmp    cx,0            ; done shifting?
  544.     je    fastd2            ; yup.
  545.     shr    edx,1            ; shift one bit
  546.     rcr    eax,1            ;  ...
  547.     loop    fastd1            ; and try again
  548. fastd2:
  549.     cmp    edx,ebx         ; umm, will the divide blow out?
  550.     jae    overd1            ;  yup.  better skip it.
  551.     div    ebx            ; do the divide
  552.     cmp    eax,0            ; did the sign flip?
  553.     jl    overd1            ;  then we overflowed
  554.     cmp    sign,0            ; is the sign reversed?
  555.     je    short divides3        ;  nope
  556.     neg    eax            ; flip the sign
  557. divides3:
  558.     push    eax            ; save the 64-bit result
  559.     pop    ax            ; low-order  16 bits
  560.     pop    dx            ; high-order 16 bits
  561.     jmp    dividereturn        ; back to common code
  562.  
  563. .8086                    ; 386-specific code ends here
  564.  
  565. slowdivide:                ; (sigh)  time to do it the hard way...
  566.  
  567.     les    ax,x            ; move X to DX:AX
  568.     mov    dx,es            ;  ...
  569.  
  570.     mov    sign,0            ; clear out the sign flag
  571.     cmp    dx,0            ; is X negative?
  572.     jge    divides4        ;  nope
  573.     not    sign            ;  yup.  flip signs
  574.     not    ax            ;   ...
  575.     not    dx            ;   ...
  576.     stc                ;   ...
  577.     adc    ax,0            ;   ...
  578.     adc    dx,0            ;   ...
  579. divides4:
  580.  
  581.     mov    cx,32            ; get ready to shift the bits
  582.     sub    cx,n            ; (shift down rather than up)
  583.     mov    byte ptr temp+4,cl    ;  ...
  584.  
  585.     mov    cx,0            ;  clear out low bits of DX:AX:CX:BX
  586.     mov    bx,0            ;  ...
  587.  
  588.     cmp    byte ptr temp+4,16    ; >= 16 bits to shift?
  589.     jl    dividex0        ;  nope
  590.     mov    bx,cx            ;  yup.  Take a short-cut
  591.     mov    cx,ax            ;   ...
  592.     mov    ax,dx            ;   ...
  593.     mov    dx,0            ;   ...
  594.     sub    byte ptr temp+4,16    ;   ...
  595. dividex0:
  596.     cmp    byte ptr temp+4,8    ; >= 8 bits to shift?
  597.     jl    dividex1        ;  nope
  598.     mov    bl,bh            ;  yup.  Take a short-cut
  599.     mov    bh,cl            ;   ...
  600.     mov    cl,ch            ;   ...
  601.     mov    ch,al            ;   ...
  602.     mov    al,ah            ;   ...
  603.     mov    ah,dl            ;   ...
  604.     mov    dl,dh            ;   ...
  605.     mov    dh,0            ;   ...
  606.     sub    byte ptr temp+4,8    ;   ...
  607. dividex1:
  608.     cmp    byte ptr temp+4,0    ; are we done yet?
  609.     je    dividex2        ;  yup
  610.     shr    dx,1            ; shift all 64 bits
  611.     rcr    ax,1            ;  ...
  612.     rcr    cx,1            ;  ...
  613.     rcr    bx,1            ;  ...
  614.     dec    byte ptr temp+4     ; decrement the shift counter
  615.     jmp    short dividex1        ;  and try again
  616. dividex2:
  617.  
  618.     les    di,y            ; move Y to SI:DI
  619.     mov    si,es            ;  ...
  620.  
  621.     cmp    si,0            ; is Y negative?
  622.     jge    divides5        ;  nope
  623.     not    sign            ;  yup.  flip signs
  624.     not    di            ;   ...
  625.     not    si            ;   ...
  626.     stc                ;   ...
  627.     adc    di,0            ;   ...
  628.     adc    si,0            ;   ...
  629. divides5:
  630.  
  631.     mov    byte ptr temp+4,33    ; main loop counter
  632.     mov    temp,0            ; results in temp
  633.     mov    word ptr temp+2,0    ;  ...
  634.  
  635. dividel1:
  636.     shl    temp,1            ; shift the result up 1
  637.     rcl    word ptr temp+2,1    ;  ...
  638.     cmp    dx,si            ; is DX:AX >= Y?
  639.     jb    dividel3        ;  nope
  640.     ja    dividel2        ;  yup
  641.     cmp    ax,di            ;  maybe
  642.     jb    dividel3        ;  nope
  643. dividel2:
  644.     cmp    byte ptr temp+4,32    ; overflow city?
  645.     jge    overd1            ;  yup.
  646.     sub    ax,di            ; subtract Y
  647.     sbb    dx,si            ;  ...
  648.     inc    temp            ; add 1 to the result
  649.     adc    word ptr temp+2,0    ;  ...
  650. dividel3:
  651.     shl    bx,1            ; shift all 64 bits
  652.     rcl    cx,1            ;  ...
  653.     rcl    ax,1            ;  ...
  654.     rcl    dx,1            ;  ...
  655.     dec    byte ptr temp+4     ; time to quit?
  656.     jnz    dividel1        ;  nope.  try again.
  657.  
  658.     mov    ax,temp         ; copy the result to DX:AX
  659.     mov    dx,word ptr temp+2    ;  ...
  660.     cmp    sign,0            ; should we negate the result?
  661.     je    divides6        ;  nope.
  662.     not    ax            ;  yup.  flip signs.
  663.     not    dx            ;   ...
  664.     mov    bx,0            ;   ...
  665.     stc                ;   ...
  666.     adc    ax,0            ;   ...
  667.     adc    dx,0            ;   ...
  668. divides6:
  669.     jmp    short dividereturn
  670.  
  671. overd1:
  672.     mov    ax,0ffffh        ; overflow value
  673.     mov    dx,07fffh        ; overflow value
  674.     mov    overflow,1        ; flag overflow
  675.  
  676. dividereturn:                ; that's all, folks!
  677.     ret
  678. divide        endp
  679.  
  680.  
  681. ; ****************** Function getakey() *****************************
  682. ; **************** Function keypressed() ****************************
  683.  
  684. ;    'getakey()' gets a key from either a "normal" or an enhanced
  685. ;    keyboard.   Returns either the vanilla ASCII code for regular
  686. ;    keys, or 1000+(the scan code) for special keys (like F1, etc)
  687. ;    Use of this routine permits the Control-Up/Down arrow keys on
  688. ;    enhanced keyboards.
  689. ;
  690. ;    The concept for this routine was "borrowed" from the MSKermit
  691. ;    SCANCHEK utility
  692. ;
  693. ;    'keypressed()' returns a zero if no keypress is outstanding,
  694. ;    and the value that 'getakey()' will return if one is.  Note
  695. ;    that you must still call 'getakey()' to flush the character.
  696. ;    As a sidebar function, calls 'help()' if appropriate, or
  697. ;    'tab_display()' if appropriate.
  698. ;    Think of 'keypressed()' as a super-'kbhit()'.
  699.  
  700. keypressed  proc
  701.     call    far ptr getkeynowait        ; check for key
  702.     jc    keypressed1            ;  got a key
  703.     sub    ax,ax                ; fast no-key return
  704.     ret
  705. keypressed1:
  706.     FRAME    <di,si,es>            ; std frame, for TC++ overlays
  707.     mov    keybuffer,ax            ; remember it for next time
  708.     cmp    ax,1059             ; help called?
  709.     jne    keypressed2            ; no help asked for.
  710.     cmp    helpmode,0            ; help enabled?
  711.     jl    keypressedx            ;  nope.
  712.     mov    keybuffer,0            ; say no key hit
  713.     xor    ax,ax
  714.     push    ax
  715.     call    far ptr help            ; help!
  716.     pop    ax
  717. ;    call    far ptr restore_active_ovly    ; help might've clobbered ovly
  718.     sub    ax,ax
  719.     jmp    short keypressedx
  720. keypressed2:
  721.     cmp    ax,9                ; TAB key hit?
  722.     je    keypressed3            ;  yup.  TAB display.
  723.     cmp    ax,1148                ; CTL_TAB key hit?
  724.     je    keypressed3            ;  yup.  TAB display.
  725.     jmp    keypressedx            ;  nope.  no TAB display.
  726. keypressed3:
  727.     cmp    tabmode,0            ; tab enabled?
  728.     je    keypressedx            ;  nope
  729.     mov    keybuffer,0            ; say no key hit
  730.     call    far ptr tab_display        ; show the TAB status
  731. ;    call    far ptr restore_active_ovly    ; tab might've clobbered ovly
  732.     sub    ax,ax
  733. keypressedx:
  734.     UNFRAME <es,si,di>            ; pop frame
  735.     ret
  736. keypressed    endp
  737.  
  738. getakeynohelp proc
  739. gknhloop:
  740.     call    far ptr getakey         ; get keystroke
  741.     cmp    ax,1059             ; help key?
  742.     je    gknhloop            ;  ignore help, none available
  743.     ret
  744. getakeynohelp endp
  745.  
  746. getakey proc
  747.         cmp     soundflag,1                     ; is the sound on?
  748.         jl      getakeyloop                     ; ok, sound is off
  749.         call    far ptr nosnd                   ; turn off sound
  750. getakeyloop:
  751.     call    far ptr getkeynowait        ; check for keystroke
  752.     jnc    getakeyloop            ;  no key, loop till we get one
  753.     ret
  754. getakey endp
  755.  
  756. getkeynowait proc
  757.     FRAME    <di,si,es>            ; std frame, for TC++ overlays
  758. getkeyn0:
  759.     cmp    keybuffer,0            ; got a key buffered?
  760.     je    getkeynobuf            ;  nope
  761.     mov    ax,keybuffer            ; key was buffered here
  762.     mov    keybuffer,0            ; clear buffer
  763.     jmp    getkeyyup            ; exit with the key
  764. getkeynobuf:
  765.     call    mouseread            ; mouse activity or savetime?
  766.     jc    getkeyn4            ;  yup, ax holds the phoney key
  767.     mov    ah,kbd_type            ; get the keyboard type
  768.     or    ah,1                ; check if a key is ready
  769.     int    16h                ; now check a key
  770.     jnz    gotkeyn             ;  got one
  771.     cmp    slides,1            ; slideshow playback active?
  772.     jne    getkeynope            ;  nope, return no key
  773.     call    far ptr slideshw        ; check next playback keystroke
  774.     cmp    ax,0                ; got one?
  775.     jne    getkeyn5            ;  yup, use it
  776. getkeynope:
  777.     clc                    ; return no key
  778.     UNFRAME <es,si,di>            ; pop frame
  779.     ret
  780. gotkeyn:                    ; got a real keyboard keystroke
  781.     mov    ah,kbd_type            ; get the keyboard type
  782.     int    16h                ; now get a key
  783.     cmp    al,0e0h             ; check: Enhanced Keyboard key?
  784.     jne    short getkeyn1            ; nope.  proceed
  785.     cmp    ah,0                ; part 2 of Enhanced Key check
  786.     je    short getkeyn1            ; failed.  normal key.
  787.     mov    al,0                ; Turn enhanced key "normal"
  788.     jmp    short getkeyn2            ; jump to common code
  789. getkeyn1:
  790.     cmp    ah,0e0h             ; check again:    Enhanced Key?
  791.     jne    short getkeyn2            ;  nope.  proceed.
  792.     mov    ah,al                ; Turn Enhanced key "normal"
  793.     mov    al,0                ;  ...
  794. getkeyn2:
  795.     cmp    al,0                ; Function Key?
  796.     jne    short getkeyn3            ;  nope.  proceed.
  797.     mov    al,ah                ; klooge into ASCII Key
  798.     mov    ah,0                ; clobber the scan code
  799.     add    ax,1000             ;  + 1000
  800.     jmp    short getkeyn4            ; go to common return
  801. getkeyn3:
  802.     mov    ah,0                ; clobber the scan code
  803. getkeyn4:                    ; got real key (not playback)
  804.     cmp    ax,9999             ; savetime from mousread?
  805.     je    getkeyn6            ;  yup, do it and don't record
  806.     cmp    slides,1            ; slideshow playback active?
  807.     jne    getkeyn5            ;  nope
  808.     cmp    ax,1bh                ; escape?
  809.     jne    getkeyn0            ;  nope, ignore the key
  810.     call    far ptr stopslideshow        ; terminate playback
  811.     jmp    short getkeyn0            ; go check for another key
  812. getkeyn5:
  813.     cmp    slides,2            ; slideshow record mode?
  814.     jne    getkeyn6            ;  nope
  815.     push    ax
  816.     call    far ptr recordshw        ; record the key
  817.     pop    ax
  818. getkeyn6:
  819.     cmp    debugflag,3000            ; color play enabled?
  820.     jne    getkeyyup            ;  nope
  821.     cmp    ax,'~'                          ; color play requested?
  822.     jne    getkeyyup            ;  nope
  823.     call    far ptr edit_text_colors    ; play
  824. ;    call    far ptr restore_active_ovly    ; might've clobbered ovly
  825.     jmp    getkeyn0            ; done playing, back around
  826. getkeyyup:
  827.     stc                    ; indicate we have a key
  828.     UNFRAME <es,si,di>            ; pop frame
  829.     ret
  830. getkeynowait endp
  831.  
  832. ; ****************** Function buzzer(int buzzertype) *******************
  833. ;
  834. ;    Sound a tone based on the value of the parameter
  835. ;
  836. ;    0 = normal completion of task
  837. ;    1 = interrupted task
  838. ;    2 = error contition
  839.  
  840. ;    "buzzer()" codes:  strings of two-word pairs
  841. ;        (frequency in cycles/sec, delay in milliseconds)
  842. ;        frequency == 0 means no sound
  843. ;        delay      == 0 means end-of-tune
  844. buzzer0     dw    1047,100    ; "normal" completion
  845.         dw    1109,100
  846.         dw    1175,100
  847.         dw    0,0
  848. buzzer1     dw    2093,100    ; "interrupted" completion
  849.         dw    1976,100
  850.         dw    1857,100
  851.         dw    0,0
  852. buzzer2     dw    40,500        ; "error" condition (razzberry)
  853.         dw    0,0
  854.  
  855. ; ***********************************************************************
  856.  
  857. buzzer    proc    uses si, buzzertype:word
  858.     cmp    soundflag,0        ; is the sound supressed?
  859.     je    buzzerreturn        ;  yup.  bail out.
  860.     mov    si, offset buzzer0    ; normal completion frequency
  861.     cmp    buzzertype,0        ; normal completion?
  862.     je    buzzerdoit        ; do it
  863.     mov    si,offset buzzer1    ; interrupted task frequency
  864.     cmp    buzzertype,1        ; interrupted task?
  865.     je    buzzerdoit        ; do it
  866.     mov    si,offset buzzer2    ; error condition frequency
  867. buzzerdoit:
  868.     mov    ax,cs:0[si]        ; get the (next) frequency
  869.     mov    bx,cs:2[si]        ; get the (next) delay
  870.     add    si,4            ; get ready for the next tone
  871.     cmp    bx,0            ; are we done?
  872.     je    buzzerreturn        ;  yup.
  873.     push    bx            ; put delay time on the stack
  874.     push    ax            ; put tone value on the stack
  875.     call    far ptr tone        ; do it
  876.     pop    ax            ; restore stack
  877.     pop    bx            ; restore stack
  878.     jmp    short buzzerdoit    ; get the next tone
  879. buzzerreturn:
  880.     ret                ; we done
  881. buzzer    endp
  882.  
  883. ; ***************** Function delay(int delaytime) ************************
  884. ;
  885. ;    performs a delay loop for 'delaytime' milliseconds
  886. ;
  887. ; ************************************************************************
  888.  
  889. delayamillisecond    proc    near    ; internal delay-a-millisecond code
  890.     mov    bx,delaycount        ; set up to burn another millisecond
  891. delayamill1:
  892.     mov    cx,delayloop        ; start up the counter
  893. delayamill2:                ;
  894.     loop    delayamill2        ; burn up some time
  895.     dec    bx            ; have we burned up a millisecond?
  896.     jnz    delayamill1        ;  nope.  try again.
  897.     ret                ; we done
  898. delayamillisecond    endp
  899.  
  900. delay    proc    uses es, delaytime:word ; delay loop (arg in milliseconds)
  901.     mov    ax,delaytime        ; get the number of milliseconds
  902.     cmp    ax,0            ; any delay time at all?
  903.     je    delayreturn        ;  nope.
  904. delayloop1:
  905.     call    delayamillisecond    ; burn up a millisecond of time
  906.     dec    ax            ; have we burned up enough m-seconds?
  907.     jnz    delayloop1        ;  nope.  try again.
  908. delayreturn:
  909.     ret                ; we done.
  910. delay    endp
  911.  
  912. ; ************** Function tone(int frequency,int delaytime) **************
  913. ;
  914. ;    buzzes the speaker with this frequency for this amount of time
  915. ;
  916. ; ************************************************************************
  917.  
  918. tone    proc    uses es, tonefrequency:word, tonedelay:word
  919.     mov    al,0b6h         ; latch to channel 2
  920.     out    43h,al            ;  ...
  921.     cmp    tonefrequency,12h    ; was there a frequency?
  922.     jbe    tonebypass        ;  nope.  delay only
  923.     mov    bx,tonefrequency    ; get the frequency value
  924.     mov    ax,0            ; ugly klooge: convert this to the
  925.     mov    dx,12h            ; divisor the 8253 wants to see
  926.     div    bx            ;  ...
  927.     out    42h,al            ; send the low value
  928.     mov    al,ah            ; then the high value
  929.     out    42h,al            ;  ...
  930.     in    al,61h            ; get the current 8255 bits
  931.     or    al,3            ; turn bits 0 and 1 on
  932.     out    61h,al            ;  ...
  933. tonebypass:
  934.     mov    ax,tonedelay        ; get the delay value
  935.     push    ax            ; set the parameter
  936.     call    far ptr delay        ; and force a delay
  937.     pop    ax            ; restore the parameter
  938.  
  939.     in    al,61h            ; get the current 8255 bits
  940.     and    al,11111100b        ; turn bits 0 and 1 off
  941.     out    61h,al
  942.  
  943.     ret                ; we done
  944. tone    endp
  945.  
  946. ; ************** Function snd(int hertz) and nosnd() **************
  947. ;
  948. ;    turn the speaker on with this frequency (snd) or off (nosnd)
  949. ;
  950. ; *****************************************************************
  951.  
  952. snd    proc    hertz:word        ;Sound the speaker
  953.     cmp    hertz, 20
  954.     jle    hertzbad
  955.     cmp    hertz, 5000
  956.     jge    hertzbad
  957.     mov    ax,0            ;Convert hertz
  958.     mov    dx, 12h         ;for use by routine
  959.     div    hertz
  960.     mov    bx, ax
  961.     mov    al,10110110b        ;Put magic number
  962.     out    43h, al         ;into timer2
  963.     mov    ax, bx            ;Pitch into AX
  964.     out    42h, al         ;LSB into timer2
  965.     mov    al, ah            ;MSB to AL then
  966.     out    42h, al         ;to timer2
  967.     in    al, 61h         ;read I/O port B into AL
  968.     or    al,3            ;turn on bits 0 and 1
  969.     out    61h,al            ;to turn on speaker
  970. hertzbad:
  971.     ret
  972. snd        endp
  973.  
  974. nosnd        proc                ;Turn off speaker
  975.     in    al, 61h         ;Read I/O port B into AL
  976.     and    al, 11111100b        ;mask lower two bits
  977.     out    61h, al         ;to turn off speaker
  978.     ret
  979. nosnd    endp
  980.  
  981.  
  982. ; ****************** Function initasmvars() *****************************
  983.  
  984. initasmvars    proc    uses es si di
  985.  
  986.      cmp    cpu,0            ; have we been called yet:
  987.      je    initasmvarsgo        ;  nope.  proceed.
  988.      jmp    initreturn        ;  yup.  no need to be here.
  989.  
  990. initasmvarsgo:
  991.     mov    ax,ds            ; save the data segment
  992.     mov    _dataseg_xx,ax        ;  for the C code
  993.  
  994.     mov    overflow,0        ; indicate no overflows so far
  995.  
  996.     mov    dx,1            ; ask for 96K of far space
  997.     mov    ax,8000h        ;  ...
  998.     push    dx            ;  ...
  999.     push    ax            ;  ...
  1000.     call    far ptr farmemalloc    ; use the assembler routine to do it
  1001.     pop    ax            ; restore the stack
  1002.     pop    ax            ;  ...
  1003.     mov    extraseg,dx        ; save the results here.
  1004.  
  1005.     call    adapter_init        ; call the video adapter init
  1006.  
  1007.                        ; first see if a mouse is installed
  1008.  
  1009.     push    es            ; (no, first check to ensure that
  1010.     mov    ax,0            ; int 33h doesn't point to 0:0)
  1011.     mov    es,ax            ; ...
  1012.     mov    ax,es:0cch        ; ...
  1013.     pop    es            ; ...
  1014.     cmp    ax,0            ; does int 33h have a non-zero value?
  1015.     je    noint33         ;  nope.  then there's no mouse.
  1016.  
  1017.      xor    ax,ax               ; function for mouse check
  1018.      int    33h               ; call mouse driver
  1019. noint33:
  1020.      mov    mouse,al           ; al holds info about mouse
  1021.  
  1022.                        ; now get the information about the kbd
  1023.      push    es               ; save ES for a tad
  1024.      mov    ax,40h               ; reload ES with BIOS data seg
  1025.      mov    es,ax               ;  ...
  1026.      mov    ah,es:96h           ; get the keyboard byte
  1027.      pop    es               ; restore ES
  1028.      and    ah,10h               ; isolate the Enhanced KBD bit
  1029.      mov    kbd_type,ah           ; and save it
  1030.  
  1031.     call    far ptr cputype     ; what kind of CPU do we have here?
  1032.     cmp    ax,0            ; protected mode of some sort?
  1033.     jge    positive        ;  nope.  proceed.
  1034.     neg    ax            ;  yup.  flip the sign.
  1035. positive:
  1036.     mov    cpu,ax            ; save the cpu type.
  1037. itsa386:
  1038.     cmp    debugflag,8088        ; say, should we pretend it's an 8088?
  1039.     jne    nodebug         ;  nope.
  1040.     mov    cpu,86            ; yup.    use 16-bit emulation.
  1041. nodebug:
  1042.     call far ptr fputype        ; what kind of an FPU do we have?
  1043.     mov    fpu,ax            ;  save the results
  1044.  
  1045.     push    es            ; save ES for a tad
  1046.     mov    ax,0            ; reset ES to BIOS data area
  1047.     mov    es,ax            ;  ...
  1048.     mov    dx,es:46ch        ; obtain the current timer value
  1049.     cmp    cpu,386            ; are we on a 386 or above?
  1050.     jb    delaystartuploop    ;  nope.  don't adjust anything
  1051.     mov    delayloop, 256        ;  yup.  slow down the timer loop
  1052. delaystartuploop:
  1053.     cmp    dx,es:46ch        ; has the timer value changed?
  1054.     je    delaystartuploop    ;  nope.  check again.
  1055.     mov    dx,es:46ch        ; obtain the current timer value again
  1056.     mov    ax,0            ; clear the delay counter
  1057.     mov    delaycount,55        ; 55 millisecs = 1/18.2 secs
  1058. delaytestloop:
  1059.     call    delayamillisecond    ; burn up a (fake) millisecond
  1060.     inc    ax            ; indicate another loop has passed
  1061.     cmp    dx,es:46ch        ; has the timer value changed?
  1062.     je    delaytestloop        ; nope.  burn up some more time.
  1063.     mov    delaycount,ax        ; save the results here
  1064.     pop    es            ; restore ES again
  1065.  
  1066. initreturn:
  1067.      ret                   ; return to caller
  1068. initasmvars endp
  1069.  
  1070.  
  1071. ; New (Apr '90) mouse code by Pieter Branderhorst follows.
  1072. ; The variable lookatmouse controls it all.  Callers of keypressed and
  1073. ; getakey should set lookatmouse to:
  1074. ;      0  ignore the mouse entirely
  1075. ;     <0  only test for left button click; if it occurs return fake key
  1076. ;        number 0-lookatmouse
  1077. ;      1  return enter key for left button, arrow keys for mouse movement,
  1078. ;        mouse sensitivity is suitable for graphics cursor
  1079. ;      2  same as 1 but sensitivity is suitable for text cursor
  1080. ;      3  specials for zoombox, left/right double-clicks generate fake
  1081. ;        keys, mouse movement generates a variety of fake keys
  1082. ;        depending on state of buttons
  1083. ; Mouse movement is accumulated & saved across calls.  Eg if mouse has been
  1084. ; moved up-right quickly, the next few calls to getakey might return:
  1085. ;      right,right,up,right,up
  1086. ; Minor jiggling of the mouse generates no keystroke, and is forgotten (not
  1087. ; accumulated with additional movement) if no additional movement in the
  1088. ; same direction occurs within a short interval.
  1089. ; Movements on angles near horiz/vert are treated as horiz/vert; the nearness
  1090. ; tolerated varies depending on mode.
  1091. ; Any movement not picked up by calling routine within a short time of mouse
  1092. ; stopping is forgotten.  (This does not apply to button pushes in modes<3.)
  1093. ; Mouseread would be more accurate if interrupt-driven, but with the usage
  1094. ; in fractint (tight getakey loops while mouse active) there's no need.
  1095.  
  1096. ; translate table for mouse movement -> fake keys
  1097. mousefkey dw   1077,1075,1080,1072  ; right,left,down,up     just movement
  1098.     dw      0,   0,1081,1073  ;         ,pgdn,pgup  + left button
  1099.     dw    1144,1142,1147,1146  ; kpad+,kpad-,cdel,cins  + rt   button
  1100.     dw    1117,1119,1118,1132  ; ctl-end,home,pgdn,pgup + mid/multi
  1101.  
  1102. DclickTime    equ 9   ; ticks within which 2nd click must occur
  1103. JitterTime    equ 6   ; idle ticks before turfing unreported mickeys
  1104. TextHSens     equ 22  ; horizontal sensitivity in text mode
  1105. TextVSens     equ 44  ; vertical sensitivity in text mode
  1106. GraphSens     equ 5   ; sensitivity in graphics mode; gets lower @ higher res
  1107. ZoomSens      equ 20  ; sensitivity for zoom box sizing/rotation
  1108. TextVHLimit   equ 6   ; treat angles < 1:6  as straight
  1109. GraphVHLimit  equ 14  ; treat angles < 1:14 as straight
  1110. ZoomVHLimit   equ 1   ; treat angles < 1:1  as straight
  1111. JitterMickeys equ 3   ; mickeys to ignore before noticing motion
  1112.  
  1113. mouseread proc near USES bx cx dx
  1114.     local    moveaxis:word
  1115.  
  1116.     ; check if it is time to do an autosave
  1117.     cmp    saveticks,0    ; autosave timer running?
  1118.     je    mouse0        ;  nope
  1119.     sub    ax,ax        ; reset ES to BIOS data area
  1120.     mov    es,ax        ;  see notes at mouse1 in similar code
  1121. tickread:
  1122.     mov    ax,es:046ch    ; obtain the current timer value
  1123.     cmp    ax,savechktime    ; a new clock tick since last check?
  1124.     je    mouse0        ;  nope, save a dozen opcodes or so
  1125.     mov    dx,es:046eh    ; high word of ticker
  1126.     cmp    ax,es:046ch    ; did a tick get counted just as we looked?
  1127.     jne    tickread    ; yep, reread both words to be safe
  1128.     mov    savechktime,ax
  1129.     sub    ax,savebase    ; calculate ticks since timer started
  1130.     sbb    dx,savebase+2
  1131.     jns    tickcompare
  1132.     add    ax,0b0h     ; wrapped past midnight, add a day
  1133.     adc    dx,018h
  1134. tickcompare:
  1135.     cmp    dx,saveticks+2    ; check if past autosave time
  1136.     jb    mouse0
  1137.     ja    ticksavetime
  1138.     cmp    ax,saveticks
  1139.     jb    mouse0
  1140. ticksavetime:            ; it is time to do a save
  1141.     mov    ax,finishrow
  1142.     cmp    ax,-1        ; waiting for the end of a row before save?
  1143.     jne    tickcheckrow    ;  yup, go check row
  1144.     cmp    calc_status,1    ; safety check, calc active?
  1145.     jne    tickdosave    ;  nope, don't check type of calc
  1146.     cmp    got_status,0    ; 1pass or 2pass?
  1147.     je    ticknoterow    ;  yup
  1148.     cmp    got_status,1    ; solid guessing?
  1149.     jne    tickdosave    ;  not 1pass, 2pass, ssg, so save immediately
  1150. ticknoterow:
  1151.     mov    ax,currow    ; note the current row
  1152.     mov    finishrow,ax    ;  ...
  1153.     jmp    short mouse0    ; and keep working for now
  1154. tickcheckrow:
  1155.     cmp    ax,currow    ; started a new row since timer went off?
  1156.     je    mouse0        ;  nope, don't do the save yet
  1157. tickdosave:
  1158.     mov    timedsave,1    ; tell mainline what's up
  1159.     mov    ax,9999     ; a dummy key value, never gets used
  1160.     jmp    mouseret
  1161.  
  1162. mouse0: ; now the mouse stuff
  1163.     cmp    mouse,-1
  1164.     jne    mouseidle    ; no mouse, that was easy
  1165.     mov    ax,lookatmouse
  1166.     cmp    ax,prevlamouse
  1167.     je    mouse1
  1168.  
  1169.     ; lookatmouse changed, reset everything
  1170.     mov    prevlamouse,ax
  1171.     mov    mbclicks,0
  1172.     mov    mbstatus,0
  1173.     mov    mhmickeys,0
  1174.     mov    mvmickeys,0
  1175.     ; note: don't use int 33 func 0 nor 21 to reset, they're SLOW
  1176.     mov    ax,06h        ; reset button counts by reading them
  1177.     mov    bx,0
  1178.     int    33h
  1179.     mov    ax,06h
  1180.     mov    bx,1
  1181.     int    33h
  1182.     mov    ax,05h
  1183.     mov    bx,0
  1184.     int    33h
  1185.     mov    ax,0Bh        ; reset motion counters by reading
  1186.     int    33h
  1187.     mov    ax,lookatmouse
  1188.  
  1189. mouse1: or    ax,ax
  1190.     jz    mouseidle    ; check nothing when lookatmouse=0
  1191.     ; following code directly accesses bios tick counter; it would be
  1192.     ; better not to rely on addr (use int 1A instead) but old PCs don't
  1193.     ; have the required int, the addr is constant in bios to date, and
  1194.     ; fractint startup already counts on it, so:
  1195.     mov    ax,0        ; reset ES to BIOS data area
  1196.     mov    es,ax        ;  ...
  1197.     mov    dx,es:46ch    ; obtain the current timer value
  1198.     cmp    dx,mousetime
  1199.     ; if timer same as last call, skip int 33s:  reduces expense and gives
  1200.     ; caller a chance to read all pending stuff and paint something
  1201.     jne    mnewtick
  1202.     cmp    lookatmouse,0    ; interested in anything other than left button?
  1203.     jl    mouseidle    ; nope, done
  1204.     jmp    mouse5
  1205.  
  1206. mouseidle:
  1207.     clc            ; tell caller no mouse activity this time
  1208.     ret
  1209.  
  1210. mnewtick: ; new tick, read buttons and motion
  1211.     mov    mousetime,dx    ; note current timer
  1212.     cmp    lookatmouse,3
  1213.     je    mouse2        ; skip button press if mode 3
  1214.  
  1215.     ; check press of left button
  1216.     mov    ax,05h        ; get button press info
  1217.     mov    bx,0        ; for left button
  1218.     int    33h
  1219.     or    bx,bx
  1220.     jnz    mleftb
  1221.     cmp    lookatmouse,0
  1222.     jl    mouseidle    ; exit if nothing but left button matters
  1223.     jmp    mouse3        ; not mode 3 so skip past button release stuff
  1224. mleftb: mov    ax,13
  1225.     cmp    lookatmouse,0
  1226.     jg    mouser        ; return fake key enter
  1227.     mov    ax,lookatmouse    ; return fake key 0-lookatmouse
  1228.     neg    ax
  1229. mouser: jmp    mouseret
  1230.  
  1231. mouse2: ; mode 3, check for double clicks
  1232.     mov    ax,06h        ; get button release info
  1233.     mov    bx,0        ; left button
  1234.     int    33h
  1235.     mov    dx,mousetime
  1236.     cmp    bx,1        ; left button released?
  1237.     jl    msnolb        ; go check timer if not
  1238.     jg    mslbgo        ; double click
  1239.     test    mbclicks,1    ; had a 1st click already?
  1240.     jnz    mslbgo        ; yup, double click
  1241.     mov    mlbtimer,dx    ; note time of 1st click
  1242.     or    mbclicks,1
  1243.     jmp    short mousrb
  1244. mslbgo: and    mbclicks,0ffh-1
  1245.     mov    ax,13        ; fake key enter
  1246.     jmp    mouseret
  1247. msnolb: sub    dx,mlbtimer    ; clear 1st click if its been too long
  1248.     cmp    dx,DclickTime
  1249.     jb    mousrb
  1250.     and    mbclicks,0ffh-1 ; forget 1st click if any
  1251.     ; next all the same code for right button
  1252. mousrb: mov    ax,06h        ; get button release info
  1253.     mov    bx,1        ; right button
  1254.     int    33h
  1255.     ; now much the same as for left
  1256.     mov    dx,mousetime
  1257.     cmp    bx,1
  1258.     jl    msnorb
  1259.     jg    msrbgo
  1260.     test    mbclicks,2
  1261.     jnz    msrbgo
  1262.     mov    mrbtimer,dx
  1263.     or    mbclicks,2
  1264.     jmp    short mouse3
  1265. msrbgo: and    mbclicks,0ffh-2
  1266.     mov    ax,1010     ; fake key ctl-enter
  1267.     jmp    mouseret
  1268. msnorb: sub    dx,mrbtimer
  1269.     cmp    dx,DclickTime
  1270.     jb    mouse3
  1271.     and    mbclicks,0ffh-2
  1272.  
  1273.     ; get buttons state, if any changed reset mickey counters
  1274. mouse3: mov    ax,03h        ; get button status
  1275.     int    33h
  1276.     and    bl,7        ; just the button bits
  1277.     cmp    bl,mbstatus    ; any changed?
  1278.     je    mouse4
  1279.     mov    mbstatus,bl    ; yup, reset stuff
  1280.     mov    mhmickeys,0
  1281.     mov    mvmickeys,0
  1282.     mov    ax,0Bh
  1283.     int    33h        ; reset driver's mickeys by reading them
  1284.  
  1285.     ; get motion counters, forget any jiggle
  1286. mouse4: mov    ax,0Bh        ; get motion counters
  1287.     int    33h
  1288.     mov    bx,mousetime    ; just to have it in a register
  1289.     cmp    cx,0        ; horiz motion?
  1290.     jne    moushm        ; yup, go accum it
  1291.     mov    ax,bx
  1292.     sub    ax,mhtimer
  1293.     cmp    ax,JitterTime    ; timeout since last horiz motion?
  1294.     jb    mousev
  1295.     mov    mhmickeys,0
  1296.     jmp    short mousev
  1297. moushm: mov    mhtimer,bx    ; note time of latest motion
  1298.     add    mhmickeys,cx
  1299.     ; same as above for vertical movement:
  1300. mousev: cmp    dx,0        ; vert motion?
  1301.     jne    mousvm
  1302.     mov    ax,bx
  1303.     sub    ax,mvtimer
  1304.     cmp    ax,JitterTime
  1305.     jb    mouse5
  1306.     mov    mvmickeys,0
  1307.     jmp    short mouse5
  1308. mousvm: mov    mvtimer,bx
  1309.     add    mvmickeys,dx
  1310.  
  1311.     ; pick the axis with largest pending movement
  1312. mouse5: mov    bx,mhmickeys
  1313.     or    bx,bx
  1314.     jns    mchkv
  1315.     neg    bx        ; make it +ve
  1316. mchkv:    mov    cx,mvmickeys
  1317.     or    cx,cx
  1318.     jns    mchkmx
  1319.     neg    cx
  1320. mchkmx: mov    moveaxis,0    ; flag that we're going horiz
  1321.     cmp    bx,cx        ; horiz>=vert?
  1322.     jge    mangle
  1323.     xchg    bx,cx        ; nope, use vert
  1324.     mov    moveaxis,1    ; flag that we're going vert
  1325.  
  1326.     ; if moving nearly horiz/vert, make it exactly horiz/vert
  1327. mangle: mov    ax,TextVHLimit
  1328.     cmp    lookatmouse,2    ; slow (text) mode?
  1329.     je    mangl2
  1330.     mov    ax,GraphVHLimit
  1331.     cmp    lookatmouse,3    ; special mode?
  1332.     jne    mangl2
  1333.     cmp    mbstatus,0    ; yup, any buttons down?
  1334.     je    mangl2
  1335.     mov    ax,ZoomVHLimit    ; yup, special zoom functions
  1336. mangl2: mul    cx        ; smaller axis * limit
  1337.     cmp    ax,bx
  1338.     ja    mchkmv        ; min*ratio <= max?
  1339.     cmp    moveaxis,0    ; yup, clear the smaller movement axis
  1340.     jne    mzeroh
  1341.     mov    mvmickeys,0
  1342.     jmp    short mchkmv
  1343. mzeroh: mov    mhmickeys,0
  1344.  
  1345.     ; pick sensitivity to use
  1346. mchkmv: cmp    lookatmouse,2    ; slow (text) mode?
  1347.     je    mchkmt
  1348.     mov    dx,ZoomSens+JitterMickeys
  1349.     cmp    lookatmouse,3    ; special mode?
  1350.     jne    mchkmg
  1351.     cmp    mbstatus,0    ; yup, any buttons down?
  1352.     jne    mchkm2        ; yup, use zoomsens
  1353. mchkmg: mov    dx,GraphSens
  1354.     mov    cx,sxdots    ; reduce sensitivity for higher res
  1355. mchkg2: cmp    cx,400        ; horiz dots >= 400?
  1356.     jl    mchkg3
  1357.     shr    cx,1        ; horiz/2
  1358.     shr    dx,1
  1359.     inc    dx        ; sensitivity/2+1
  1360.     jmp    short mchkg2
  1361. mchkg3: add    dx,JitterMickeys
  1362.     jmp    short mchkm2
  1363. mchkmt: mov    dx,TextVSens+JitterMickeys
  1364.     cmp    moveaxis,0
  1365.     jne    mchkm2
  1366.     mov    dx,TextHSens+JitterMickeys ; slower on X axis than Y
  1367.  
  1368.     ; is largest movement past threshold?
  1369. mchkm2: cmp    bx,dx
  1370.     jge    mmove
  1371.     jmp    mouseidle    ; no movement past threshold, return nothing
  1372.  
  1373.     ; set bx for right/left/down/up, and reduce the pending mickeys
  1374. mmove:    sub    dx,JitterMickeys
  1375.     cmp    moveaxis,0
  1376.     jne    mmovev
  1377.     cmp    mhmickeys,0
  1378.     jl    mmovh2
  1379.     sub    mhmickeys,dx    ; horiz, right
  1380.     mov    bx,0
  1381.     jmp    short mmoveb
  1382. mmovh2: add    mhmickeys,dx    ; horiz, left
  1383.     mov    bx,2
  1384.     jmp    short mmoveb
  1385. mmovev: cmp    mvmickeys,0
  1386.     jl    mmovv2
  1387.     sub    mvmickeys,dx    ; vert, down
  1388.     mov    bx,4
  1389.     jmp    short mmoveb
  1390. mmovv2: add    mvmickeys,dx    ; vert, up
  1391.     mov    bx,6
  1392.  
  1393.     ; modify bx if a button is being held down
  1394. mmoveb: cmp    lookatmouse,3
  1395.     jne    mmovek        ; only modify in mode 3
  1396.     cmp    mbstatus,1
  1397.     jne    mmovb2
  1398.     add    bx,8        ; modify by left button
  1399.     jmp    short mmovek
  1400. mmovb2: cmp    mbstatus,2
  1401.     jne    mmovb3
  1402.     add    bx,16        ; modify by rb
  1403.     jmp    short mmovek
  1404. mmovb3: cmp    mbstatus,0
  1405.     je    mmovek
  1406.     add    bx,24        ; modify by middle or multiple
  1407.  
  1408.     ; finally, get the fake key number
  1409. mmovek: mov    ax,mousefkey[bx]
  1410.  
  1411. mouseret:
  1412.     stc
  1413.     ret
  1414. mouseread endp
  1415.  
  1416.  
  1417. ; long readticker() returns current bios ticker value
  1418.  
  1419. readticker proc uses es
  1420.     sub    ax,ax        ; reset ES to BIOS data area
  1421.     mov    es,ax        ;  see notes at mouse1 in similar code
  1422. tickread:
  1423.     mov    ax,es:046ch    ; obtain the current timer value
  1424.     mov    dx,es:046eh    ; high word of ticker
  1425.     cmp    ax,es:046ch    ; did a tick get counted just as we looked?
  1426.     jne    tickread    ; yep, reread both words to be safe
  1427.     ret
  1428. readticker endp
  1429.  
  1430.  
  1431. ;===============================================================
  1432. ;
  1433. ; CPUTYPE.ASM : C-callable functions cputype() and ndptype() adapted
  1434. ; by Lee Daniel Crocker from code appearing in the late PC Tech Journal,
  1435. ; August 1987 and November 1987.  PC Tech Journal was a Ziff-Davis
  1436. ; Publication.    Code herein is copyrighted and used with permission.
  1437. ;
  1438. ; The function cputype() returns an integer value based on what kind
  1439. ; of CPU it found, as follows:
  1440. ;
  1441. ;    Value    CPU Type
  1442. ;    =====    ========
  1443. ;    86    8086, 8088, V20, or V30
  1444. ;    186    80186 or 80188
  1445. ;    286    80286
  1446. ;    386    80386 or 80386sx
  1447. ;    -286    80286 in protected mode
  1448. ;    -386    80386 or 80386sx in protected or 32-bit address mode
  1449. ;
  1450. ; The function ndptype() returns an integer based on the type of NDP
  1451. ; it found, as follows:
  1452. ;
  1453. ;    Value    NDP Type
  1454. ;    =====    ========
  1455. ;    0    No NDP found
  1456. ;    87    8087
  1457. ;    287    80287
  1458. ;    387    80387
  1459. ;
  1460. ; No provisions are made for the 80486 CPU/FPU or Weitek FPA chips.
  1461. ;
  1462. ; Neither function takes any arguments or affects any external storage,
  1463. ; so there should be no memory-model dependencies.
  1464.  
  1465. .286P
  1466. .code
  1467.  
  1468. cputype proc
  1469.     push    bp
  1470.  
  1471.     push    sp            ; 86/186 will push SP-2;
  1472.     pop    ax            ; 286/386 will push SP.
  1473.     cmp    ax, sp
  1474.     jz    not86            ; If equal, SP was pushed
  1475.     mov    ax, 186
  1476.     mov    cl, 32            ;   186 uses count mod 32 = 0;
  1477.     shl    ax, cl            ;   86 shifts 32 so ax = 0
  1478.     jnz    exit            ; Non-zero: no shift, so 186
  1479.     mov    ax, 86            ; Zero: shifted out all bits
  1480.     jmp    short exit
  1481. not86:
  1482.     pushf                ; Test 16 or 32 operand size:
  1483.     mov    ax, sp            ;   Pushed 2 or 4 bytes of flags?
  1484.     popf
  1485.     inc    ax
  1486.     inc    ax
  1487.     cmp    ax, sp            ;   Did pushf change SP by 2?
  1488.     jnz    is32bit         ;   If not, then 4 bytes of flags
  1489. is16bit:
  1490.     sub    sp, 6            ; Is it 286 or 386 in 16-bit mode?
  1491.     mov    bp, sp            ; Allocate stack space for GDT pointer
  1492.     sgdt    fword ptr [bp]
  1493.     add    sp, 4            ; Discard 2 words of GDT pointer
  1494.     pop    ax            ; Get third word
  1495.     inc    ah            ; 286 stores -1, 386 stores 0 or 1
  1496.     jnz    is386
  1497. is286:
  1498.     mov    ax, 286
  1499.     jmp    short testprot        ; Check for protected mode
  1500. is32bit:
  1501.     db    66h            ; 16-bit override in 32-bit mode
  1502. is386:
  1503.     mov    ax, 386
  1504. testprot:
  1505.     smsw    cx            ; Protected?  Machine status -> CX
  1506.     ror    cx,1            ; Protection bit -> carry flag
  1507.     jnc    exit            ; Real mode if no carry
  1508.     neg    ax            ; Protected:  return neg value
  1509. exit:
  1510.     pop    bp
  1511.     ret
  1512. cputype endp
  1513.  
  1514. .data
  1515.  
  1516. control dw    0            ; Temp storage for 8087 control
  1517.                     ;   and status registers
  1518. .code
  1519.  
  1520. fputype proc
  1521.     push    bp
  1522.  
  1523.     fninit                ; Defaults to 64-bit mantissa
  1524.     mov    byte ptr control+1, 0
  1525.     fnstcw    control         ; Store control word over 0
  1526. ;    dw    3ed9h            ; (klooge to avoid the MASM \e switch)
  1527. ;    dw    offset control        ; ((equates to the above 'fnstcw' cmd))
  1528.     mov    ah, byte ptr control+1    ; Test contents of byte written
  1529.     cmp    ah, 03h         ; Test for 64-bit precision flags
  1530.     je    gotone            ; Got one!  Now let's find which
  1531.     xor    ax, ax
  1532.     jmp    short fexit        ; No NDP found
  1533. gotone:
  1534.     and    control, not 0080h    ; IEM = 0 (interrupts on)
  1535.     fldcw    control
  1536.     fdisi                ; Disable ints; 287/387 will ignore
  1537.     fstcw    control
  1538.     test    control, 0080h
  1539.     jz    not87            ; Got 287/387; keep testing
  1540.     mov    ax, 87
  1541.     jmp    short freset
  1542. not87:
  1543.     finit
  1544.     fld1
  1545.     fldz
  1546.     fdiv                ; Divide 1/0 to create infinity
  1547.     fld    st
  1548.     fchs                ; Push -infinity on stack
  1549.     fcompp                ; Compare +-infinity
  1550.     fstsw    control
  1551.     mov    ax, control
  1552.     sahf
  1553.     jnz    got387            ; 387 will compare correctly
  1554.     mov    ax, 287
  1555.     jmp    short freset
  1556. got387:                 ; Only one left (until 487/Weitek
  1557.     mov    ax, 387         ;   test is added)
  1558. freset:
  1559.     fninit                ; in case tests have had strange
  1560.     finit                ; side-effects, reset
  1561. fexit:
  1562.     pop    bp
  1563.     ret
  1564. fputype endp
  1565.  
  1566. ; ************************* Far Segment RAM Support **************************
  1567. ;
  1568. ;
  1569. ;    farptr = (char far *)farmemalloc(long bytestoalloc);
  1570. ;    (void)farmemfree(farptr);
  1571. ;
  1572. ;    alternatives to Microsoft/TurboC routines
  1573. ;
  1574. ;
  1575. .8086
  1576.  
  1577. farmemalloc    proc    uses es, bytestoallocate:dword
  1578.     les    bx,bytestoallocate    ; get the # of bytes into DX:BX
  1579.     mov    dx,es            ;  ...
  1580.     add    bx,15            ; round up to next paragraph boundary
  1581.     adc    dx,0            ;  ...
  1582.     shr    dx,1            ; convert to paragraphs
  1583.     rcr    bx,1            ;  ...
  1584.     shr    dx,1            ;  ...
  1585.     rcr    bx,1            ;  ...
  1586.     shr    dx,1            ;  ...
  1587.     rcr    bx,1            ;  ...
  1588.     shr    dx,1            ;  ...
  1589.     rcr    bx,1            ;  ...
  1590.     cmp    dx,0            ; ensure that we don't want > 1MB
  1591.     jne    farmemallocfailed    ;  bail out if we do
  1592.     mov    ah,48h            ; invoke DOS to allocate memory
  1593.     int    21h            ;  ...
  1594.     jc    farmemallocfailed    ; bail out on failure
  1595.     mov    dx,ax            ; set up DX:AX as far address
  1596.     mov    ax,0            ;  ...
  1597.     jmp    short farmemallocreturn ; and return
  1598. farmemallocfailed:
  1599.     mov    ax,0            ; (load up with a failed response)
  1600.     mov    dx,0            ;  ...
  1601. farmemallocreturn:
  1602.     ret                ; we done.
  1603. farmemalloc    endp
  1604.  
  1605. farmemfree    proc    uses es, farptr:dword
  1606.     les    ax,farptr        ; get the segment into ES
  1607.     mov    ah,49h            ; invoke DOS to free the segment
  1608.     int    21h            ;  ...
  1609.     ret
  1610. farmemfree    endp
  1611.  
  1612. erasesegment    proc    uses es di si, segaddress:word, segvalue:word
  1613.     mov    ax,segaddress        ; load up the segment address
  1614.     mov    es,ax            ;  ...
  1615.     mov    di,0            ; start at the beginning
  1616.     mov    ax,segvalue        ; use this value
  1617.     mov    cx,8000h        ; over the entire segment
  1618.     rep    stosw            ; do it
  1619.     ret                ; we done
  1620. erasesegment    endp
  1621.  
  1622.  
  1623. farread proc uses ds, handle:word, buf:dword, len:word
  1624.     mov    ah, 03Fh
  1625.     mov    bx, [handle]
  1626.     mov    cx, [len]
  1627.     lds    dx, [buf]
  1628.     int    21h
  1629.     jnc    farreaddone
  1630.     mov    ax, -1
  1631. farreaddone:
  1632.     ret
  1633. farread endp
  1634.  
  1635.  
  1636. farwrite proc uses ds, handle:word, buf:dword, len:word
  1637.     mov    ah, 040h
  1638.     mov    bx, [handle]
  1639.     mov    cx, [len]
  1640.     lds    dx, [buf]
  1641.     int    21h
  1642.     jnc    farwritedone
  1643.     mov    ax, -1
  1644. farwritedone:
  1645.     ret
  1646. farwrite endp
  1647.  
  1648.  
  1649. ; Convert segment:offset to equiv pointer with minimum possible offset
  1650. normalize proc p: dword
  1651. ;    mov    ax, [word ptr p]
  1652. ;    mov    dx, [word ptr p+2]
  1653.     les    ax, p
  1654.     mov    dx, es
  1655.     mov    bx, ax
  1656.     shr    bx, 1
  1657.     shr    bx, 1
  1658.     shr    bx, 1
  1659.     shr    bx, 1
  1660.     and    ax, 0Fh
  1661.     add    dx, bx
  1662.     ret
  1663. normalize endp
  1664.  
  1665.  
  1666. ; *************** Far string/memory functions *********
  1667. ;    far_strlen ( char far *);
  1668. ;    far_strcpy ( char far *, char far *);
  1669. ;    far_strcmp ( char far *, char far *);
  1670. ;    far_stricmp( char far *, char far *);
  1671. ;    far_strnicmp(char far *, char far *, int);
  1672. ;    far_strcat ( char far *, char far *);
  1673. ;    far_memset ( char far *, char far,   int);
  1674. ;    far_memcpy ( char far *, char far *, int);
  1675. ;    far_memcmp ( char far *, char far *, int);
  1676. ;    far_memicmp( char far *, char far *, int);
  1677.  
  1678. ;    xxxfar_routines are called internally with:
  1679. ;        ds:si pointing to the source
  1680. ;        es:di pointing to the destination
  1681. ;        cx    containing a byte count
  1682. ;        al    contining  a character (set) value
  1683. ;        (and they destroy registers willy-nilly)
  1684.  
  1685. xxxfar_memlen    proc    near    ; return string length - INCLUDING the 0
  1686.     mov    ax,0
  1687.     mov    cx,1024
  1688.     repne    scasb
  1689.     sub    cx,1024
  1690.     neg    cx
  1691.     ret
  1692. xxxfar_memlen    endp
  1693.  
  1694. xxxfar_memcmp    proc    near    ; compare two strings - length in CX
  1695.     mov    ax,0
  1696.     rep    cmpsb
  1697.     jz    wedone
  1698.     mov    ax,1
  1699. wedone: ret
  1700. xxxfar_memcmp    endp
  1701.  
  1702. xxxfar_memicmp    proc    near    ; compare two caseless strings - length in CX
  1703.     mov    ax,0
  1704.     cmp    cx,0
  1705.     je    wedone
  1706.     dec    si
  1707.     dec    di
  1708. loop1:    inc    si
  1709.     inc    di
  1710.     mov    al,es:[di]
  1711.     mov    ah,ds:[si]
  1712.     cmp    al,ah
  1713.     je    loop2
  1714.     cmp    al,'A'
  1715.     jb    lower1
  1716.     cmp    al,'Z'
  1717.     ja    lower1
  1718.     add    al,20h
  1719. lower1: cmp    ah,'A'
  1720.     jb    lower2
  1721.     cmp    ah,'Z'
  1722.     ja    lower2
  1723.     add    ah,20h
  1724. lower2: cmp    al,ah
  1725.     jne    uneql
  1726. loop2:    loop    loop1
  1727.     mov    ax,0
  1728.     jmp    short wedone
  1729. uneql:    mov    ax,1
  1730. wedone: ret
  1731. xxxfar_memicmp    endp
  1732.  
  1733.  
  1734. far_strlen proc uses ds es di si, fromaddr:dword
  1735.     les    di,fromaddr        ; point to start-of-string
  1736.     call    xxxfar_memlen        ; find the string length
  1737.     mov    ax,cx            ; return len
  1738.     dec    ax            ; don't count null
  1739.     ret                ; we done.
  1740. far_strlen endp
  1741.  
  1742. far_strnicmp proc    uses ds es di si, toaddr:dword, fromaddr:dword, len:word
  1743.     les    di,fromaddr        ; point to start-of-string
  1744.     call    xxxfar_memlen        ; find the string length
  1745.     cmp    cx,len            ; source less than or equal to len?
  1746.     jle    cxbigger        ; yup - use cx
  1747.     mov    cx,len            ; nope - use len
  1748. cxbigger:
  1749.     les    di,toaddr        ; get the dest string
  1750.     lds    si,fromaddr        ; get the source string
  1751.     call    xxxfar_memicmp        ; compare them
  1752.     ret                ; we done.
  1753. far_strnicmp endp
  1754.  
  1755. far_strcpy proc uses ds es di si, toaddr:dword, fromaddr:dword
  1756.     les    di,fromaddr        ; point to start-of-string
  1757.     call    xxxfar_memlen        ; find the string length
  1758.     les    di,toaddr        ; now move to here
  1759.     lds    si,fromaddr        ; from here
  1760.     rep    movsb            ; move them
  1761.     ret                ; we done.
  1762. far_strcpy endp
  1763.  
  1764. far_strcmp proc uses ds es di si, toaddr:dword, fromaddr:dword
  1765.     les    di,fromaddr        ; point to start-of-string
  1766.     call    xxxfar_memlen        ; find the string length
  1767.     les    di,toaddr        ; now compare to here
  1768.     lds    si,fromaddr        ; compare here
  1769.     call    xxxfar_memcmp        ; compare them
  1770.     ret                ; we done.
  1771. far_strcmp endp
  1772.  
  1773. far_stricmp proc    uses ds es di si, toaddr:dword, fromaddr:dword
  1774.     les    di,fromaddr        ; point to start-of-string
  1775.     call    xxxfar_memlen        ; find the string length
  1776.     les    di,toaddr        ; get the dest string
  1777.     lds    si,fromaddr        ; get the source string
  1778.     call    xxxfar_memicmp        ; compare them
  1779.     ret                ; we done.
  1780. far_stricmp endp
  1781.  
  1782. far_strcat proc uses ds es di si, toaddr:dword, fromaddr:dword
  1783.     les    di,fromaddr        ; point to start-of-string
  1784.     call    xxxfar_memlen        ; find the string length
  1785.     push    cx            ; save it
  1786.     les    di,toaddr        ; point to start-of-string
  1787.     call    xxxfar_memlen        ; find the string length
  1788.     les    di,toaddr        ; now move to here
  1789.     add    di,cx            ; but start at the end of string
  1790.     dec    di            ; (less the EOS zero)
  1791.     lds    si,fromaddr        ; from here
  1792.     pop    cx            ; get the string length
  1793.     rep    movsb            ; move them
  1794.     ret                ; we done.
  1795. far_strcat endp
  1796.  
  1797. far_memset proc uses es di, toaddr:dword, fromvalue:word, slength:word
  1798.     mov    ax,fromvalue        ; get the value to store
  1799.     mov    cx,slength        ; get the store length
  1800.     les    di,toaddr        ; now move to here
  1801.     rep    stosb            ; store them
  1802.     ret                ; we done.
  1803. far_memset endp
  1804.  
  1805. far_memcpy proc uses ds es di si, toaddr:dword, fromaddr:dword, slength:word
  1806.     mov    cx,slength        ; get the move length
  1807.     les    di,toaddr        ; now move to here
  1808.     lds    si,fromaddr        ; from here
  1809.     rep    movsb            ; move them
  1810.     ret                ; we done.
  1811. far_memcpy endp
  1812.  
  1813. far_memcmp proc uses ds es di si, toaddr:dword, fromaddr:dword, slength:word
  1814.     mov    cx,slength        ; get the compare length
  1815.     les    di,toaddr        ; now compare to here
  1816.     lds    si,fromaddr        ; compare here
  1817.     call    xxxfar_memcmp        ; compare them
  1818.     ret                ; we done.
  1819. far_memcmp endp
  1820.  
  1821. far_memicmp proc uses ds es di si, toaddr:dword, fromaddr:dword, slength:word
  1822.     mov    cx,slength        ; get the compare length
  1823.     les    di,toaddr        ; get the dest string
  1824.     lds    si,fromaddr        ; get the source string
  1825.     call    xxxfar_memicmp        ; compare them
  1826.     ret                ; we done.
  1827. far_memicmp endp
  1828.  
  1829. disable proc                ; disable interrupts
  1830.     cli
  1831.     ret
  1832. disable endp
  1833.  
  1834. enable    proc                ; re-enable interrupts
  1835.     sti
  1836.     ret
  1837. enable    endp
  1838.  
  1839. ; *************** Expanded Memory Manager Support Routines ******************
  1840. ;        for use with LIM 3.2 or 4.0 Expanded Memory
  1841. ;
  1842. ;    farptr = emmquery()    ; Query presence of EMM and initialize EMM code
  1843. ;                ; returns EMM FAR Address, or 0 if no EMM
  1844. ;    freepages = emmgetfree(); Returns the number of pages (1 page = 16K)
  1845. ;                ; not already allocated for something else
  1846. ;    handle = emmallocate(pages)    ; allocate EMM pages (1 page = 16K)
  1847. ;                ; returns handle # if OK, or else 0
  1848. ;    emmdeallocate(handle)    ; return EMM pages to system - MUST BE CALLED
  1849. ;                ; or allocated EMM memory fills up
  1850. ;    emmgetpage(page, handle); get an EMM page (actually, links the EMM
  1851. ;                ; page to the EMM Segment ADDR, saving any
  1852. ;                ; prior page in the process)
  1853. ;    emmclearpage(page, handle) ; performs an 'emmgetpage()' and then clears
  1854. ;                ; it out (quickly) to zeroes with a 'REP STOSW'
  1855.  
  1856. .8086
  1857.  
  1858. .DATA
  1859.  
  1860. emm_name    db    'EMMXXXX0',0    ; device driver for EMM
  1861. emm_segment    dw    0        ; EMM page frame segment
  1862. emm_zeroflag    db    0        ; klooge flag for handle==0
  1863.  
  1864. .CODE
  1865.  
  1866. emmquery    proc
  1867.     mov    ah,3dh            ; function 3dh = open file
  1868.     mov    al,0            ;  read only
  1869.     mov    dx,offset emm_name    ; DS:DX = address of name of EMM
  1870.     int    21h            ; open it
  1871.     jc    emmqueryfailed        ;  oops.  no EMM.
  1872.  
  1873.     mov    bx,ax            ; BX = handle for EMM
  1874.     mov    ah,44h            ; function 44h = IOCTL
  1875.     mov    al,7            ; get outo. status
  1876.     mov    cx,0            ; CX = # of bytes to read
  1877.     int    21h            ; do it.
  1878.     push    ax            ; save the IOCTL handle.
  1879.  
  1880.     mov    ah,3eh            ; function 3H = close
  1881.     int    21h            ; BX still cintains handle
  1882.     pop    ax            ; restore AX for the status query
  1883.     jc    emmqueryfailed        ; huh?    close FAILED?
  1884.  
  1885.     or    al,al            ; was the status 0?
  1886.     jz    emmqueryfailed        ; well then, it wasn't EMM!
  1887.  
  1888.     mov    ah,40h            ; query EMM: hardware ok?
  1889.     int    67h            ; EMM call
  1890.     cmp    ah,0            ; is it ok?
  1891.     jne    emmqueryfailed        ; if not, fail
  1892.  
  1893.     mov    ah,41h            ; query EMM: Get Page Frame Segment
  1894.     int    67h            ; EMM call
  1895.     cmp    ah,0            ; is it ok?
  1896.     jne    emmqueryfailed        ; if not, fail
  1897.     mov    emm_segment,bx        ; save page frame segment
  1898.     mov    dx,bx            ; return page frame address
  1899.     mov    ax,0            ;  ...
  1900.     jmp    short    emmqueryreturn    ; we done.
  1901.  
  1902. emmqueryfailed:
  1903.     mov    ax,0            ; return 0 (no EMM found)
  1904.     mov    dx,0            ;  ...
  1905. emmqueryreturn:
  1906.     ret                ; we done.
  1907. emmquery    endp
  1908.  
  1909. emmgetfree    proc            ; get # of free EMM pages
  1910.     mov    ah,42h            ; EMM call: get total and free pages
  1911.     int    67h            ; EMM call
  1912.     cmp    ah,0            ; did we suceed?
  1913.     jne    emmgetfreefailed    ;  nope.  return 0 free pages
  1914.     mov    ax,bx            ; else return # of free pages
  1915.     jmp    emmgetfreereturn    ; we done.
  1916. emmgetfreefailed:
  1917.     mov    ax,0            ; failure mode
  1918. emmgetfreereturn:
  1919.     ret                ; we done
  1920. emmgetfree    endp
  1921.  
  1922. emmallocate    proc    pages:word    ; allocate EMM pages
  1923.     mov    bx,pages        ; BX = # of 16K pages
  1924.     mov    ah,43h            ; ask for the memory
  1925.     int    67h            ; EMM call
  1926.     mov    emm_zeroflag,0        ; clear the klooge flag
  1927.     cmp    ah,0            ; did the call work?
  1928.     jne    emmallocatebad        ;  nope.
  1929.     mov    ax,dx            ; yup.    save the handle here
  1930.     cmp    ax,0            ; was the handle a zero?
  1931.     jne    emmallocatereturn    ;  yup.  no kloogy fixes
  1932.     mov    emm_zeroflag,1        ; oops.  set an internal flag
  1933.     mov    ax,1234         ; and make up a dummy handle.
  1934.     jmp    short    emmallocatereturn ; and return
  1935. emmallocatebad:
  1936.     mov    ax,0            ; indicate no handle
  1937. emmallocatereturn:
  1938.     ret                ; we done.
  1939. emmallocate    endp
  1940.  
  1941. emmdeallocate    proc    emm_handle:word ; De-allocate EMM memory
  1942. emmdeallocatestart:
  1943.     mov    dx,emm_handle        ; get the EMM handle
  1944.     cmp    dx,1234         ; was it our special klooge value?
  1945.     jne    emmdeallocatecontinue    ;  nope.  proceed.
  1946.     cmp    emm_zeroflag,1        ; was it really a zero handle?
  1947.     jne    emmdeallocatecontinue    ;  nope.  proceed.
  1948.     mov    dx,0            ; yup.    use zero instead.
  1949. emmdeallocatecontinue:
  1950.     mov    ah,45h            ; EMM function: deallocate
  1951.     int    67h            ; EMM call
  1952.     cmp    ah,0            ; did it work?
  1953.     jne    emmdeallocatestart    ; well then, try it again!
  1954. emmdeallocatereturn:
  1955.     ret                ; we done
  1956. emmdeallocate    endp
  1957.  
  1958. emmgetpage    proc    pagenum:word, emm_handle:word    ; get EMM page
  1959.     mov    bx,pagenum        ; BX = page numper
  1960.     mov    dx,emm_handle        ; DX = EMM handle
  1961.     cmp    dx,1234         ; was it our special klooge value?
  1962.     jne    emmgetpagecontinue    ;  nope.  proceed.
  1963.     cmp    emm_zeroflag,1        ; was it really a zero handle?
  1964.     jne    emmgetpagecontinue    ;  nope.  proceed.
  1965.     mov    dx,0            ; yup.    use zero instead.
  1966. emmgetpagecontinue:
  1967.     mov    ah,44h            ; EMM call: get page
  1968.     mov    al,0            ; get it into page 0
  1969.     int    67h            ; EMM call
  1970.     ret                ; we done
  1971. emmgetpage    endp
  1972.  
  1973. emmclearpage    proc    pagenum:word, emm_handle:word    ; clear EMM page
  1974.     mov    bx,pagenum        ; BX = page numper
  1975.     mov    dx,emm_handle        ; DX = EMM handle
  1976.     cmp    dx,1234         ; was it our special klooge value?
  1977.     jne    emmclearpagecontinue    ;  nope.  proceed.
  1978.     cmp    emm_zeroflag,1        ; was it really a zero handle?
  1979.     jne    emmclearpagecontinue    ;  nope.  proceed.
  1980.     mov    dx,0            ; yup.    use zero instead.
  1981. emmclearpagecontinue:
  1982.     mov    ah,44h            ; EMM call: get page
  1983.     mov    al,0            ; get it into page 0
  1984.     int    67h            ; EMM call
  1985.     mov    ax,emm_segment        ; get EMM segment into ES
  1986.     push    es            ;  ...
  1987.     mov    es,ax            ;  ...
  1988.     mov    di,0            ; start at offset 0
  1989.     mov    cx,8192         ; for 16K (in words)
  1990.     mov    ax,0            ; clear out EMM segment to zeroes
  1991.     rep    stosw            ; clear the page
  1992.     pop    es            ; restore ES
  1993.     ret                ; we done
  1994. emmclearpage    endp
  1995.  
  1996. ; *************** Extended Memory Manager Support Routines ******************
  1997. ;          for use XMS 2.0 and later Extended Memory
  1998. ;
  1999. ;    xmmquery()        ; Query presence of XMM and initialize XMM code
  2000. ;                ; returns 0 if no XMM
  2001. ;    xmmlongest()        ; return size of largest available
  2002. ;                ; XMM block in Kbytes (or zero if none)
  2003. ;    handle = xmmallocate(Kbytes)    ; allocate XMM block in Kbytes
  2004. ;                    ; returns handle # if OK, or else 0
  2005. ;    xmmreallocate(handle, Kbytes)    ; change size of handle's block
  2006. ;                ; to size of Kbytes.  Returns 0 if failed
  2007. ;    xmmdeallocate(handle)    ; return XMM block to system - MUST BE CALLED
  2008. ;                ; or allocated XMM memory is not released.
  2009. ;    xmmmoveextended(&MoveStruct)    ; Moves a block of memory to or
  2010. ;                ; from extended memory.  Returns 1 if OK
  2011. ;                ; else returns 0.
  2012. ; The structure format for use with xmmoveextended is:
  2013. ;
  2014. ;    ASM             |        C
  2015. ;--------------------------------+----------------------------------------
  2016. ; XMM_Move    struc         |    struct XMM_Move
  2017. ;                 |    {
  2018. ;   Length      dd ?         |        unsigned long   Length;
  2019. ;   SourceHandle  dw ?         |        unsigned int    SourceHandle;
  2020. ;   SourceOffset  dd ?         |        unsigned long   SourceOffset;
  2021. ;   DestHandle      dw ?         |        unsigned int    DestHandle;
  2022. ;   DestOffset      dd ?         |        unsigned long   DestOffset;
  2023. ; XMM_Move    ends         |    };
  2024. ;                 |
  2025. ;
  2026. ; Please refer to XMS spec version 2.0 for further information.
  2027.  
  2028. .data
  2029. xmscontrol    dd dword ptr (0) ; Address of driver's control function
  2030.  
  2031. .code
  2032. xmmquery    proc
  2033.     mov    ax,4300h        ; Is an XMS driver installed?
  2034.     int    2fh
  2035.     cmp    al, 80h         ; Did it succeed?
  2036.     jne    xmmqueryfailed        ; No
  2037.     mov    ax,4310h        ; Get control function address
  2038.     int    2fh
  2039.     mov    word ptr [xmscontrol], bx   ; Put address in xmscontrol
  2040.     mov    word ptr [xmscontrol+2], es ; ...
  2041.     mov    ah,00h            ; Get version number
  2042.     call [xmscontrol]
  2043.     cmp    ax,0200h        ; Is 2.00 or higher?
  2044.     jge  xmmquerydone        ; Yes
  2045. xmmqueryfailed:
  2046.     mov    ax,0            ; return failure
  2047.     mov    dx,0
  2048. xmmquerydone:
  2049.     ret
  2050. xmmquery    endp
  2051.  
  2052.  
  2053. xmmlongest    proc            ; query length of largest avail block
  2054.     mov    ah, 08h            ;
  2055.     call    [xmscontrol]
  2056.     mov    dx, 0
  2057.     ret
  2058. xmmlongest    endp
  2059.  
  2060.  
  2061. xmmallocate    proc    ksize:word
  2062.     mov    ah,09h            ; Allocate extended memory block
  2063.     mov    dx,ksize        ; size of block in Kbytes
  2064.     call    [xmscontrol]
  2065.     cmp    ax,0001h        ; did it succeed?
  2066.     jne    xmmallocatefail     ; nope
  2067.     mov    ax,dx            ; Put handle here
  2068.     jmp    short xmmallocatedone
  2069. xmmallocatefail:
  2070.     mov    ax, 0            ; Indicate failure;
  2071. xmmallocatedone:
  2072.     ret
  2073. xmmallocate    endp
  2074.  
  2075.  
  2076. xmmreallocate    proc    handle:word, newsize:word
  2077.     mov    ah, 0Fh            ; Change size of extended mem block
  2078.     mov    dx, handle        ; handle of block to reallocate
  2079.     mov    bx, newsize             ; new size for block
  2080.     call    [xmscontrol]
  2081.     cmp    ax, 0001h        ; one indicates success
  2082.     je    xmmreallocdone
  2083.     mov    ax, 0            ; we return zero for failure
  2084. xmmreallocdone:
  2085.     ret
  2086. xmmreallocate    endp
  2087.  
  2088.  
  2089. xmmdeallocate    proc    xmm_handle:word
  2090.     mov    ah,0ah            ; Deallocate extended memory block
  2091.     mov    dx, xmm_handle        ; Give it handle
  2092.     call    [xmscontrol]
  2093.     ret
  2094. xmmdeallocate    endp
  2095.  
  2096. xmmmoveextended proc uses si, MoveStruct:word
  2097.  
  2098.     ; Call the XMS MoveExtended function.
  2099.     mov    ah,0Bh
  2100.     mov    si,MoveStruct            ; the move structure.
  2101.     call    [xmscontrol]            ;
  2102.  
  2103. ; The call to xmscontrol returns a 1 in AX if successful, 0 otherwise.
  2104.  
  2105.     ret
  2106.  
  2107. xmmmoveextended endp
  2108.  
  2109.  
  2110. ; ********************* IIT FPU Chip Support Routines ******************
  2111. ;             for use with 2C87 and 3C87 FPU chips
  2112. ;
  2113. ; load_mat(double matrix[16])     ; Load a 4x4 matrix of doubles into IIT
  2114. ;                 ; IIT registers
  2115. ;
  2116. ; mult_vec_iit(double vector[3]) ; Multiply matrix times vector. Routine
  2117. ;                 ; is not completely general - makes use of
  2118. ;                 ; the fact that Fractint 3D vectors always
  2119. ;                 ; have a fourth component of 1. Only three
  2120. ;                 ; array elements are actually accessed.
  2121. ;                 ; Source and target vectors are the same.
  2122. ;
  2123. ; IITCoPro()             ; Detect IIT chip - return 1. Do not call
  2124. ;                 ; unless at least a 287 already detected.
  2125. ;
  2126. ; Code adapted by Tim Wegner from IIT documentation and detect routine
  2127. ; sent by Jonathan Osuch and modified by Charles Marslett -- 01/29/91
  2128. ;
  2129. ; The following routines were provided by IIT to implement a semaphore
  2130. ; system to protect the IIT extra registers from multi-tasking:
  2131. ;
  2132. ; F4x4Check()              ; returns 1 if semaphore TSR loaded           
  2133. ; F4x4Lock()             ; returns 1 if semaphore free and locks
  2134. ; F4x4Free()             ; frees locked semaphore
  2135.  
  2136. .286
  2137. .287
  2138. .data
  2139. one dq 1.0
  2140. .code
  2141. ;
  2142. ;load_mat(double *array)
  2143. ;
  2144. load_mat proc    array:WORD
  2145.     finit
  2146.     db    0DBh,0EBh    ; select register set 0
  2147.     fwait
  2148.     mov    bx, array
  2149.     fld    QWORD PTR [bx+64 ] ; load row 3
  2150.     fld    QWORD PTR [bx+72 ]
  2151.     fld    QWORD PTR [bx+80 ]
  2152.     fld    QWORD PTR [bx+88 ]
  2153.     fld    QWORD PTR [bx+96 ] ; load row 4
  2154.     fld    QWORD PTR [bx+104]
  2155.     fld    QWORD PTR [bx+112]
  2156.     fld    QWORD PTR [bx+120]
  2157.  
  2158.     finit
  2159.     db    0DBh,0EAh    ; select register set 1
  2160.     fld    QWORD PTR [bx+0 ] ; load row 1
  2161.     fld    QWORD PTR [bx+8 ]
  2162.     fld    QWORD PTR [bx+16]
  2163.     fld    QWORD PTR [bx+24]
  2164.     fld    QWORD PTR [bx+32] ; load row 2
  2165.     fld    QWORD PTR [bx+40]
  2166.     fld    QWORD PTR [bx+48]
  2167.     fld    QWORD PTR [bx+56]
  2168.  
  2169.     finit
  2170.     db    0DBh,0E8h    ; select register set 0
  2171.     fwait
  2172.     ret
  2173. load_mat endp
  2174.  
  2175. .code
  2176. ;
  2177. ;mult_vec_iit(vector)
  2178. ;
  2179. mult_vec_iit proc uses bx, vector:WORD
  2180.     mov    bx,vector
  2181.     fld    one ; last component always 1 in fractint
  2182. ;    fld    QWORD PTR [bx+24 ] ; 4
  2183.     fld    QWORD PTR [bx+16 ] ; 3
  2184.     fld    QWORD PTR [bx+8  ] ; 2
  2185.     fld    QWORD PTR [bx+0  ] ; 1
  2186.  
  2187.     db    0DBh,0F1h    ; multiply the column vector
  2188.     fwait
  2189.     fstp    QWORD PTR [bx+0  ] ; 1
  2190.     fstp    QWORD PTR [bx+8  ] ; 2
  2191.     fstp    QWORD PTR [bx+16 ] ; 3
  2192. ;    fstp    QWORD PTR [bx+24 ] ; 4 vectors length 3 in Fractint
  2193.  
  2194.     fwait
  2195.     ret
  2196. mult_vec_iit endp
  2197.  
  2198. ;
  2199. ; IITCoPro()
  2200. ;
  2201.  
  2202. .data
  2203. testdata db 0FFh,0FFh,00h,00h,00h,00h,00h,00h,00h,00h
  2204. .code
  2205.  
  2206. IITCoPro proc
  2207.      finit
  2208.      fld   tbyte ptr testdata
  2209.      fstp  tbyte ptr temp
  2210.      fwait
  2211.      mov   ax,word ptr temp
  2212.      or    ax,ax             ; test for 1st word of result zero
  2213.      mov   ax,1             ; return 1 if next branch taken
  2214.      jz    must_be_IIT         ;  result was zero, is IIT
  2215.      xor   ax,ax             ; return 0
  2216. must_be_IIT:
  2217.      ret
  2218. IITCoPro endp
  2219.  
  2220. .8086
  2221. .8087
  2222.  
  2223. eIIT2fService        equ    0C0h        ; user services: 0C0h - 0FFh
  2224.  
  2225. ; services provided by int 2F
  2226. eInstallationCheck    equ    0        ; <== must be zero
  2227. eSetSemaphore        equ    1
  2228. eClearSemaphore        equ    2
  2229.  
  2230. .code
  2231. F4x4Check    PROC    FAR
  2232. ; IIT F4x4 semaphore installation check
  2233.     
  2234.     inc    bp
  2235.     push    bp
  2236.     mov    bp, sp
  2237.     mov    ax, (eIIT2fService SHL 8) + eInstallationCheck
  2238.     mov    bx, 'II'
  2239.     mov    cx, 'Ts'
  2240.     mov    dx, 'em'
  2241.     int    2Fh
  2242.     cmp    ax, (eIIT2fService SHL 8) + 0FFh
  2243.     jne    not_installed
  2244.     cmp    bx, 'OK'
  2245.     jne    not_installed
  2246.     cmp    cx, ' I'
  2247.     jne    not_installed
  2248.     cmp    dx, 'IT'
  2249.     jne    not_installed
  2250. installed:
  2251.     mov    ax, 1
  2252.     pop    bp
  2253.     dec    bp
  2254.     ret
  2255. not_installed:
  2256.     xor    ax, ax
  2257.     pop    bp
  2258.     dec    bp
  2259.     ret
  2260. F4x4Check    ENDP
  2261.  
  2262.  
  2263. F4x4Lock    PROC    FAR
  2264.     inc    bp
  2265.     push    bp
  2266.     mov    bp, sp
  2267.     mov    ax, (eIIT2fService SHL 8) + eSetSemaphore 
  2268.     mov    bx, 'II'
  2269.     mov    cx, 'Ts'
  2270.     mov    dx, 'em'
  2271.     int    2Fh
  2272.     pop    bp
  2273.     dec    bp
  2274.     ret
  2275. F4x4Lock    ENDP
  2276.  
  2277.  
  2278. F4x4Free    PROC    FAR
  2279.     inc    bp
  2280.     push    bp
  2281.     mov    bp, sp
  2282.     mov    ax, (eIIT2fService SHL 8) + eClearSemaphore
  2283.     mov    bx, 'II'
  2284.     mov    cx, 'Ts'
  2285.     mov    dx, 'em'
  2286.     int    2Fh
  2287.     pop    bp
  2288.     dec    bp
  2289.     ret
  2290. F4x4Free    ENDP
  2291.     END
  2292.  
  2293.