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

  1.     page    60,126
  2.  
  3. ; Name        vidirect -- Read or write rectangle directly from or to
  4. ;                video adapter
  5. ;
  6. ; Synopsis    ercode = vidirect(pfrom_ads,pto_ads,num_rows,row_length,
  7. ;               scn_width,attr,option);
  8. ;
  9. ;        int ercode     Error code:  1 if option is unknown,
  10. ;                          0 if okay.
  11. ;        ADS *pfrom_ads     Pointer to ADS structure containing
  12. ;                 address of source buffer
  13. ;        ADS *pto_ads     Pointer to ADS structure containing
  14. ;                 address of destination buffer
  15. ;        int num_rows     Number of rows in the rectangle
  16. ;        int row_length     Width of the rectangle (in columns)
  17. ;        int scn_width     Twice the screen width (80 or 160 bytes)
  18. ;        int attr     Foreground and background attributes
  19. ;                 (only used for certain options)
  20. ;        int option     Style and direction of I/O; also
  21. ;                 whether interference is to be prevented.
  22. ;                 See discussion and table below.
  23. ;
  24. ; Description    This function reads or writes a rectangular buffer
  25. ;        directly from or to the video memory on an IBM
  26. ;        Color/Graphics Adapter, Monochrome Adapter, or
  27. ;        compatible adapter.  It can do so with attention to the
  28. ;        horizontal and vertical retrace intervals for maximum
  29. ;        throughput without interference.  The characters are
  30. ;        transferred row by row.
  31. ;
  32. ;        The buffer addresses are stored in ADS structures
  33. ;        pointed to by pfrom_ads and pto_ads.  Either or both of
  34. ;        these may point to physical locations in video memory,
  35. ;        depending on the value of option.
  36. ;
  37. ;        Several styles of I/O are available.  They are selected
  38. ;        by the value of option.  Bit 15 of option, if set,
  39. ;        indicates that the transfer is to be performed without
  40. ;        regard to interference.  The remaining bits of option
  41. ;        form an integer which should be selected from the
  42. ;        following table:
  43. ;
  44. ;        Option    Description
  45. ;        ------    -----------
  46. ;           0    Write characters to the screen without attributes.
  47. ;            *pfrom_ads points to a buffer of characters.
  48. ;           1    Write characters to the screen with attributes.
  49. ;            *pfrom_ads points to a buffer of (char,attr) pairs.
  50. ;           2    Write characters to the screen with constant
  51. ;                attribute specified by attr.
  52. ;            *pfrom_ads points to a buffer of characters.
  53. ;           3    Fill screen rectangle with constant char & attr.
  54. ;            *pfrom_ads points to the character.
  55. ;            Attr specifies the attribute.
  56. ;           4    Fill screen rectangle with constant attribute
  57. ;                without altering characters displayed.
  58. ;            pfrom_ads is ignored.
  59. ;           5    Read characters from screen without attributes.
  60. ;            **pto_ads is filled with the characters.  (Note
  61. ;                that no trailing NUL is added.)
  62. ;           6    Read characters from screen with attributes.
  63. ;            **pto_ads is filled with (char,attr) pairs.
  64. ;           7    Copy window (with attributes) upward or directly
  65. ;                leftward on same display page.
  66. ;           8    Copy window (without attributes) upward or
  67. ;                directly leftward on same display page.
  68. ;           9    Copy window (with attributes) downward or
  69. ;                directly rightward on same display page.
  70. ;          10    Copy window (without attributes) downward or
  71. ;                directly rightward on same display page.
  72. ;          11    Copy window (with attributes) to non-overlapping
  73. ;                window or to another display page.
  74. ;          12    Copy window (without attributes) to non-
  75. ;                overlapping window or to another display page.
  76. ;
  77. ;        Beware:  None of the arguments is checked for
  78. ;        correctness (except that option is checked for
  79. ;        out-of-range values)!
  80. ;
  81. ;        If bit 15 of option is clear (indicating protection
  82. ;        against interference), interrupts will be left enabled
  83. ;        when this procedure exits.
  84. ;
  85. ; Returns    ercode          Error code:  1 if option is unknown,
  86. ;                           0 if okay.
  87. ;        **pto_ads      Data in destination buffer
  88. ;
  89. ; Version    3.0 (C)Copyright Blaise Computing Inc.    1986
  90.  
  91.     name     vidirect
  92.  
  93. video_status_port   equ     3DAh
  94.  
  95.     LONGPROG  = 0              ; initialize constants for
  96.     LONGDATA  = 0              ; Pass1 of the assembler
  97.  
  98.     include   compiler.mac          ; Specifies the C compiler
  99.  
  100.     if LAT200 or LAT210 or LAT300
  101.     include  dos.mac
  102.     LONGPROG = LPROG
  103.     LONGDATA = LDATA
  104.  
  105.     pseg
  106.     public     vidirect
  107.     if     LPROG
  108.     x   equ  4            ; parameter offset
  109.     else
  110.     x   equ  2
  111.     endif
  112.     if     LDATA
  113.     a   equ  2            ; Location of parameters
  114.     b   equ  6            ; on the stack.  Offset is
  115.     c   equ  10            ; pointer size.
  116.     d   equ  12
  117.     e   equ  14
  118.     f   equ  16
  119.     g   equ  18
  120.     else
  121.     a   equ  2
  122.     b   equ  4
  123.     c   equ  6
  124.     d   equ  8
  125.     e   equ  10
  126.     f   equ  12
  127.     g   equ  14
  128.     endif
  129.     endif
  130.  
  131.  
  132.     if MSC300
  133.     include  dos.mac
  134.     LONGPROG = LPROG
  135.     LONGDATA = LDATA
  136.  
  137.     pseg     vidirect
  138.     public     _vidirect
  139.     if     LPROG
  140.     x   equ  4            ; parameter offset
  141.     else
  142.     x   equ  2
  143.     endif
  144.  
  145.     if     LDATA
  146.     a   equ  2            ; Location of parameters
  147.     b   equ  6            ; on the stack.  Offset is
  148.     c   equ  10            ; pointer size.
  149.     d   equ  12
  150.     e   equ  14
  151.     f   equ  16
  152.     g   equ  18
  153.     else
  154.     a   equ  2
  155.     b   equ  4
  156.     c   equ  6
  157.     d   equ  8
  158.     e   equ  10
  159.     f   equ  12
  160.     g   equ  14
  161.     endif
  162.     endif
  163.  
  164. ;   Symbols to aid in referencing arguments on stack
  165. ;   (Some of these items are also used as local variables.):
  166.  
  167. pfrom_ads   equ           [bp + x + a]
  168. pto_ads     equ           [bp + x + b]
  169. num_rows    equ     byte  ptr [bp + x + c]
  170. row_length  equ     byte  ptr [bp + x + d]
  171. scn_width   equ     word  ptr [bp + x + e]
  172. attr        equ     byte  ptr [bp + x + f]
  173. option        equ     word  ptr [bp + x + g]
  174.  
  175. ;   Macro to wait until the VERY BEGINNING of a horizontal
  176. ;   retrace interval.  On exit, interrupts are off.
  177.  
  178. await_horiz_retrace macro
  179.     local    await_display_on,await_display_off
  180.     sti
  181.     nop
  182.     cli
  183. await_display_on:
  184.     in    al,dx
  185.     shr    al,1
  186.     jc    await_display_on
  187. await_display_off:
  188.     in    al,dx
  189.     shr    al,1
  190.     jnc    await_display_off
  191.     endm
  192.  
  193. ;   Here are macros to define procedures for each style of I/O as
  194. ;    specified by "option".
  195. ;
  196. ;   Each style of I/O requires four procedures:
  197. ;    (1) a specialized setup procedure ("setup_proc") (which may
  198. ;        do nothing)
  199. ;    (2) a procedure ("slow_proc") to do the entire read or write
  200. ;        while avoiding video interference;
  201. ;    (3) a procedure ("fast_proc") to do the entire read or write
  202. ;        without regard to interference (i.e., as fast as possible);
  203. ;    (4) a procedure ("newrow_proc") to adjust address registers
  204. ;        (SI and/or DI) and the column counter (CX) for a new row
  205. ;        of the rectangle.
  206. ;   These four procedures will be addressed through table "option_table"
  207. ;    defined below.
  208. ;   Since these procedures must therefore be highly regular in form,
  209. ;    the following macros aid in defining the beginning and end of
  210. ;    each.
  211.  
  212. begin_setup_proc    macro   o_name  ;; Define beginning of setup procedure
  213. o_name&_setup_proc  proc    near
  214.             endm
  215.  
  216. end_setup_proc        macro   o_name  ;; Define end of setup procedure
  217.             ret
  218. o_name&_setup_proc  endp
  219.             endm
  220.  
  221.                 ; On entry to each slow_proc,
  222.                 ;
  223.                 ;    BX = address of option_table entry;
  224.                 ;    CX = number of cells in one row;
  225.                 ;    DX = I/O address of video status port.
  226.                 ;
  227.                 ; Also, usually
  228.                 ;    DS:SI = address of first source byte;
  229.                 ;    ES:DI = address of first target byte.
  230.                 ;
  231.                 ; Additional entry conditions are satisfied
  232.                 ;    by the setup_proc.
  233.  
  234. begin_slow_proc     macro   o_name,save_bx_flag
  235. o_name&_slow_proc   proc    near
  236.             ifidn   <save_bx_flag>,<save_bx>
  237.             push    bx        ;; Some options must use BX as
  238.                     ;; additional storage space, so
  239.                     ;; save BX if requested.
  240.             endif
  241. o_name&_single_loop:            ;; Begin loop that transfers one
  242.                     ;; character cell.
  243.             endm
  244.  
  245. end_slow_proc        macro   o_name,save_bx_flag
  246.             local   done
  247.             loop    o_name&_single_loop ;; Transfer one more cell
  248.                         ;; in this row.
  249.  
  250.             sti
  251.             ifidn   <save_bx_flag>,<save_bx>
  252.             pop    bx        ;; Restore BX if it was used for
  253.                     ;; temporary data.
  254.             endif
  255.             dec     num_rows
  256.             jz        done
  257.             call    cs:[bx].do_newrow    ;; Use BX to invoke
  258.                         ;; proper newrow_proc.
  259.             jmp     short o_name&_slow_proc ;; Do next row.
  260. done:            ret                 ;; Done.
  261. o_name&_slow_proc   endp
  262.             endm
  263.  
  264.                 ; Each fast_proc has the same entry
  265.                 ; conditions as slow_proc.
  266.  
  267. begin_fast_proc     macro   o_name
  268. o_name&_fast_proc   proc    near
  269.             endm
  270.  
  271. end_fast_proc        macro   o_name
  272.             local   done
  273.  
  274.             dec     num_rows
  275.             jz        done
  276.             call    cs:[bx].do_newrow    ;; Use BX to invoke
  277.                         ;; proper newrow_proc.
  278.             jmp     short o_name&_fast_proc
  279. done:            ret
  280. o_name&_fast_proc   endp
  281.             endm
  282.  
  283.                 ; Each newrow_proc assumes CH == 0.
  284.  
  285. begin_newrow_proc   macro   o_name
  286. o_name&_newrow_proc proc    near
  287.             mov     cl,row_length
  288.             endm
  289.  
  290. end_newrow_proc     macro   o_name
  291.             ret
  292. o_name&_newrow_proc endp
  293.             endm
  294.  
  295. ; *************************************************************
  296. ; *********   DEFINE FOUR PROCEDURES FOR EACH OPTION **********
  297. ; *************************************************************
  298. ;
  299. ;   (Note:  many of these definitions just refer to others using "equ".)
  300.  
  301. ;   W_CHARS:    Write characters to the screen without arguments.
  302.  
  303.     begin_setup_proc    w_chars     ; No setup needed.
  304.     end_setup_proc    w_chars
  305.  
  306.     begin_slow_proc    w_chars
  307.      await_horiz_retrace
  308.      movsb
  309.      inc     di
  310.     end_slow_proc    w_chars
  311.  
  312.     begin_fast_proc    w_chars
  313. w_chars_fast_loop:
  314.      movsb
  315.      inc     di
  316.      loop     w_chars_fast_loop
  317.     end_fast_proc    w_chars
  318.  
  319.     begin_newrow_proc    w_chars
  320.      sub     di,cx
  321.      sub     di,cx
  322.      add     di,scn_width
  323.     end_newrow_proc    w_chars
  324.  
  325. ;   W_CHAR_ATTRS:  Write characters & attributes to the screen.
  326.  
  327. w_char_attrs_setup_proc     equ     w_chars_setup_proc ; No setup needed.
  328.  
  329.     begin_slow_proc    w_char_attrs,save_bx
  330.      lodsw                ; Insufficient time for a
  331.      mov     bx,ax            ; whole "movsw", so do the job
  332.      await_horiz_retrace        ; by a "lodsw" plus a "stosw".
  333.      mov     ax,bx
  334.      stosw
  335.     end_slow_proc    w_char_attrs,save_bx
  336.  
  337.     begin_fast_proc    w_char_attrs
  338.      rep  movsw
  339.     end_fast_proc    w_char_attrs
  340.  
  341. w_char_attrs_newrow_proc    equ     w_chars_newrow_proc
  342.  
  343. ;   CONST_ATTR:  Write characters to screen with constant attribute
  344. ;         specified in AH.
  345.  
  346.     begin_setup_proc    const_attr
  347.      mov     ah,attr
  348.     end_setup_proc    const_attr
  349.  
  350.  
  351.     begin_slow_proc    const_attr,save_bx
  352.      lodsb                ; Insufficient time for "lodsb"
  353.      mov     bl,al            ; plus "stosw", so do the job
  354.      await_horiz_retrace        ; in halves.
  355.      mov     al,bl
  356.      stosw
  357.     end_slow_proc    const_attr,save_bx
  358.  
  359.     begin_fast_proc    const_attr
  360. const_attr_fast_loop:
  361.      lodsb
  362.      stosw
  363.      loop     const_attr_fast_loop
  364.     end_fast_proc    const_attr
  365.  
  366. const_attr_newrow_proc        equ     w_chars_newrow_proc
  367.  
  368. ;   CONST_CHAR_ATTR:  Fill rectangle with constant character
  369. ;              and attribute (stored in SI).
  370.  
  371.     begin_setup_proc    const_char_attr
  372.      mov    ah,attr
  373.      mov    al,[si]
  374.      mov    si,ax         ; Keep char & attr in SI for the loop
  375.     end_setup_proc    const_char_attr
  376.  
  377.     begin_slow_proc    const_char_attr
  378.      await_horiz_retrace
  379.      mov    ax,si
  380.      stosw
  381.     end_slow_proc    const_char_attr
  382.  
  383.     begin_fast_proc    const_char_attr
  384.      mov    ax,si
  385.      rep    stosw
  386.     end_fast_proc    const_char_attr
  387.  
  388. const_char_attr_newrow_proc   equ   w_chars_newrow_proc
  389.  
  390. ;   CONST_ATTR_ONLY:  Change attribute on rectangle to value
  391. ;              stored in AH.
  392.  
  393.     begin_setup_proc    const_attr_only
  394.      mov    ah,attr
  395.     end_setup_proc    const_attr_only
  396.  
  397.     begin_slow_proc    const_attr_only
  398.      await_horiz_retrace
  399.      inc    di
  400.      mov    al,ah
  401.      stosb
  402.     end_slow_proc    const_attr_only
  403.  
  404.     begin_fast_proc    const_attr_only
  405.      mov    al,ah
  406. const_attr_only_fast_loop:
  407.      inc    di
  408.      stosb
  409.      loop    const_attr_only_fast_loop
  410.     end_fast_proc    const_attr_only
  411.  
  412. const_attr_only_newrow_proc   equ   w_chars_newrow_proc
  413.  
  414. ;   R_CHARS:  Read only characters from screen.
  415.  
  416. r_chars_setup_proc     equ     w_chars_setup_proc ; No setup needed.
  417.  
  418.     begin_slow_proc    r_chars
  419.      await_horiz_retrace
  420.      movsb
  421.      inc    si
  422.     end_slow_proc    r_chars
  423.  
  424.     begin_fast_proc    r_chars
  425. r_chars_fast_loop:
  426.      movsb
  427.      inc    si
  428.      loop    r_chars_fast_loop
  429.     end_fast_proc    r_chars
  430.  
  431.     begin_newrow_proc    r_chars
  432.      sub    si,cx
  433.      sub    si,cx
  434.      add    si,scn_width
  435.     end_newrow_proc    r_chars
  436.  
  437. ;   R_CHAR_ATTRS:  Read characters & attributes from screen.
  438.  
  439. r_char_attrs_setup_proc     equ     w_chars_setup_proc ; No setup needed.
  440.  
  441.     begin_slow_proc    r_char_attrs
  442.      await_horiz_retrace
  443.      movsw
  444.     end_slow_proc    r_char_attrs
  445.  
  446. r_char_attrs_fast_proc        equ     w_char_attrs_fast_proc
  447. r_char_attrs_newrow_proc    equ     r_chars_newrow_proc
  448.  
  449. ;   UP_CHAR_ATTRS:  Copy characters & attributes upward on the screen
  450. ;            or directly to the left.  Start at upper left
  451. ;            corner of rectangle.
  452.  
  453. up_char_attrs_setup_proc     equ     w_chars_setup_proc ; No setup needed.
  454.  
  455.     begin_slow_proc    up_char_attrs,save_bx
  456.      await_horiz_retrace
  457.      lodsw                ; Insufficient time for a
  458.      mov     bx,ax            ; whole "movsw", so do the job
  459.      await_horiz_retrace        ; by a "lodsw" plus a "stosw".
  460.      mov     ax,bx
  461.      stosw
  462.     end_slow_proc    up_char_attrs,save_bx
  463.  
  464. up_char_attrs_fast_proc      equ     w_char_attrs_fast_proc
  465.  
  466.     begin_newrow_proc    up_char_attrs
  467.      sub    si,cx
  468.      sub    si,cx
  469.      add    si,scn_width
  470.      sub    di,cx
  471.      sub    di,cx
  472.      add    di,scn_width
  473.     end_newrow_proc    up_char_attrs
  474.  
  475. ;   UP_CHAR_ATTRS:  Copy only characters upward on the screen
  476. ;            or directly to the left.  Start at upper left
  477. ;            corner of rectangle.
  478.  
  479. up_chars_setup_proc    equ    w_chars_setup_proc ; No setup needed.
  480.  
  481.     begin_slow_proc    up_chars
  482.     await_horiz_retrace
  483.     movsb
  484.     inc    si
  485.     inc    di
  486.     end_slow_proc    up_chars
  487.  
  488.     begin_fast_proc    up_chars
  489. up_chars_fast_loop:
  490.     movsb
  491.     inc    si
  492.     inc    di
  493.     loop    up_chars_fast_loop
  494.     end_fast_proc    up_chars
  495.  
  496. up_chars_newrow_proc    equ    up_char_attrs_newrow_proc
  497.  
  498. ;   DOWN_CHAR_ATTRS:  Copy characters & attributes downward on the screen
  499. ;              or directly to the right.  Start at lower right
  500. ;              corner of rectangle.
  501.  
  502.     begin_setup_proc    down_char_attrs
  503.     mov    al,num_rows    ; Adjust SI and DI to point
  504.     dec    al        ; at the last character in the
  505.     mov    cx,scn_width    ; rectangle instead of the first.
  506.     mul    cl
  507.     mov    cl,row_length
  508.     mov    ch,0
  509.     dec    cx
  510.     add    cx,cx
  511.     add    ax,cx
  512.  
  513.     add    si,ax
  514.     add    di,ax
  515.     end_setup_proc    down_char_attrs
  516.  
  517. down_char_attrs_slow_proc   equ     up_char_attrs_slow_proc
  518. down_char_attrs_fast_proc   equ     up_char_attrs_fast_proc
  519.  
  520.     begin_newrow_proc    down_char_attrs
  521.     add    si,cx
  522.     add    si,cx
  523.     sub    si,scn_width
  524.     add    di,cx
  525.     add    di,cx
  526.     sub    di,scn_width
  527.     end_newrow_proc    down_char_attrs
  528.  
  529. ;   DOWN_CHARS:  Copy only characters downward on the screen
  530. ;         or directly to the right.  Start at lower right
  531. ;         corner of rectangle.
  532.  
  533. down_chars_setup_proc      equ      down_char_attrs_setup_proc
  534.  
  535.     begin_slow_proc    down_chars
  536.     await_horiz_retrace
  537.     movsb
  538.     dec    si
  539.     dec    di
  540.     end_slow_proc    down_chars
  541.  
  542.     begin_fast_proc    down_chars
  543. down_chars_fast_loop:
  544.     movsb
  545.     dec    si
  546.     dec    di
  547.     loop    down_chars_fast_loop
  548.     end_fast_proc    down_chars
  549.  
  550. down_chars_newrow_proc        equ     down_char_attrs_newrow_proc
  551.  
  552. ; *******************************************************************
  553. ; **********   DEFINE THE TABLE OF OPTION INFORMATION  **************
  554. ; *******************************************************************
  555.  
  556. ;   The following is the format of the table entry for one option:
  557.  
  558. option_entry    struc
  559. do_setup    dw    ?   ; Address of setup_proc.
  560. do_slow     dw    ?   ; Address of slow_proc.
  561. do_fast     dw    ?   ; Address of fast_proc.
  562. do_newrow    dw    ?   ; Address of newrow_proc.
  563. dirflag     db    ?   ; Direction:  down or up.
  564. option_entry    ends
  565.  
  566. ;   Values for dirflag:
  567.  
  568. down    equ    1        ; Decreasing addresses (up the screen)
  569. up    equ    0        ; Increasing addresses (down the screen)
  570.  
  571. ;   Macro to create one table entry:
  572.  
  573. o_e        macro    o_name,dir
  574. option_entry <o_name&_setup_proc,o_name&_slow_proc,o_name&_fast_proc,o_name&_newrow_proc,dir>
  575.         endm
  576.  
  577. ;   The table itself follows.  It's part of the code segment.
  578.  
  579. option_table:
  580.         o_e    w_chars,up         ; 0
  581. end_of_first    label    option_entry
  582.         o_e    w_char_attrs,up      ; 1
  583.         o_e    const_attr,up         ; 2
  584.         o_e    const_char_attr,up   ; 3
  585.         o_e    const_attr_only,up   ; 4
  586.         o_e    r_chars,up         ; 5
  587.         o_e    r_char_attrs,up      ; 6
  588.         o_e    up_char_attrs,up     ; 7
  589.         o_e    up_chars,up         ; 8
  590.         o_e    down_char_attrs,down ; 9
  591.         o_e    down_chars,down      ; 10
  592.         o_e    up_char_attrs,up     ; 11
  593.         o_e    up_chars,up         ; 12
  594.  
  595.                 ; (Note that options 11 and 12 just
  596.                 ; refer to the procedures used by options
  597.                 ; 7 and 8, respectively, because the
  598.                 ; same algorithms suffice.)
  599.  
  600. end_of_table    label    option_entry
  601.  
  602. size_of_entry    equ    end_of_first - option_table
  603. table_length    equ    (end_of_table - option_table)/size_of_entry
  604.  
  605.  
  606. ; ********************************************************************
  607. ; **********************   BEGIN THE ACTUAL CODE   *******************
  608. ; ********************************************************************
  609.  
  610.     if    MSC300
  611.     if    LONGPROG
  612. _vidirect   proc    far
  613.     else
  614. _vidirect   proc    near
  615.     endif
  616.     else
  617.     if    LONGPROG
  618. vidirect    proc    far
  619.     else
  620. vidirect    proc    near
  621.     endif
  622.     endif
  623.  
  624.     push    bp           ; Save the frame pointer
  625.     mov    bp,sp
  626.     if    MSC300
  627.     push    di           ; Save register variables
  628.     push    si
  629.     endif
  630.  
  631.     push    ds
  632.     push    es
  633.  
  634.     mov    ax,option
  635.     mov    dx,ax            ; Spare copy of raw option
  636.     and    ax,7fffh        ; Ignore bit 15 of option.
  637.  
  638.     cmp    ax,table_length     ; Check for option value beyond table.
  639.     jb    option_ok
  640.  
  641. bad_option:
  642.     mov    ax,1            ; Invalid option
  643.     jmp    exit
  644.  
  645. option_ok:
  646.     mov    cl,size_of_entry    ; Convert option value into address
  647.     mul    cl            ; of proper table entry.
  648.     jc    bad_option
  649.     add    ax,offset option_table
  650.     mov    option,ax        ; Save address of table entry.
  651.  
  652.     mov    bx,ax            ; Set DF according to dirflag value
  653.     cmp    cs:[bx].dirflag,down; in table.
  654.     je    downward
  655.  
  656.     cld
  657.     jmp    short have_set_direction
  658.  
  659. downward:
  660.     std
  661.  
  662. have_set_direction:
  663.  
  664.     if    LONGDATA
  665.     lds    bx,pto_ads
  666.     assume    ds:nothing
  667.     else
  668.     mov    bx,pto_ads
  669.     endif
  670.     les    di,[bx]         ; Put target address into ES:DI.
  671.     assume    es:nothing
  672.  
  673.     if    LONGDATA
  674.     lds    bx,pfrom_ads
  675.     assume    ds:nothing
  676.     else
  677.     mov    bx,pfrom_ads
  678.     endif
  679.     lds    si,[bx]         ; Put source address into DS:SI.
  680.     assume    ds:nothing
  681.  
  682.     mov    bx,option
  683.  
  684.     call    cs:[bx].do_setup    ; Do specialized portion
  685.                     ; of setup.
  686.  
  687.                     ; Complete the setup.
  688.     mov    cl,row_length
  689.     mov    ch,0
  690.     test    dh,80h
  691.     mov    dx,video_status_port
  692.     jnz    no_waiting
  693.  
  694.     call    cs:[bx].do_slow     ; Invoke slow_proc.
  695.     jmp    short done
  696.  
  697. no_waiting:
  698.     call    cs:[bx].do_fast     ; Invoke fast_proc.
  699.  
  700. done:
  701.     xor    ax,ax            ; Successful completion
  702.  
  703. exit:
  704.     sti
  705.     pop    es
  706.     pop    ds
  707.  
  708.     if    MSC300
  709.     pop    si            ; Recover index registers
  710.     pop    di            ; (used for register variables)
  711.     cld                ; MSC 3 expects DF cleared
  712.     endif
  713.  
  714.     pop    bp            ; Get the original frame pointer.
  715.     ret
  716.  
  717.     if    MSC300
  718. _vidirect   endp
  719.     else
  720. vidirect    endp
  721.     endif
  722.  
  723.     if LAT200 or LAT210 or LAT300
  724.     endps
  725.     endif
  726.  
  727.     if MSC300
  728.     endps        vidirect
  729.     endif
  730.  
  731.     end
  732.